breezyshim 0.7.15

Rust shim around the Breezy Python API
Documentation
//! Debian specific functionality.
//!
//! This module provides functionality for working with Debian packages.
//!
//! It mostly wraps the `breezy.plugins.debian` module from the Breezy VCS.
pub mod apt;
/// Module for working with Debian commit messages and tagging.
pub mod debcommit;
/// Module for working with Debian directory structures.
pub mod directory;
/// Module defining errors specific to Debian functionality.
pub mod error;
/// Module for importing Debian source packages (dsc files).
pub mod import_dsc;
/// Module for merging upstream changes into Debian packages.
pub mod merge_upstream;
pub mod release;
/// Module for working with upstream sources in Debian packages.
pub mod upstream;
/// Module for checking if a Debian package in version control is up to date with the archive.
pub mod vcs_up_to_date;

/// Default directory for building Debian packages.
pub const DEFAULT_BUILD_DIR: &str = "../build-area";
/// Default directory for orig tarballs.
pub const DEFAULT_ORIG_DIR: &str = "..";
/// Default directory for build results.
pub const DEFAULT_RESULT_DIR: &str = "..";

use crate::branch::PyBranch;
use crate::debian::error::Error as DebianError;
use crate::error::Error;
use crate::tree::PyTree;
use crate::workingtree::PyWorkingTree;
use std::collections::HashMap;
use std::path::PathBuf;

use pyo3::exceptions::PyValueError;
use pyo3::prelude::*;
use pyo3::types::PyDict;

/// Represents different Debian-based distributions/vendors.
///
/// This enum is used to differentiate between various Debian-based
/// distributions when working with packages.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum Vendor {
    /// The Debian distribution.
    Debian,
    /// The Ubuntu distribution.
    Ubuntu,
    /// The Kali Linux distribution.
    Kali,
}

impl std::fmt::Display for Vendor {
    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
        match self {
            Vendor::Debian => write!(f, "debian"),
            Vendor::Ubuntu => write!(f, "ubuntu"),
            Vendor::Kali => write!(f, "kali"),
        }
    }
}

impl std::str::FromStr for Vendor {
    type Err = String;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        match s {
            "debian" => Ok(Vendor::Debian),
            "ubuntu" => Ok(Vendor::Ubuntu),
            "kali" => Ok(Vendor::Kali),
            _ => Err(format!("Invalid vendor: {}", s)),
        }
    }
}

impl<'a, 'py> FromPyObject<'a, 'py> for Vendor {
    type Error = PyErr;

    fn extract(ob: Borrowed<'a, 'py, PyAny>) -> PyResult<Self> {
        let vendor = ob.extract::<String>()?;
        match vendor.as_str() {
            "debian" => Ok(Vendor::Debian),
            "ubuntu" => Ok(Vendor::Ubuntu),
            "kali" => Ok(Vendor::Kali),
            _ => Err(PyValueError::new_err((format!(
                "Invalid vendor: {}",
                vendor
            ),))),
        }
    }
}

/// Kinds of upstream version handling.
///
/// This enum represents the different ways an upstream version can be handled,
/// particularly when determining version numbers for packages.
#[derive(Debug, Clone, PartialEq, Eq, std::hash::Hash, Default)]
pub enum VersionKind {
    /// Automatically determine the kind of version.
    #[default]
    Auto,
    /// Use snapshot versioning (typically includes a revision identifier).
    Snapshot,
    /// Use release versioning (clean version without revision identifiers).
    Release,
}

impl std::str::FromStr for VersionKind {
    type Err = String;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        match s {
            "auto" => Ok(VersionKind::Auto),
            "snapshot" => Ok(VersionKind::Snapshot),
            "release" => Ok(VersionKind::Release),
            _ => Err(format!("Invalid version kind: {}", s)),
        }
    }
}

impl<'py> IntoPyObject<'py> for VersionKind {
    type Target = PyAny;
    type Output = Bound<'py, PyAny>;
    type Error = PyErr;

    fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
        let s = match self {
            VersionKind::Auto => "auto",
            VersionKind::Snapshot => "snapshot",
            VersionKind::Release => "release",
        };
        Ok(s.into_pyobject(py)?.into_any())
    }
}

impl std::fmt::Display for VersionKind {
    fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
        match self {
            VersionKind::Auto => write!(f, "auto"),
            VersionKind::Snapshot => write!(f, "snapshot"),
            VersionKind::Release => write!(f, "release"),
        }
    }
}

impl<'a, 'py> FromPyObject<'a, 'py> for VersionKind {
    type Error = PyErr;

    fn extract(ob: Borrowed<'a, 'py, PyAny>) -> PyResult<Self> {
        let kind = ob.extract::<String>()?;
        match kind.as_str() {
            "auto" => Ok(VersionKind::Auto),
            "snapshot" => Ok(VersionKind::Snapshot),
            "release" => Ok(VersionKind::Release),
            _ => Err(PyValueError::new_err((format!(
                "Invalid version kind: {}",
                kind
            ),))),
        }
    }
}

/// Kind of tarball in a Debian source package.
///
/// Debian source packages can include multiple tarballs: the main orig tarball
/// and additional component tarballs. This enum represents those types.
#[derive(Debug, Clone, PartialEq, Eq, std::hash::Hash)]
pub enum TarballKind {
    /// The main original upstream tarball.
    Orig,
    /// An additional component tarball with the specified component name.
    Additional(String),
}

impl serde::ser::Serialize for TarballKind {
    fn serialize<S: serde::ser::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
        match self {
            TarballKind::Orig => serializer.serialize_none(),
            TarballKind::Additional(kind) => serializer.serialize_some(kind),
        }
    }
}

impl<'a> serde::de::Deserialize<'a> for TarballKind {
    fn deserialize<D: serde::de::Deserializer<'a>>(deserializer: D) -> Result<Self, D::Error> {
        let kind = Option::<String>::deserialize(deserializer)?;
        Ok(kind.into())
    }
}

impl From<Option<String>> for TarballKind {
    fn from(kind: Option<String>) -> Self {
        match kind {
            Some(kind) => TarballKind::Additional(kind),
            None => TarballKind::Orig,
        }
    }
}

impl From<TarballKind> for Option<String> {
    fn from(kind: TarballKind) -> Self {
        match kind {
            TarballKind::Orig => None,
            TarballKind::Additional(kind) => Some(kind),
        }
    }
}

impl<'a, 'py> FromPyObject<'a, 'py> for TarballKind {
    type Error = PyErr;

    fn extract(ob: Borrowed<'a, 'py, PyAny>) -> PyResult<Self> {
        let kind = ob.extract::<Option<String>>()?;
        Ok(kind.into())
    }
}

impl<'py> IntoPyObject<'py> for TarballKind {
    type Target = PyAny;
    type Output = Bound<'py, PyAny>;
    type Error = PyErr;

    fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
        let o: Option<String> = self.into();
        Ok(o.into_pyobject(py)?.into_any())
    }
}

/// Helper function to build a Debian package.
///
/// # Arguments
/// * `local_tree` - The working tree containing the Debian package
/// * `subpath` - Path to the debian directory within the tree
/// * `branch` - Branch containing the package
/// * `target_dir` - Directory to store build results
/// * `builder` - Name of the build tool to use
/// * `guess_upstream_branch_url` - Whether to guess the upstream branch URL
/// * `apt_repo` - Optional APT repository to use
///
/// # Returns
/// A map of result file types to their paths, or an error
pub fn build_helper(
    local_tree: &dyn PyWorkingTree,
    subpath: &std::path::Path,
    branch: &dyn PyBranch,
    target_dir: &std::path::Path,
    builder: &str,
    guess_upstream_branch_url: bool,
    apt_repo: Option<&dyn apt::Apt>,
) -> Result<HashMap<String, PathBuf>, DebianError> {
    Python::attach(|py| -> PyResult<HashMap<String, PathBuf>> {
        let locals = PyDict::new(py);
        locals.set_item("local_tree", local_tree.to_object(py))?;
        locals.set_item("subpath", subpath.to_string_lossy().to_string())?;
        locals.set_item("branch", branch.to_object(py))?;
        locals.set_item("target_dir", target_dir.to_string_lossy().to_string())?;
        locals.set_item("builder", builder)?;
        locals.set_item("guess_upstream_branch_url", guess_upstream_branch_url)?;

        if let Some(apt_repo) = apt_repo {
            locals.set_item("apt", apt_repo.as_pyobject())?;
        }

        py.import("breezy.plugins.debian.cmds")?
            .call_method1("_build_helper", (locals,))?
            .extract()
    })
    .map_err(DebianError::from)
}

/// Return the name of the debian tag for the given tree and branch.
///
/// # Arguments
/// * `tree` - The tree to get the debian tag name for.
/// * `branch` - The branch to get the debian tag name for.
/// * `subpath` - The subpath to get the debian tag name for.
/// * `vendor` - The vendor to get the debian tag name for.
///
/// # Returns
/// The name of the debian tag.
pub fn tree_debian_tag_name(
    tree: &dyn PyTree,
    branch: &dyn PyBranch,
    subpath: Option<&std::path::Path>,
    vendor: Option<Vendor>,
) -> Result<String, Error> {
    Python::attach(|py| {
        let result = py.import("breezy.plugins.debian")?.call_method1(
            "tree_debian_tag_name",
            (
                tree.to_object(py),
                branch.to_object(py),
                subpath,
                vendor.map(|v| v.to_string()),
            ),
        )?;

        Ok(result.extract()?)
    })
}

// TODO(jelmer): deduplicate this with the suite_to_distribution function
// in debian-analyzer
/// Infer the distribution from a suite.
///
/// When passed the name of a suite (anything in the distributions field of
/// a changelog) it will infer the distribution from that (i.e. Debian or
/// Ubuntu).
///
/// # Arguments
/// * `suite`: the string containing the suite
///
/// # Returns
/// Vendor or None if the distribution cannot be inferred.
pub fn suite_to_distribution(suite: &str) -> Option<Vendor> {
    Python::attach(|py| -> PyResult<Option<Vendor>> {
        let result = py
            .import("breezy.plugins.debian.util")?
            .call_method1("suite_to_distribution", (suite,))?;

        result.extract()
    })
    .unwrap()
}