mise 2026.5.0

Dev tools, env vars, and tasks in one CLI
use std::path::{Path, PathBuf};

use eyre::Result;

use crate::deps::rule::DepsProviderConfig;
use crate::deps::{DepsCommand, DepsProvider};

use super::ProviderBase;

/// Deps provider for PHP Composer (composer.lock)
#[derive(Debug)]
pub struct ComposerDepsProvider {
    base: ProviderBase,
}

impl ComposerDepsProvider {
    pub fn new(project_root: &Path, config: DepsProviderConfig) -> Self {
        Self {
            base: ProviderBase::new("composer", project_root, config),
        }
    }
}

impl DepsProvider for ComposerDepsProvider {
    fn base(&self) -> &ProviderBase {
        &self.base
    }

    fn sources(&self) -> Vec<PathBuf> {
        let root = self.base.config_root();
        vec![root.join("composer.lock"), root.join("composer.json")]
    }

    fn outputs(&self) -> Vec<PathBuf> {
        vec![self.base.config_root().join("vendor")]
    }

    fn install_command(&self) -> Result<DepsCommand> {
        if let Some(run) = &self.base.config.run {
            return DepsCommand::from_string(run, &self.base.project_root, &self.base.config);
        }

        Ok(DepsCommand {
            program: "composer".to_string(),
            args: vec!["install".to_string()],
            env: self.base.config.env.clone(),
            cwd: Some(self.base.config_root()),
            description: self
                .base
                .config
                .description
                .clone()
                .unwrap_or_else(|| "composer install".to_string()),
        })
    }

    fn is_applicable(&self) -> bool {
        self.base.config_root().join("composer.lock").exists()
    }
}