use anyhow::Result;
#[derive(
Debug, Clone, Hash, Eq, PartialEq, PartialOrd, Ord, serde::Serialize, serde::Deserialize,
)]
pub struct VersionError(String);
impl VersionError {
pub fn from_missing_version() -> Self {
Self("Missing version number".to_string())
}
pub fn from_parse_error(raw_version_number: &str) -> Self {
Self(format!("Version parse error: {}", raw_version_number))
}
pub fn message(&self) -> String {
self.0.clone()
}
}
pub type VersionParseResult = std::result::Result<String, VersionError>;
#[derive(
Clone, Debug, Hash, Eq, PartialEq, PartialOrd, Ord, serde::Serialize, serde::Deserialize,
)]
pub struct Dependency {
pub name: String,
pub version: VersionParseResult,
}
pub trait DependenciesCollection: Sized {
fn registry_host_name(&self) -> &String;
fn dependencies(&self) -> &Vec<Dependency>;
}
#[derive(Clone, Debug, Hash, Eq, PartialEq, serde::Serialize, serde::Deserialize)]
pub struct PackageDependencies {
pub package_version: VersionParseResult,
pub registry_host_name: String,
pub dependencies: Vec<Dependency>,
}
impl DependenciesCollection for PackageDependencies {
fn registry_host_name(&self) -> &String {
&self.registry_host_name
}
fn dependencies(&self) -> &Vec<Dependency> {
&self.dependencies
}
}
#[derive(Clone, Debug, Hash, Eq, PartialEq, serde::Serialize, serde::Deserialize)]
pub struct FileDefinedDependencies {
pub path: std::path::PathBuf,
pub registry_host_name: String,
pub dependencies: Vec<Dependency>,
}
impl DependenciesCollection for FileDefinedDependencies {
fn registry_host_name(&self) -> &String {
&self.registry_host_name
}
fn dependencies(&self) -> &Vec<Dependency> {
&self.dependencies
}
}
#[derive(Debug, Clone, Hash, Eq, PartialEq, serde::Serialize, serde::Deserialize)]
pub struct RegistryPackageMetadata {
pub registry_host_name: String,
pub human_url: String,
pub artifact_url: String,
pub is_primary: bool,
pub package_version: String,
}
#[derive(Debug, Clone, Default, Hash, Eq, PartialEq, serde::Serialize, serde::Deserialize)]
pub struct ReviewTargetPolicy {
pub excluded_exact_paths: Vec<String>,
}
impl ReviewTargetPolicy {
pub fn excludes_exact_path(&self, package_relative_path: &str) -> bool {
self.excluded_exact_paths
.iter()
.any(|excluded_path| excluded_path == package_relative_path)
}
pub fn excludes_path(&self, package_relative_path: &std::path::Path) -> bool {
self.excludes_exact_path(&package_relative_path_string(package_relative_path))
}
}
fn package_relative_path_string(package_relative_path: &std::path::Path) -> String {
if package_relative_path.as_os_str().is_empty() {
return ".".to_string();
}
package_relative_path
.iter()
.map(|component| component.to_string_lossy().into_owned())
.collect::<Vec<_>>()
.join("/")
}
pub trait FromLib: Extension + Send + Sync {
fn new() -> Self
where
Self: Sized;
}
pub trait FromProcess: Extension + Send + Sync {
fn from_process(
process_path: &std::path::PathBuf,
extension_config_path: &std::path::PathBuf,
) -> Result<Self>
where
Self: Sized;
}
pub trait Extension: Send + Sync {
fn name(&self) -> String;
fn registries(&self) -> Vec<String>;
fn review_target_policy(&self) -> ReviewTargetPolicy {
ReviewTargetPolicy::default()
}
fn identify_package_dependencies(
&self,
package_name: &str,
package_version: &Option<&str>,
extension_args: &Vec<String>,
) -> Result<Vec<PackageDependencies>>;
fn identify_file_defined_dependencies(
&self,
working_directory: &std::path::PathBuf,
extension_args: &Vec<String>,
) -> Result<Vec<FileDefinedDependencies>>;
fn registries_package_metadata(
&self,
package_name: &str,
package_version: &Option<&str>,
) -> Result<Vec<RegistryPackageMetadata>>;
}