Merge branch 'mm/line-log-cleanup' into next

The `git log -L` implementation has been refactored to use the
standard diff output pipeline, enabling pickaxe and diff-filter to
work as expected. Additionally, metadata-only diff formats like
--raw and --name-only are now supported with -L.

* mm/line-log-cleanup:
  line-log: allow non-patch diff formats with -L
  line-log: integrate -L output with the standard log-tree pipeline
  revision: move -L setup before output_format-to-diff derivation
This commit is contained in:
Junio C Hamano
2026-06-04 08:14:24 +09:00
8 changed files with 120 additions and 56 deletions

View File

@@ -8,12 +8,14 @@
give zero or one positive revision arguments, and
_<start>_ and _<end>_ (or _<funcname>_) must exist in the starting revision.
You can specify this option more than once. Implies `--patch`.
Patch output can be suppressed using `--no-patch`, but other diff formats
(namely `--raw`, `--numstat`, `--shortstat`, `--dirstat`, `--summary`,
`--name-only`, `--name-status`, `--check`) are not currently implemented.
Patch output can be suppressed using `--no-patch`.
Non-patch diff formats `--raw`, `--name-only`, `--name-status`,
and `--summary` are supported. Diff stat formats
(`--stat`, `--numstat`, `--shortstat`, `--dirstat`) are not
currently implemented.
+
Patch formatting options such as `--word-diff`, `--color-moved`,
`--no-prefix`, and whitespace options (`-w`, `-b`) are supported,
as are pickaxe options (`-S`, `-G`).
as are pickaxe options (`-S`, `-G`) and `--diff-filter`.
+
include::line-range-format.adoc[]

View File

@@ -13,7 +13,6 @@
#include "revision.h"
#include "xdiff-interface.h"
#include "strbuf.h"
#include "log-tree.h"
#include "line-log.h"
#include "setup.h"
#include "strvec.h"
@@ -1004,29 +1003,18 @@ static int process_all_files(struct line_log_data **range_out,
return changed;
}
int line_log_print(struct rev_info *rev, struct commit *commit)
void line_log_queue_pairs(struct rev_info *rev, struct commit *commit)
{
show_log(rev);
if (!(rev->diffopt.output_format & DIFF_FORMAT_NO_OUTPUT)) {
struct line_log_data *range = lookup_line_range(rev, commit);
struct line_log_data *r;
const char *prefix = diff_line_prefix(&rev->diffopt);
struct line_log_data *range = lookup_line_range(rev, commit);
struct line_log_data *r;
fprintf(rev->diffopt.file, "%s\n", prefix);
for (r = range; r; r = r->next) {
if (r->pair) {
struct diff_filepair *p =
diff_filepair_dup(r->pair);
p->line_ranges = &r->ranges;
diff_q(&diff_queued_diff, p);
}
for (r = range; r; r = r->next) {
if (r->pair) {
struct diff_filepair *p = diff_filepair_dup(r->pair);
p->line_ranges = &r->ranges;
diff_q(&diff_queued_diff, p);
}
diffcore_std(&rev->diffopt);
diff_flush(&rev->diffopt);
}
return 1;
}
static int bloom_filter_check(struct rev_info *rev,

View File

@@ -46,7 +46,7 @@ int line_log_filter(struct rev_info *rev);
int line_log_process_ranges_arbitrary_commit(struct rev_info *rev,
struct commit *commit);
int line_log_print(struct rev_info *rev, struct commit *commit);
void line_log_queue_pairs(struct rev_info *rev, struct commit *commit);
void line_log_free(struct rev_info *rev);

View File

@@ -1105,6 +1105,12 @@ static int log_tree_diff(struct rev_info *opt, struct commit *commit, struct log
if (!all_need_diff && !opt->merges_need_diff)
return 0;
if (opt->line_level_traverse) {
line_log_queue_pairs(opt, commit);
log_tree_diff_flush(opt);
return !opt->loginfo;
}
parse_commit_or_die(commit);
oid = get_commit_tree_oid(commit);
@@ -1179,10 +1185,6 @@ int log_tree_commit(struct rev_info *opt, struct commit *commit)
opt->loginfo = &log;
opt->diffopt.no_free = 1;
/* NEEDSWORK: no restoring of no_free? Why? */
if (opt->line_level_traverse)
return line_log_print(opt, commit);
if (opt->track_linear && !opt->linear && !opt->reverse_output_stage)
fprintf(opt->diffopt.file, "\n%s\n", opt->break_bar);
shown = log_tree_diff(opt, commit, &log);

View File

@@ -3118,6 +3118,14 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, struct s
object_context_release(&oc);
}
if (revs->line_level_traverse) {
if (want_ancestry(revs))
revs->limited = 1;
revs->topo_order = 1;
if (!revs->diffopt.output_format)
revs->diffopt.output_format = DIFF_FORMAT_PATCH;
}
/* Did the user ask for any diff output? Run the diff! */
if (revs->diffopt.output_format & ~DIFF_FORMAT_NO_OUTPUT)
revs->diff = 1;
@@ -3131,14 +3139,6 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, struct s
if (revs->diffopt.objfind)
revs->simplify_history = 0;
if (revs->line_level_traverse) {
if (want_ancestry(revs))
revs->limited = 1;
revs->topo_order = 1;
if (!revs->diffopt.output_format)
revs->diffopt.output_format = DIFF_FORMAT_PATCH;
}
if (revs->topo_order && !generation_numbers_enabled(the_repository))
revs->limited = 1;
@@ -3189,8 +3189,12 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, struct s
die(_("the option '%s' requires '%s'"), "--grep-reflog", "--walk-reflogs");
if (revs->line_level_traverse &&
(revs->diffopt.output_format & ~(DIFF_FORMAT_PATCH | DIFF_FORMAT_NO_OUTPUT)))
die(_("-L does not yet support diff formats besides -p and -s"));
(revs->full_diff ||
(revs->diffopt.output_format &
~(DIFF_FORMAT_PATCH | DIFF_FORMAT_NO_OUTPUT |
DIFF_FORMAT_RAW | DIFF_FORMAT_NAME |
DIFF_FORMAT_NAME_STATUS | DIFF_FORMAT_SUMMARY))))
die(_("-L does not yet support the requested diff format"));
if (revs->expand_tabs_in_log < 0)
revs->expand_tabs_in_log = revs->expand_tabs_in_log_default;

View File

@@ -155,8 +155,45 @@ test_expect_success '-p shows the default patch output' '
test_cmp expect actual
'
test_expect_success '--raw is forbidden' '
test_must_fail git log -L1,24:b.c --raw
test_expect_success '--raw shows mode, oid, status and path' '
git log -L1,24:b.c --raw --format= >actual &&
test_grep "^:100644 100644 [0-9a-f]\{7\} [0-9a-f]\{7\} M b.c$" actual &&
test_grep ! "^diff --git" actual &&
test_grep ! "^@@" actual
'
test_expect_success '--name-only shows path' '
git log -L1,24:b.c --name-only --format= >actual &&
test_grep "^b.c$" actual &&
test_grep ! "^diff --git" actual &&
test_grep ! "^@@" actual
'
test_expect_success '--name-status shows status and path' '
git log -L1,24:b.c --name-status --format= >actual &&
test_grep "^M b.c$" actual &&
test_grep ! "^diff --git" actual &&
test_grep ! "^@@" actual
'
test_expect_success '--stat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --stat 2>err &&
test_grep "does not yet support" err
'
test_expect_success '--numstat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --numstat 2>err &&
test_grep "does not yet support" err
'
test_expect_success '--shortstat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --shortstat 2>err &&
test_grep "does not yet support" err
'
test_expect_success '--dirstat is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --dirstat 2>err &&
test_grep "does not yet support" err
'
test_expect_success 'setup for checking fancy rename following' '
@@ -368,7 +405,6 @@ test_expect_success '-L diff output includes index and new file mode' '
test_expect_success '-L with --word-diff' '
cat >expect <<-\EOF &&
diff --git a/file.c b/file.c
--- a/file.c
+++ b/file.c
@@ -377,7 +413,6 @@ test_expect_success '-L with --word-diff' '
{
return [-F2;-]{+F2 + 2;+}
}
diff --git a/file.c b/file.c
new file mode 100644
--- /dev/null
@@ -433,7 +468,6 @@ test_expect_success 'show line-log with graph' '
null_blob=$(test_oid zero | cut -c1-7) &&
qz_to_tab_space >expect <<-EOF &&
* $head_oid Modify func2() in file.c
|Z
| diff --git a/file.c b/file.c
| index $head_blob_old..$head_blob_new 100644
| --- a/file.c
@@ -445,7 +479,6 @@ test_expect_success 'show line-log with graph' '
| + return F2 + 2;
| }
* $root_oid Add func1() and func2() in file.c
ZZ
diff --git a/file.c b/file.c
new file mode 100644
index $null_blob..$root_blob
@@ -494,23 +527,17 @@ test_expect_success '-L --find-object does not crash with merge and rename' '
--find-object=$(git rev-parse HEAD:file) >actual
'
# Commit-level filtering with pickaxe does not yet work for -L.
# show_log() prints the commit header before diffcore_std() runs
# pickaxe, so commits cannot be suppressed even when no diff pairs
# survive filtering. Fixing this would require deferring show_log()
# until after diffcore_std(), which is a larger restructuring of the
# log-tree output pipeline.
test_expect_failure '-L -G should filter commits by pattern' '
test_expect_success '-L -G should filter commits by pattern' '
git log --format="%s" --no-patch -L 1,1:file -G "nomatch" >actual &&
test_must_be_empty actual
'
test_expect_failure '-L -S should filter commits by pattern' '
test_expect_success '-L -S should filter commits by pattern' '
git log --format="%s" --no-patch -L 1,1:file -S "nomatch" >actual &&
test_must_be_empty actual
'
test_expect_failure '-L --find-object should filter commits by object' '
test_expect_success '-L --find-object should filter commits by object' '
git log --format="%s" --no-patch -L 1,1:file \
--find-object=$ZERO_OID >actual &&
test_must_be_empty actual
@@ -711,4 +738,47 @@ test_expect_success '-L with -G filters to diff-text matches' '
grep "F2 + 2" actual
'
test_expect_success '-L with --diff-filter=M excludes root commit' '
git checkout parent-oids &&
git log -L:func2:file.c --diff-filter=M --format=%s --no-patch >actual &&
# Root commit is an Add (A), not a Modify (M), so it should
# be excluded; only the modification commit remains.
echo "Modify func2() in file.c" >expect &&
test_cmp expect actual
'
test_expect_success '-L with --diff-filter=A shows only root commit' '
git checkout parent-oids &&
git log -L:func2:file.c --diff-filter=A --format=%s --no-patch >actual &&
echo "Add func1() and func2() in file.c" >expect &&
test_cmp expect actual
'
test_expect_success '-L with -S suppresses non-matching commits' '
git checkout parent-oids &&
git log -L:func2:file.c -S "F2 + 2" --format=%s --no-patch >actual &&
# Only the commit that changes the count of "F2 + 2" should appear.
echo "Modify func2() in file.c" >expect &&
test_cmp expect actual
'
test_expect_success '--full-diff is not yet supported with -L' '
test_must_fail git log -L1,24:b.c --full-diff 2>err &&
test_grep "does not yet support" err
'
test_expect_success '-L --oneline has no extra blank line before diff' '
git checkout parent-oids &&
git log --oneline -L:func2:file.c -1 >actual &&
# Oneline header on line 1, diff starts immediately on line 2
sed -n 2p actual >line2 &&
test_grep "^diff --git" line2
'
test_expect_success '--summary shows new file on root commit' '
git checkout parent-oids &&
git log -L:func2:file.c --summary --format= >actual &&
test_grep "create mode 100644 file.c" actual
'
test_done

View File

@@ -5,7 +5,6 @@ Date: Fri Apr 12 16:16:24 2013 +0200
Merge across the rename
commit 6ce3c4ff690136099bb17e1a8766b75764726ea7
Author: Thomas Rast <trast@student.ethz.ch>
Date: Thu Feb 28 10:49:50 2013 +0100

View File

@@ -5,7 +5,6 @@ Date: Fri Apr 12 16:16:24 2013 +0200
Merge across the rename
commit 4f7a58195a92c400e28a2354328587f1ff14fb77f5cf894536f17ccbc72931b9
Author: Thomas Rast <trast@student.ethz.ch>
Date: Thu Feb 28 10:49:50 2013 +0100