Skip to content

Pathspec support for magic words #1132

Open
@darius-sas

Description

@darius-sas

Hi all, I'm using git2-rs's diff API interface and I'm trying to filter out some paths using pathspec.

From Git's user manual, pathspecs should support "magic words", which would allow me to write something like this :(exclude)*.toml to exclude all toml files when computing the diff. This is important to me because it allows to save a lot of computation time wasted on files I do not care about (and have to filter out manually after the diff).

When using Git, this works as expected, however, I can't replicate the same behaviour using git2-rs from my Rust application.

Any idea on how to workaround on this issue? Am I using the APIs wrongly? Or even pointers on how to implement this in the library (I'm a newbie to Rust, but I might give it a try...).

Here's an example Rust application that showcases the problem.

use std::{env::args, error::Error};

use git2::{DiffFindOptions, DiffOptions, ObjectType, Repository};

pub fn diff(
    repo: &Repository,
    parent: &str,
    child: &str,
    pathspecs: &[String],
) -> Result<(), git2::Error> {
    let parent = repo.revparse_single(parent)?;
    let child = repo.revparse_single(child)?;
    let p_obj = parent.peel(ObjectType::Tree)?;
    let c_obj = child.peel(ObjectType::Tree)?;
    let p_tree = p_obj.as_tree().unwrap();
    let c_tree = c_obj.as_tree().unwrap();
    let mut rename_opts = DiffFindOptions::new();
    rename_opts.renames(true);
    rename_opts.rename_limit(200);
    rename_opts.renames_from_rewrites(true);
    let mut opts = DiffOptions::new();
    opts.update_index(false);
    opts.context_lines(0);
    for p in pathspecs {
        opts.pathspec(p);
    }
    let mut diff = repo.diff_tree_to_tree(Some(p_tree), Some(c_tree), Some(&mut opts))?;
    diff.find_similar(Some(&mut rename_opts))?;
    for delta in diff.deltas() {
        println!("{}", delta.old_file().path().unwrap().to_str().unwrap());
    }
    Ok(())
}

fn main() -> Result<(), Box<dyn Error>> {
    let mut args = args();
    if args.len() <= 3 {
        println!("Usage: <repo> <sha> <sha> [pathspec...]");
        return Err("Wrong number of arguments".into());
    }
    args.next();
    let repo = args.next().expect("Repository must be provided");
    let sha_parent = args.next().expect("Parent sha must be provided");
    let sha_child = args.next().expect("Child sha must be provided");
    let pathspecs = args.collect::<Vec<String>>();
    let repo = Repository::open(repo)?;
    diff(&repo, &sha_parent, &sha_child, pathspecs.as_slice())?;
    Ok(())
}

To illustrate the problem, you can follow these steps (after setting up the sample cargo app above):

  1. Run Git on an example repository
git clone https://github.com/rust-lang/git2-rs /tmp/git2-rs
git -C /tmp/git2-rs/ diff --name-only d1ae3b6c 7b7be439 ":(exclude)*.toml"

Which outputs

libgit2-sys/lib.rs
src/repo.rs

Note that also Cargo.toml and git2-curl/Cargo.toml have changed, but they are not part of the diff.

  1. Run sample app
cargo run -- /tmp/git2-rs/ d1ae3b6c 7b7be439 ":(exclude)*.toml"

No output is display at all if you provide a pathspec with any magic word. In this case, the expected output is the two .rs files seen above.

Morover, combinations like these seem to fail too:

cargo run -- /tmp/git2-rs/ d1ae3b6c 7b7be439 "*.toml" ":(exclude)*.toml"
# Outputs
Cargo.toml
git2-curl/Cargo.toml

Git correctly prints out nothing.

Thank you! Looking forward to your answers :)

Metadata

Metadata

Assignees

No one assigned

    Labels

    upstreamAn upstream issue with libgit2

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions