Mercurial > hg-stable
view rust/rhg/src/commands/files.rs @ 52182:fa58f4f97337 stable tip
ci: shard the test run on mac os X
This should comes with some benefit:
- spread the load across more runner,
- reduce the real-time CI run,
- reduce the "retry" run when we need them.
We start with the Mac jobs, but that would be tremendously useful for Windows
too.
For linux, we need to reduce the startup overhead for this to be worth it.
Building smaller image and speeding up clone should help with that.
author | Pierre-Yves David <pierre-yves.david@octobus.net> |
---|---|
date | Fri, 08 Nov 2024 17:08:11 +0100 |
parents | 7c105b953ca4 |
children |
line wrap: on
line source
use crate::error::CommandError; use crate::ui::{ print_narrow_sparse_warnings, relative_paths, RelativePaths, Ui, }; use crate::utils::path_utils::RelativizePaths; use clap::Arg; use hg::filepatterns::parse_pattern_args; use hg::matchers::IntersectionMatcher; use hg::narrow; use hg::operations::list_revset_tracked_files; use hg::repo::Repo; use hg::utils::files::get_bytes_from_os_str; use hg::utils::filter_map_results; use hg::utils::hg_path::HgPath; use rayon::prelude::*; pub const HELP_TEXT: &str = " List tracked files. Returns 0 on success. "; pub fn args() -> clap::Command { clap::command!("files") .arg( Arg::new("rev") .help("search the repository as it is in REV") .short('r') .long("revision") .value_name("REV"), ) .arg( Arg::new("file") .value_parser(clap::value_parser!(std::ffi::OsString)) .help("show only these files") .action(clap::ArgAction::Append), ) .about(HELP_TEXT) } pub fn run(invocation: &crate::CliInvocation) -> Result<(), CommandError> { let relative_paths = match relative_paths(invocation.config)? { RelativePaths::Legacy => true, RelativePaths::Bool(v) => v, }; let args = invocation.subcommand_args; let rev = args.get_one::<String>("rev"); let repo = invocation.repo?; // 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", )); } let (matcher, narrow_warnings) = narrow::matcher(repo)?; print_narrow_sparse_warnings(&narrow_warnings, &[], invocation.ui, repo)?; let matcher = match args.get_many::<std::ffi::OsString>("file") { None => matcher, Some(files) => { let patterns: Vec<Vec<u8>> = files .filter(|s| !s.is_empty()) .map(get_bytes_from_os_str) .collect(); for file in &patterns { if file.starts_with(b"set:") { return Err(CommandError::unsupported("fileset")); } } let cwd = hg::utils::current_dir()?; let root = repo.working_directory_path(); let ignore_patterns = parse_pattern_args(patterns, &cwd, root)?; let files_matcher = hg::matchers::PatternMatcher::new(ignore_patterns)?; Box::new(IntersectionMatcher::new( Box::new(files_matcher), matcher, )) } }; if let Some(rev) = rev { let files = list_revset_tracked_files(repo, rev, matcher) .map_err(|e| (e, rev.as_ref()))?; display_files( invocation.ui, repo, relative_paths, files.iter().map::<Result<_, CommandError>, _>(|f| { let (f, _, _) = f?; Ok(f) }), ) } else { // The dirstate always reflects the sparse narrowspec. let dirstate = repo.dirstate_map()?; let files_res: Result<Vec<_>, _> = filter_map_results(dirstate.iter(), |(path, entry)| { Ok(if entry.tracked() && matcher.matches(path) { Some(path) } else { None }) }) .collect(); let mut files = files_res?; files.par_sort_unstable(); display_files( invocation.ui, repo, relative_paths, files.into_iter().map::<Result<_, CommandError>, _>(Ok), ) } } fn display_files<'a, E>( ui: &Ui, repo: &Repo, relative_paths: bool, files: impl IntoIterator<Item = Result<&'a HgPath, E>>, ) -> Result<(), CommandError> where CommandError: From<E>, { let mut stdout = ui.stdout_buffer(); let mut any = false; let relativize = RelativizePaths::new(repo)?; for result in files { let path = result?; if relative_paths { stdout.write_all(&relativize.relativize(path))?; } else { stdout.write_all(path.as_bytes())?; } stdout.write_all(b"\n")?; any = true; } stdout.flush()?; if any { Ok(()) } else { Err(CommandError::Unsuccessful) } }