Merge branch 'js/range-diff'
"git tbdiff" that lets us compare individual patches in two iterations of a topic has been rewritten and made into a built-in command. * js/range-diff: (21 commits) range-diff: use dim/bold cues to improve dual color mode range-diff: make --dual-color the default mode range-diff: left-pad patch numbers completion: support `git range-diff` range-diff: populate the man page range-diff --dual-color: skip white-space warnings range-diff: offer to dual-color the diffs diff: add an internal option to dual-color diffs of diffs color: add the meta color GIT_COLOR_REVERSE range-diff: use color for the commit pairs range-diff: add tests range-diff: do not show "function names" in hunk headers range-diff: adjust the output of the commit pairs range-diff: suppress the diff headers range-diff: indent the diffs just like tbdiff range-diff: right-trim commit messages range-diff: also show the diff between patches range-diff: improve the order of the shown commits range-diff: first rudimentary implementation Introduce `range-diff` to compare iterations of a topic branch ...
This commit is contained in:
commit
81eab6871e
1
.gitignore
vendored
1
.gitignore
vendored
@ -113,6 +113,7 @@
|
|||||||
/git-pull
|
/git-pull
|
||||||
/git-push
|
/git-push
|
||||||
/git-quiltimport
|
/git-quiltimport
|
||||||
|
/git-range-diff
|
||||||
/git-read-tree
|
/git-read-tree
|
||||||
/git-rebase
|
/git-rebase
|
||||||
/git-rebase--am
|
/git-rebase--am
|
||||||
|
@ -1225,8 +1225,10 @@ color.diff.<slot>::
|
|||||||
(highlighting whitespace errors), `oldMoved` (deleted lines),
|
(highlighting whitespace errors), `oldMoved` (deleted lines),
|
||||||
`newMoved` (added lines), `oldMovedDimmed`, `oldMovedAlternative`,
|
`newMoved` (added lines), `oldMovedDimmed`, `oldMovedAlternative`,
|
||||||
`oldMovedAlternativeDimmed`, `newMovedDimmed`, `newMovedAlternative`
|
`oldMovedAlternativeDimmed`, `newMovedDimmed`, `newMovedAlternative`
|
||||||
and `newMovedAlternativeDimmed` (See the '<mode>'
|
`newMovedAlternativeDimmed` (See the '<mode>'
|
||||||
setting of '--color-moved' in linkgit:git-diff[1] for details).
|
setting of '--color-moved' in linkgit:git-diff[1] for details),
|
||||||
|
`contextDimmed`, `oldDimmed`, `newDimmed`, `contextBold`,
|
||||||
|
`oldBold`, and `newBold` (see linkgit:git-range-diff[1] for details).
|
||||||
|
|
||||||
color.decorate.<slot>::
|
color.decorate.<slot>::
|
||||||
Use customized color for 'git log --decorate' output. `<slot>` is one
|
Use customized color for 'git log --decorate' output. `<slot>` is one
|
||||||
|
252
Documentation/git-range-diff.txt
Normal file
252
Documentation/git-range-diff.txt
Normal file
@ -0,0 +1,252 @@
|
|||||||
|
git-range-diff(1)
|
||||||
|
=================
|
||||||
|
|
||||||
|
NAME
|
||||||
|
----
|
||||||
|
git-range-diff - Compare two commit ranges (e.g. two versions of a branch)
|
||||||
|
|
||||||
|
SYNOPSIS
|
||||||
|
--------
|
||||||
|
[verse]
|
||||||
|
'git range-diff' [--color=[<when>]] [--no-color] [<diff-options>]
|
||||||
|
[--no-dual-color] [--creation-factor=<factor>]
|
||||||
|
( <range1> <range2> | <rev1>...<rev2> | <base> <rev1> <rev2> )
|
||||||
|
|
||||||
|
DESCRIPTION
|
||||||
|
-----------
|
||||||
|
|
||||||
|
This command shows the differences between two versions of a patch
|
||||||
|
series, or more generally, two commit ranges (ignoring merge commits).
|
||||||
|
|
||||||
|
To that end, it first finds pairs of commits from both commit ranges
|
||||||
|
that correspond with each other. Two commits are said to correspond when
|
||||||
|
the diff between their patches (i.e. the author information, the commit
|
||||||
|
message and the commit diff) is reasonably small compared to the
|
||||||
|
patches' size. See ``Algorithm`` below for details.
|
||||||
|
|
||||||
|
Finally, the list of matching commits is shown in the order of the
|
||||||
|
second commit range, with unmatched commits being inserted just after
|
||||||
|
all of their ancestors have been shown.
|
||||||
|
|
||||||
|
|
||||||
|
OPTIONS
|
||||||
|
-------
|
||||||
|
--no-dual-color::
|
||||||
|
When the commit diffs differ, `git range-diff` recreates the
|
||||||
|
original diffs' coloring, and adds outer -/+ diff markers with
|
||||||
|
the *background* being red/green to make it easier to see e.g.
|
||||||
|
when there was a change in what exact lines were added.
|
||||||
|
+
|
||||||
|
Additionally, the commit diff lines that are only present in the first commit
|
||||||
|
range are shown "dimmed" (this can be overridden using the `color.diff.<slot>`
|
||||||
|
config setting where `<slot>` is one of `contextDimmed`, `oldDimmed` and
|
||||||
|
`newDimmed`), and the commit diff lines that are only present in the second
|
||||||
|
commit range are shown in bold (which can be overridden using the config
|
||||||
|
settings `color.diff.<slot>` with `<slot>` being one of `contextBold`,
|
||||||
|
`oldBold` or `newBold`).
|
||||||
|
+
|
||||||
|
This is known to `range-diff` as "dual coloring". Use `--no-dual-color`
|
||||||
|
to revert to color all lines according to the outer diff markers
|
||||||
|
(and completely ignore the inner diff when it comes to color).
|
||||||
|
|
||||||
|
--creation-factor=<percent>::
|
||||||
|
Set the creation/deletion cost fudge factor to `<percent>`.
|
||||||
|
Defaults to 60. Try a larger value if `git range-diff` erroneously
|
||||||
|
considers a large change a total rewrite (deletion of one commit
|
||||||
|
and addition of another), and a smaller one in the reverse case.
|
||||||
|
See the ``Algorithm`` section below for an explanation why this is
|
||||||
|
needed.
|
||||||
|
|
||||||
|
<range1> <range2>::
|
||||||
|
Compare the commits specified by the two ranges, where
|
||||||
|
`<range1>` is considered an older version of `<range2>`.
|
||||||
|
|
||||||
|
<rev1>...<rev2>::
|
||||||
|
Equivalent to passing `<rev2>..<rev1>` and `<rev1>..<rev2>`.
|
||||||
|
|
||||||
|
<base> <rev1> <rev2>::
|
||||||
|
Equivalent to passing `<base>..<rev1>` and `<base>..<rev2>`.
|
||||||
|
Note that `<base>` does not need to be the exact branch point
|
||||||
|
of the branches. Example: after rebasing a branch `my-topic`,
|
||||||
|
`git range-diff my-topic@{u} my-topic@{1} my-topic` would
|
||||||
|
show the differences introduced by the rebase.
|
||||||
|
|
||||||
|
`git range-diff` also accepts the regular diff options (see
|
||||||
|
linkgit:git-diff[1]), most notably the `--color=[<when>]` and
|
||||||
|
`--no-color` options. These options are used when generating the "diff
|
||||||
|
between patches", i.e. to compare the author, commit message and diff of
|
||||||
|
corresponding old/new commits. There is currently no means to tweak the
|
||||||
|
diff options passed to `git log` when generating those patches.
|
||||||
|
|
||||||
|
|
||||||
|
CONFIGURATION
|
||||||
|
-------------
|
||||||
|
This command uses the `diff.color.*` and `pager.range-diff` settings
|
||||||
|
(the latter is on by default).
|
||||||
|
See linkgit:git-config[1].
|
||||||
|
|
||||||
|
|
||||||
|
EXAMPLES
|
||||||
|
--------
|
||||||
|
|
||||||
|
When a rebase required merge conflicts to be resolved, compare the changes
|
||||||
|
introduced by the rebase directly afterwards using:
|
||||||
|
|
||||||
|
------------
|
||||||
|
$ git range-diff @{u} @{1} @
|
||||||
|
------------
|
||||||
|
|
||||||
|
|
||||||
|
A typical output of `git range-diff` would look like this:
|
||||||
|
|
||||||
|
------------
|
||||||
|
-: ------- > 1: 0ddba11 Prepare for the inevitable!
|
||||||
|
1: c0debee = 2: cab005e Add a helpful message at the start
|
||||||
|
2: f00dbal ! 3: decafe1 Describe a bug
|
||||||
|
@@ -1,3 +1,3 @@
|
||||||
|
Author: A U Thor <author@example.com>
|
||||||
|
|
||||||
|
-TODO: Describe a bug
|
||||||
|
+Describe a bug
|
||||||
|
@@ -324,5 +324,6
|
||||||
|
This is expected.
|
||||||
|
|
||||||
|
-+What is unexpected is that it will also crash.
|
||||||
|
++Unexpectedly, it also crashes. This is a bug, and the jury is
|
||||||
|
++still out there how to fix it best. See ticket #314 for details.
|
||||||
|
|
||||||
|
Contact
|
||||||
|
3: bedead < -: ------- TO-UNDO
|
||||||
|
------------
|
||||||
|
|
||||||
|
In this example, there are 3 old and 3 new commits, where the developer
|
||||||
|
removed the 3rd, added a new one before the first two, and modified the
|
||||||
|
commit message of the 2nd commit as well its diff.
|
||||||
|
|
||||||
|
When the output goes to a terminal, it is color-coded by default, just
|
||||||
|
like regular `git diff`'s output. In addition, the first line (adding a
|
||||||
|
commit) is green, the last line (deleting a commit) is red, the second
|
||||||
|
line (with a perfect match) is yellow like the commit header of `git
|
||||||
|
show`'s output, and the third line colors the old commit red, the new
|
||||||
|
one green and the rest like `git show`'s commit header.
|
||||||
|
|
||||||
|
A naive color-coded diff of diffs is actually a bit hard to read,
|
||||||
|
though, as it colors the entire lines red or green. The line that added
|
||||||
|
"What is unexpected" in the old commit, for example, is completely red,
|
||||||
|
even if the intent of the old commit was to add something.
|
||||||
|
|
||||||
|
To help with that, `range` uses the `--dual-color` mode by default. In
|
||||||
|
this mode, the diff of diffs will retain the original diff colors, and
|
||||||
|
prefix the lines with -/+ markers that have their *background* red or
|
||||||
|
green, to make it more obvious that they describe how the diff itself
|
||||||
|
changed.
|
||||||
|
|
||||||
|
|
||||||
|
Algorithm
|
||||||
|
---------
|
||||||
|
|
||||||
|
The general idea is this: we generate a cost matrix between the commits
|
||||||
|
in both commit ranges, then solve the least-cost assignment.
|
||||||
|
|
||||||
|
The cost matrix is populated thusly: for each pair of commits, both
|
||||||
|
diffs are generated and the "diff of diffs" is generated, with 3 context
|
||||||
|
lines, then the number of lines in that diff is used as cost.
|
||||||
|
|
||||||
|
To avoid false positives (e.g. when a patch has been removed, and an
|
||||||
|
unrelated patch has been added between two iterations of the same patch
|
||||||
|
series), the cost matrix is extended to allow for that, by adding
|
||||||
|
fixed-cost entries for wholesale deletes/adds.
|
||||||
|
|
||||||
|
Example: Let commits `1--2` be the first iteration of a patch series and
|
||||||
|
`A--C` the second iteration. Let's assume that `A` is a cherry-pick of
|
||||||
|
`2,` and `C` is a cherry-pick of `1` but with a small modification (say,
|
||||||
|
a fixed typo). Visualize the commits as a bipartite graph:
|
||||||
|
|
||||||
|
------------
|
||||||
|
1 A
|
||||||
|
|
||||||
|
2 B
|
||||||
|
|
||||||
|
C
|
||||||
|
------------
|
||||||
|
|
||||||
|
We are looking for a "best" explanation of the new series in terms of
|
||||||
|
the old one. We can represent an "explanation" as an edge in the graph:
|
||||||
|
|
||||||
|
|
||||||
|
------------
|
||||||
|
1 A
|
||||||
|
/
|
||||||
|
2 --------' B
|
||||||
|
|
||||||
|
C
|
||||||
|
------------
|
||||||
|
|
||||||
|
This explanation comes for "free" because there was no change. Similarly
|
||||||
|
`C` could be explained using `1`, but that comes at some cost c>0
|
||||||
|
because of the modification:
|
||||||
|
|
||||||
|
------------
|
||||||
|
1 ----. A
|
||||||
|
| /
|
||||||
|
2 ----+---' B
|
||||||
|
|
|
||||||
|
`----- C
|
||||||
|
c>0
|
||||||
|
------------
|
||||||
|
|
||||||
|
In mathematical terms, what we are looking for is some sort of a minimum
|
||||||
|
cost bipartite matching; `1` is matched to `C` at some cost, etc. The
|
||||||
|
underlying graph is in fact a complete bipartite graph; the cost we
|
||||||
|
associate with every edge is the size of the diff between the two
|
||||||
|
commits' patches. To explain also new commits, we introduce dummy nodes
|
||||||
|
on both sides:
|
||||||
|
|
||||||
|
------------
|
||||||
|
1 ----. A
|
||||||
|
| /
|
||||||
|
2 ----+---' B
|
||||||
|
|
|
||||||
|
o `----- C
|
||||||
|
c>0
|
||||||
|
o o
|
||||||
|
|
||||||
|
o o
|
||||||
|
------------
|
||||||
|
|
||||||
|
The cost of an edge `o--C` is the size of `C`'s diff, modified by a
|
||||||
|
fudge factor that should be smaller than 100%. The cost of an edge
|
||||||
|
`o--o` is free. The fudge factor is necessary because even if `1` and
|
||||||
|
`C` have nothing in common, they may still share a few empty lines and
|
||||||
|
such, possibly making the assignment `1--C`, `o--o` slightly cheaper
|
||||||
|
than `1--o`, `o--C` even if `1` and `C` have nothing in common. With the
|
||||||
|
fudge factor we require a much larger common part to consider patches as
|
||||||
|
corresponding.
|
||||||
|
|
||||||
|
The overall time needed to compute this algorithm is the time needed to
|
||||||
|
compute n+m commit diffs and then n*m diffs of patches, plus the time
|
||||||
|
needed to compute the least-cost assigment between n and m diffs. Git
|
||||||
|
uses an implementation of the Jonker-Volgenant algorithm to solve the
|
||||||
|
assignment problem, which has cubic runtime complexity. The matching
|
||||||
|
found in this case will look like this:
|
||||||
|
|
||||||
|
------------
|
||||||
|
1 ----. A
|
||||||
|
| /
|
||||||
|
2 ----+---' B
|
||||||
|
.--+-----'
|
||||||
|
o -' `----- C
|
||||||
|
c>0
|
||||||
|
o ---------- o
|
||||||
|
|
||||||
|
o ---------- o
|
||||||
|
------------
|
||||||
|
|
||||||
|
|
||||||
|
SEE ALSO
|
||||||
|
--------
|
||||||
|
linkgit:git-log[1]
|
||||||
|
|
||||||
|
GIT
|
||||||
|
---
|
||||||
|
Part of the linkgit:git[1] suite
|
3
Makefile
3
Makefile
@ -876,6 +876,7 @@ LIB_OBJS += gpg-interface.o
|
|||||||
LIB_OBJS += graph.o
|
LIB_OBJS += graph.o
|
||||||
LIB_OBJS += grep.o
|
LIB_OBJS += grep.o
|
||||||
LIB_OBJS += hashmap.o
|
LIB_OBJS += hashmap.o
|
||||||
|
LIB_OBJS += linear-assignment.o
|
||||||
LIB_OBJS += help.o
|
LIB_OBJS += help.o
|
||||||
LIB_OBJS += hex.o
|
LIB_OBJS += hex.o
|
||||||
LIB_OBJS += ident.o
|
LIB_OBJS += ident.o
|
||||||
@ -931,6 +932,7 @@ LIB_OBJS += progress.o
|
|||||||
LIB_OBJS += prompt.o
|
LIB_OBJS += prompt.o
|
||||||
LIB_OBJS += protocol.o
|
LIB_OBJS += protocol.o
|
||||||
LIB_OBJS += quote.o
|
LIB_OBJS += quote.o
|
||||||
|
LIB_OBJS += range-diff.o
|
||||||
LIB_OBJS += reachable.o
|
LIB_OBJS += reachable.o
|
||||||
LIB_OBJS += read-cache.o
|
LIB_OBJS += read-cache.o
|
||||||
LIB_OBJS += reflog-walk.o
|
LIB_OBJS += reflog-walk.o
|
||||||
@ -1069,6 +1071,7 @@ BUILTIN_OBJS += builtin/prune-packed.o
|
|||||||
BUILTIN_OBJS += builtin/prune.o
|
BUILTIN_OBJS += builtin/prune.o
|
||||||
BUILTIN_OBJS += builtin/pull.o
|
BUILTIN_OBJS += builtin/pull.o
|
||||||
BUILTIN_OBJS += builtin/push.o
|
BUILTIN_OBJS += builtin/push.o
|
||||||
|
BUILTIN_OBJS += builtin/range-diff.o
|
||||||
BUILTIN_OBJS += builtin/read-tree.o
|
BUILTIN_OBJS += builtin/read-tree.o
|
||||||
BUILTIN_OBJS += builtin/rebase--helper.o
|
BUILTIN_OBJS += builtin/rebase--helper.o
|
||||||
BUILTIN_OBJS += builtin/receive-pack.o
|
BUILTIN_OBJS += builtin/receive-pack.o
|
||||||
|
@ -201,6 +201,7 @@ extern int cmd_prune(int argc, const char **argv, const char *prefix);
|
|||||||
extern int cmd_prune_packed(int argc, const char **argv, const char *prefix);
|
extern int cmd_prune_packed(int argc, const char **argv, const char *prefix);
|
||||||
extern int cmd_pull(int argc, const char **argv, const char *prefix);
|
extern int cmd_pull(int argc, const char **argv, const char *prefix);
|
||||||
extern int cmd_push(int argc, const char **argv, const char *prefix);
|
extern int cmd_push(int argc, const char **argv, const char *prefix);
|
||||||
|
extern int cmd_range_diff(int argc, const char **argv, const char *prefix);
|
||||||
extern int cmd_read_tree(int argc, const char **argv, const char *prefix);
|
extern int cmd_read_tree(int argc, const char **argv, const char *prefix);
|
||||||
extern int cmd_rebase__helper(int argc, const char **argv, const char *prefix);
|
extern int cmd_rebase__helper(int argc, const char **argv, const char *prefix);
|
||||||
extern int cmd_receive_pack(int argc, const char **argv, const char *prefix);
|
extern int cmd_receive_pack(int argc, const char **argv, const char *prefix);
|
||||||
|
116
builtin/range-diff.c
Normal file
116
builtin/range-diff.c
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
#include "cache.h"
|
||||||
|
#include "builtin.h"
|
||||||
|
#include "parse-options.h"
|
||||||
|
#include "range-diff.h"
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
static const char * const builtin_range_diff_usage[] = {
|
||||||
|
N_("git range-diff [<options>] <old-base>..<old-tip> <new-base>..<new-tip>"),
|
||||||
|
N_("git range-diff [<options>] <old-tip>...<new-tip>"),
|
||||||
|
N_("git range-diff [<options>] <base> <old-tip> <new-tip>"),
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct strbuf *output_prefix_cb(struct diff_options *opt, void *data)
|
||||||
|
{
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
int cmd_range_diff(int argc, const char **argv, const char *prefix)
|
||||||
|
{
|
||||||
|
int creation_factor = 60;
|
||||||
|
struct diff_options diffopt = { NULL };
|
||||||
|
int simple_color = -1;
|
||||||
|
struct option options[] = {
|
||||||
|
OPT_INTEGER(0, "creation-factor", &creation_factor,
|
||||||
|
N_("Percentage by which creation is weighted")),
|
||||||
|
OPT_BOOL(0, "no-dual-color", &simple_color,
|
||||||
|
N_("color both diff and diff-between-diffs")),
|
||||||
|
OPT_END()
|
||||||
|
};
|
||||||
|
int i, j, res = 0;
|
||||||
|
struct strbuf four_spaces = STRBUF_INIT;
|
||||||
|
struct strbuf range1 = STRBUF_INIT, range2 = STRBUF_INIT;
|
||||||
|
|
||||||
|
git_config(git_diff_ui_config, NULL);
|
||||||
|
|
||||||
|
diff_setup(&diffopt);
|
||||||
|
diffopt.output_format = DIFF_FORMAT_PATCH;
|
||||||
|
diffopt.flags.suppress_diff_headers = 1;
|
||||||
|
diffopt.output_prefix = output_prefix_cb;
|
||||||
|
strbuf_addstr(&four_spaces, " ");
|
||||||
|
diffopt.output_prefix_data = &four_spaces;
|
||||||
|
|
||||||
|
argc = parse_options(argc, argv, NULL, options,
|
||||||
|
builtin_range_diff_usage, PARSE_OPT_KEEP_UNKNOWN |
|
||||||
|
PARSE_OPT_KEEP_DASHDASH | PARSE_OPT_KEEP_ARGV0);
|
||||||
|
|
||||||
|
for (i = j = 1; i < argc && strcmp("--", argv[i]); ) {
|
||||||
|
int c = diff_opt_parse(&diffopt, argv + i, argc - i, prefix);
|
||||||
|
|
||||||
|
if (!c)
|
||||||
|
argv[j++] = argv[i++];
|
||||||
|
else
|
||||||
|
i += c;
|
||||||
|
}
|
||||||
|
while (i < argc)
|
||||||
|
argv[j++] = argv[i++];
|
||||||
|
argc = j;
|
||||||
|
diff_setup_done(&diffopt);
|
||||||
|
|
||||||
|
/* Make sure that there are no unparsed options */
|
||||||
|
argc = parse_options(argc, argv, NULL,
|
||||||
|
options + ARRAY_SIZE(options) - 1, /* OPT_END */
|
||||||
|
builtin_range_diff_usage, 0);
|
||||||
|
|
||||||
|
if (simple_color < 1) {
|
||||||
|
if (!simple_color)
|
||||||
|
/* force color when --dual-color was used */
|
||||||
|
diffopt.use_color = 1;
|
||||||
|
diffopt.flags.dual_color_diffed_diffs = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (argc == 2) {
|
||||||
|
if (!strstr(argv[0], ".."))
|
||||||
|
die(_("no .. in range: '%s'"), argv[0]);
|
||||||
|
strbuf_addstr(&range1, argv[0]);
|
||||||
|
|
||||||
|
if (!strstr(argv[1], ".."))
|
||||||
|
die(_("no .. in range: '%s'"), argv[1]);
|
||||||
|
strbuf_addstr(&range2, argv[1]);
|
||||||
|
} else if (argc == 3) {
|
||||||
|
strbuf_addf(&range1, "%s..%s", argv[0], argv[1]);
|
||||||
|
strbuf_addf(&range2, "%s..%s", argv[0], argv[2]);
|
||||||
|
} else if (argc == 1) {
|
||||||
|
const char *b = strstr(argv[0], "..."), *a = argv[0];
|
||||||
|
int a_len;
|
||||||
|
|
||||||
|
if (!b) {
|
||||||
|
error(_("single arg format must be symmetric range"));
|
||||||
|
usage_with_options(builtin_range_diff_usage, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
a_len = (int)(b - a);
|
||||||
|
if (!a_len) {
|
||||||
|
a = "HEAD";
|
||||||
|
a_len = strlen(a);
|
||||||
|
}
|
||||||
|
b += 3;
|
||||||
|
if (!*b)
|
||||||
|
b = "HEAD";
|
||||||
|
strbuf_addf(&range1, "%s..%.*s", b, a_len, a);
|
||||||
|
strbuf_addf(&range2, "%.*s..%s", a_len, a, b);
|
||||||
|
} else {
|
||||||
|
error(_("need two commit ranges"));
|
||||||
|
usage_with_options(builtin_range_diff_usage, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
res = show_range_diff(range1.buf, range2.buf, creation_factor,
|
||||||
|
&diffopt);
|
||||||
|
|
||||||
|
strbuf_release(&range1);
|
||||||
|
strbuf_release(&range2);
|
||||||
|
strbuf_release(&four_spaces);
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
7
color.h
7
color.h
@ -36,6 +36,12 @@ struct strbuf;
|
|||||||
#define GIT_COLOR_BOLD_BLUE "\033[1;34m"
|
#define GIT_COLOR_BOLD_BLUE "\033[1;34m"
|
||||||
#define GIT_COLOR_BOLD_MAGENTA "\033[1;35m"
|
#define GIT_COLOR_BOLD_MAGENTA "\033[1;35m"
|
||||||
#define GIT_COLOR_BOLD_CYAN "\033[1;36m"
|
#define GIT_COLOR_BOLD_CYAN "\033[1;36m"
|
||||||
|
#define GIT_COLOR_FAINT_RED "\033[2;31m"
|
||||||
|
#define GIT_COLOR_FAINT_GREEN "\033[2;32m"
|
||||||
|
#define GIT_COLOR_FAINT_YELLOW "\033[2;33m"
|
||||||
|
#define GIT_COLOR_FAINT_BLUE "\033[2;34m"
|
||||||
|
#define GIT_COLOR_FAINT_MAGENTA "\033[2;35m"
|
||||||
|
#define GIT_COLOR_FAINT_CYAN "\033[2;36m"
|
||||||
#define GIT_COLOR_BG_RED "\033[41m"
|
#define GIT_COLOR_BG_RED "\033[41m"
|
||||||
#define GIT_COLOR_BG_GREEN "\033[42m"
|
#define GIT_COLOR_BG_GREEN "\033[42m"
|
||||||
#define GIT_COLOR_BG_YELLOW "\033[43m"
|
#define GIT_COLOR_BG_YELLOW "\033[43m"
|
||||||
@ -44,6 +50,7 @@ struct strbuf;
|
|||||||
#define GIT_COLOR_BG_CYAN "\033[46m"
|
#define GIT_COLOR_BG_CYAN "\033[46m"
|
||||||
#define GIT_COLOR_FAINT "\033[2m"
|
#define GIT_COLOR_FAINT "\033[2m"
|
||||||
#define GIT_COLOR_FAINT_ITALIC "\033[2;3m"
|
#define GIT_COLOR_FAINT_ITALIC "\033[2;3m"
|
||||||
|
#define GIT_COLOR_REVERSE "\033[7m"
|
||||||
|
|
||||||
/* A special value meaning "no color selected" */
|
/* A special value meaning "no color selected" */
|
||||||
#define GIT_COLOR_NIL "NIL"
|
#define GIT_COLOR_NIL "NIL"
|
||||||
|
@ -139,6 +139,7 @@ git-prune-packed plumbingmanipulators
|
|||||||
git-pull mainporcelain remote
|
git-pull mainporcelain remote
|
||||||
git-push mainporcelain remote
|
git-push mainporcelain remote
|
||||||
git-quiltimport foreignscminterface
|
git-quiltimport foreignscminterface
|
||||||
|
git-range-diff mainporcelain
|
||||||
git-read-tree plumbingmanipulators
|
git-read-tree plumbingmanipulators
|
||||||
git-rebase mainporcelain history
|
git-rebase mainporcelain history
|
||||||
git-receive-pack synchelpers
|
git-receive-pack synchelpers
|
||||||
|
@ -1976,6 +1976,20 @@ _git_push ()
|
|||||||
__git_complete_remote_or_refspec
|
__git_complete_remote_or_refspec
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_git_range_diff ()
|
||||||
|
{
|
||||||
|
case "$cur" in
|
||||||
|
--*)
|
||||||
|
__gitcomp "
|
||||||
|
--creation-factor= --no-dual-color
|
||||||
|
$__git_diff_common_options
|
||||||
|
"
|
||||||
|
return
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
__git_complete_revlist
|
||||||
|
}
|
||||||
|
|
||||||
_git_rebase ()
|
_git_rebase ()
|
||||||
{
|
{
|
||||||
__git_find_repo_path
|
__git_find_repo_path
|
||||||
|
103
diff.c
103
diff.c
@ -70,6 +70,12 @@ static char diff_colors[][COLOR_MAXLEN] = {
|
|||||||
GIT_COLOR_BOLD_YELLOW, /* NEW_MOVED ALTERNATIVE */
|
GIT_COLOR_BOLD_YELLOW, /* NEW_MOVED ALTERNATIVE */
|
||||||
GIT_COLOR_FAINT, /* NEW_MOVED_DIM */
|
GIT_COLOR_FAINT, /* NEW_MOVED_DIM */
|
||||||
GIT_COLOR_FAINT_ITALIC, /* NEW_MOVED_ALTERNATIVE_DIM */
|
GIT_COLOR_FAINT_ITALIC, /* NEW_MOVED_ALTERNATIVE_DIM */
|
||||||
|
GIT_COLOR_FAINT, /* CONTEXT_DIM */
|
||||||
|
GIT_COLOR_FAINT_RED, /* OLD_DIM */
|
||||||
|
GIT_COLOR_FAINT_GREEN, /* NEW_DIM */
|
||||||
|
GIT_COLOR_BOLD, /* CONTEXT_BOLD */
|
||||||
|
GIT_COLOR_BOLD_RED, /* OLD_BOLD */
|
||||||
|
GIT_COLOR_BOLD_GREEN, /* NEW_BOLD */
|
||||||
};
|
};
|
||||||
|
|
||||||
static const char *color_diff_slots[] = {
|
static const char *color_diff_slots[] = {
|
||||||
@ -89,6 +95,12 @@ static const char *color_diff_slots[] = {
|
|||||||
[DIFF_FILE_NEW_MOVED_ALT] = "newMovedAlternative",
|
[DIFF_FILE_NEW_MOVED_ALT] = "newMovedAlternative",
|
||||||
[DIFF_FILE_NEW_MOVED_DIM] = "newMovedDimmed",
|
[DIFF_FILE_NEW_MOVED_DIM] = "newMovedDimmed",
|
||||||
[DIFF_FILE_NEW_MOVED_ALT_DIM] = "newMovedAlternativeDimmed",
|
[DIFF_FILE_NEW_MOVED_ALT_DIM] = "newMovedAlternativeDimmed",
|
||||||
|
[DIFF_CONTEXT_DIM] = "contextDimmed",
|
||||||
|
[DIFF_FILE_OLD_DIM] = "oldDimmed",
|
||||||
|
[DIFF_FILE_NEW_DIM] = "newDimmed",
|
||||||
|
[DIFF_CONTEXT_BOLD] = "contextBold",
|
||||||
|
[DIFF_FILE_OLD_BOLD] = "oldBold",
|
||||||
|
[DIFF_FILE_NEW_BOLD] = "newBold",
|
||||||
};
|
};
|
||||||
|
|
||||||
static NORETURN void die_want_option(const char *option_name)
|
static NORETURN void die_want_option(const char *option_name)
|
||||||
@ -611,14 +623,18 @@ static void check_blank_at_eof(mmfile_t *mf1, mmfile_t *mf2,
|
|||||||
ecbdata->blank_at_eof_in_postimage = (at - l2) + 1;
|
ecbdata->blank_at_eof_in_postimage = (at - l2) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void emit_line_0(struct diff_options *o, const char *set, const char *reset,
|
static void emit_line_0(struct diff_options *o,
|
||||||
|
const char *set, unsigned reverse, const char *reset,
|
||||||
int first, const char *line, int len)
|
int first, const char *line, int len)
|
||||||
{
|
{
|
||||||
int has_trailing_newline, has_trailing_carriage_return;
|
int has_trailing_newline, has_trailing_carriage_return;
|
||||||
int nofirst;
|
int nofirst;
|
||||||
FILE *file = o->file;
|
FILE *file = o->file;
|
||||||
|
|
||||||
|
if (first)
|
||||||
fputs(diff_line_prefix(o), file);
|
fputs(diff_line_prefix(o), file);
|
||||||
|
else if (!len)
|
||||||
|
return;
|
||||||
|
|
||||||
if (len == 0) {
|
if (len == 0) {
|
||||||
has_trailing_newline = (first == '\n');
|
has_trailing_newline = (first == '\n');
|
||||||
@ -636,8 +652,10 @@ static void emit_line_0(struct diff_options *o, const char *set, const char *res
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (len || !nofirst) {
|
if (len || !nofirst) {
|
||||||
|
if (reverse && want_color(o->use_color))
|
||||||
|
fputs(GIT_COLOR_REVERSE, file);
|
||||||
fputs(set, file);
|
fputs(set, file);
|
||||||
if (!nofirst)
|
if (first && !nofirst)
|
||||||
fputc(first, file);
|
fputc(first, file);
|
||||||
fwrite(line, len, 1, file);
|
fwrite(line, len, 1, file);
|
||||||
fputs(reset, file);
|
fputs(reset, file);
|
||||||
@ -651,7 +669,7 @@ static void emit_line_0(struct diff_options *o, const char *set, const char *res
|
|||||||
static void emit_line(struct diff_options *o, const char *set, const char *reset,
|
static void emit_line(struct diff_options *o, const char *set, const char *reset,
|
||||||
const char *line, int len)
|
const char *line, int len)
|
||||||
{
|
{
|
||||||
emit_line_0(o, set, reset, line[0], line+1, len-1);
|
emit_line_0(o, set, 0, reset, line[0], line+1, len-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum diff_symbol {
|
enum diff_symbol {
|
||||||
@ -1170,7 +1188,8 @@ static void dim_moved_lines(struct diff_options *o)
|
|||||||
|
|
||||||
static void emit_line_ws_markup(struct diff_options *o,
|
static void emit_line_ws_markup(struct diff_options *o,
|
||||||
const char *set, const char *reset,
|
const char *set, const char *reset,
|
||||||
const char *line, int len, char sign,
|
const char *line, int len,
|
||||||
|
const char *set_sign, char sign,
|
||||||
unsigned ws_rule, int blank_at_eof)
|
unsigned ws_rule, int blank_at_eof)
|
||||||
{
|
{
|
||||||
const char *ws = NULL;
|
const char *ws = NULL;
|
||||||
@ -1181,14 +1200,20 @@ static void emit_line_ws_markup(struct diff_options *o,
|
|||||||
ws = NULL;
|
ws = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ws)
|
if (!ws && !set_sign)
|
||||||
emit_line_0(o, set, reset, sign, line, len);
|
emit_line_0(o, set, 0, reset, sign, line, len);
|
||||||
else if (blank_at_eof)
|
else if (!ws) {
|
||||||
|
/* Emit just the prefix, then the rest. */
|
||||||
|
emit_line_0(o, set_sign ? set_sign : set, !!set_sign, reset,
|
||||||
|
sign, "", 0);
|
||||||
|
emit_line_0(o, set, 0, reset, 0, line, len);
|
||||||
|
} else if (blank_at_eof)
|
||||||
/* Blank line at EOF - paint '+' as well */
|
/* Blank line at EOF - paint '+' as well */
|
||||||
emit_line_0(o, ws, reset, sign, line, len);
|
emit_line_0(o, ws, 0, reset, sign, line, len);
|
||||||
else {
|
else {
|
||||||
/* Emit just the prefix, then the rest. */
|
/* Emit just the prefix, then the rest. */
|
||||||
emit_line_0(o, set, reset, sign, "", 0);
|
emit_line_0(o, set_sign ? set_sign : set, !!set_sign, reset,
|
||||||
|
sign, "", 0);
|
||||||
ws_check_emit(line, len, ws_rule,
|
ws_check_emit(line, len, ws_rule,
|
||||||
o->file, set, reset, ws);
|
o->file, set, reset, ws);
|
||||||
}
|
}
|
||||||
@ -1198,7 +1223,7 @@ static void emit_diff_symbol_from_struct(struct diff_options *o,
|
|||||||
struct emitted_diff_symbol *eds)
|
struct emitted_diff_symbol *eds)
|
||||||
{
|
{
|
||||||
static const char *nneof = " No newline at end of file\n";
|
static const char *nneof = " No newline at end of file\n";
|
||||||
const char *context, *reset, *set, *meta, *fraginfo;
|
const char *context, *reset, *set, *set_sign, *meta, *fraginfo;
|
||||||
struct strbuf sb = STRBUF_INIT;
|
struct strbuf sb = STRBUF_INIT;
|
||||||
|
|
||||||
enum diff_symbol s = eds->s;
|
enum diff_symbol s = eds->s;
|
||||||
@ -1211,7 +1236,7 @@ static void emit_diff_symbol_from_struct(struct diff_options *o,
|
|||||||
context = diff_get_color_opt(o, DIFF_CONTEXT);
|
context = diff_get_color_opt(o, DIFF_CONTEXT);
|
||||||
reset = diff_get_color_opt(o, DIFF_RESET);
|
reset = diff_get_color_opt(o, DIFF_RESET);
|
||||||
putc('\n', o->file);
|
putc('\n', o->file);
|
||||||
emit_line_0(o, context, reset, '\\',
|
emit_line_0(o, context, 0, reset, '\\',
|
||||||
nneof, strlen(nneof));
|
nneof, strlen(nneof));
|
||||||
break;
|
break;
|
||||||
case DIFF_SYMBOL_SUBMODULE_HEADER:
|
case DIFF_SYMBOL_SUBMODULE_HEADER:
|
||||||
@ -1238,7 +1263,18 @@ static void emit_diff_symbol_from_struct(struct diff_options *o,
|
|||||||
case DIFF_SYMBOL_CONTEXT:
|
case DIFF_SYMBOL_CONTEXT:
|
||||||
set = diff_get_color_opt(o, DIFF_CONTEXT);
|
set = diff_get_color_opt(o, DIFF_CONTEXT);
|
||||||
reset = diff_get_color_opt(o, DIFF_RESET);
|
reset = diff_get_color_opt(o, DIFF_RESET);
|
||||||
emit_line_ws_markup(o, set, reset, line, len, ' ',
|
set_sign = NULL;
|
||||||
|
if (o->flags.dual_color_diffed_diffs) {
|
||||||
|
char c = !len ? 0 : line[0];
|
||||||
|
|
||||||
|
if (c == '+')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_NEW);
|
||||||
|
else if (c == '@')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FRAGINFO);
|
||||||
|
else if (c == '-')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_OLD);
|
||||||
|
}
|
||||||
|
emit_line_ws_markup(o, set, reset, line, len, set_sign, ' ',
|
||||||
flags & (DIFF_SYMBOL_CONTENT_WS_MASK), 0);
|
flags & (DIFF_SYMBOL_CONTENT_WS_MASK), 0);
|
||||||
break;
|
break;
|
||||||
case DIFF_SYMBOL_PLUS:
|
case DIFF_SYMBOL_PLUS:
|
||||||
@ -1265,7 +1301,23 @@ static void emit_diff_symbol_from_struct(struct diff_options *o,
|
|||||||
set = diff_get_color_opt(o, DIFF_FILE_NEW);
|
set = diff_get_color_opt(o, DIFF_FILE_NEW);
|
||||||
}
|
}
|
||||||
reset = diff_get_color_opt(o, DIFF_RESET);
|
reset = diff_get_color_opt(o, DIFF_RESET);
|
||||||
emit_line_ws_markup(o, set, reset, line, len, '+',
|
if (!o->flags.dual_color_diffed_diffs)
|
||||||
|
set_sign = NULL;
|
||||||
|
else {
|
||||||
|
char c = !len ? 0 : line[0];
|
||||||
|
|
||||||
|
set_sign = set;
|
||||||
|
if (c == '-')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_OLD_BOLD);
|
||||||
|
else if (c == '@')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FRAGINFO);
|
||||||
|
else if (c == '+')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_NEW_BOLD);
|
||||||
|
else
|
||||||
|
set = diff_get_color_opt(o, DIFF_CONTEXT_BOLD);
|
||||||
|
flags &= ~DIFF_SYMBOL_CONTENT_WS_MASK;
|
||||||
|
}
|
||||||
|
emit_line_ws_markup(o, set, reset, line, len, set_sign, '+',
|
||||||
flags & DIFF_SYMBOL_CONTENT_WS_MASK,
|
flags & DIFF_SYMBOL_CONTENT_WS_MASK,
|
||||||
flags & DIFF_SYMBOL_CONTENT_BLANK_LINE_EOF);
|
flags & DIFF_SYMBOL_CONTENT_BLANK_LINE_EOF);
|
||||||
break;
|
break;
|
||||||
@ -1293,7 +1345,22 @@ static void emit_diff_symbol_from_struct(struct diff_options *o,
|
|||||||
set = diff_get_color_opt(o, DIFF_FILE_OLD);
|
set = diff_get_color_opt(o, DIFF_FILE_OLD);
|
||||||
}
|
}
|
||||||
reset = diff_get_color_opt(o, DIFF_RESET);
|
reset = diff_get_color_opt(o, DIFF_RESET);
|
||||||
emit_line_ws_markup(o, set, reset, line, len, '-',
|
if (!o->flags.dual_color_diffed_diffs)
|
||||||
|
set_sign = NULL;
|
||||||
|
else {
|
||||||
|
char c = !len ? 0 : line[0];
|
||||||
|
|
||||||
|
set_sign = set;
|
||||||
|
if (c == '+')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_NEW_DIM);
|
||||||
|
else if (c == '@')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FRAGINFO);
|
||||||
|
else if (c == '-')
|
||||||
|
set = diff_get_color_opt(o, DIFF_FILE_OLD_DIM);
|
||||||
|
else
|
||||||
|
set = diff_get_color_opt(o, DIFF_CONTEXT_DIM);
|
||||||
|
}
|
||||||
|
emit_line_ws_markup(o, set, reset, line, len, set_sign, '-',
|
||||||
flags & DIFF_SYMBOL_CONTENT_WS_MASK, 0);
|
flags & DIFF_SYMBOL_CONTENT_WS_MASK, 0);
|
||||||
break;
|
break;
|
||||||
case DIFF_SYMBOL_WORDS_PORCELAIN:
|
case DIFF_SYMBOL_WORDS_PORCELAIN:
|
||||||
@ -1484,6 +1551,7 @@ static void emit_hunk_header(struct emit_callback *ecbdata,
|
|||||||
const char *frag = diff_get_color(ecbdata->color_diff, DIFF_FRAGINFO);
|
const char *frag = diff_get_color(ecbdata->color_diff, DIFF_FRAGINFO);
|
||||||
const char *func = diff_get_color(ecbdata->color_diff, DIFF_FUNCINFO);
|
const char *func = diff_get_color(ecbdata->color_diff, DIFF_FUNCINFO);
|
||||||
const char *reset = diff_get_color(ecbdata->color_diff, DIFF_RESET);
|
const char *reset = diff_get_color(ecbdata->color_diff, DIFF_RESET);
|
||||||
|
const char *reverse = ecbdata->color_diff ? GIT_COLOR_REVERSE : "";
|
||||||
static const char atat[2] = { '@', '@' };
|
static const char atat[2] = { '@', '@' };
|
||||||
const char *cp, *ep;
|
const char *cp, *ep;
|
||||||
struct strbuf msgbuf = STRBUF_INIT;
|
struct strbuf msgbuf = STRBUF_INIT;
|
||||||
@ -1504,6 +1572,8 @@ static void emit_hunk_header(struct emit_callback *ecbdata,
|
|||||||
ep += 2; /* skip over @@ */
|
ep += 2; /* skip over @@ */
|
||||||
|
|
||||||
/* The hunk header in fraginfo color */
|
/* The hunk header in fraginfo color */
|
||||||
|
if (ecbdata->opt->flags.dual_color_diffed_diffs)
|
||||||
|
strbuf_addstr(&msgbuf, reverse);
|
||||||
strbuf_addstr(&msgbuf, frag);
|
strbuf_addstr(&msgbuf, frag);
|
||||||
strbuf_add(&msgbuf, line, ep - line);
|
strbuf_add(&msgbuf, line, ep - line);
|
||||||
strbuf_addstr(&msgbuf, reset);
|
strbuf_addstr(&msgbuf, reset);
|
||||||
@ -3397,13 +3467,16 @@ static void builtin_diff(const char *name_a,
|
|||||||
memset(&xpp, 0, sizeof(xpp));
|
memset(&xpp, 0, sizeof(xpp));
|
||||||
memset(&xecfg, 0, sizeof(xecfg));
|
memset(&xecfg, 0, sizeof(xecfg));
|
||||||
memset(&ecbdata, 0, sizeof(ecbdata));
|
memset(&ecbdata, 0, sizeof(ecbdata));
|
||||||
|
if (o->flags.suppress_diff_headers)
|
||||||
|
lbl[0] = NULL;
|
||||||
ecbdata.label_path = lbl;
|
ecbdata.label_path = lbl;
|
||||||
ecbdata.color_diff = want_color(o->use_color);
|
ecbdata.color_diff = want_color(o->use_color);
|
||||||
ecbdata.ws_rule = whitespace_rule(name_b);
|
ecbdata.ws_rule = whitespace_rule(name_b);
|
||||||
if (ecbdata.ws_rule & WS_BLANK_AT_EOF)
|
if (ecbdata.ws_rule & WS_BLANK_AT_EOF)
|
||||||
check_blank_at_eof(&mf1, &mf2, &ecbdata);
|
check_blank_at_eof(&mf1, &mf2, &ecbdata);
|
||||||
ecbdata.opt = o;
|
ecbdata.opt = o;
|
||||||
ecbdata.header = header.len ? &header : NULL;
|
if (header.len && !o->flags.suppress_diff_headers)
|
||||||
|
ecbdata.header = &header;
|
||||||
xpp.flags = o->xdl_opts;
|
xpp.flags = o->xdl_opts;
|
||||||
xpp.anchors = o->anchors;
|
xpp.anchors = o->anchors;
|
||||||
xpp.anchors_nr = o->anchors_nr;
|
xpp.anchors_nr = o->anchors_nr;
|
||||||
|
10
diff.h
10
diff.h
@ -94,6 +94,8 @@ struct diff_flags {
|
|||||||
unsigned funccontext:1;
|
unsigned funccontext:1;
|
||||||
unsigned default_follow_renames:1;
|
unsigned default_follow_renames:1;
|
||||||
unsigned stat_with_summary:1;
|
unsigned stat_with_summary:1;
|
||||||
|
unsigned suppress_diff_headers:1;
|
||||||
|
unsigned dual_color_diffed_diffs:1;
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline void diff_flags_or(struct diff_flags *a,
|
static inline void diff_flags_or(struct diff_flags *a,
|
||||||
@ -246,7 +248,13 @@ enum color_diff {
|
|||||||
DIFF_FILE_NEW_MOVED = 13,
|
DIFF_FILE_NEW_MOVED = 13,
|
||||||
DIFF_FILE_NEW_MOVED_ALT = 14,
|
DIFF_FILE_NEW_MOVED_ALT = 14,
|
||||||
DIFF_FILE_NEW_MOVED_DIM = 15,
|
DIFF_FILE_NEW_MOVED_DIM = 15,
|
||||||
DIFF_FILE_NEW_MOVED_ALT_DIM = 16
|
DIFF_FILE_NEW_MOVED_ALT_DIM = 16,
|
||||||
|
DIFF_CONTEXT_DIM = 17,
|
||||||
|
DIFF_FILE_OLD_DIM = 18,
|
||||||
|
DIFF_FILE_NEW_DIM = 19,
|
||||||
|
DIFF_CONTEXT_BOLD = 20,
|
||||||
|
DIFF_FILE_OLD_BOLD = 21,
|
||||||
|
DIFF_FILE_NEW_BOLD = 22,
|
||||||
};
|
};
|
||||||
const char *diff_get_color(int diff_use_color, enum color_diff ix);
|
const char *diff_get_color(int diff_use_color, enum color_diff ix);
|
||||||
#define diff_get_color_opt(o, ix) \
|
#define diff_get_color_opt(o, ix) \
|
||||||
|
1
git.c
1
git.c
@ -520,6 +520,7 @@ static struct cmd_struct commands[] = {
|
|||||||
{ "prune-packed", cmd_prune_packed, RUN_SETUP },
|
{ "prune-packed", cmd_prune_packed, RUN_SETUP },
|
||||||
{ "pull", cmd_pull, RUN_SETUP | NEED_WORK_TREE },
|
{ "pull", cmd_pull, RUN_SETUP | NEED_WORK_TREE },
|
||||||
{ "push", cmd_push, RUN_SETUP },
|
{ "push", cmd_push, RUN_SETUP },
|
||||||
|
{ "range-diff", cmd_range_diff, RUN_SETUP | USE_PAGER },
|
||||||
{ "read-tree", cmd_read_tree, RUN_SETUP | SUPPORT_SUPER_PREFIX},
|
{ "read-tree", cmd_read_tree, RUN_SETUP | SUPPORT_SUPER_PREFIX},
|
||||||
{ "rebase--helper", cmd_rebase__helper, RUN_SETUP | NEED_WORK_TREE },
|
{ "rebase--helper", cmd_rebase__helper, RUN_SETUP | NEED_WORK_TREE },
|
||||||
{ "receive-pack", cmd_receive_pack },
|
{ "receive-pack", cmd_receive_pack },
|
||||||
|
201
linear-assignment.c
Normal file
201
linear-assignment.c
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
/*
|
||||||
|
* Based on: Jonker, R., & Volgenant, A. (1987). <i>A shortest augmenting path
|
||||||
|
* algorithm for dense and sparse linear assignment problems</i>. Computing,
|
||||||
|
* 38(4), 325-340.
|
||||||
|
*/
|
||||||
|
#include "cache.h"
|
||||||
|
#include "linear-assignment.h"
|
||||||
|
|
||||||
|
#define COST(column, row) cost[(column) + column_count * (row)]
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The parameter `cost` is the cost matrix: the cost to assign column j to row
|
||||||
|
* i is `cost[j + column_count * i].
|
||||||
|
*/
|
||||||
|
void compute_assignment(int column_count, int row_count, int *cost,
|
||||||
|
int *column2row, int *row2column)
|
||||||
|
{
|
||||||
|
int *v, *d;
|
||||||
|
int *free_row, free_count = 0, saved_free_count, *pred, *col;
|
||||||
|
int i, j, phase;
|
||||||
|
|
||||||
|
memset(column2row, -1, sizeof(int) * column_count);
|
||||||
|
memset(row2column, -1, sizeof(int) * row_count);
|
||||||
|
ALLOC_ARRAY(v, column_count);
|
||||||
|
|
||||||
|
/* column reduction */
|
||||||
|
for (j = column_count - 1; j >= 0; j--) {
|
||||||
|
int i1 = 0;
|
||||||
|
|
||||||
|
for (i = 1; i < row_count; i++)
|
||||||
|
if (COST(j, i1) > COST(j, i))
|
||||||
|
i1 = i;
|
||||||
|
v[j] = COST(j, i1);
|
||||||
|
if (row2column[i1] == -1) {
|
||||||
|
/* row i1 unassigned */
|
||||||
|
row2column[i1] = j;
|
||||||
|
column2row[j] = i1;
|
||||||
|
} else {
|
||||||
|
if (row2column[i1] >= 0)
|
||||||
|
row2column[i1] = -2 - row2column[i1];
|
||||||
|
column2row[j] = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* reduction transfer */
|
||||||
|
ALLOC_ARRAY(free_row, row_count);
|
||||||
|
for (i = 0; i < row_count; i++) {
|
||||||
|
int j1 = row2column[i];
|
||||||
|
if (j1 == -1)
|
||||||
|
free_row[free_count++] = i;
|
||||||
|
else if (j1 < -1)
|
||||||
|
row2column[i] = -2 - j1;
|
||||||
|
else {
|
||||||
|
int min = COST(!j1, i) - v[!j1];
|
||||||
|
for (j = 1; j < column_count; j++)
|
||||||
|
if (j != j1 && min > COST(j, i) - v[j])
|
||||||
|
min = COST(j, i) - v[j];
|
||||||
|
v[j1] -= min;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (free_count ==
|
||||||
|
(column_count < row_count ? row_count - column_count : 0)) {
|
||||||
|
free(v);
|
||||||
|
free(free_row);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* augmenting row reduction */
|
||||||
|
for (phase = 0; phase < 2; phase++) {
|
||||||
|
int k = 0;
|
||||||
|
|
||||||
|
saved_free_count = free_count;
|
||||||
|
free_count = 0;
|
||||||
|
while (k < saved_free_count) {
|
||||||
|
int u1, u2;
|
||||||
|
int j1 = 0, j2, i0;
|
||||||
|
|
||||||
|
i = free_row[k++];
|
||||||
|
u1 = COST(j1, i) - v[j1];
|
||||||
|
j2 = -1;
|
||||||
|
u2 = INT_MAX;
|
||||||
|
for (j = 1; j < column_count; j++) {
|
||||||
|
int c = COST(j, i) - v[j];
|
||||||
|
if (u2 > c) {
|
||||||
|
if (u1 < c) {
|
||||||
|
u2 = c;
|
||||||
|
j2 = j;
|
||||||
|
} else {
|
||||||
|
u2 = u1;
|
||||||
|
u1 = c;
|
||||||
|
j2 = j1;
|
||||||
|
j1 = j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (j2 < 0) {
|
||||||
|
j2 = j1;
|
||||||
|
u2 = u1;
|
||||||
|
}
|
||||||
|
|
||||||
|
i0 = column2row[j1];
|
||||||
|
if (u1 < u2)
|
||||||
|
v[j1] -= u2 - u1;
|
||||||
|
else if (i0 >= 0) {
|
||||||
|
j1 = j2;
|
||||||
|
i0 = column2row[j1];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i0 >= 0) {
|
||||||
|
if (u1 < u2)
|
||||||
|
free_row[--k] = i0;
|
||||||
|
else
|
||||||
|
free_row[free_count++] = i0;
|
||||||
|
}
|
||||||
|
row2column[i] = j1;
|
||||||
|
column2row[j1] = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* augmentation */
|
||||||
|
saved_free_count = free_count;
|
||||||
|
ALLOC_ARRAY(d, column_count);
|
||||||
|
ALLOC_ARRAY(pred, column_count);
|
||||||
|
ALLOC_ARRAY(col, column_count);
|
||||||
|
for (free_count = 0; free_count < saved_free_count; free_count++) {
|
||||||
|
int i1 = free_row[free_count], low = 0, up = 0, last, k;
|
||||||
|
int min, c, u1;
|
||||||
|
|
||||||
|
for (j = 0; j < column_count; j++) {
|
||||||
|
d[j] = COST(j, i1) - v[j];
|
||||||
|
pred[j] = i1;
|
||||||
|
col[j] = j;
|
||||||
|
}
|
||||||
|
|
||||||
|
j = -1;
|
||||||
|
do {
|
||||||
|
last = low;
|
||||||
|
min = d[col[up++]];
|
||||||
|
for (k = up; k < column_count; k++) {
|
||||||
|
j = col[k];
|
||||||
|
c = d[j];
|
||||||
|
if (c <= min) {
|
||||||
|
if (c < min) {
|
||||||
|
up = low;
|
||||||
|
min = c;
|
||||||
|
}
|
||||||
|
col[k] = col[up];
|
||||||
|
col[up++] = j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (k = low; k < up; k++)
|
||||||
|
if (column2row[col[k]] == -1)
|
||||||
|
goto update;
|
||||||
|
|
||||||
|
/* scan a row */
|
||||||
|
do {
|
||||||
|
int j1 = col[low++];
|
||||||
|
|
||||||
|
i = column2row[j1];
|
||||||
|
u1 = COST(j1, i) - v[j1] - min;
|
||||||
|
for (k = up; k < column_count; k++) {
|
||||||
|
j = col[k];
|
||||||
|
c = COST(j, i) - v[j] - u1;
|
||||||
|
if (c < d[j]) {
|
||||||
|
d[j] = c;
|
||||||
|
pred[j] = i;
|
||||||
|
if (c == min) {
|
||||||
|
if (column2row[j] == -1)
|
||||||
|
goto update;
|
||||||
|
col[k] = col[up];
|
||||||
|
col[up++] = j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while (low != up);
|
||||||
|
} while (low == up);
|
||||||
|
|
||||||
|
update:
|
||||||
|
/* updating of the column pieces */
|
||||||
|
for (k = 0; k < last; k++) {
|
||||||
|
int j1 = col[k];
|
||||||
|
v[j1] += d[j1] - min;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* augmentation */
|
||||||
|
do {
|
||||||
|
if (j < 0)
|
||||||
|
BUG("negative j: %d", j);
|
||||||
|
i = pred[j];
|
||||||
|
column2row[j] = i;
|
||||||
|
SWAP(j, row2column[i]);
|
||||||
|
} while (i1 != i);
|
||||||
|
}
|
||||||
|
|
||||||
|
free(col);
|
||||||
|
free(pred);
|
||||||
|
free(d);
|
||||||
|
free(v);
|
||||||
|
free(free_row);
|
||||||
|
}
|
22
linear-assignment.h
Normal file
22
linear-assignment.h
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef LINEAR_ASSIGNMENT_H
|
||||||
|
#define LINEAR_ASSIGNMENT_H
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Compute an assignment of columns -> rows (and vice versa) such that every
|
||||||
|
* column is assigned to at most one row (and vice versa) minimizing the
|
||||||
|
* overall cost.
|
||||||
|
*
|
||||||
|
* The parameter `cost` is the cost matrix: the cost to assign column j to row
|
||||||
|
* i is `cost[j + column_count * i].
|
||||||
|
*
|
||||||
|
* The arrays column2row and row2column will be populated with the respective
|
||||||
|
* assignments (-1 for unassigned, which can happen only if column_count !=
|
||||||
|
* row_count).
|
||||||
|
*/
|
||||||
|
void compute_assignment(int column_count, int row_count, int *cost,
|
||||||
|
int *column2row, int *row2column);
|
||||||
|
|
||||||
|
/* The maximal cost in the cost matrix (to prevent integer overflows). */
|
||||||
|
#define COST_MAX (1<<16)
|
||||||
|
|
||||||
|
#endif
|
435
range-diff.c
Normal file
435
range-diff.c
Normal file
@ -0,0 +1,435 @@
|
|||||||
|
#include "cache.h"
|
||||||
|
#include "range-diff.h"
|
||||||
|
#include "string-list.h"
|
||||||
|
#include "run-command.h"
|
||||||
|
#include "argv-array.h"
|
||||||
|
#include "hashmap.h"
|
||||||
|
#include "xdiff-interface.h"
|
||||||
|
#include "linear-assignment.h"
|
||||||
|
#include "diffcore.h"
|
||||||
|
#include "commit.h"
|
||||||
|
#include "pretty.h"
|
||||||
|
#include "userdiff.h"
|
||||||
|
|
||||||
|
struct patch_util {
|
||||||
|
/* For the search for an exact match */
|
||||||
|
struct hashmap_entry e;
|
||||||
|
const char *diff, *patch;
|
||||||
|
|
||||||
|
int i, shown;
|
||||||
|
int diffsize;
|
||||||
|
size_t diff_offset;
|
||||||
|
/* the index of the matching item in the other branch, or -1 */
|
||||||
|
int matching;
|
||||||
|
struct object_id oid;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Reads the patches into a string list, with the `util` field being populated
|
||||||
|
* as struct object_id (will need to be free()d).
|
||||||
|
*/
|
||||||
|
static int read_patches(const char *range, struct string_list *list)
|
||||||
|
{
|
||||||
|
struct child_process cp = CHILD_PROCESS_INIT;
|
||||||
|
FILE *in;
|
||||||
|
struct strbuf buf = STRBUF_INIT, line = STRBUF_INIT;
|
||||||
|
struct patch_util *util = NULL;
|
||||||
|
int in_header = 1;
|
||||||
|
|
||||||
|
argv_array_pushl(&cp.args, "log", "--no-color", "-p", "--no-merges",
|
||||||
|
"--reverse", "--date-order", "--decorate=no",
|
||||||
|
"--no-abbrev-commit", range,
|
||||||
|
NULL);
|
||||||
|
cp.out = -1;
|
||||||
|
cp.no_stdin = 1;
|
||||||
|
cp.git_cmd = 1;
|
||||||
|
|
||||||
|
if (start_command(&cp))
|
||||||
|
return error_errno(_("could not start `log`"));
|
||||||
|
in = fdopen(cp.out, "r");
|
||||||
|
if (!in) {
|
||||||
|
error_errno(_("could not read `log` output"));
|
||||||
|
finish_command(&cp);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (strbuf_getline(&line, in) != EOF) {
|
||||||
|
const char *p;
|
||||||
|
|
||||||
|
if (skip_prefix(line.buf, "commit ", &p)) {
|
||||||
|
if (util) {
|
||||||
|
string_list_append(list, buf.buf)->util = util;
|
||||||
|
strbuf_reset(&buf);
|
||||||
|
}
|
||||||
|
util = xcalloc(sizeof(*util), 1);
|
||||||
|
if (get_oid(p, &util->oid)) {
|
||||||
|
error(_("could not parse commit '%s'"), p);
|
||||||
|
free(util);
|
||||||
|
string_list_clear(list, 1);
|
||||||
|
strbuf_release(&buf);
|
||||||
|
strbuf_release(&line);
|
||||||
|
fclose(in);
|
||||||
|
finish_command(&cp);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
util->matching = -1;
|
||||||
|
in_header = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (starts_with(line.buf, "diff --git")) {
|
||||||
|
in_header = 0;
|
||||||
|
strbuf_addch(&buf, '\n');
|
||||||
|
if (!util->diff_offset)
|
||||||
|
util->diff_offset = buf.len;
|
||||||
|
strbuf_addbuf(&buf, &line);
|
||||||
|
} else if (in_header) {
|
||||||
|
if (starts_with(line.buf, "Author: ")) {
|
||||||
|
strbuf_addbuf(&buf, &line);
|
||||||
|
strbuf_addstr(&buf, "\n\n");
|
||||||
|
} else if (starts_with(line.buf, " ")) {
|
||||||
|
strbuf_rtrim(&line);
|
||||||
|
strbuf_addbuf(&buf, &line);
|
||||||
|
strbuf_addch(&buf, '\n');
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
} else if (starts_with(line.buf, "@@ "))
|
||||||
|
strbuf_addstr(&buf, "@@");
|
||||||
|
else if (!line.buf[0] || starts_with(line.buf, "index "))
|
||||||
|
/*
|
||||||
|
* A completely blank (not ' \n', which is context)
|
||||||
|
* line is not valid in a diff. We skip it
|
||||||
|
* silently, because this neatly handles the blank
|
||||||
|
* separator line between commits in git-log
|
||||||
|
* output.
|
||||||
|
*
|
||||||
|
* We also want to ignore the diff's `index` lines
|
||||||
|
* because they contain exact blob hashes in which
|
||||||
|
* we are not interested.
|
||||||
|
*/
|
||||||
|
continue;
|
||||||
|
else
|
||||||
|
strbuf_addbuf(&buf, &line);
|
||||||
|
|
||||||
|
strbuf_addch(&buf, '\n');
|
||||||
|
util->diffsize++;
|
||||||
|
}
|
||||||
|
fclose(in);
|
||||||
|
strbuf_release(&line);
|
||||||
|
|
||||||
|
if (util)
|
||||||
|
string_list_append(list, buf.buf)->util = util;
|
||||||
|
strbuf_release(&buf);
|
||||||
|
|
||||||
|
if (finish_command(&cp))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int patch_util_cmp(const void *dummy, const struct patch_util *a,
|
||||||
|
const struct patch_util *b, const char *keydata)
|
||||||
|
{
|
||||||
|
return strcmp(a->diff, keydata ? keydata : b->diff);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void find_exact_matches(struct string_list *a, struct string_list *b)
|
||||||
|
{
|
||||||
|
struct hashmap map;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
hashmap_init(&map, (hashmap_cmp_fn)patch_util_cmp, NULL, 0);
|
||||||
|
|
||||||
|
/* First, add the patches of a to a hash map */
|
||||||
|
for (i = 0; i < a->nr; i++) {
|
||||||
|
struct patch_util *util = a->items[i].util;
|
||||||
|
|
||||||
|
util->i = i;
|
||||||
|
util->patch = a->items[i].string;
|
||||||
|
util->diff = util->patch + util->diff_offset;
|
||||||
|
hashmap_entry_init(util, strhash(util->diff));
|
||||||
|
hashmap_add(&map, util);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now try to find exact matches in b */
|
||||||
|
for (i = 0; i < b->nr; i++) {
|
||||||
|
struct patch_util *util = b->items[i].util, *other;
|
||||||
|
|
||||||
|
util->i = i;
|
||||||
|
util->patch = b->items[i].string;
|
||||||
|
util->diff = util->patch + util->diff_offset;
|
||||||
|
hashmap_entry_init(util, strhash(util->diff));
|
||||||
|
other = hashmap_remove(&map, util, NULL);
|
||||||
|
if (other) {
|
||||||
|
if (other->matching >= 0)
|
||||||
|
BUG("already assigned!");
|
||||||
|
|
||||||
|
other->matching = i;
|
||||||
|
util->matching = other->i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hashmap_free(&map, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void diffsize_consume(void *data, char *line, unsigned long len)
|
||||||
|
{
|
||||||
|
(*(int *)data)++;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int diffsize(const char *a, const char *b)
|
||||||
|
{
|
||||||
|
xpparam_t pp = { 0 };
|
||||||
|
xdemitconf_t cfg = { 0 };
|
||||||
|
mmfile_t mf1, mf2;
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
mf1.ptr = (char *)a;
|
||||||
|
mf1.size = strlen(a);
|
||||||
|
mf2.ptr = (char *)b;
|
||||||
|
mf2.size = strlen(b);
|
||||||
|
|
||||||
|
cfg.ctxlen = 3;
|
||||||
|
if (!xdi_diff_outf(&mf1, &mf2, diffsize_consume, &count, &pp, &cfg))
|
||||||
|
return count;
|
||||||
|
|
||||||
|
error(_("failed to generate diff"));
|
||||||
|
return COST_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void get_correspondences(struct string_list *a, struct string_list *b,
|
||||||
|
int creation_factor)
|
||||||
|
{
|
||||||
|
int n = a->nr + b->nr;
|
||||||
|
int *cost, c, *a2b, *b2a;
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
ALLOC_ARRAY(cost, st_mult(n, n));
|
||||||
|
ALLOC_ARRAY(a2b, n);
|
||||||
|
ALLOC_ARRAY(b2a, n);
|
||||||
|
|
||||||
|
for (i = 0; i < a->nr; i++) {
|
||||||
|
struct patch_util *a_util = a->items[i].util;
|
||||||
|
|
||||||
|
for (j = 0; j < b->nr; j++) {
|
||||||
|
struct patch_util *b_util = b->items[j].util;
|
||||||
|
|
||||||
|
if (a_util->matching == j)
|
||||||
|
c = 0;
|
||||||
|
else if (a_util->matching < 0 && b_util->matching < 0)
|
||||||
|
c = diffsize(a_util->diff, b_util->diff);
|
||||||
|
else
|
||||||
|
c = COST_MAX;
|
||||||
|
cost[i + n * j] = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
c = a_util->matching < 0 ?
|
||||||
|
a_util->diffsize * creation_factor / 100 : COST_MAX;
|
||||||
|
for (j = b->nr; j < n; j++)
|
||||||
|
cost[i + n * j] = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (j = 0; j < b->nr; j++) {
|
||||||
|
struct patch_util *util = b->items[j].util;
|
||||||
|
|
||||||
|
c = util->matching < 0 ?
|
||||||
|
util->diffsize * creation_factor / 100 : COST_MAX;
|
||||||
|
for (i = a->nr; i < n; i++)
|
||||||
|
cost[i + n * j] = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = a->nr; i < n; i++)
|
||||||
|
for (j = b->nr; j < n; j++)
|
||||||
|
cost[i + n * j] = 0;
|
||||||
|
|
||||||
|
compute_assignment(n, n, cost, a2b, b2a);
|
||||||
|
|
||||||
|
for (i = 0; i < a->nr; i++)
|
||||||
|
if (a2b[i] >= 0 && a2b[i] < b->nr) {
|
||||||
|
struct patch_util *a_util = a->items[i].util;
|
||||||
|
struct patch_util *b_util = b->items[a2b[i]].util;
|
||||||
|
|
||||||
|
a_util->matching = a2b[i];
|
||||||
|
b_util->matching = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(cost);
|
||||||
|
free(a2b);
|
||||||
|
free(b2a);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void output_pair_header(struct diff_options *diffopt,
|
||||||
|
int patch_no_width,
|
||||||
|
struct strbuf *buf,
|
||||||
|
struct strbuf *dashes,
|
||||||
|
struct patch_util *a_util,
|
||||||
|
struct patch_util *b_util)
|
||||||
|
{
|
||||||
|
struct object_id *oid = a_util ? &a_util->oid : &b_util->oid;
|
||||||
|
struct commit *commit;
|
||||||
|
char status;
|
||||||
|
const char *color_reset = diff_get_color_opt(diffopt, DIFF_RESET);
|
||||||
|
const char *color_old = diff_get_color_opt(diffopt, DIFF_FILE_OLD);
|
||||||
|
const char *color_new = diff_get_color_opt(diffopt, DIFF_FILE_NEW);
|
||||||
|
const char *color_commit = diff_get_color_opt(diffopt, DIFF_COMMIT);
|
||||||
|
const char *color;
|
||||||
|
|
||||||
|
if (!dashes->len)
|
||||||
|
strbuf_addchars(dashes, '-',
|
||||||
|
strlen(find_unique_abbrev(oid,
|
||||||
|
DEFAULT_ABBREV)));
|
||||||
|
|
||||||
|
if (!b_util) {
|
||||||
|
color = color_old;
|
||||||
|
status = '<';
|
||||||
|
} else if (!a_util) {
|
||||||
|
color = color_new;
|
||||||
|
status = '>';
|
||||||
|
} else if (strcmp(a_util->patch, b_util->patch)) {
|
||||||
|
color = color_commit;
|
||||||
|
status = '!';
|
||||||
|
} else {
|
||||||
|
color = color_commit;
|
||||||
|
status = '=';
|
||||||
|
}
|
||||||
|
|
||||||
|
strbuf_reset(buf);
|
||||||
|
strbuf_addstr(buf, status == '!' ? color_old : color);
|
||||||
|
if (!a_util)
|
||||||
|
strbuf_addf(buf, "%*s: %s ", patch_no_width, "-", dashes->buf);
|
||||||
|
else
|
||||||
|
strbuf_addf(buf, "%*d: %s ", patch_no_width, a_util->i + 1,
|
||||||
|
find_unique_abbrev(&a_util->oid, DEFAULT_ABBREV));
|
||||||
|
|
||||||
|
if (status == '!')
|
||||||
|
strbuf_addf(buf, "%s%s", color_reset, color);
|
||||||
|
strbuf_addch(buf, status);
|
||||||
|
if (status == '!')
|
||||||
|
strbuf_addf(buf, "%s%s", color_reset, color_new);
|
||||||
|
|
||||||
|
if (!b_util)
|
||||||
|
strbuf_addf(buf, " %*s: %s", patch_no_width, "-", dashes->buf);
|
||||||
|
else
|
||||||
|
strbuf_addf(buf, " %*d: %s", patch_no_width, b_util->i + 1,
|
||||||
|
find_unique_abbrev(&b_util->oid, DEFAULT_ABBREV));
|
||||||
|
|
||||||
|
commit = lookup_commit_reference(the_repository, oid);
|
||||||
|
if (commit) {
|
||||||
|
if (status == '!')
|
||||||
|
strbuf_addf(buf, "%s%s", color_reset, color);
|
||||||
|
|
||||||
|
strbuf_addch(buf, ' ');
|
||||||
|
pp_commit_easy(CMIT_FMT_ONELINE, commit, buf);
|
||||||
|
}
|
||||||
|
strbuf_addf(buf, "%s\n", color_reset);
|
||||||
|
|
||||||
|
fwrite(buf->buf, buf->len, 1, stdout);
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct userdiff_driver no_func_name = {
|
||||||
|
.funcname = { "$^", 0 }
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct diff_filespec *get_filespec(const char *name, const char *p)
|
||||||
|
{
|
||||||
|
struct diff_filespec *spec = alloc_filespec(name);
|
||||||
|
|
||||||
|
fill_filespec(spec, &null_oid, 0, 0644);
|
||||||
|
spec->data = (char *)p;
|
||||||
|
spec->size = strlen(p);
|
||||||
|
spec->should_munmap = 0;
|
||||||
|
spec->is_stdin = 1;
|
||||||
|
spec->driver = &no_func_name;
|
||||||
|
|
||||||
|
return spec;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void patch_diff(const char *a, const char *b,
|
||||||
|
struct diff_options *diffopt)
|
||||||
|
{
|
||||||
|
diff_queue(&diff_queued_diff,
|
||||||
|
get_filespec("a", a), get_filespec("b", b));
|
||||||
|
|
||||||
|
diffcore_std(diffopt);
|
||||||
|
diff_flush(diffopt);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void output(struct string_list *a, struct string_list *b,
|
||||||
|
struct diff_options *diffopt)
|
||||||
|
{
|
||||||
|
struct strbuf buf = STRBUF_INIT, dashes = STRBUF_INIT;
|
||||||
|
int patch_no_width = decimal_width(1 + (a->nr > b->nr ? a->nr : b->nr));
|
||||||
|
int i = 0, j = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We assume the user is really more interested in the second argument
|
||||||
|
* ("newer" version). To that end, we print the output in the order of
|
||||||
|
* the RHS (the `b` parameter). To put the LHS (the `a` parameter)
|
||||||
|
* commits that are no longer in the RHS into a good place, we place
|
||||||
|
* them once we have shown all of their predecessors in the LHS.
|
||||||
|
*/
|
||||||
|
|
||||||
|
while (i < a->nr || j < b->nr) {
|
||||||
|
struct patch_util *a_util, *b_util;
|
||||||
|
a_util = i < a->nr ? a->items[i].util : NULL;
|
||||||
|
b_util = j < b->nr ? b->items[j].util : NULL;
|
||||||
|
|
||||||
|
/* Skip all the already-shown commits from the LHS. */
|
||||||
|
while (i < a->nr && a_util->shown)
|
||||||
|
a_util = ++i < a->nr ? a->items[i].util : NULL;
|
||||||
|
|
||||||
|
/* Show unmatched LHS commit whose predecessors were shown. */
|
||||||
|
if (i < a->nr && a_util->matching < 0) {
|
||||||
|
output_pair_header(diffopt, patch_no_width,
|
||||||
|
&buf, &dashes, a_util, NULL);
|
||||||
|
i++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Show unmatched RHS commits. */
|
||||||
|
while (j < b->nr && b_util->matching < 0) {
|
||||||
|
output_pair_header(diffopt, patch_no_width,
|
||||||
|
&buf, &dashes, NULL, b_util);
|
||||||
|
b_util = ++j < b->nr ? b->items[j].util : NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Show matching LHS/RHS pair. */
|
||||||
|
if (j < b->nr) {
|
||||||
|
a_util = a->items[b_util->matching].util;
|
||||||
|
output_pair_header(diffopt, patch_no_width,
|
||||||
|
&buf, &dashes, a_util, b_util);
|
||||||
|
if (!(diffopt->output_format & DIFF_FORMAT_NO_OUTPUT))
|
||||||
|
patch_diff(a->items[b_util->matching].string,
|
||||||
|
b->items[j].string, diffopt);
|
||||||
|
a_util->shown = 1;
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
strbuf_release(&buf);
|
||||||
|
strbuf_release(&dashes);
|
||||||
|
}
|
||||||
|
|
||||||
|
int show_range_diff(const char *range1, const char *range2,
|
||||||
|
int creation_factor, struct diff_options *diffopt)
|
||||||
|
{
|
||||||
|
int res = 0;
|
||||||
|
|
||||||
|
struct string_list branch1 = STRING_LIST_INIT_DUP;
|
||||||
|
struct string_list branch2 = STRING_LIST_INIT_DUP;
|
||||||
|
|
||||||
|
if (read_patches(range1, &branch1))
|
||||||
|
res = error(_("could not parse log for '%s'"), range1);
|
||||||
|
if (!res && read_patches(range2, &branch2))
|
||||||
|
res = error(_("could not parse log for '%s'"), range2);
|
||||||
|
|
||||||
|
if (!res) {
|
||||||
|
find_exact_matches(&branch1, &branch2);
|
||||||
|
get_correspondences(&branch1, &branch2, creation_factor);
|
||||||
|
output(&branch1, &branch2, diffopt);
|
||||||
|
}
|
||||||
|
|
||||||
|
string_list_clear(&branch1, 1);
|
||||||
|
string_list_clear(&branch2, 1);
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
9
range-diff.h
Normal file
9
range-diff.h
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
#ifndef RANGE_DIFF_H
|
||||||
|
#define RANGE_DIFF_H
|
||||||
|
|
||||||
|
#include "diff.h"
|
||||||
|
|
||||||
|
int show_range_diff(const char *range1, const char *range2,
|
||||||
|
int creation_factor, struct diff_options *diffopt);
|
||||||
|
|
||||||
|
#endif
|
1
t/.gitattributes
vendored
1
t/.gitattributes
vendored
@ -1,6 +1,7 @@
|
|||||||
t[0-9][0-9][0-9][0-9]/* -whitespace
|
t[0-9][0-9][0-9][0-9]/* -whitespace
|
||||||
/diff-lib/* eol=lf
|
/diff-lib/* eol=lf
|
||||||
/t0110/url-* binary
|
/t0110/url-* binary
|
||||||
|
/t3206/* eol=lf
|
||||||
/t3900/*.txt eol=lf
|
/t3900/*.txt eol=lf
|
||||||
/t3901/*.txt eol=lf
|
/t3901/*.txt eol=lf
|
||||||
/t4034/*/* eol=lf
|
/t4034/*/* eol=lf
|
||||||
|
145
t/t3206-range-diff.sh
Executable file
145
t/t3206-range-diff.sh
Executable file
@ -0,0 +1,145 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
test_description='range-diff tests'
|
||||||
|
|
||||||
|
. ./test-lib.sh
|
||||||
|
|
||||||
|
# Note that because of the range-diff's heuristics, test_commit does more
|
||||||
|
# harm than good. We need some real history.
|
||||||
|
|
||||||
|
test_expect_success 'setup' '
|
||||||
|
git fast-import < "$TEST_DIRECTORY"/t3206/history.export
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'simple A..B A..C (unmodified)' '
|
||||||
|
git range-diff --no-color master..topic master..unmodified \
|
||||||
|
>actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: 35b9b25 s/5/A/
|
||||||
|
2: fccce22 = 2: de345ab s/4/A/
|
||||||
|
3: 147e64e = 3: 9af6654 s/11/B/
|
||||||
|
4: a63e992 = 4: 2901f77 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'simple B...C (unmodified)' '
|
||||||
|
git range-diff --no-color topic...unmodified >actual &&
|
||||||
|
# same "expected" as above
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'simple A B C (unmodified)' '
|
||||||
|
git range-diff --no-color master topic unmodified >actual &&
|
||||||
|
# same "expected" as above
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'trivial reordering' '
|
||||||
|
git range-diff --no-color master topic reordered >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: aca177a s/5/A/
|
||||||
|
3: 147e64e = 2: 14ad629 s/11/B/
|
||||||
|
4: a63e992 = 3: ee58208 s/12/B/
|
||||||
|
2: fccce22 = 4: 307b27a s/4/A/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'removed a commit' '
|
||||||
|
git range-diff --no-color master topic removed >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: 7657159 s/5/A/
|
||||||
|
2: fccce22 < -: ------- s/4/A/
|
||||||
|
3: 147e64e = 2: 43d84d3 s/11/B/
|
||||||
|
4: a63e992 = 3: a740396 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'added a commit' '
|
||||||
|
git range-diff --no-color master topic added >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: 2716022 s/5/A/
|
||||||
|
2: fccce22 = 2: b62accd s/4/A/
|
||||||
|
-: ------- > 3: df46cfa s/6/A/
|
||||||
|
3: 147e64e = 4: 3e64548 s/11/B/
|
||||||
|
4: a63e992 = 5: 12b4063 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'new base, A B C' '
|
||||||
|
git range-diff --no-color master topic rebased >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: cc9c443 s/5/A/
|
||||||
|
2: fccce22 = 2: c5d9641 s/4/A/
|
||||||
|
3: 147e64e = 3: 28cc2b6 s/11/B/
|
||||||
|
4: a63e992 = 4: 5628ab7 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'new base, B...C' '
|
||||||
|
# this syntax includes the commits from master!
|
||||||
|
git range-diff --no-color topic...rebased >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
-: ------- > 1: a31b12e unrelated
|
||||||
|
1: 4de457d = 2: cc9c443 s/5/A/
|
||||||
|
2: fccce22 = 3: c5d9641 s/4/A/
|
||||||
|
3: 147e64e = 4: 28cc2b6 s/11/B/
|
||||||
|
4: a63e992 = 5: 5628ab7 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'changed commit' '
|
||||||
|
git range-diff --no-color topic...changed >actual &&
|
||||||
|
cat >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: a4b3333 s/5/A/
|
||||||
|
2: fccce22 = 2: f51d370 s/4/A/
|
||||||
|
3: 147e64e ! 3: 0559556 s/11/B/
|
||||||
|
@@ -10,7 +10,7 @@
|
||||||
|
9
|
||||||
|
10
|
||||||
|
-11
|
||||||
|
-+B
|
||||||
|
++BB
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
4: a63e992 ! 4: d966c5c s/12/B/
|
||||||
|
@@ -8,7 +8,7 @@
|
||||||
|
@@
|
||||||
|
9
|
||||||
|
10
|
||||||
|
- B
|
||||||
|
+ BB
|
||||||
|
-12
|
||||||
|
+B
|
||||||
|
13
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success 'changed message' '
|
||||||
|
git range-diff --no-color topic...changed-message >actual &&
|
||||||
|
sed s/Z/\ /g >expected <<-EOF &&
|
||||||
|
1: 4de457d = 1: f686024 s/5/A/
|
||||||
|
2: fccce22 ! 2: 4ab067d s/4/A/
|
||||||
|
@@ -2,6 +2,8 @@
|
||||||
|
Z
|
||||||
|
Z s/4/A/
|
||||||
|
Z
|
||||||
|
+ Also a silly comment here!
|
||||||
|
+
|
||||||
|
Zdiff --git a/file b/file
|
||||||
|
Z--- a/file
|
||||||
|
Z+++ b/file
|
||||||
|
3: 147e64e = 3: b9cb956 s/11/B/
|
||||||
|
4: a63e992 = 4: 8add5f1 s/12/B/
|
||||||
|
EOF
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_done
|
604
t/t3206/history.export
Normal file
604
t/t3206/history.export
Normal file
@ -0,0 +1,604 @@
|
|||||||
|
blob
|
||||||
|
mark :1
|
||||||
|
data 51
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
4
|
||||||
|
5
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
11
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
reset refs/heads/removed
|
||||||
|
commit refs/heads/removed
|
||||||
|
mark :2
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374424921 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374484724 +0200
|
||||||
|
data 8
|
||||||
|
initial
|
||||||
|
M 100644 :1 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :3
|
||||||
|
data 51
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
4
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
11
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/topic
|
||||||
|
mark :4
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :5
|
||||||
|
data 51
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
11
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/topic
|
||||||
|
mark :6
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :4
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :7
|
||||||
|
data 50
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/topic
|
||||||
|
mark :8
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :6
|
||||||
|
M 100644 :7 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :9
|
||||||
|
data 49
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
B
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/topic
|
||||||
|
mark :10
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :8
|
||||||
|
M 100644 :9 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :11
|
||||||
|
data 10
|
||||||
|
unrelated
|
||||||
|
|
||||||
|
commit refs/heads/master
|
||||||
|
mark :12
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485127 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485127 +0200
|
||||||
|
data 10
|
||||||
|
unrelated
|
||||||
|
from :2
|
||||||
|
M 100644 :11 otherfile
|
||||||
|
|
||||||
|
commit refs/heads/rebased
|
||||||
|
mark :13
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485137 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :12
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/rebased
|
||||||
|
mark :14
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485138 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :13
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
commit refs/heads/rebased
|
||||||
|
mark :15
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485138 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :14
|
||||||
|
M 100644 :7 file
|
||||||
|
|
||||||
|
commit refs/heads/rebased
|
||||||
|
mark :16
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485138 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :15
|
||||||
|
M 100644 :9 file
|
||||||
|
|
||||||
|
commit refs/heads/added
|
||||||
|
mark :17
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485341 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/added
|
||||||
|
mark :18
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485341 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :17
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :19
|
||||||
|
data 51
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
A
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
11
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/added
|
||||||
|
mark :20
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485186 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485341 +0200
|
||||||
|
data 7
|
||||||
|
s/6/A/
|
||||||
|
from :18
|
||||||
|
M 100644 :19 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :21
|
||||||
|
data 50
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
A
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/added
|
||||||
|
mark :22
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485341 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :20
|
||||||
|
M 100644 :21 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :23
|
||||||
|
data 49
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
A
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
B
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/added
|
||||||
|
mark :24
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485341 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :22
|
||||||
|
M 100644 :23 file
|
||||||
|
|
||||||
|
commit refs/heads/reordered
|
||||||
|
mark :25
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485350 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :26
|
||||||
|
data 50
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
4
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/reordered
|
||||||
|
mark :27
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485350 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :25
|
||||||
|
M 100644 :26 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :28
|
||||||
|
data 49
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
4
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
B
|
||||||
|
B
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/reordered
|
||||||
|
mark :29
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485350 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :27
|
||||||
|
M 100644 :28 file
|
||||||
|
|
||||||
|
commit refs/heads/reordered
|
||||||
|
mark :30
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485350 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :29
|
||||||
|
M 100644 :9 file
|
||||||
|
|
||||||
|
commit refs/heads/changed
|
||||||
|
mark :31
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485507 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/changed
|
||||||
|
mark :32
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485507 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :31
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :33
|
||||||
|
data 51
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
BB
|
||||||
|
12
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/changed
|
||||||
|
mark :34
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485507 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :32
|
||||||
|
M 100644 :33 file
|
||||||
|
|
||||||
|
blob
|
||||||
|
mark :35
|
||||||
|
data 50
|
||||||
|
1
|
||||||
|
2
|
||||||
|
3
|
||||||
|
A
|
||||||
|
A
|
||||||
|
6
|
||||||
|
7
|
||||||
|
8
|
||||||
|
9
|
||||||
|
10
|
||||||
|
BB
|
||||||
|
B
|
||||||
|
13
|
||||||
|
14
|
||||||
|
15
|
||||||
|
16
|
||||||
|
17
|
||||||
|
18
|
||||||
|
19
|
||||||
|
20
|
||||||
|
|
||||||
|
commit refs/heads/changed
|
||||||
|
mark :36
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485507 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :34
|
||||||
|
M 100644 :35 file
|
||||||
|
|
||||||
|
commit refs/heads/changed-message
|
||||||
|
mark :37
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485530 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/changed-message
|
||||||
|
mark :38
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485530 +0200
|
||||||
|
data 35
|
||||||
|
s/4/A/
|
||||||
|
|
||||||
|
Also a silly comment here!
|
||||||
|
from :37
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
commit refs/heads/changed-message
|
||||||
|
mark :39
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485536 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :38
|
||||||
|
M 100644 :7 file
|
||||||
|
|
||||||
|
commit refs/heads/changed-message
|
||||||
|
mark :40
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485536 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :39
|
||||||
|
M 100644 :9 file
|
||||||
|
|
||||||
|
commit refs/heads/unmodified
|
||||||
|
mark :41
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485631 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/unmodified
|
||||||
|
mark :42
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485024 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485631 +0200
|
||||||
|
data 7
|
||||||
|
s/4/A/
|
||||||
|
from :41
|
||||||
|
M 100644 :5 file
|
||||||
|
|
||||||
|
commit refs/heads/unmodified
|
||||||
|
mark :43
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485632 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :42
|
||||||
|
M 100644 :7 file
|
||||||
|
|
||||||
|
commit refs/heads/unmodified
|
||||||
|
mark :44
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374485632 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :43
|
||||||
|
M 100644 :9 file
|
||||||
|
|
||||||
|
commit refs/heads/removed
|
||||||
|
mark :45
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485014 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374486061 +0200
|
||||||
|
data 7
|
||||||
|
s/5/A/
|
||||||
|
from :2
|
||||||
|
M 100644 :3 file
|
||||||
|
|
||||||
|
commit refs/heads/removed
|
||||||
|
mark :46
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485036 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374486061 +0200
|
||||||
|
data 8
|
||||||
|
s/11/B/
|
||||||
|
from :45
|
||||||
|
M 100644 :26 file
|
||||||
|
|
||||||
|
commit refs/heads/removed
|
||||||
|
mark :47
|
||||||
|
author Thomas Rast <trast@inf.ethz.ch> 1374485044 +0200
|
||||||
|
committer Thomas Rast <trast@inf.ethz.ch> 1374486061 +0200
|
||||||
|
data 8
|
||||||
|
s/12/B/
|
||||||
|
from :46
|
||||||
|
M 100644 :28 file
|
||||||
|
|
||||||
|
reset refs/heads/removed
|
||||||
|
from :47
|
||||||
|
|
Loading…
Reference in New Issue
Block a user