use std::collections::HashMap;
use std::path::Path;
use std::process::Stdio;
use async_trait::async_trait;
use eyre::bail;
use super::{InstallOpts, PackageRequest, PackageState, PackageStatus, SystemPackageManager};
use crate::result::Result;
use crate::system::sudo;
pub struct AptManager {}
impl AptManager {
pub fn new() -> Self {
Self {}
}
fn lists_missing(&self) -> bool {
let lists = Path::new("/var/lib/apt/lists");
!crate::file::ls(lists).unwrap_or_default().iter().any(|p| {
p.file_name()
.map(|f| f.to_string_lossy().contains("_Packages"))
.unwrap_or(false)
})
}
fn update(&self, opts: &InstallOpts) -> Result<()> {
let args = vec!["update".to_string()];
if opts.dry_run {
miseprintln!(
"{}",
sudo::argv_with_env("apt-get", &args, &debian_frontend()).join(" ")
);
return Ok(());
}
sudo::run("apt-get", &args, &debian_frontend())
}
}
fn debian_frontend() -> Vec<(String, String)> {
vec![("DEBIAN_FRONTEND".to_string(), "noninteractive".to_string())]
}
fn dpkg_name(name: &str) -> &str {
name.split(':').next().unwrap_or(name)
}
fn parse_dpkg_query(output: &str, requests: &[PackageRequest]) -> Vec<PackageStatus> {
let mut installed: HashMap<String, Vec<&str>> = HashMap::new();
for line in output.lines() {
let mut parts = line.split('\t');
if let (Some(name), Some(status), Some(version)) =
(parts.next(), parts.next(), parts.next())
{
if status != "installed" {
continue;
}
if let Some(arch) = parts.next() {
installed
.entry(format!("{name}:{arch}"))
.or_default()
.push(version);
}
installed.entry(name.to_string()).or_default().push(version);
}
}
requests
.iter()
.map(|req| {
let state = match installed.get(&req.name) {
Some(versions) => match &req.version {
Some(requested) if !versions.contains(&requested.as_str()) => {
PackageState::VersionMismatch {
installed: versions[0].to_string(),
}
}
Some(requested) => PackageState::Installed {
version: requested.clone(),
},
None => PackageState::Installed {
version: versions[0].to_string(),
},
},
None => PackageState::Missing,
};
PackageStatus {
request: req.clone(),
state,
}
})
.collect()
}
#[async_trait]
impl SystemPackageManager for AptManager {
fn name(&self) -> &'static str {
"apt"
}
fn is_available(&self) -> bool {
cfg!(target_os = "linux") && crate::file::which("apt-get").is_some()
}
fn unavailable_reason(&self) -> String {
if cfg!(target_os = "linux") {
"apt-get not found".to_string()
} else {
"only available on linux".to_string()
}
}
async fn installed(&self, pkgs: &[PackageRequest]) -> Result<Vec<PackageStatus>> {
if pkgs.is_empty() {
return Ok(vec![]);
}
let mut args = vec![
"-W".to_string(),
"-f=${Package}\\t${db:Status-Status}\\t${Version}\\t${Architecture}\\n".to_string(),
];
args.extend(pkgs.iter().map(|p| dpkg_name(&p.name).to_string()));
debug!("$ dpkg-query {}", args.join(" "));
let output = tokio::process::Command::new("dpkg-query")
.args(&args)
.stdin(Stdio::null())
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.output()
.await?;
if !output.status.success() && output.status.code() != Some(1) {
bail!(
"dpkg-query failed: {}",
String::from_utf8_lossy(&output.stderr).trim()
);
}
let stdout = String::from_utf8_lossy(&output.stdout);
Ok(parse_dpkg_query(&stdout, pkgs))
}
async fn install(&self, pkgs: &[PackageRequest], opts: &InstallOpts) -> Result<()> {
if opts.update || self.lists_missing() {
self.update(opts)?;
}
let mut args = vec!["install".to_string(), "-y".to_string(), "--".to_string()];
args.extend(pkgs.iter().map(|p| match &p.version {
Some(v) => format!("{}={v}", p.name),
None => p.name.clone(),
}));
if opts.dry_run {
miseprintln!(
"{}",
sudo::argv_with_env("apt-get", &args, &debian_frontend()).join(" ")
);
return Ok(());
}
sudo::run("apt-get", &args, &debian_frontend())
}
async fn upgrade(&self, pkgs: &[PackageRequest], opts: &InstallOpts) -> Result<()> {
self.update(opts)?;
let mut args = vec![
"install".to_string(),
"-y".to_string(),
"--only-upgrade".to_string(),
"--".to_string(),
];
args.extend(pkgs.iter().map(|p| match &p.version {
Some(v) => format!("{}={v}", p.name),
None => p.name.clone(),
}));
if opts.dry_run {
miseprintln!(
"{}",
sudo::argv_with_env("apt-get", &args, &debian_frontend()).join(" ")
);
return Ok(());
}
sudo::run("apt-get", &args, &debian_frontend())
}
}
#[cfg(test)]
mod tests {
use super::*;
fn req(name: &str, version: Option<&str>) -> PackageRequest {
PackageRequest {
name: name.to_string(),
version: version.map(str::to_string),
}
}
#[test]
fn test_dpkg_name() {
assert_eq!(dpkg_name("gcc"), "gcc");
assert_eq!(dpkg_name("gcc:arm64"), "gcc");
}
#[test]
fn test_parse_dpkg_query() {
let requests = vec![
req("bc", None),
req("nonexistent", None),
req("removed-pkg", None),
req("curl", Some("9.9.9")),
];
let output = "bc\tinstalled\t1.07.1-3\tamd64\nremoved-pkg\tdeinstall\t2.0\tamd64\ncurl\tinstalled\t8.5.0-2\tamd64\n";
let statuses = parse_dpkg_query(output, &requests);
assert_eq!(
statuses[0].state,
PackageState::Installed {
version: "1.07.1-3".to_string()
}
);
assert_eq!(statuses[1].state, PackageState::Missing);
assert_eq!(statuses[2].state, PackageState::Missing);
assert_eq!(
statuses[3].state,
PackageState::VersionMismatch {
installed: "8.5.0-2".to_string()
}
);
}
#[test]
fn test_parse_dpkg_query_multiarch_bare_name() {
let requests = vec![req("libssl3", None)];
for output in [
"libssl3\tinstalled\t3.0.2\tamd64\nlibssl3\tdeinstall\t3.0.1\ti386\n",
"libssl3\tdeinstall\t3.0.1\ti386\nlibssl3\tinstalled\t3.0.2\tamd64\n",
] {
let statuses = parse_dpkg_query(output, &requests);
assert_eq!(
statuses[0].state,
PackageState::Installed {
version: "3.0.2".to_string()
}
);
}
}
#[test]
fn test_parse_dpkg_query_multiarch_bare_name_versioned() {
let requests = vec![req("libssl3", Some("3.0.2"))];
for output in [
"libssl3\tinstalled\t3.0.1\ti386\nlibssl3\tinstalled\t3.0.2\tamd64\n",
"libssl3\tinstalled\t3.0.2\tamd64\nlibssl3\tinstalled\t3.0.1\ti386\n",
] {
let statuses = parse_dpkg_query(output, &requests);
assert_eq!(
statuses[0].state,
PackageState::Installed {
version: "3.0.2".to_string()
}
);
}
}
#[test]
fn test_parse_dpkg_query_arch_qualified() {
let requests = vec![req("gcc:arm64", None), req("gcc:amd64", None)];
let output = "gcc\tinstalled\t12.3\tarm64\n";
let statuses = parse_dpkg_query(output, &requests);
assert_eq!(
statuses[0].state,
PackageState::Installed {
version: "12.3".to_string()
}
);
assert_eq!(statuses[1].state, PackageState::Missing);
}
}