You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
170 lines
3.2 KiB
170 lines
3.2 KiB
#!/usr/bin/perl -w |
|
# |
|
# Copyright (c) 2005 Junio C Hamano |
|
# |
|
# Read .git/FETCH_HEAD and make a human readable merge message |
|
# by grouping branches and tags together to form a single line. |
|
|
|
use strict; |
|
|
|
my @src; |
|
my %src; |
|
sub andjoin { |
|
my ($label, $labels, $stuff) = @_; |
|
my $l = scalar @$stuff; |
|
my $m = ''; |
|
if ($l == 0) { |
|
return (); |
|
} |
|
if ($l == 1) { |
|
$m = "$label$stuff->[0]"; |
|
} |
|
else { |
|
$m = ("$labels" . |
|
join (', ', @{$stuff}[0..$l-2]) . |
|
" and $stuff->[-1]"); |
|
} |
|
return ($m); |
|
} |
|
|
|
sub repoconfig { |
|
my $fh; |
|
my $val; |
|
eval { |
|
open $fh, '-|', 'git-repo-config', '--get', 'merge.summary' |
|
or die "$!"; |
|
($val) = <$fh>; |
|
close $fh; |
|
}; |
|
return $val; |
|
} |
|
|
|
sub mergebase { |
|
my ($other) = @_; |
|
my $fh; |
|
open $fh, '-|', 'git-merge-base', '--all', 'HEAD', $other or die "$!"; |
|
my (@mb) = map { chomp; $_ } <$fh>; |
|
close $fh or die "$!"; |
|
return @mb; |
|
} |
|
|
|
sub shortlog { |
|
my ($tip, $limit, @base) = @_; |
|
my ($fh, @result); |
|
open $fh, '-|', ('git-log', "--max-count=$limit", '--topo-order', |
|
'--pretty=oneline', $tip, map { "^$_" } @base) |
|
or die "$!"; |
|
while (<$fh>) { |
|
s/^[0-9a-f]{40}\s+//; |
|
push @result, $_; |
|
} |
|
close $fh or die "$!"; |
|
return @result; |
|
} |
|
|
|
my @origin = (); |
|
while (<>) { |
|
my ($bname, $tname, $gname, $src, $sha1, $origin); |
|
chomp; |
|
s/^([0-9a-f]*) //; |
|
$sha1 = $1; |
|
next if (/^not-for-merge/); |
|
s/^ //; |
|
if (s/ of (.*)$//) { |
|
$src = $1; |
|
} else { |
|
# Pulling HEAD |
|
$src = $_; |
|
$_ = 'HEAD'; |
|
} |
|
if (! exists $src{$src}) { |
|
push @src, $src; |
|
$src{$src} = { |
|
BRANCH => [], |
|
TAG => [], |
|
GENERIC => [], |
|
# &1 == has HEAD. |
|
# &2 == has others. |
|
HEAD_STATUS => 0, |
|
}; |
|
} |
|
if (/^branch (.*)$/) { |
|
$origin = $1; |
|
push @{$src{$src}{BRANCH}}, $1; |
|
$src{$src}{HEAD_STATUS} |= 2; |
|
} |
|
elsif (/^tag (.*)$/) { |
|
$origin = $_; |
|
push @{$src{$src}{TAG}}, $1; |
|
$src{$src}{HEAD_STATUS} |= 2; |
|
} |
|
elsif (/^HEAD$/) { |
|
$origin = $src; |
|
$src{$src}{HEAD_STATUS} |= 1; |
|
} |
|
else { |
|
push @{$src{$src}{GENERIC}}, $_; |
|
$src{$src}{HEAD_STATUS} |= 2; |
|
$origin = $src; |
|
} |
|
if ($src eq '.' || $src eq $origin) { |
|
$origin =~ s/^'(.*)'$/$1/; |
|
push @origin, [$sha1, "$origin"]; |
|
} |
|
else { |
|
push @origin, [$sha1, "$origin of $src"]; |
|
} |
|
} |
|
|
|
my @msg; |
|
for my $src (@src) { |
|
if ($src{$src}{HEAD_STATUS} == 1) { |
|
# Only HEAD is fetched, nothing else. |
|
push @msg, $src; |
|
next; |
|
} |
|
my @this; |
|
if ($src{$src}{HEAD_STATUS} == 3) { |
|
# HEAD is fetched among others. |
|
push @this, andjoin('', '', ['HEAD']); |
|
} |
|
push @this, andjoin("branch ", "branches ", |
|
$src{$src}{BRANCH}); |
|
push @this, andjoin("tag ", "tags ", |
|
$src{$src}{TAG}); |
|
push @this, andjoin("commit ", "commits ", |
|
$src{$src}{GENERIC}); |
|
my $this = join(', ', @this); |
|
if ($src ne '.') { |
|
$this .= " of $src"; |
|
} |
|
push @msg, $this; |
|
} |
|
print "Merge ", join("; ", @msg), "\n"; |
|
|
|
if (!repoconfig) { |
|
exit(0); |
|
} |
|
|
|
# We limit the merge message to the latst 20 or so per each branch. |
|
my $limit = 20; |
|
|
|
for (@origin) { |
|
my ($sha1, $name) = @$_; |
|
my @mb = mergebase($sha1); |
|
my @log = shortlog($sha1, $limit + 1, @mb); |
|
if ($limit + 1 <= @log) { |
|
print "\n* $name: (" . scalar(@log) . " commits)\n"; |
|
} |
|
else { |
|
print "\n* $name:\n"; |
|
} |
|
my $cnt = 0; |
|
for my $log (@log) { |
|
if ($limit < ++$cnt) { |
|
print " ...\n"; |
|
last; |
|
} |
|
print " $log"; |
|
} |
|
}
|
|
|