From 874ee11d82b2828f1414a9a6ed32ed6a8486e6f4 Mon Sep 17 00:00:00 2001 From: Johannes Altmanninger Date: Mon, 4 Oct 2021 22:01:20 +0200 Subject: [PATCH] Fix blame when opened from an initial diff view (#1135) When running "tig show 529182c -- README.adoc", blames were started on HEAD instead of 529182c. Fix this by reading the commit line of the displayed diff. This also works when multiple commits are shown ("tig show --all"). Using %(commit) instead of parsing the commit line did unfortunately not work because, because %(commit) can contain the wrong SHA for when returning from a blame view to a stash view, see test/blame/stash-test. Sadly, stashes don't contain commit lines, so we need to find out their SHAs in another way. Once we fix %(commit), we can use it, but for now this commit adds a workaround to use the old "view->vid", which seems to be set in the cases where display the output from "git stash show". Also, when blame is requested on a context line, use the current commit rather than its parent. This does not change behavior since a context line can never be blamed on the current commit, but this is consistent with the stage view, which only uses HEAD for deletions and the index for others. [ja: add test and commit message] Co-authored-by: Thomas Koutcher --- src/diff.c | 39 +++++++++++++--------------- test/blame/initial-diff-test | 28 +++++++++++++++++++++ test/blame/stash-test | 49 ++++++++++++++++++++++++++++++++++++ 3 files changed, 95 insertions(+), 21 deletions(-) create mode 100755 test/blame/initial-diff-test create mode 100755 test/blame/stash-test diff --git a/src/diff.c b/src/diff.c index 8d2f8ab07..4b1bb5a84 100644 --- a/src/diff.c +++ b/src/diff.c @@ -635,6 +635,8 @@ diff_get_lineno(struct view *view, struct line *line, bool old) static enum request diff_trace_origin(struct view *view, struct line *line) { + struct line *commit_line = find_prev_line_by_type(view, line, LINE_COMMIT); + char id[SIZEOF_REV]; struct line *diff = find_prev_line_by_type(view, line, LINE_DIFF_HEADER); struct line *chunk = find_prev_line_by_type(view, line, LINE_DIFF_CHUNK); const char *chunk_data; @@ -645,7 +647,7 @@ diff_trace_origin(struct view *view, struct line *line) struct blame_header header; struct blame_commit commit; - if (!diff || !chunk || chunk == line) { + if (!diff || !chunk || chunk == line || diff < commit_line) { report("The line to trace must be inside a diff chunk"); return REQ_NONE; } @@ -686,27 +688,25 @@ diff_trace_origin(struct view *view, struct line *line) } } - if (chunk_marker == '+') - string_copy(ref, view->vid); + if (commit_line) + string_copy_rev_from_commit_line(id, box_text(commit_line)); else - string_format(ref, "%s^", view->vid); + string_copy(id, view->vid); - if (string_rev_is_null(ref)) { - string_ncopy(view->env->file, file, strlen(file)); - string_copy(view->env->ref, ""); - view->env->goto_lineno = lineno - 1; - - } else { - if (!diff_blame_line(ref, file, lineno, &header, &commit)) { - report("Failed to read blame data"); - return REQ_NONE; - } + if (chunk_marker == '-') + string_format(ref, "%s^", id); + else + string_copy(ref, id); - string_ncopy(view->env->file, commit.filename, strlen(commit.filename)); - string_copy(view->env->ref, header.id); - view->env->goto_lineno = header.orig_lineno - 1; + if (!diff_blame_line(ref, file, lineno, &header, &commit)) { + report("Failed to read blame data"); + return REQ_NONE; } + string_ncopy(view->env->file, commit.filename, strlen(commit.filename)); + string_copy(view->env->ref, header.id); + view->env->goto_lineno = header.orig_lineno - 1; + return REQ_VIEW_BLAME; } @@ -791,10 +791,7 @@ diff_request(struct view *view, enum request request, struct line *line) return diff_common_enter(view, request, line); case REQ_REFRESH: - if (string_rev_is_null(view->vid)) - refresh_view(view); - else - reload_view(view); + reload_view(view); return REQ_NONE; default: diff --git a/test/blame/initial-diff-test b/test/blame/initial-diff-test new file mode 100755 index 000000000..8bef3a750 --- /dev/null +++ b/test/blame/initial-diff-test @@ -0,0 +1,28 @@ +#!/bin/sh + +. libtest.sh +. libgit.sh + +steps ' + :save-display initial-diff.screen + :20 # Move to a deleted line. + :view-blame + :scroll-right + :save-display blame-deleted-line.screen +' + +in_work_dir create_repo_from_tgz "$base_dir/files/scala-js-benchmarks.tgz" + +test_tig show a1dcf1a + +sed -i.old -n /asInstanceOf/p *.screen + +assert_equals 'initial-diff.screen' < file + git add file + git commit -m "Initial commit" + echo "changed line" > file + git stash +} + +LINES=3 test_tig + +assert_equals 'diff.screen' <