Skip to content

Commit

Permalink
feat: gix attribute query as something similar to git check-attrs.
Browse files Browse the repository at this point in the history
  • Loading branch information
Byron committed May 9, 2023
1 parent 450212e commit bfdcb14
Show file tree
Hide file tree
Showing 4 changed files with 116 additions and 0 deletions.
66 changes: 66 additions & 0 deletions gitoxide-core/src/repository/attributes.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
use std::io;

use anyhow::bail;
use gix::prelude::FindExt;

use crate::OutputFormat;

pub mod query {
use crate::OutputFormat;

pub struct Options {
pub format: OutputFormat,
pub statistics: bool,
}
}

pub fn query(
repo: gix::Repository,
pathspecs: impl Iterator<Item = gix::path::Spec>,
mut out: impl io::Write,
mut err: impl io::Write,
query::Options { format, statistics }: query::Options,
) -> anyhow::Result<()> {
if format != OutputFormat::Human {
bail!("JSON output isn't implemented yet");
}

let index = repo.index()?;
let mut cache = repo.attributes(
&index,
gix::worktree::cache::state::attributes::Source::WorktreeThenIdMapping,
gix::worktree::cache::state::ignore::Source::IdMapping,
None,
)?;

let prefix = repo.prefix().expect("worktree - we have an index by now")?;
let mut matches = cache.attribute_matches();

for mut spec in pathspecs {
for path in spec.apply_prefix(&prefix).items() {
let is_dir = gix::path::from_bstr(path).metadata().ok().map(|m| m.is_dir());
let entry = cache.at_entry(path, is_dir, |oid, buf| repo.objects.find_blob(oid, buf))?;

if !entry.matching_attributes(&mut matches) {
continue;
}
for m in matches.iter() {
writeln!(
out,
"{}:{}:{}\t{}\t{}",
m.location.source.map(|p| p.to_string_lossy()).unwrap_or_default(),
m.location.sequence_number,
m.pattern,
path,
m.assignment
)?;
}
}
}

if let Some(stats) = statistics.then(|| cache.take_statistics()) {
out.flush()?;
writeln!(err, "{:#?}", stats).ok();
}
Ok(())
}
1 change: 1 addition & 0 deletions gitoxide-core/src/repository/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ pub mod commit;
pub mod config;
mod credential;
pub use credential::function as credential;
pub mod attributes;
#[cfg(feature = "blocking-client")]
pub mod clone;
pub mod exclude;
Expand Down
29 changes: 29 additions & 0 deletions src/plumbing/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ use gitoxide_core as core;
use gitoxide_core::pack::verify;
use gix::bstr::io::BufReadExt;

use crate::plumbing::options::attributes;
use crate::{
plumbing::{
options::{commit, config, credential, exclude, free, index, mailmap, odb, revision, tree, Args, Subcommands},
Expand Down Expand Up @@ -831,6 +832,34 @@ pub fn main() -> Result<()> {
},
),
},
Subcommands::Attributes(cmd) => match cmd {
attributes::Subcommands::Query { statistics, pathspecs } => prepare_and_run(
"attributes-query",
verbose,
progress,
progress_keep_open,
None,
move |_progress, out, err| {
use gix::bstr::ByteSlice;
core::repository::attributes::query(
repository(Mode::Strict)?,
if pathspecs.is_empty() {
Box::new(
stdin_or_bail()?
.byte_lines()
.filter_map(Result::ok)
.filter_map(|line| gix::path::Spec::from_bytes(line.as_bstr())),
) as Box<dyn Iterator<Item = gix::path::Spec>>
} else {
Box::new(pathspecs.into_iter())
},
out,
err,
core::repository::attributes::query::Options { format, statistics },
)
},
),
},
Subcommands::Exclude(cmd) => match cmd {
exclude::Subcommands::Query {
patterns,
Expand Down
20 changes: 20 additions & 0 deletions src/plumbing/options/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,9 @@ pub enum Subcommands {
/// Interact with the remote hosts.
#[cfg(any(feature = "gitoxide-core-async-client", feature = "gitoxide-core-blocking-client"))]
Remote(remote::Platform),
/// Interact with the attribute files like .gitattributes.
#[clap(subcommand, visible_alias = "attrs")]
Attributes(attributes::Subcommands),
/// Interact with the exclude files like .gitignore.
#[clap(subcommand)]
Exclude(exclude::Subcommands),
Expand Down Expand Up @@ -440,6 +443,23 @@ pub mod revision {
}
}

pub mod attributes {
use crate::shared::AsPathSpec;

#[derive(Debug, clap::Subcommand)]
pub enum Subcommands {
/// List all attributes of the given path-specs and display the result similar to `git check-attr`.
Query {
/// Print various statistics to stderr
#[clap(long, short = 's')]
statistics: bool,
/// The git path specifications to list attributes for, or unset to read from stdin one per line.
#[clap(value_parser = AsPathSpec)]
pathspecs: Vec<gix::path::Spec>,
},
}
}

pub mod exclude {
use std::ffi::OsString;

Expand Down

0 comments on commit bfdcb14

Please sign in to comment.