files.rs
110 lines
| 3.3 KiB
| application/rls-services+xml
|
RustLexer
Simon Sapin
|
r47163 | use crate::error::CommandError; | ||
Antoine Cezar
|
r45923 | use crate::ui::Ui; | ||
Simon Sapin
|
r49284 | use crate::utils::path_utils::RelativizePaths; | ||
Simon Sapin
|
r47251 | use clap::Arg; | ||
Simon Sapin
|
r49165 | use hg::errors::HgError; | ||
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", | ||||
)); | ||||
} | ||||
Simon Sapin
|
r47250 | if let Some(rev) = rev { | ||
Arseniy Alekseyev
|
r49238 | if repo.has_narrow() { | ||
return Err(CommandError::unsupported( | ||||
"rhg files -r <rev> is not supported in narrow clones", | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r50534 | let files = list_rev_tracked_files(repo, rev) | ||
.map_err(|e| (e, rev.as_ref()))?; | ||||
Simon Sapin
|
r47335 | display_files(invocation.ui, repo, files.iter()) | ||
Simon Sapin
|
r47250 | } else { | ||
Arseniy Alekseyev
|
r49238 | // The dirstate always reflects the sparse narrowspec, so if | ||
// we only have sparse without narrow all is fine. | ||||
// If we have narrow, then [hg files] needs to check if | ||||
// the store narrowspec is in sync with the one of the dirstate, | ||||
// so we can't support that without explicit code. | ||||
if repo.has_narrow() { | ||||
return Err(CommandError::unsupported( | ||||
"rhg files is not supported in narrow clones", | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r50875 | let dirstate = repo.dirstate_map()?; | ||
let files_res: Result<Vec<_>, _> = | ||||
filter_map_results(dirstate.iter(), |(path, entry)| { | ||||
Ok(if entry.tracked() { Some(path) } else { None }) | ||||
}) | ||||
.collect(); | ||||
let mut files = files_res?; | ||||
files.par_sort_unstable(); | ||||
Simon Sapin
|
r49165 | display_files(invocation.ui, repo, files.into_iter().map(Ok)) | ||
Antoine Cezar
|
r45923 | } | ||
} | ||||
Antoine Cezar
|
r46106 | |||
Simon Sapin
|
r47250 | fn display_files<'a>( | ||
ui: &Ui, | ||||
repo: &Repo, | ||||
Simon Sapin
|
r49165 | files: impl IntoIterator<Item = Result<&'a HgPath, HgError>>, | ||
Simon Sapin
|
r47250 | ) -> Result<(), CommandError> { | ||
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 | } | ||