files.rs
110 lines
| 3.1 KiB
| application/rls-services+xml
|
RustLexer
Simon Sapin
|
r47163 | use crate::error::CommandError; | ||
Raphaël Gomès
|
r50879 | use crate::ui::{print_narrow_sparse_warnings, Ui}; | ||
Simon Sapin
|
r49284 | use crate::utils::path_utils::RelativizePaths; | ||
Simon Sapin
|
r47251 | use clap::Arg; | ||
Raphaël Gomès
|
r50879 | use hg::narrow; | ||
Simon Sapin
|
r47165 | use hg::operations::list_rev_tracked_files; | ||
Simon Sapin
|
r46782 | use hg::repo::Repo; | ||
Raphaël Gomès
|
r50875 | use hg::utils::filter_map_results; | ||
Pulkit Goyal
|
r48988 | use hg::utils::hg_path::HgPath; | ||
Raphaël Gomès
|
r50875 | use rayon::prelude::*; | ||
Antoine Cezar
|
r45923 | |||
pub const HELP_TEXT: &str = " | ||||
List tracked files. | ||||
Returns 0 on success. | ||||
"; | ||||
Raphaël Gomès
|
r50534 | pub fn args() -> clap::Command { | ||
clap::command!("files") | ||||
Simon Sapin
|
r47251 | .arg( | ||
Raphaël Gomès
|
r50534 | Arg::new("rev") | ||
Simon Sapin
|
r47251 | .help("search the repository as it is in REV") | ||
Raphaël Gomès
|
r50534 | .short('r') | ||
.long("revision") | ||||
.value_name("REV"), | ||||
Simon Sapin
|
r47251 | ) | ||
.about(HELP_TEXT) | ||||
} | ||||
Simon Sapin
|
r47334 | pub fn run(invocation: &crate::CliInvocation) -> Result<(), CommandError> { | ||
Simon Sapin
|
r47473 | let relative = invocation.config.get(b"ui", b"relative-paths"); | ||
if relative.is_some() { | ||||
return Err(CommandError::unsupported( | ||||
"non-default ui.relative-paths", | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r50534 | let rev = invocation.subcommand_args.get_one::<String>("rev"); | ||
Antoine Cezar
|
r45923 | |||
Simon Sapin
|
r47335 | let repo = invocation.repo?; | ||
Arseniy Alekseyev
|
r49238 | |||
// It seems better if this check is removed: this would correspond to | ||||
// automatically enabling the extension if the repo requires it. | ||||
// However we need this check to be in sync with vanilla hg so hg tests | ||||
// pass. | ||||
if repo.has_sparse() | ||||
&& invocation.config.get(b"extensions", b"sparse").is_none() | ||||
{ | ||||
return Err(CommandError::unsupported( | ||||
"repo is using sparse, but sparse extension is not enabled", | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r50880 | let (narrow_matcher, narrow_warnings) = narrow::matcher(repo)?; | ||
print_narrow_sparse_warnings(&narrow_warnings, &[], invocation.ui, repo)?; | ||||
Simon Sapin
|
r47250 | if let Some(rev) = rev { | ||
Raphaël Gomès
|
r50880 | let files = list_rev_tracked_files(repo, rev, narrow_matcher) | ||
Raphaël Gomès
|
r50534 | .map_err(|e| (e, rev.as_ref()))?; | ||
Simon Sapin
|
r47335 | display_files(invocation.ui, repo, files.iter()) | ||
Simon Sapin
|
r47250 | } else { | ||
Raphaël Gomès
|
r50879 | // The dirstate always reflects the sparse narrowspec. | ||
Raphaël Gomès
|
r50875 | let dirstate = repo.dirstate_map()?; | ||
let files_res: Result<Vec<_>, _> = | ||||
filter_map_results(dirstate.iter(), |(path, entry)| { | ||||
Raphaël Gomès
|
r50879 | Ok(if entry.tracked() && narrow_matcher.matches(path) { | ||
Some(path) | ||||
} else { | ||||
None | ||||
}) | ||||
Raphaël Gomès
|
r50875 | }) | ||
.collect(); | ||||
let mut files = files_res?; | ||||
files.par_sort_unstable(); | ||||
Raphaël Gomès
|
r50879 | display_files( | ||
invocation.ui, | ||||
repo, | ||||
files.into_iter().map::<Result<_, CommandError>, _>(Ok), | ||||
) | ||||
Antoine Cezar
|
r45923 | } | ||
} | ||||
Antoine Cezar
|
r46106 | |||
Raphaël Gomès
|
r50878 | fn display_files<'a, E>( | ||
Simon Sapin
|
r47250 | ui: &Ui, | ||
repo: &Repo, | ||||
Raphaël Gomès
|
r50878 | files: impl IntoIterator<Item = Result<&'a HgPath, E>>, | ||
) -> Result<(), CommandError> | ||||
where | ||||
CommandError: From<E>, | ||||
{ | ||||
Simon Sapin
|
r47250 | let mut stdout = ui.stdout_buffer(); | ||
Pulkit Goyal
|
r48988 | let mut any = false; | ||
Simon Sapin
|
r47687 | |||
Simon Sapin
|
r49284 | let relativize = RelativizePaths::new(repo)?; | ||
for result in files { | ||||
let path = result?; | ||||
stdout.write_all(&relativize.relativize(path))?; | ||||
stdout.write_all(b"\n")?; | ||||
Pulkit Goyal
|
r48988 | any = true; | ||
Simon Sapin
|
r49284 | } | ||
Simon Sapin
|
r47250 | stdout.flush()?; | ||
Simon Sapin
|
r47479 | if any { | ||
Ok(()) | ||||
} else { | ||||
Err(CommandError::Unsuccessful) | ||||
} | ||||
Antoine Cezar
|
r46108 | } | ||