use clap_complete::ArgValueCandidates;
use clap_complete::ArgValueCompleter;
use futures::TryStreamExt as _;
use futures::future::try_join_all;
use indexmap::IndexSet;
use itertools::Itertools as _;
use jj_lib::copies::CopyRecords;
use jj_lib::merge::Diff;
use jj_lib::repo::Repo as _;
use jj_lib::rewrite::merge_commit_trees;
use tracing::instrument;
use crate::cli_util::CommandHelper;
use crate::cli_util::RevisionArg;
use crate::cli_util::print_unmatched_explicit_paths;
use crate::cli_util::short_commit_hash;
use crate::command_error::CommandError;
use crate::command_error::user_error;
use crate::complete;
use crate::diff_util::DiffFormatArgs;
use crate::diff_util::get_copy_records;
use crate::diff_util::show_templated;
use crate::ui::Ui;
#[derive(clap::Args, Clone, Debug)]
#[command(mut_arg("ignore_all_space", |a| a.short('w')))]
#[command(mut_arg("ignore_space_change", |a| a.short('b')))]
pub(crate) struct DiffArgs {
#[arg(long, short, value_name = "REVSETS", alias = "revision")]
#[arg(add = ArgValueCompleter::new(complete::revset_expression_all))]
revisions: Option<Vec<RevisionArg>>,
#[arg(long, short, conflicts_with = "revisions", value_name = "REVSET")]
#[arg(add = ArgValueCompleter::new(complete::revset_expression_all))]
from: Option<RevisionArg>,
#[arg(long, short, conflicts_with = "revisions", value_name = "REVSET")]
#[arg(add = ArgValueCompleter::new(complete::revset_expression_all))]
to: Option<RevisionArg>,
#[arg(value_name = "FILESETS", value_hint = clap::ValueHint::AnyPath)]
#[arg(add = ArgValueCompleter::new(complete::modified_revision_or_range_files))]
paths: Vec<String>,
#[arg(
long,
short = 'T',
conflicts_with_all = ["short-format", "long-format", "tool"],
help_heading = "Diff Formatting Options",
)]
#[arg(add = ArgValueCandidates::new(complete::template_aliases))]
template: Option<String>,
#[command(flatten)]
format: DiffFormatArgs,
}
#[instrument(skip_all)]
pub(crate) async fn cmd_diff(
ui: &mut Ui,
command: &CommandHelper,
args: &DiffArgs,
) -> Result<(), CommandError> {
let workspace_command = command.workspace_helper(ui)?;
let repo = workspace_command.repo();
let fileset_expression = workspace_command.parse_file_patterns(ui, &args.paths)?;
let matcher = fileset_expression.to_matcher();
let from_tree;
let to_tree;
let mut copy_records = CopyRecords::default();
if args.from.is_some() || args.to.is_some() {
let resolve_revision = async |r: &Option<RevisionArg>| {
workspace_command
.resolve_single_rev(ui, r.as_ref().unwrap_or(&RevisionArg::AT))
.await
};
let from = resolve_revision(&args.from).await?;
let to = resolve_revision(&args.to).await?;
from_tree = from.tree();
to_tree = to.tree();
let records = get_copy_records(repo.store(), from.id(), to.id(), &matcher).await?;
copy_records.add_records(records);
} else {
let revision_args = args
.revisions
.as_deref()
.unwrap_or(std::slice::from_ref(&RevisionArg::AT));
let revisions_evaluator = workspace_command.parse_union_revsets(ui, revision_args)?;
let target_expression = revisions_evaluator.expression();
let mut gaps_revset = workspace_command
.attach_revset_evaluator(
target_expression
.roots()
.range(&target_expression.heads())
.minus(target_expression),
)
.evaluate_to_commit_ids()?;
if let Some(commit_id) = gaps_revset.try_next().await? {
return Err(
user_error("Cannot diff revsets with gaps in.").hinted(format!(
"Revision {} would need to be in the set.",
short_commit_hash(&commit_id)
)),
);
}
let heads: Vec<_> = workspace_command
.attach_revset_evaluator(target_expression.heads())
.evaluate_to_commits()?
.try_collect()
.await?;
let roots: Vec<_> = workspace_command
.attach_revset_evaluator(target_expression.roots())
.evaluate_to_commits()?
.try_collect()
.await?;
let parents: IndexSet<_> = try_join_all(roots.iter().map(|c| c.parents()))
.await?
.into_iter()
.flatten()
.collect();
let parents = parents.into_iter().collect_vec();
from_tree = merge_commit_trees(repo.as_ref(), &parents).await?;
to_tree = merge_commit_trees(repo.as_ref(), &heads).await?;
for p in &parents {
for to in &heads {
let records = get_copy_records(repo.store(), p.id(), to.id(), &matcher).await?;
copy_records.add_records(records);
}
}
}
let maybe_template;
let diff_renderer;
if let Some(text) = &args.template {
let language = workspace_command.commit_template_language();
let template = workspace_command
.parse_template(ui, &language, text)?
.labeled(["diff"]);
maybe_template = Some(template);
diff_renderer = workspace_command.diff_renderer(vec![]);
} else {
maybe_template = None;
diff_renderer = workspace_command.diff_renderer_for(&args.format)?;
}
ui.request_pager();
if let Some(template) = &maybe_template {
let tree_diff = from_tree.diff_stream_with_copies(&to_tree, &matcher, ©_records);
show_templated(ui.stdout_formatter().as_mut(), tree_diff, template).await?;
}
diff_renderer
.show_diff(
ui,
ui.stdout_formatter().as_mut(),
Diff::new(&from_tree, &to_tree),
&matcher,
©_records,
ui.term_width(),
)
.await?;
print_unmatched_explicit_paths(
ui,
&workspace_command,
&fileset_expression,
[&from_tree, &to_tree],
)?;
Ok(())
}