rippy-cli 0.2.0

A shell command safety hook for AI coding tools (Claude Code, Cursor, Gemini CLI) — Rust rewrite of Dippy
Documentation
use std::path::PathBuf;

use crate::resolve::{EnvLookup, VarLookup};

/// External environment dependencies for the analysis pipeline.
///
/// Groups all values that come from the OS environment so they can be
/// overridden in tests without manipulating env vars. Production code
/// creates this via [`Environment::from_system`]; tests use the builder
/// methods to inject specific values.
pub struct Environment {
    /// Home directory (`$HOME`). Used for `~/.rippy/config`, `~/.claude/settings`.
    /// `None` skips all home-based lookups.
    pub home: Option<PathBuf>,

    /// Working directory for the analysis (usually `std::env::current_dir()`).
    pub working_directory: PathBuf,

    /// Variable lookup for static expansion resolution.
    /// Defaults to `EnvLookup` (real `std::env::var`).
    pub var_lookup: Box<dyn VarLookup>,

    /// Whether the command originates from a remote context (e.g. `docker exec`).
    pub remote: bool,

    /// Emit tracing to stderr.
    pub verbose: bool,
}

impl Environment {
    /// Build from the real system environment.
    #[must_use]
    pub fn from_system(working_directory: PathBuf, remote: bool, verbose: bool) -> Self {
        Self {
            home: std::env::var_os("HOME").map(PathBuf::from),
            working_directory,
            var_lookup: Box::new(EnvLookup),
            remote,
            verbose,
        }
    }

    /// Build an isolated environment for tests: no home directory, no remote,
    /// no verbose, real env-var lookup for variable resolution.
    #[must_use]
    pub fn for_test(working_directory: PathBuf) -> Self {
        Self {
            home: None,
            working_directory,
            var_lookup: Box::new(EnvLookup),
            remote: false,
            verbose: false,
        }
    }

    /// Override the home directory (builder pattern).
    #[must_use]
    pub fn with_home(mut self, home: Option<PathBuf>) -> Self {
        self.home = home;
        self
    }

    /// Override the variable lookup (builder pattern).
    #[must_use]
    pub fn with_var_lookup(mut self, var_lookup: Box<dyn VarLookup>) -> Self {
        self.var_lookup = var_lookup;
        self
    }
}