use clap_complete::ArgValueCandidates;
use clap_complete::ArgValueCompleter;
use jj_lib::annotate::FileAnnotation;
use jj_lib::annotate::FileAnnotator;
use jj_lib::annotate::LineOrigin;
use jj_lib::repo::Repo;
use jj_lib::revset::RevsetExpression;
use tracing::instrument;
use crate::cli_util::CommandHelper;
use crate::cli_util::RevisionArg;
use crate::command_error::CommandError;
use crate::command_error::user_error;
use crate::commit_templater::AnnotationLine;
use crate::complete;
use crate::templater::TemplateRenderer;
use crate::ui::Ui;
#[derive(clap::Args, Clone, Debug)]
pub(crate) struct FileAnnotateArgs {
#[arg(value_hint = clap::ValueHint::AnyPath)]
#[arg(add = ArgValueCompleter::new(complete::all_revision_files))]
path: String,
#[arg(long, short, value_name = "REVSET")]
#[arg(add = ArgValueCompleter::new(complete::revset_expression_all))]
revision: Option<RevisionArg>,
#[arg(long, short = 'T')]
#[arg(add = ArgValueCandidates::new(complete::template_aliases))]
template: Option<String>,
}
#[instrument(skip_all)]
pub(crate) async fn cmd_file_annotate(
ui: &mut Ui,
command: &CommandHelper,
args: &FileAnnotateArgs,
) -> Result<(), CommandError> {
let workspace_command = command.workspace_helper(ui)?;
let repo = workspace_command.repo();
let starting_commit = workspace_command
.resolve_single_rev(ui, args.revision.as_ref().unwrap_or(&RevisionArg::AT))
.await?;
let file_path = workspace_command.parse_file_path(&args.path)?;
let file_value = starting_commit.tree().path_value(&file_path).await?;
let ui_path = workspace_command.format_file_path(&file_path);
if file_value.is_absent() {
return Err(user_error(format!("No such path: {ui_path}")));
}
if file_value.to_file_merge().is_none() {
return Err(user_error(format!(
"Path exists but is not a regular file: {ui_path}"
)));
}
let template_text = match &args.template {
Some(value) => value.clone(),
None => workspace_command
.settings()
.get_string("templates.file_annotate")?,
};
let language = workspace_command.commit_template_language();
let template = workspace_command.parse_template(ui, &language, &template_text)?;
let mut annotator = FileAnnotator::from_commit(&starting_commit, &file_path).await?;
annotator
.compute(repo.as_ref(), &RevsetExpression::all())
.await?;
let annotation = annotator.to_annotation();
render_file_annotation(repo.as_ref(), ui, &template, &annotation).await?;
Ok(())
}
async fn render_file_annotation(
repo: &dyn Repo,
ui: &mut Ui,
template_render: &TemplateRenderer<'_, AnnotationLine>,
annotation: &FileAnnotation,
) -> Result<(), CommandError> {
ui.request_pager();
let mut formatter = ui.stdout_formatter();
let mut last_id = None;
let default_line_origin = LineOrigin {
commit_id: repo.store().root_commit_id().clone(),
line_number: 0,
};
for (line_number, (line_origin, content)) in annotation.line_origins().enumerate() {
let line_origin = line_origin.unwrap_or(&default_line_origin);
let commit = repo
.store()
.get_commit_async(&line_origin.commit_id)
.await?;
let first_line_in_hunk = last_id != Some(&line_origin.commit_id);
let annotation_line = AnnotationLine {
commit,
content: content.to_owned(),
line_number: line_number + 1,
original_line_number: line_origin.line_number + 1,
first_line_in_hunk,
};
template_render.format(&annotation_line, formatter.as_mut())?;
last_id = Some(&line_origin.commit_id);
}
Ok(())
}