use crate::lossy::{Relations, SourceRelation};
use deb822_fast::{FromDeb822, FromDeb822Paragraph, ToDeb822, ToDeb822Paragraph};
fn deserialize_yesno(s: &str) -> Result<bool, String> {
match s {
"yes" => Ok(true),
"no" => Ok(false),
_ => Err(format!("invalid value for yesno: {}", s)),
}
}
fn serialize_yesno(b: &bool) -> String {
if *b {
"yes".to_string()
} else {
"no".to_string()
}
}
fn deserialize_components(value: &str) -> Result<Vec<String>, String> {
Ok(value.split_whitespace().map(|s| s.to_string()).collect())
}
fn join_whitespace(components: &[String]) -> String {
components.join(" ")
}
fn deserialize_architectures(value: &str) -> Result<Vec<String>, String> {
Ok(value.split_whitespace().map(|s| s.to_string()).collect())
}
#[derive(Debug, Clone, PartialEq, Eq, ToDeb822, FromDeb822)]
pub struct Release {
#[deb822(field = "Codename")]
pub codename: Option<String>,
#[deb822(
field = "Components",
deserialize_with = deserialize_components,
serialize_with = join_whitespace
)]
pub components: Vec<String>,
#[deb822(
field = "Architectures",
deserialize_with = deserialize_architectures,
serialize_with = join_whitespace
)]
pub architectures: Vec<String>,
#[deb822(field = "Description")]
pub description: Option<String>,
#[deb822(field = "Origin")]
pub origin: Option<String>,
#[deb822(field = "Label")]
pub label: Option<String>,
#[deb822(field = "Suite")]
pub suite: Option<String>,
#[deb822(field = "Version")]
pub version: Option<String>,
#[deb822(field = "Date")]
pub date: Option<String>,
#[deb822(field = "NotAutomatic", deserialize_with = deserialize_yesno, serialize_with = serialize_yesno)]
pub not_automatic: Option<bool>,
#[deb822(field = "ButAutomaticUpgrades", deserialize_with = deserialize_yesno, serialize_with = serialize_yesno)]
pub but_automatic_upgrades: Option<bool>,
#[deb822(field = "Acquire-By-Hash", deserialize_with = deserialize_yesno, serialize_with = serialize_yesno)]
pub acquire_by_hash: Option<bool>,
#[deb822(field = "MD5Sum", deserialize_with = deserialize_checksums::<crate::fields::Md5Checksum>, serialize_with = serialize_checksums::<crate::fields::Md5Checksum>)]
pub checksums_md5: Option<Vec<crate::fields::Md5Checksum>>,
#[deb822(field = "SHA1", deserialize_with = deserialize_checksums::<crate::fields::Sha1Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha1Checksum>)]
pub checksums_sha1: Option<Vec<crate::fields::Sha1Checksum>>,
#[deb822(field = "SHA256", deserialize_with = deserialize_checksums::<crate::fields::Sha256Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha256Checksum>)]
pub checksums_sha256: Option<Vec<crate::fields::Sha256Checksum>>,
#[deb822(field = "SHA512", deserialize_with = deserialize_checksums::<crate::fields::Sha512Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha512Checksum>)]
pub checksums_sha512: Option<Vec<crate::fields::Sha512Checksum>>,
}
impl std::str::FromStr for Release {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let para = s
.parse::<deb822_fast::Paragraph>()
.map_err(|e| e.to_string())?;
FromDeb822Paragraph::from_paragraph(¶)
}
}
impl std::fmt::Display for Release {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let para: deb822_fast::Paragraph = self.to_paragraph();
write!(f, "{}", para)
}
}
fn deserialize_binaries(value: &str) -> Result<Vec<String>, String> {
Ok(value.split(",").map(|s| s.trim().to_string()).collect())
}
fn deserialize_testsuite_triggers(value: &str) -> Result<Vec<String>, String> {
Ok(value.split(",").map(|s| s.trim().to_string()).collect())
}
fn join_lines(components: &[String]) -> String {
components.join("\n")
}
fn deserialize_package_list(value: &str) -> Result<Vec<String>, String> {
Ok(value.split('\n').map(|s| s.to_string()).collect())
}
fn deserialize_checksums<T>(value: &str) -> Result<Vec<T>, String>
where
T: std::str::FromStr,
T::Err: std::fmt::Display,
{
value
.lines()
.map(|s| s.trim())
.filter(|s| !s.is_empty())
.map(|s| s.parse::<T>().map_err(|e| e.to_string()))
.collect()
}
fn serialize_checksums<T: std::fmt::Display>(checksums: &[T]) -> String {
checksums
.iter()
.map(|c| format!(" {}", c))
.collect::<Vec<_>>()
.join("\n")
}
#[derive(Debug, Clone, PartialEq, Eq, ToDeb822, FromDeb822)]
pub struct Source {
#[deb822(field = "Directory")]
pub directory: String,
#[deb822(field = "Description")]
pub description: Option<String>,
#[deb822(field = "Version")]
pub version: debversion::Version,
#[deb822(field = "Package")]
pub package: String,
#[deb822(field = "Binary", deserialize_with = deserialize_binaries, serialize_with = join_whitespace)]
pub binaries: Option<Vec<String>>,
#[deb822(field = "Maintainer")]
pub maintainer: Option<String>,
#[deb822(field = "Build-Depends")]
pub build_depends: Option<Relations>,
#[deb822(field = "Build-Depends-Indep")]
pub build_depends_indep: Option<Relations>,
#[deb822(field = "Build-Depends-Arch")]
pub build_depends_arch: Option<Relations>,
#[deb822(field = "Build-Conflicts")]
pub build_conflicts: Option<Relations>,
#[deb822(field = "Build-Conflicts-Indep")]
pub build_conflicts_indep: Option<Relations>,
#[deb822(field = "Build-Conflicts-Arch")]
pub build_conflicts_arch: Option<Relations>,
#[deb822(field = "Standards-Version")]
pub standards_version: Option<String>,
#[deb822(field = "Homepage")]
pub homepage: Option<String>,
#[deb822(field = "Autobuild")]
pub autobuild: Option<bool>,
#[deb822(field = "Testsuite")]
pub testsuite: Option<String>,
#[deb822(field = "Testsuite-Triggers", deserialize_with = deserialize_testsuite_triggers, serialize_with = join_whitespace)]
pub testsuite_triggers: Option<Vec<String>>,
#[deb822(field = "Vcs-Browser")]
pub vcs_browser: Option<String>,
#[deb822(field = "Vcs-Git")]
pub vcs_git: Option<String>,
#[deb822(field = "Vcs-Bzr")]
pub vcs_bzr: Option<String>,
#[deb822(field = "Vcs-Hg")]
pub vcs_hg: Option<String>,
#[deb822(field = "Vcs-Svn")]
pub vcs_svn: Option<String>,
#[deb822(field = "Vcs-Darcs")]
pub vcs_darcs: Option<String>,
#[deb822(field = "Vcs-Cvs")]
pub vcs_cvs: Option<String>,
#[deb822(field = "Vcs-Arch")]
pub vcs_arch: Option<String>,
#[deb822(field = "Vcs-Mtn")]
pub vcs_mtn: Option<String>,
#[deb822(field = "Dgit")]
pub dgit: Option<crate::fields::DgitInfo>,
#[deb822(field = "Priority")]
pub priority: Option<crate::fields::Priority>,
#[deb822(field = "Section")]
pub section: Option<String>,
#[deb822(field = "Format")]
pub format: Option<String>,
#[deb822(field = "Architecture")]
pub architecture: Option<String>,
#[deb822(field = "Package-List", deserialize_with = deserialize_package_list, serialize_with = join_lines)]
pub package_list: Vec<String>,
#[deb822(field = "Files", deserialize_with = deserialize_checksums::<crate::fields::Md5Checksum>, serialize_with = serialize_checksums::<crate::fields::Md5Checksum>)]
pub files: Option<Vec<crate::fields::Md5Checksum>>,
#[deb822(field = "Checksums-Sha1", deserialize_with = deserialize_checksums::<crate::fields::Sha1Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha1Checksum>)]
pub checksums_sha1: Option<Vec<crate::fields::Sha1Checksum>>,
#[deb822(field = "Checksums-Sha256", deserialize_with = deserialize_checksums::<crate::fields::Sha256Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha256Checksum>)]
pub checksums_sha256: Option<Vec<crate::fields::Sha256Checksum>>,
#[deb822(field = "Checksums-Sha512", deserialize_with = deserialize_checksums::<crate::fields::Sha512Checksum>, serialize_with = serialize_checksums::<crate::fields::Sha512Checksum>)]
pub checksums_sha512: Option<Vec<crate::fields::Sha512Checksum>>,
}
impl std::str::FromStr for Source {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let para = s
.parse::<deb822_fast::Paragraph>()
.map_err(|e| e.to_string())?;
FromDeb822Paragraph::from_paragraph(¶)
}
}
impl std::fmt::Display for Source {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let para: deb822_fast::Paragraph = self.to_paragraph();
write!(f, "{}", para)
}
}
#[derive(Debug, Clone, PartialEq, Eq, ToDeb822, FromDeb822)]
pub struct Package {
#[deb822(field = "Package")]
pub name: String,
#[deb822(field = "Version")]
pub version: debversion::Version,
#[deb822(field = "Source")]
pub source: Option<SourceRelation>,
#[deb822(field = "Architecture")]
pub architecture: String,
#[deb822(field = "Maintainer")]
pub maintainer: Option<String>,
#[deb822(field = "Installed-Size")]
pub installed_size: Option<usize>,
#[deb822(field = "Depends")]
pub depends: Option<Relations>,
#[deb822(field = "Pre-Depends")]
pub pre_depends: Option<Relations>,
#[deb822(field = "Recommends")]
pub recommends: Option<Relations>,
#[deb822(field = "Suggests")]
pub suggests: Option<Relations>,
#[deb822(field = "Enhances")]
pub enhances: Option<Relations>,
#[deb822(field = "Breaks")]
pub breaks: Option<Relations>,
#[deb822(field = "Conflicts")]
pub conflicts: Option<Relations>,
#[deb822(field = "Provides")]
pub provides: Option<Relations>,
#[deb822(field = "Replaces")]
pub replaces: Option<Relations>,
#[deb822(field = "Built-Using")]
pub built_using: Option<Relations>,
#[deb822(field = "Static-Built-Using")]
pub static_built_using: Option<Relations>,
#[deb822(field = "Description")]
pub description: Option<String>,
#[deb822(field = "Homepage")]
pub homepage: Option<String>,
#[deb822(field = "Priority")]
pub priority: Option<crate::fields::Priority>,
#[deb822(field = "Section")]
pub section: Option<String>,
#[deb822(field = "Essential", deserialize_with = deserialize_yesno, serialize_with = serialize_yesno)]
pub essential: Option<bool>,
#[deb822(field = "Tag")]
pub tag: Option<String>,
#[deb822(field = "Size")]
pub size: Option<usize>,
#[deb822(field = "Filename")]
pub filename: Option<String>,
#[deb822(field = "MD5sum")]
pub md5sum: Option<String>,
#[deb822(field = "SHA1")]
pub sha1: Option<String>,
#[deb822(field = "SHA256")]
pub sha256: Option<String>,
#[deb822(field = "SHA512")]
pub sha512: Option<String>,
#[deb822(field = "Description-MD5")]
pub description_md5: Option<String>,
}
impl std::str::FromStr for Package {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let para = s
.parse::<deb822_fast::Paragraph>()
.map_err(|e| e.to_string())?;
FromDeb822Paragraph::from_paragraph(¶)
}
}
impl std::fmt::Display for Package {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
let para: deb822_fast::Paragraph = self.to_paragraph();
write!(f, "{}", para)
}
}
#[cfg(test)]
mod tests {
use super::*;
use deb822_fast::Paragraph;
use deb822_fast::ToDeb822Paragraph;
#[test]
fn test_release() {
let release = Release {
codename: Some("focal".to_string()),
components: vec!["main".to_string(), "restricted".to_string()],
architectures: vec!["amd64".to_string(), "arm64".to_string()],
description: Some("Ubuntu 20.04 LTS".to_string()),
origin: Some("Ubuntu".to_string()),
label: Some("Ubuntu".to_string()),
suite: Some("focal".to_string()),
version: Some("20.04".to_string()),
date: Some("Thu, 23 Apr 2020 17:19:19 UTC".to_string()),
not_automatic: Some(false),
but_automatic_upgrades: Some(true),
acquire_by_hash: Some(true),
checksums_md5: None,
checksums_sha1: None,
checksums_sha256: None,
checksums_sha512: None,
};
let deb822 = r#"Codename: focal
Components: main restricted
Architectures: amd64 arm64
Description: Ubuntu 20.04 LTS
Origin: Ubuntu
Label: Ubuntu
Suite: focal
Version: 20.04
Date: Thu, 23 Apr 2020 17:19:19 UTC
NotAutomatic: no
ButAutomaticUpgrades: yes
Acquire-By-Hash: yes
"#;
let para = deb822.parse::<Paragraph>().unwrap();
let release: deb822_fast::Paragraph = release.to_paragraph();
assert_eq!(release, para);
}
#[test]
fn test_package() {
let package = r#"Package: apt
Version: 2.1.10
Architecture: amd64
Maintainer: APT Development Team <apt@lists.debian.org>
Installed-Size: 3524
Depends: libc6 (>= 2.14), libgcc1
Pre-Depends: dpkg (>= 1.15.6)
Recommends: gnupg
Suggests: apt-doc, aptitude | synaptic | wajig
"#;
let package: Package = package.parse().unwrap();
assert_eq!(package.name, "apt");
assert_eq!(package.version, "2.1.10".parse().unwrap());
assert_eq!(package.architecture, "amd64");
}
#[test]
fn test_package_essential() {
let package = r#"Package: base-files
Version: 11.1
Architecture: amd64
Essential: yes
"#;
let package: Package = package.parse().unwrap();
assert_eq!(package.name, "base-files");
assert_eq!(package.essential, Some(true));
}
#[test]
fn test_package_essential_no() {
let package = r#"Package: apt
Version: 2.1.10
Architecture: amd64
Essential: no
"#;
let package: Package = package.parse().unwrap();
assert_eq!(package.name, "apt");
assert_eq!(package.essential, Some(false));
}
#[test]
fn test_package_with_different_source() {
let package = r#"Package: apt
Source: not-apt (1.1.5)
Version: 2.1.10
Architecture: amd64
Maintainer: APT Development Team <apt@lists.debian.org>
Installed-Size: 3524
Depends: libc6 (>= 2.14), libgcc1
Pre-Depends: dpkg (>= 1.15.6)
Recommends: gnupg
Suggests: apt-doc, aptitude | synaptic | wajig
"#;
let package: Package = package.parse().unwrap();
assert_eq!(package.name, "apt");
assert_eq!(package.version, "2.1.10".parse().unwrap());
assert_eq!(package.architecture, "amd64");
assert_eq!(
package.source,
Some(SourceRelation {
name: "not-apt".to_string(),
version: Some("1.1.5".parse().unwrap())
})
);
}
#[test]
fn test_source() {
let source = r#"Package: abinit
Binary: abinit, abinit-doc, abinit-data
Version: 9.10.4-3
Maintainer: Debichem Team <debichem-devel@lists.alioth.debian.org>
Uploaders: Andreas Tille <tille@debian.org>, Michael Banck <mbanck@debian.org>
Build-Depends: debhelper (>= 11), gfortran, liblapack-dev, python3, graphviz, markdown, ghostscript, help2man, libfftw3-dev, libhdf5-dev, libnetcdff-dev, libssl-dev, libxc-dev, mpi-default-dev, python3-dev, python3-numpy, python3-pandas, python3-yaml, texlive-latex-extra, texlive-fonts-recommended, texlive-extra-utils, texlive-pstricks, texlive-publishers, texlive-luatex
Architecture: any all
Standards-Version: 3.9.8
Format: 3.0 (quilt)
Files:
843550cbd14395c0b9408158a91a239c 2464 abinit_9.10.4-3.dsc
a323f11fbd4a7d0f461d99c931903b5c 130747285 abinit_9.10.4.orig.tar.gz
27c12d3dac5cd105cebaa2af4247e807 15068 abinit_9.10.4-3.debian.tar.xz
Vcs-Browser: https://salsa.debian.org/debichem-team/abinit
Vcs-Git: https://salsa.debian.org/debichem-team/abinit.git
Checksums-Sha256:
c3c217b14bc5705a1d8930a2e7fcef58e64beaa22abc213e2eacc7d5537ef840 2464 abinit_9.10.4-3.dsc
6bf3c276c333956f722761f189f2b4324e150c8a50470ecb72ee07cc1c457d48 130747285 abinit_9.10.4.orig.tar.gz
80c4fb7575d67f3167d7c34fd59477baf839810d0b863e19f1dd9fea1bc0b3b5 15068 abinit_9.10.4-3.debian.tar.xz
Homepage: http://www.abinit.org/
Package-List:
abinit deb science optional arch=any
abinit-data deb science optional arch=all
abinit-doc deb doc optional arch=all
Testsuite: autopkgtest
Testsuite-Triggers: python3, python3-numpy, python3-pandas, python3-yaml
Directory: pool/main/a/abinit
Priority: optional
Section: science
"#;
let source: Source = source.parse().unwrap();
assert_eq!(source.package, "abinit");
assert_eq!(source.version, "9.10.4-3".parse().unwrap());
assert_eq!(
source.binaries,
Some(vec![
"abinit".to_string(),
"abinit-doc".to_string(),
"abinit-data".to_string()
])
);
let build_depends = source.build_depends.as_ref();
let build_depends: Vec<_> = build_depends.iter().collect();
let build_depends = build_depends[0];
let expected_build_depends = &[
"debhelper",
"gfortran",
"liblapack-dev",
"python3",
"graphviz",
"markdown",
"ghostscript",
"help2man",
"libfftw3-dev",
"libhdf5-dev",
"libnetcdff-dev",
"libssl-dev",
"libxc-dev",
"mpi-default-dev",
"python3-dev",
"python3-numpy",
"python3-pandas",
"python3-yaml",
"texlive-latex-extra",
"texlive-fonts-recommended",
"texlive-extra-utils",
"texlive-pstricks",
"texlive-publishers",
"texlive-luatex",
];
assert_eq!(build_depends.len(), expected_build_depends.len());
assert_eq!(build_depends[0][0].name, expected_build_depends[0]);
assert_eq!(
build_depends[build_depends.len() - 1][0].name,
expected_build_depends[build_depends.len() - 1]
);
assert_eq!(
source.testsuite_triggers,
Some(
["python3", "python3-numpy", "python3-pandas", "python3-yaml"]
.into_iter()
.map(String::from)
.collect()
)
);
}
#[test]
fn test_source_with_dgit() {
let source = r#"Package: test-pkg
Version: 1.0-1
Maintainer: Test Maintainer <test@example.com>
Dgit: c1370424e2404d3c22bd09c828d4b28d81d897ad debian archive/debian/1.1.0 https://git.dgit.debian.org/test-pkg
Directory: pool/main/t/test-pkg
Package-List:
abinit deb science optional arch=any
abinit-data deb science optional arch=all
abinit-doc deb doc optional arch=all
"#;
let source: Source = source.parse().unwrap();
assert_eq!(source.package, "test-pkg");
let dgit = source.dgit.as_ref().unwrap();
assert_eq!(dgit.commit, "c1370424e2404d3c22bd09c828d4b28d81d897ad");
assert_eq!(dgit.suite, "debian");
assert_eq!(dgit.git_ref, "archive/debian/1.1.0");
assert_eq!(dgit.url, "https://git.dgit.debian.org/test-pkg");
}
#[test]
fn test_source_checksums() {
let source = r#"Package: hello
Version: 1.0-1
Directory: pool/main/h/hello
Package-List:
hello deb misc optional arch=any
Files:
acbd18db4cc2f85cedef654fccc4a4d8 1234 hello_1.0.orig.tar.gz
Checksums-Sha256:
2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824 1234 hello_1.0.orig.tar.gz
"#;
let source: Source = source.parse().unwrap();
assert_eq!(source.package, "hello");
let files = source.files.as_ref().unwrap();
assert_eq!(files.len(), 1);
assert_eq!(files[0].md5sum, "acbd18db4cc2f85cedef654fccc4a4d8");
assert_eq!(files[0].size, 1234);
assert_eq!(files[0].filename, "hello_1.0.orig.tar.gz");
let sha256 = source.checksums_sha256.as_ref().unwrap();
assert_eq!(sha256.len(), 1);
assert_eq!(
sha256[0].sha256,
"2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824"
);
}
#[test]
fn test_release_checksums() {
let release_str = r#"Suite: stable
Codename: stable
Architectures: amd64
Components: main
MD5Sum:
acbd18db4cc2f85cedef654fccc4a4d8 1234 main/binary-amd64/Packages
SHA256:
2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824 1234 main/binary-amd64/Packages
"#;
let release: Release = release_str.parse().unwrap();
assert_eq!(release.suite, Some("stable".to_string()));
let md5 = release.checksums_md5.as_ref().unwrap();
assert_eq!(md5.len(), 1);
assert_eq!(md5[0].filename, "main/binary-amd64/Packages");
assert_eq!(md5[0].size, 1234);
let sha256 = release.checksums_sha256.as_ref().unwrap();
assert_eq!(sha256.len(), 1);
assert_eq!(sha256[0].filename, "main/binary-amd64/Packages");
}
}