3 # List people who might be interested in a patch. Useful as the argument to
4 # git-send-email --cc-cmd option, and in other situations.
6 # Usage: git contacts <file | rev-list option> ...
12 my $since = '5-years-ago';
14 my $labels_rx = qr/Signed-off-by|Reviewed-by|Acked-by|Cc/i;
18 my ($name, $email) = @_;
19 return "$name <$email>";
23 my ($commit, $data) = @_;
24 my $contacts = $commit->{contacts
};
26 for (split(/^/m, $data)) {
28 if (/^author ([^<>]+) <(\S+)> .+$/) {
29 $contacts->{format_contact
($1, $2)} = 1;
33 } elsif (/^$labels_rx:\s+([^<>]+)\s+<(\S+?)>$/o) {
34 $contacts->{format_contact
($1, $2)} = 1;
41 return unless %$commits;
42 my $pid = open2
my $reader, my $writer, qw(git cat-file --batch);
43 for my $id (keys(%$commits)) {
44 print $writer "$id\n";
46 if ($line =~ /^([0-9a-f]{40}) commit (\d+)/) {
47 my ($cid, $len) = ($1, $2);
48 die "expected $id but got $cid\n" unless $id eq $cid;
50 # cat-file emits newline after data, so read len+1
51 read $reader, $data, $len + 1;
52 parse_commit
($commits->{$id}, $data);
58 die "git-cat-file error: $?\n" if $?
;
62 my ($commits, $source, $start, $len, $from) = @_;
63 $len = 1 unless defined($len);
66 qw(git blame --porcelain -C), '-L', "$start,+$len",
67 '--since', $since, "$from^", '--', $source or die;
69 if (/^([0-9a-f]{40}) \d+ \d+ \d+$/) {
71 $commits->{$id} = { id
=> $id, contacts
=> {} }
80 my ($commits, $id, $f) = @_;
83 if (/^From ([0-9a-f]{40}) Mon Sep 17 00:00:00 2001$/) {
88 if (m{^--- (?:a/(.+)|/dev/null)$}) {
91 die "Cannot parse hunk source: $_\n";
92 } elsif (/^@@ -(\d+)(?:,(\d+))?/ && $source) {
93 get_blame
($commits, $source, $1, $2, $id);
99 my ($commits, $file) = @_;
100 open my $f, '<', $file or die "read failure: $file: $!\n";
101 scan_patches
($commits, undef, $f);
108 qw(git rev-parse --revs-only --default HEAD --symbolic), @args
116 return @revs if scalar(@revs) != 1;
117 return "^$revs[0]", 'HEAD' unless $revs[0] =~ /^-/;
118 return $revs[0], 'HEAD';
122 my ($commits, $args) = @_;
123 my @revs = parse_rev_args
(@
$args);
124 open my $f, '-|', qw(git rev-list --reverse), @revs or die;
129 open my $g, '-|', qw(git show -C --oneline), $id or die;
130 scan_patches
($commits, $id, $g);
136 sub mailmap_contacts
{
139 my $pid = open2
my $reader, my $writer, qw(git check-mailmap --stdin);
140 for my $contact (keys(%$contacts)) {
141 print $writer "$contact\n";
142 my $canonical = <$reader>;
144 $mapped{$canonical} += $contacts->{$contact};
149 die "git-check-mailmap error: $?\n" if $?
;
154 die "No input revisions or patch files\n";
157 my (@files, @rev_args);
168 scan_patch_file
(\
%commits, $_);
171 scan_rev_args
(\
%commits, \
@rev_args)
173 import_commits
(\
%commits);
176 for my $commit (values %commits) {
177 for my $contact (keys %{$commit->{contacts
}}) {
178 $contacts->{$contact}++;
181 $contacts = mailmap_contacts
($contacts);
183 my $ncommits = scalar(keys %commits);
184 for my $contact (keys %$contacts) {
185 my $percent = $contacts->{$contact} * 100 / $ncommits;
186 next if $percent < $min_percent;