files.rs
101 lines
| 3.1 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; | ||
use hg::operations::Dirstate; | ||||
Simon Sapin
|
r46782 | use hg::repo::Repo; | ||
Pulkit Goyal
|
r48988 | use hg::utils::hg_path::HgPath; | ||
Antoine Cezar
|
r45923 | |||
pub const HELP_TEXT: &str = " | ||||
List tracked files. | ||||
Returns 0 on success. | ||||
"; | ||||
Simon Sapin
|
r47251 | pub fn args() -> clap::App<'static, 'static> { | ||
clap::SubCommand::with_name("files") | ||||
.arg( | ||||
Arg::with_name("rev") | ||||
.help("search the repository as it is in REV") | ||||
.short("-r") | ||||
.long("--revision") | ||||
.value_name("REV") | ||||
.takes_value(true), | ||||
) | ||||
.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", | ||||
)); | ||||
} | ||||
Simon Sapin
|
r47334 | let rev = invocation.subcommand_args.value_of("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", | ||||
)); | ||||
} | ||||
Simon Sapin
|
r47335 | let files = list_rev_tracked_files(repo, rev).map_err(|e| (e, rev))?; | ||
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", | ||||
)); | ||||
} | ||||
Simon Sapin
|
r47335 | let distate = Dirstate::new(repo)?; | ||
Simon Sapin
|
r47250 | let files = distate.tracked_files()?; | ||
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 | } | ||