rustquty-core 0.2.0

Core library for rustquty, a local-first quality scanner for Rust projects
Documentation
//! Audit collector — runs `cargo audit`.

use super::{Collector, CollectorError, CollectorOutput};
use crate::context::Context;
use std::process::Command;

pub struct AuditCollector;

impl AuditCollector {
    pub fn new() -> Self {
        Self
    }

    fn parse_json_output(&self, stdout: &str) -> (u32, u32) {
        let mut vulnerability_count = 0u32;
        let mut critical_count = 0u32;

        if let Ok(json) = serde_json::from_str::<serde_json::Value>(stdout) {
            if let Some(found) = json.get("vulnerabilities").and_then(|v| v.get("found")) {
                vulnerability_count = found.as_u64().unwrap_or(0) as u32;
            }
            if let Some(list) = json.get("vulnerabilities").and_then(|v| v.get("list"))
                && let Some(arr) = list.as_array()
            {
                for item in arr {
                    if let Some(severity) = item.get("severity").and_then(|s| s.as_str())
                        && severity.eq_ignore_ascii_case("critical")
                    {
                        critical_count += 1;
                    }
                }
            }
        }

        (vulnerability_count, critical_count)
    }
}

impl Collector for AuditCollector {
    fn name(&self) -> &'static str {
        "audit"
    }

    fn is_available(&self) -> bool {
        Command::new("cargo-audit")
            .arg("--version")
            .output()
            .map(|o| o.status.success())
            .unwrap_or(false)
    }

    fn collect(&self, ctx: &Context) -> Result<CollectorOutput, CollectorError> {
        let start = std::time::Instant::now();
        let output = Command::new("cargo-audit")
            .args(["--json"])
            .current_dir(&ctx.workspace_root)
            .output()
            .map_err(|e| CollectorError::IoError(e.to_string()))?;

        let duration_ms = start.elapsed().as_millis() as u64;
        let stdout = String::from_utf8_lossy(&output.stdout).to_string();
        let stderr = String::from_utf8_lossy(&output.stderr).to_string();

        let (vulnerability_count, _critical_count) = self.parse_json_output(&stdout);
        let status = if vulnerability_count == 0 {
            crate::schema::CollectorStatus::Pass
        } else {
            crate::schema::CollectorStatus::Fail
        };

        Ok(CollectorOutput {
            status,
            duration_ms,
            stdout,
            stderr,
        })
    }
}

impl Default for AuditCollector {
    fn default() -> Self {
        Self::new()
    }
}

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

    #[test]
    fn test_parse_json_output_no_vulns() {
        let collector = AuditCollector::new();
        let json = r#"{"vulnerabilities":{"found":0,"list":[]}}"#;
        let (vuln, critical) = collector.parse_json_output(json);
        assert_eq!(vuln, 0);
        assert_eq!(critical, 0);
    }

    #[test]
    fn test_parse_json_output_with_vulns() {
        let collector = AuditCollector::new();
        let json = r#"{"vulnerabilities":{"found":2,"list":[{"id":"RUSTSEC-0001","severity":"High"},{"id":"RUSTSEC-0002","severity":"critical"}]}}"#;
        let (vuln, critical) = collector.parse_json_output(json);
        assert_eq!(vuln, 2);
        assert_eq!(critical, 1);
    }
}