rust/rhg/src/commands/files.rs
author Simon Sapin <simon.sapin@octobus.net>
Mon, 08 Mar 2021 08:35:43 +0100
changeset 46740 97ac588b6d9e
parent 46739 c184b490da37
child 46745 63bfcddddac1
permissions -rw-r--r--
rhg: Don’t make repository path absolute too early Some error messages want to include a relative path, which affects the output of some tests. Differential Revision: https://phab.mercurial-scm.org/D10138

use crate::error::CommandError;
use crate::ui::Ui;
use clap::Arg;
use hg::operations::list_rev_tracked_files;
use hg::operations::Dirstate;
use hg::repo::Repo;
use hg::utils::current_dir;
use hg::utils::files::{get_bytes_from_path, relativize_path};
use hg::utils::hg_path::{HgPath, HgPathBuf};

pub const HELP_TEXT: &str = "
List tracked files.

Returns 0 on success.
";

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)
}

pub fn run(invocation: &crate::CliInvocation) -> Result<(), CommandError> {
    let relative = invocation.config.get(b"ui", b"relative-paths");
    if relative.is_some() {
        return Err(CommandError::unsupported(
            "non-default ui.relative-paths",
        ));
    }

    let rev = invocation.subcommand_args.value_of("rev");

    let repo = invocation.repo?;
    if let Some(rev) = rev {
        let files = list_rev_tracked_files(repo, rev).map_err(|e| (e, rev))?;
        display_files(invocation.ui, repo, files.iter())
    } else {
        let distate = Dirstate::new(repo)?;
        let files = distate.tracked_files()?;
        display_files(invocation.ui, repo, files)
    }
}

fn display_files<'a>(
    ui: &Ui,
    repo: &Repo,
    files: impl IntoIterator<Item = &'a HgPath>,
) -> Result<(), CommandError> {
    let cwd = HgPathBuf::from(get_bytes_from_path(hg::utils::current_dir()?));
    let working_directory = repo.working_directory_path();
    let working_directory = current_dir()?.join(working_directory); // Make it absolute
    let working_directory =
        HgPathBuf::from(get_bytes_from_path(working_directory));

    let mut stdout = ui.stdout_buffer();

    for file in files {
        let file = working_directory.join(file);
        stdout.write_all(relativize_path(&file, &cwd).as_ref())?;
        stdout.write_all(b"\n")?;
    }
    stdout.flush()?;
    Ok(())
}