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))
}
}
impl std::fmt::Display for VersionError {
fn fmt(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
formatter.write_str(&self.0)
}
}
impl std::error::Error for VersionError {}
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) -> &str;
fn dependencies(&self) -> &[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) -> &str {
&self.registry_host_name
}
fn dependencies(&self) -> &[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) -> &str {
&self.registry_host_name
}
fn dependencies(&self) -> &[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::Path,
extension_config_path: &std::path::Path,
) -> 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: &[String],
) -> Result<Vec<PackageDependencies>>;
fn identify_file_defined_dependencies(
&self,
working_directory: &std::path::Path,
extension_args: &[String],
) -> Result<Vec<FileDefinedDependencies>>;
fn registries_package_metadata(
&self,
package_name: &str,
package_version: &Option<&str>,
) -> Result<Vec<RegistryPackageMetadata>>;
}