initial prototype complete

This commit is contained in:
2024-07-24 19:29:18 +02:00
parent b11271cf82
commit 610f94d1d4
10 changed files with 313 additions and 141 deletions

View File

@@ -6,7 +6,8 @@ use clap::{Args, Parser, Subcommand};
author = "Michał Czyż", author = "Michał Czyż",
version = "0.1.0", version = "0.1.0",
about = "A declarative package manager for Arch Linux", about = "A declarative package manager for Arch Linux",
long_about = "Arch Helper is a declarative package management tool for Arch Linux. It leverages paru or other package managers for seamless integration.")] long_about = "Arch Helper is a declarative package management tool for Arch Linux. It leverages paru or other package managers for seamless integration."
)]
pub struct Cli { pub struct Cli {
#[command(subcommand)] #[command(subcommand)]
pub command: Option<Commands>, pub command: Option<Commands>,
@@ -14,25 +15,25 @@ pub struct Cli {
#[derive(Subcommand)] #[derive(Subcommand)]
pub enum Commands { pub enum Commands {
#[command(about = "Install packages")] #[command(alias = "i", about = "Install packages")]
Install(PackageList), Install(PackageList),
#[command(about = "Upgrade packages")] #[command(alias = "u", about = "Upgrade packages")]
Upgrade { Upgrade {
#[arg(help = "Don't prompt for confirmation", default_value_t = false)] #[arg(help = "Don't prompt for confirmation", default_value_t = false)]
noconfirm: bool noconfirm: bool,
}, },
#[command(about = "Synchronize packages")] #[command(alias = "s", about = "Synchronize packages")]
Sync { Sync {
#[arg(help = "Don't prompt for confirmation", default_value_t = false)] #[arg(help = "Don't prompt for confirmation", default_value_t = false)]
noconfirm: bool noconfirm: bool,
}, },
#[command(about = "Remove packages")] #[command(alias = "r", about = "Remove packages")]
Remove(PackageList), Remove(PackageList),
#[command(about = "Find packages")] #[command(alias = "f", about = "Find packages")]
Find(Query), Find(Query),
} }

View File

@@ -1,6 +1,7 @@
use std::{ use std::{
fs::File, fs::{File, OpenOptions},
io::{prelude::*, BufReader}, path::PathBuf, io::{prelude::*, BufReader},
path::PathBuf,
}; };
// pub fn read_config(path: &str) -> Vec<String> { // pub fn read_config(path: &str) -> Vec<String> {
@@ -15,9 +16,31 @@ pub fn read_packages(path: PathBuf) -> Vec<String> {
let file = File::open(path).expect("Failed to open file"); let file = File::open(path).expect("Failed to open file");
let buf = BufReader::new(file); let buf = BufReader::new(file);
buf.lines().map(|l| l.expect("Failed to read line")).collect() buf.lines()
.map(|l| l.expect("Failed to read line"))
.collect()
} }
// pub fn write_packages(path: &str, content: &str) { pub fn append_package(path: PathBuf, package: &str) {
// todo!(); let mut file = OpenOptions::new()
// } .write(true)
.append(true)
.open(path)
.expect("Failed to open file");
if let Err(err) = writeln!(file, "{}", package) {
eprintln!("Couldn't write to file: {}", err);
}
}
pub fn write_packages(path: PathBuf, content: &str) {
let mut file = OpenOptions::new()
.write(true)
.truncate(true)
.open(path)
.expect("Failed to open file");
if let Err(err) = writeln!(file, "{}", content) {
eprintln!("Couldn't write to file: {}", err);
}
}

View File

@@ -3,8 +3,8 @@ use cli::{PackageList, Query};
use colored::Colorize; use colored::Colorize;
mod cli; mod cli;
mod packages;
mod file; mod file;
mod packages;
fn main() { fn main() {
let cli = cli::Cli::parse(); let cli = cli::Cli::parse();

View File

@@ -4,7 +4,11 @@ use std::process::Command;
const PACKAGE_MANAGER: &str = "paru"; const PACKAGE_MANAGER: &str = "paru";
pub fn find(query: Vec<String>) -> Result<(), Box<dyn std::error::Error>> { pub fn find(query: Vec<String>) -> Result<(), Box<dyn std::error::Error>> {
println!("{} {}", "::".bold().green(), "Looking for package...".bold()); println!(
"{} {}",
"::".bold().green(),
"Looking for package...".bold()
);
if query.is_empty() { if query.is_empty() {
return Err("No query provided".into()); return Err("No query provided".into());

View File

@@ -1,6 +1,62 @@
use crate::{file, packages::get_package_path};
use colored::Colorize; use colored::Colorize;
use std::io::Write;
use std::process::{Command, Stdio};
pub fn install(_package: Vec<String>) -> Result<(), Box<dyn std::error::Error>> { const PACKAGE_MANAGER: &str = "paru";
println!("{} {}", "::".bold().green(), "Installing packages...".bold());
todo!(); pub fn install(new_packages: Vec<String>) -> Result<(), Box<dyn std::error::Error>> {
println!(
"{} {}",
"::".bold().green(),
"Installing packages...".bold()
);
let packages = file::read_packages(get_package_path());
let packages = packages
.into_iter()
.filter(|p| !p.contains("#") && !p.is_empty())
.collect::<Vec<String>>();
let mut child = Command::new(PACKAGE_MANAGER)
.arg("--color")
.arg("always")
.arg("-S")
.arg("--needed")
// .arg(noconfirm)
.arg("-")
.stdin(Stdio::piped())
.stdout(Stdio::inherit())
.stderr(Stdio::inherit())
.spawn()
.expect("Failed to execute command");
if let Some(mut stdin) = child.stdin.take() {
for new_package in new_packages.clone() {
writeln!(stdin, "{}", new_package).unwrap();
}
}
let status = child.wait().expect("Failed to wait on child");
if !status.success() {
return Err("Failed to install packages".into());
}
println!("{} {}", "::".bold().green(), "Packages installed".bold());
println!(
"{} {}",
"::".bold().blue(),
"Updating package index...".bold()
);
for new_package in new_packages {
if !packages.contains(&new_package) {
file::append_package(get_package_path(), &new_package);
}
}
Ok(())
} }

View File

@@ -1,20 +1,20 @@
use colored::Colorize; use colored::Colorize;
use std::path::PathBuf;
use std::io::{self, Write}; use std::io::{self, Write};
use std::path::PathBuf;
pub mod find;
pub mod install;
pub mod rebuild; pub mod rebuild;
pub mod remove;
pub mod sync; pub mod sync;
pub mod upgrade; pub mod upgrade;
pub mod install;
pub mod remove;
pub mod find;
pub use find::find;
pub use install::install;
pub use rebuild::rebuild; pub use rebuild::rebuild;
pub use remove::remove;
pub use sync::sync; pub use sync::sync;
pub use upgrade::upgrade; pub use upgrade::upgrade;
pub use install::install;
pub use remove::remove;
pub use find::find;
fn get_package_path() -> PathBuf { fn get_package_path() -> PathBuf {
let home_dir = std::env::var("HOME").unwrap(); let home_dir = std::env::var("HOME").unwrap();
@@ -23,7 +23,11 @@ fn get_package_path() -> PathBuf {
} }
fn ask_confirmation() -> Result<bool, io::Error> { fn ask_confirmation() -> Result<bool, io::Error> {
print!("{} {}", "::".bold().blue(), "Do you want to continue? [Y/n] "); print!(
"{} {}",
"::".bold().blue(),
"Do you want to continue? [Y/n] "
);
io::stdout().flush()?; io::stdout().flush()?;
let mut input = String::new(); let mut input = String::new();

View File

@@ -1,6 +1,6 @@
use colored::Colorize; use colored::Colorize;
use std::process::{Command, Stdio};
use std::io::Write; use std::io::Write;
use std::process::{Command, Stdio};
use crate::file; use crate::file;
use crate::packages::{ask_confirmation, get_package_path}; use crate::packages::{ask_confirmation, get_package_path};
@@ -8,7 +8,11 @@ use crate::packages::{ask_confirmation, get_package_path};
const PACKAGE_MANAGER: &str = "paru"; const PACKAGE_MANAGER: &str = "paru";
pub fn rebuild(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> { pub fn rebuild(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> {
println!("{} {}", "::".bold().green(), "Upgrading & syncing packages...".bold()); println!(
"{} {}",
"::".bold().green(),
"Upgrading & syncing packages...".bold()
);
if !ask_confirmation()? { if !ask_confirmation()? {
return Err("Operation aborted".into()); return Err("Operation aborted".into());
@@ -16,11 +20,16 @@ pub fn rebuild(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> {
let packages = file::read_packages(get_package_path()); let packages = file::read_packages(get_package_path());
let packages = packages.into_iter() let packages = packages
.into_iter()
.filter(|p| !p.contains("#") && !p.is_empty()) .filter(|p| !p.contains("#") && !p.is_empty())
.collect::<Vec<String>>(); .collect::<Vec<String>>();
let noconfirm = if noconfirm { "--noconfirm" } else { "--confirm" }; let noconfirm = if noconfirm {
"--noconfirm"
} else {
"--confirm"
};
let mut child = Command::new(PACKAGE_MANAGER) let mut child = Command::new(PACKAGE_MANAGER)
.arg("--color") .arg("--color")
@@ -47,6 +56,10 @@ pub fn rebuild(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> {
return Err("Failed to upgrade & sync packages".into()); return Err("Failed to upgrade & sync packages".into());
} }
println!("{} {}", "::".bold().green(), "Packages upgraded & synced".bold()); println!(
"{} {}",
"::".bold().green(),
"Packages upgraded & synced".bold()
);
Ok(()) Ok(())
} }

View File

@@ -1,6 +1,58 @@
use crate::{file, packages::get_package_path};
use colored::Colorize; use colored::Colorize;
use std::io::Write;
use std::process::{Command, Stdio};
pub fn remove(_package: Vec<String>) -> Result<(), Box<dyn std::error::Error>> { const PACKAGE_MANAGER: &str = "paru";
pub fn remove(unwanted_packages: Vec<String>) -> Result<(), Box<dyn std::error::Error>> {
println!("{} {}", "::".bold().green(), "Removing packages...".bold()); println!("{} {}", "::".bold().green(), "Removing packages...".bold());
todo!();
let packages = file::read_packages(get_package_path());
let mut packages = packages
.into_iter()
.filter(|p| !p.contains("#") && !p.is_empty())
.collect::<Vec<String>>();
let mut child = Command::new(PACKAGE_MANAGER)
.arg("--color")
.arg("always")
.arg("-R")
// .arg("--needed")
// .arg(noconfirm)
.arg("-")
.stdin(Stdio::piped())
.stdout(Stdio::inherit())
.stderr(Stdio::inherit())
.spawn()
.expect("Failed to execute command");
if let Some(mut stdin) = child.stdin.take() {
for unwanted_package in unwanted_packages.clone() {
writeln!(stdin, "{}", unwanted_package).unwrap();
}
}
let status = child.wait().expect("Failed to wait on child");
if !status.success() {
return Err("Failed to remove packages".into());
}
println!("{} {}", "::".bold().green(), "Packages removed".bold());
println!(
"{} {}",
"::".bold().blue(),
"Updating package index...".bold()
);
for unwanted_package in unwanted_packages {
packages.retain(|p| *p != unwanted_package);
}
file::write_packages(get_package_path(), &packages.join("\n"));
Ok(())
} }

View File

@@ -1,22 +1,31 @@
use colored::Colorize; use colored::Colorize;
use std::process::{Command, Stdio};
use std::io::Write; use std::io::Write;
use std::process::{Command, Stdio};
use crate::file; use crate::file;
use crate::packages::get_package_path; use crate::packages::{ask_confirmation, get_package_path};
const PACKAGE_MANAGER: &str = "paru"; const PACKAGE_MANAGER: &str = "paru";
pub fn sync(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> { pub fn sync(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> {
println!("{} {}", "::".bold().green(), "Syncing packages...".bold()); println!("{} {}", "::".bold().green(), "Syncing packages...".bold());
if !ask_confirmation()? {
return Err("Operation aborted".into());
}
let packages = file::read_packages(get_package_path()); let packages = file::read_packages(get_package_path());
let packages = packages.into_iter() let packages = packages
.into_iter()
.filter(|p| !p.contains("#") && !p.is_empty()) .filter(|p| !p.contains("#") && !p.is_empty())
.collect::<Vec<String>>(); .collect::<Vec<String>>();
let noconfirm = if noconfirm { "--noconfirm" } else { "--confirm" }; let noconfirm = if noconfirm {
"--noconfirm"
} else {
"--confirm"
};
let mut child = Command::new(PACKAGE_MANAGER) let mut child = Command::new(PACKAGE_MANAGER)
.arg("--color") .arg("--color")

View File

@@ -1,12 +1,22 @@
use colored::Colorize; use colored::Colorize;
use std::process::Command; use std::process::Command;
use crate::packages::ask_confirmation;
const PACKAGE_MANAGER: &str = "paru"; const PACKAGE_MANAGER: &str = "paru";
pub fn upgrade(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> { pub fn upgrade(noconfirm: bool) -> Result<(), Box<dyn std::error::Error>> {
println!("{} {}", "::".bold().green(), "Upgrading packages...".bold()); println!("{} {}", "::".bold().green(), "Upgrading packages...".bold());
let noconfirm = if noconfirm { "--noconfirm" } else { "--confirm" }; if !ask_confirmation()? {
return Err("Operation aborted".into());
}
let noconfirm = if noconfirm {
"--noconfirm"
} else {
"--confirm"
};
let mut child = Command::new(PACKAGE_MANAGER) let mut child = Command::new(PACKAGE_MANAGER)
.arg("--color") .arg("--color")