use crate::managers::{Package, PackageManager};
use ratatui::DefaultTerminal;
use std::collections::{HashMap, HashSet};
use std::process::Command;
pub struct AptManager;
impl PackageManager for AptManager {
fn name(&self) -> &str {
"APT (Debian/Ubuntu)"
}
fn search(&self, query: &str) -> Vec<Package> {
if query.is_empty() {
return Vec::new();
}
let output = Command::new("apt-cache")
.args(&["search", query])
.output()
.ok();
if let Some(output) = output {
let stdout = String::from_utf8_lossy(&output.stdout);
stdout
.lines()
.filter_map(|line| {
let parts: Vec<&str> = line.splitn(2, " - ").collect();
if parts.len() >= 1 {
let name = parts[0].trim().to_string();
let desc = parts.get(1).unwrap_or(&"").trim().to_string();
let score = crate::fuzzy::fuzzy_match(query, &name);
Some(Package {
provider: "apt".to_string(),
name,
version: "".to_string(),
description: desc,
score,
})
} else {
None
}
})
.filter(|p| p.score > 0.01)
.collect()
} else {
Vec::new()
}
}
fn get_installed(&self) -> HashSet<String> {
let output = Command::new("dpkg-query")
.args(&["-W", "-f=${Package}\n"])
.output()
.ok();
if let Some(output) = output {
let stdout = String::from_utf8_lossy(&output.stdout);
stdout.lines().map(|s| s.trim().to_string()).collect()
} else {
HashSet::new()
}
}
fn get_installed_details(&self) -> Vec<Package> {
let output = Command::new("dpkg-query")
.args(&["-W", "-f=${Package}\t${Version}\t${Description}\n"])
.output()
.ok();
if let Some(output) = output {
let stdout = String::from_utf8_lossy(&output.stdout);
stdout
.lines()
.filter_map(|line| {
let parts: Vec<&str> = line.split('\t').collect();
if parts.len() >= 2 {
Some(Package {
provider: "apt/local".to_string(),
name: parts[0].to_string(),
version: parts[1].to_string(),
description: parts.get(2).unwrap_or(&"").to_string(),
score: 1.0,
})
} else {
None
}
})
.collect()
} else {
Vec::new()
}
}
fn get_updates(&self) -> Vec<Package> {
let output = Command::new("apt")
.args(&["list", "--upgradable"])
.output()
.ok();
if let Some(output) = output {
let stdout = String::from_utf8_lossy(&output.stdout);
stdout
.lines()
.skip(1)
.filter_map(|line| {
let parts: Vec<&str> = line.split('/').collect();
if parts.len() >= 1 {
Some(Package {
provider: "apt/update".to_string(),
name: parts[0].to_string(),
version: "Update available".to_string(),
description: "".to_string(),
score: 1.0,
})
} else {
None
}
})
.collect()
} else {
Vec::new()
}
}
fn get_details(&self, pkg: &str, _provider: &str) -> Option<HashMap<String, String>> {
{
let cache = crate::managers::DETAILS_CACHE.lock().unwrap();
if let Some(cached) = cache.get(pkg) {
return Some(cached.clone());
}
}
let output = Command::new("apt-cache")
.args(&["show", pkg])
.output()
.ok()?;
let stdout = String::from_utf8_lossy(&output.stdout);
let mut out = HashMap::new();
out.insert("Info".to_string(), stdout.to_string());
{
let mut cache = crate::managers::DETAILS_CACHE.lock().unwrap();
cache.insert(pkg.to_string(), out.clone());
}
Some(out)
}
fn install(
&self,
terminal: &mut DefaultTerminal,
pkgs: &HashSet<String>,
) -> Result<(), Box<dyn std::error::Error>> {
let mut args = vec!["apt", "install", "-y"];
let pkg_refs: Vec<&str> = pkgs.iter().map(|s| s.as_str()).collect();
args.extend(pkg_refs);
crate::execute_external_command(terminal, "sudo", &args)?;
Ok(())
}
fn remove(
&self,
terminal: &mut DefaultTerminal,
pkgs: &HashSet<String>,
) -> Result<(), Box<dyn std::error::Error>> {
let mut args = vec!["apt", "remove", "-y"];
let pkg_refs: Vec<&str> = pkgs.iter().map(|s| s.as_str()).collect();
args.extend(pkg_refs);
crate::execute_external_command(terminal, "sudo", &args)?;
Ok(())
}
fn system_upgrade(
&self,
terminal: &mut DefaultTerminal,
) -> Result<(), Box<dyn std::error::Error>> {
crate::execute_external_command(terminal, "sudo", &["apt", "upgrade", "-y"])?;
Ok(())
}
fn refresh_databases(
&self,
terminal: &mut DefaultTerminal,
) -> Result<(), Box<dyn std::error::Error>> {
crate::execute_external_command(terminal, "sudo", &["apt", "update"])?;
Ok(())
}
}