vtcode_commons/walk.rs
1//! Shared directory walker helpers built on the `ignore` crate.
2//!
3//! All file traversal in vtcode should go through these builders so that
4//! `.gitignore`, `.ignore`, `.git/exclude`, and the centralized exclusion
5//! constants are applied consistently.
6
7use ignore::{DirEntry, WalkBuilder};
8use std::path::Path;
9
10use crate::exclusions::DEFAULT_EXCLUDED_DIRS;
11
12/// Build a multi-threaded [`WalkBuilder`] with sensible defaults.
13///
14/// - Respects `.gitignore`, `.ignore`, `.git/exclude`, and parent ignore files
15/// - Does not follow symlinks
16/// - Uses the `ignore` crate's default thread pool
17///
18/// Callers that need to prune additional directories should use
19/// [`filter_entry`](WalkBuilder::filter_entry) with [`is_excluded_dir`].
20pub fn build_default_walker(root: &Path) -> WalkBuilder {
21 let mut builder = WalkBuilder::new(root);
22 apply_defaults(&mut builder);
23 builder
24}
25
26/// Build a single-threaded [`WalkBuilder`] with the same defaults as
27/// [`build_default_walker`].
28///
29/// Use this in synchronous contexts where spawning the `ignore` crate's
30/// thread pool would be wasteful (e.g., inside `spawn_blocking` closures
31/// that already run on a dedicated thread).
32pub fn build_walker_single_threaded(root: &Path) -> WalkBuilder {
33 let mut builder = WalkBuilder::new(root);
34 builder.threads(1);
35 apply_defaults(&mut builder);
36 builder
37}
38
39fn apply_defaults(builder: &mut WalkBuilder) {
40 // Respect all standard ignore-file mechanisms.
41 builder.git_ignore(true);
42 builder.git_global(true);
43 builder.git_exclude(true);
44 builder.ignore(true);
45 builder.parents(true);
46
47 // Do not follow symlinks by default.
48 builder.follow_links(false);
49
50 // Do not skip hidden files by default. The `ignore` crate skips them
51 // by default, but the previous traversal code did not. Callers that
52 // want to hide dotfiles should filter them explicitly.
53 builder.hidden(false);
54}
55
56/// Returns `true` if `entry` is a directory whose name appears in
57/// [`DEFAULT_EXCLUDED_DIRS`].
58///
59/// Intended for use inside [`WalkBuilder::filter_entry`] closures:
60///
61/// ```ignore
62/// builder.filter_entry(|entry| !vtcode_commons::walk::is_excluded_dir(entry));
63/// ```
64pub fn is_excluded_dir(entry: &DirEntry) -> bool {
65 if !entry.file_type().is_some_and(|ft| ft.is_dir()) {
66 return false;
67 }
68
69 entry
70 .file_name()
71 .to_str()
72 .is_some_and(|name| DEFAULT_EXCLUDED_DIRS.contains(&name))
73}