blob: c2600227a332b39172c2e6b2e7c092ee81701da4 [file] [log] [blame]
use crate::bstr::{BStr, BString};
use crate::util::OwnedOrStaticAtomicBool;
use crate::worktree::IndexPersistedOrInMemory;
use crate::{config, dirwalk, Repository};
use std::sync::atomic::AtomicBool;
impl Repository {
/// Return default options suitable for performing a directory walk on this repository.
///
/// Used in conjunction with [`dirwalk()`](Self::dirwalk())
pub fn dirwalk_options(&self) -> Result<dirwalk::Options, config::boolean::Error> {
Ok(dirwalk::Options::from_fs_caps(self.filesystem_options()?))
}
/// Perform a directory walk configured with `options` under control of the `delegate`. Use `patterns` to
/// further filter entries. `should_interrupt` is polled to see if an interrupt is requested, causing an
/// error to be returned instead.
///
/// The `index` is used to determine if entries are tracked, and for excludes and attributes
/// lookup. Note that items will only count as tracked if they have the [`gix_index::entry::Flags::UPTODATE`]
/// flag set.
///
/// See [`gix_dir::walk::delegate::Collect`] for a delegate that collects all seen entries.
pub fn dirwalk(
&self,
index: &gix_index::State,
patterns: impl IntoIterator<Item = impl AsRef<BStr>>,
should_interrupt: &AtomicBool,
options: dirwalk::Options,
delegate: &mut dyn gix_dir::walk::Delegate,
) -> Result<dirwalk::Outcome<'_>, dirwalk::Error> {
let _span = gix_trace::coarse!("gix::dirwalk");
let workdir = self.work_dir().ok_or(dirwalk::Error::MissingWorkDir)?;
let mut excludes = self.excludes(
index,
None,
crate::worktree::stack::state::ignore::Source::WorktreeThenIdMappingIfNotSkipped,
)?;
let mut pathspec = self.pathspec(
options.empty_patterns_match_prefix, /* empty patterns match prefix */
patterns,
true, /* inherit ignore case */
index,
crate::worktree::stack::state::attributes::Source::WorktreeThenIdMapping,
)?;
let git_dir_realpath =
crate::path::realpath_opts(self.git_dir(), self.current_dir(), crate::path::realpath::MAX_SYMLINKS)?;
let fs_caps = self.filesystem_options()?;
let accelerate_lookup = fs_caps.ignore_case.then(|| index.prepare_icase_backing());
let (outcome, traversal_root) = gix_dir::walk(
workdir,
gix_dir::walk::Context {
should_interrupt: Some(should_interrupt),
git_dir_realpath: git_dir_realpath.as_ref(),
current_dir: self.current_dir(),
index,
ignore_case_index_lookup: accelerate_lookup.as_ref(),
pathspec: &mut pathspec.search,
pathspec_attributes: &mut |relative_path, case, is_dir, out| {
let stack = pathspec
.stack
.as_mut()
.expect("can only be called if attributes are used in patterns");
stack
.set_case(case)
.at_entry(relative_path, Some(is_dir), &self.objects)
.map_or(false, |platform| platform.matching_attributes(out))
},
excludes: Some(&mut excludes.inner),
objects: &self.objects,
explicit_traversal_root: (!options.empty_patterns_match_prefix).then_some(workdir),
},
options.into(),
delegate,
)?;
Ok(dirwalk::Outcome {
dirwalk: outcome,
traversal_root,
excludes,
pathspec,
})
}
/// Create an iterator over a running traversal, which stops if the iterator is dropped. All arguments
/// are the same as in [`dirwalk()`](Self::dirwalk).
///
/// `should_interrupt` should be set to `Default::default()` if it is supposed to be unused.
/// Otherwise, it can be created by passing a `&'static AtomicBool`, `&Arc<AtomicBool>` or `Arc<AtomicBool>`.
pub fn dirwalk_iter(
&self,
index: impl Into<IndexPersistedOrInMemory>,
patterns: impl IntoIterator<Item = impl Into<BString>>,
should_interrupt: OwnedOrStaticAtomicBool,
options: dirwalk::Options,
) -> Result<dirwalk::Iter, dirwalk::iter::Error> {
dirwalk::Iter::new(
self,
index.into(),
patterns.into_iter().map(Into::into).collect(),
should_interrupt,
options,
)
}
}