depup-cli 0.2.1

Check dependency versions across Maven and npm ecosystems
//! Tool version resolver trait and registry.
//!
//! Provides an extensible mechanism for resolving non-Maven version properties
//! found in POM files (e.g., `version.node`, `version.pnpm`). Each resolver
//! declares the property name patterns it handles and performs its own
//! registry lookup.

use std::future::Future;
use std::pin::Pin;
use std::sync::Arc;

use anyhow::Result;

use crate::maven::discovery::VersionProperty;
use crate::maven::node::NodeResolver;
use crate::maven::pm_versions::PmVersionsResolver;
use crate::model::CheckResult;

/// Trait for resolving tool version properties against their respective registries.
pub trait ToolVersionResolver: Send + Sync {
    /// Returns the property name patterns this resolver handles.
    fn patterns(&self) -> &[&str];
    /// Returns a display label for progress reporting.
    fn label(&self, property: &VersionProperty) -> String;
    /// Resolves the property's current value against the latest available version.
    fn resolve<'a>(
        &'a self,
        property: &'a VersionProperty,
        source: &'a str,
    ) -> Pin<Box<dyn Future<Output = Result<CheckResult>> + Send + 'a>>;
}

/// Registry of all available tool version resolvers.
/// Matches orphan POM properties to the appropriate resolver by pattern.
pub struct ToolResolverRegistry {
    resolvers: Vec<Arc<dyn ToolVersionResolver>>,
}

impl ToolResolverRegistry {
    pub fn new(stable: bool) -> Self {
        Self {
            resolvers: vec![
                Arc::new(NodeResolver::new(stable)),
                Arc::new(PmVersionsResolver::new()),
            ],
        }
    }

    /// Finds the first resolver whose patterns include the given property name.
    pub fn find(&self, property_name: &str) -> Option<Arc<dyn ToolVersionResolver>> {
        self.resolvers
            .iter()
            .find(|c| c.patterns().contains(&property_name))
            .cloned()
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn find_node_resolver() {
        let registry = ToolResolverRegistry::new(false);
        let resolver = registry.find("version.node");
        assert!(resolver.is_some());
        assert!(resolver.unwrap().patterns().contains(&"version.node"));
    }

    #[test]
    fn find_pm_resolver_for_npm() {
        let registry = ToolResolverRegistry::new(false);
        let resolver = registry.find("version.npm");
        assert!(resolver.is_some());
        assert!(resolver.unwrap().patterns().contains(&"version.npm"));
    }

    #[test]
    fn find_returns_none_for_non_tool_property() {
        let registry = ToolResolverRegistry::new(false);
        assert!(registry.find("version.junit").is_none());
    }

    #[test]
    fn find_returns_none_for_random_property() {
        let registry = ToolResolverRegistry::new(false);
        assert!(registry.find("random.property").is_none());
    }
}