gobby-code 0.9.1

Fast Rust CLI for Gobby's code index — AST-aware search, symbol navigation, and dependency graph
Documentation
//! Content hashing for incremental indexing.
//! Ports logic from src/gobby/code_index/hasher.py.

use std::path::Path;

/// SHA-256 hash of entire file contents.
pub fn file_content_hash(path: &Path) -> anyhow::Result<String> {
    Ok(gobby_core::indexing::file_content_hash(path)?)
}

/// SHA-256 hash of a byte slice (symbol source).
pub fn symbol_content_hash(source: &[u8], start: usize, end: usize) -> anyhow::Result<String> {
    let slice = source.get(start..end).ok_or_else(|| {
        anyhow::anyhow!(
            "invalid byte range {}..{} for source len {}",
            start,
            end,
            source.len()
        )
    })?;
    Ok(gobby_core::indexing::content_hash(slice))
}

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

    #[test]
    fn file_content_hash_delegates_to_gobby_core() {
        let tmp = tempfile::NamedTempFile::new().expect("tempfile");
        std::fs::write(tmp.path(), b"hash me\n").expect("write file");

        let actual = file_content_hash(tmp.path()).expect("hash via wrapper");
        let expected =
            gobby_core::indexing::file_content_hash(tmp.path()).expect("hash via gobby-core");
        assert_eq!(actual, expected);

        let source = include_str!("hasher.rs");
        let delegate = ["gobby_core", "::indexing::file_content_hash"].concat();
        let local_buffer = format!("let mut buf = [0u8; {}]", 64 * 1024);
        assert!(source.contains(&delegate));
        assert!(!source.contains(&local_buffer));
    }
}