diff options
Diffstat (limited to 'scripts/get_maintainer.pl')
-rwxr-xr-x | scripts/get_maintainer.pl | 871 |
1 files changed, 596 insertions, 275 deletions
diff --git a/scripts/get_maintainer.pl b/scripts/get_maintainer.pl index 1ae8c50f1908..f51176039ff5 100755 --- a/scripts/get_maintainer.pl +++ b/scripts/get_maintainer.pl | |||
@@ -13,7 +13,7 @@ | |||
13 | use strict; | 13 | use strict; |
14 | 14 | ||
15 | my $P = $0; | 15 | my $P = $0; |
16 | my $V = '0.25'; | 16 | my $V = '0.26-beta3'; |
17 | 17 | ||
18 | use Getopt::Long qw(:config no_auto_abbrev); | 18 | use Getopt::Long qw(:config no_auto_abbrev); |
19 | 19 | ||
@@ -27,6 +27,7 @@ my $email_git_penguin_chiefs = 0; | |||
27 | my $email_git = 0; | 27 | my $email_git = 0; |
28 | my $email_git_all_signature_types = 0; | 28 | my $email_git_all_signature_types = 0; |
29 | my $email_git_blame = 0; | 29 | my $email_git_blame = 0; |
30 | my $email_git_blame_signatures = 1; | ||
30 | my $email_git_fallback = 1; | 31 | my $email_git_fallback = 1; |
31 | my $email_git_min_signatures = 1; | 32 | my $email_git_min_signatures = 1; |
32 | my $email_git_max_maintainers = 5; | 33 | my $email_git_max_maintainers = 5; |
@@ -51,9 +52,12 @@ my $pattern_depth = 0; | |||
51 | my $version = 0; | 52 | my $version = 0; |
52 | my $help = 0; | 53 | my $help = 0; |
53 | 54 | ||
55 | my $vcs_used = 0; | ||
56 | |||
54 | my $exit = 0; | 57 | my $exit = 0; |
55 | 58 | ||
56 | my %shortlog_buffer; | 59 | my %commit_author_hash; |
60 | my %commit_signer_hash; | ||
57 | 61 | ||
58 | my @penguin_chief = (); | 62 | my @penguin_chief = (); |
59 | push(@penguin_chief, "Linus Torvalds:torvalds\@linux-foundation.org"); | 63 | push(@penguin_chief, "Linus Torvalds:torvalds\@linux-foundation.org"); |
@@ -77,7 +81,6 @@ my @signature_tags = (); | |||
77 | push(@signature_tags, "Signed-off-by:"); | 81 | push(@signature_tags, "Signed-off-by:"); |
78 | push(@signature_tags, "Reviewed-by:"); | 82 | push(@signature_tags, "Reviewed-by:"); |
79 | push(@signature_tags, "Acked-by:"); | 83 | push(@signature_tags, "Acked-by:"); |
80 | my $signaturePattern = "\(" . join("|", @signature_tags) . "\)"; | ||
81 | 84 | ||
82 | # rfc822 email address - preloaded methods go here. | 85 | # rfc822 email address - preloaded methods go here. |
83 | my $rfc822_lwsp = "(?:(?:\\r\\n)?[ \\t])"; | 86 | my $rfc822_lwsp = "(?:(?:\\r\\n)?[ \\t])"; |
@@ -90,29 +93,62 @@ my %VCS_cmds; | |||
90 | my %VCS_cmds_git = ( | 93 | my %VCS_cmds_git = ( |
91 | "execute_cmd" => \&git_execute_cmd, | 94 | "execute_cmd" => \&git_execute_cmd, |
92 | "available" => '(which("git") ne "") && (-d ".git")', | 95 | "available" => '(which("git") ne "") && (-d ".git")', |
93 | "find_signers_cmd" => "git log --no-color --since=\$email_git_since -- \$file", | 96 | "find_signers_cmd" => |
94 | "find_commit_signers_cmd" => "git log --no-color -1 \$commit", | 97 | "git log --no-color --since=\$email_git_since " . |
95 | "find_commit_author_cmd" => "git log -1 --format=\"%an <%ae>\" \$commit", | 98 | '--format="GitCommit: %H%n' . |
99 | 'GitAuthor: %an <%ae>%n' . | ||
100 | 'GitDate: %aD%n' . | ||
101 | 'GitSubject: %s%n' . | ||
102 | '%b%n"' . | ||
103 | " -- \$file", | ||
104 | "find_commit_signers_cmd" => | ||
105 | "git log --no-color " . | ||
106 | '--format="GitCommit: %H%n' . | ||
107 | 'GitAuthor: %an <%ae>%n' . | ||
108 | 'GitDate: %aD%n' . | ||
109 | 'GitSubject: %s%n' . | ||
110 | '%b%n"' . | ||
111 | " -1 \$commit", | ||
112 | "find_commit_author_cmd" => | ||
113 | "git log --no-color " . | ||
114 | '--format="GitCommit: %H%n' . | ||
115 | 'GitAuthor: %an <%ae>%n' . | ||
116 | 'GitDate: %aD%n' . | ||
117 | 'GitSubject: %s%n"' . | ||
118 | " -1 \$commit", | ||
96 | "blame_range_cmd" => "git blame -l -L \$diff_start,+\$diff_length \$file", | 119 | "blame_range_cmd" => "git blame -l -L \$diff_start,+\$diff_length \$file", |
97 | "blame_file_cmd" => "git blame -l \$file", | 120 | "blame_file_cmd" => "git blame -l \$file", |
98 | "commit_pattern" => "^commit [0-9a-f]{40,40}", | 121 | "commit_pattern" => "^GitCommit: ([0-9a-f]{40,40})", |
99 | "blame_commit_pattern" => "^([0-9a-f]+) ", | 122 | "blame_commit_pattern" => "^([0-9a-f]+) ", |
100 | "shortlog_cmd" => "git log --no-color --oneline --since=\$email_git_since --author=\"\$email\" -- \$file" | 123 | "author_pattern" => "^GitAuthor: (.*)", |
124 | "subject_pattern" => "^GitSubject: (.*)", | ||
101 | ); | 125 | ); |
102 | 126 | ||
103 | my %VCS_cmds_hg = ( | 127 | my %VCS_cmds_hg = ( |
104 | "execute_cmd" => \&hg_execute_cmd, | 128 | "execute_cmd" => \&hg_execute_cmd, |
105 | "available" => '(which("hg") ne "") && (-d ".hg")', | 129 | "available" => '(which("hg") ne "") && (-d ".hg")', |
106 | "find_signers_cmd" => | 130 | "find_signers_cmd" => |
107 | "hg log --date=\$email_hg_since" . | 131 | "hg log --date=\$email_hg_since " . |
108 | " --template='commit {node}\\n{desc}\\n' -- \$file", | 132 | "--template='HgCommit: {node}\\n" . |
109 | "find_commit_signers_cmd" => "hg log --template='{desc}\\n' -r \$commit", | 133 | "HgAuthor: {author}\\n" . |
110 | "find_commit_author_cmd" => "hg log -l 1 --template='{author}\\n' -r \$commit", | 134 | "HgSubject: {desc}\\n'" . |
135 | " -- \$file", | ||
136 | "find_commit_signers_cmd" => | ||
137 | "hg log " . | ||
138 | "--template='HgSubject: {desc}\\n'" . | ||
139 | " -r \$commit", | ||
140 | "find_commit_author_cmd" => | ||
141 | "hg log " . | ||
142 | "--template='HgCommit: {node}\\n" . | ||
143 | "HgAuthor: {author}\\n" . | ||
144 | "HgSubject: {desc|firstline}\\n'" . | ||
145 | " -r \$commit", | ||
111 | "blame_range_cmd" => "", # not supported | 146 | "blame_range_cmd" => "", # not supported |
112 | "blame_file_cmd" => "hg blame -c \$file", | 147 | "blame_file_cmd" => "hg blame -n \$file", |
113 | "commit_pattern" => "^commit [0-9a-f]{40,40}", | 148 | "commit_pattern" => "^HgCommit: ([0-9a-f]{40,40})", |
114 | "blame_commit_pattern" => "^([0-9a-f]+):", | 149 | "blame_commit_pattern" => "^([ 0-9a-f]+):", |
115 | "shortlog_cmd" => "ht log --date=\$email_hg_since" | 150 | "author_pattern" => "^HgAuthor: (.*)", |
151 | "subject_pattern" => "^HgSubject: (.*)", | ||
116 | ); | 152 | ); |
117 | 153 | ||
118 | my $conf = which_conf(".get_maintainer.conf"); | 154 | my $conf = which_conf(".get_maintainer.conf"); |
@@ -146,6 +182,7 @@ if (!GetOptions( | |||
146 | 'git!' => \$email_git, | 182 | 'git!' => \$email_git, |
147 | 'git-all-signature-types!' => \$email_git_all_signature_types, | 183 | 'git-all-signature-types!' => \$email_git_all_signature_types, |
148 | 'git-blame!' => \$email_git_blame, | 184 | 'git-blame!' => \$email_git_blame, |
185 | 'git-blame-signatures!' => \$email_git_blame_signatures, | ||
149 | 'git-fallback!' => \$email_git_fallback, | 186 | 'git-fallback!' => \$email_git_fallback, |
150 | 'git-chief-penguins!' => \$email_git_penguin_chiefs, | 187 | 'git-chief-penguins!' => \$email_git_penguin_chiefs, |
151 | 'git-min-signatures=i' => \$email_git_min_signatures, | 188 | 'git-min-signatures=i' => \$email_git_min_signatures, |
@@ -193,13 +230,9 @@ if (-t STDIN && !@ARGV) { | |||
193 | die "$P: missing patchfile or -f file - use --help if necessary\n"; | 230 | die "$P: missing patchfile or -f file - use --help if necessary\n"; |
194 | } | 231 | } |
195 | 232 | ||
196 | if ($output_separator ne ", ") { | 233 | $output_multiline = 0 if ($output_separator ne ", "); |
197 | $output_multiline = 0; | 234 | $output_rolestats = 1 if ($interactive); |
198 | } | 235 | $output_roles = 1 if ($output_rolestats); |
199 | |||
200 | if ($output_rolestats) { | ||
201 | $output_roles = 1; | ||
202 | } | ||
203 | 236 | ||
204 | if ($sections) { | 237 | if ($sections) { |
205 | $email = 0; | 238 | $email = 0; |
@@ -227,12 +260,6 @@ if (!top_of_kernel_tree($lk_path)) { | |||
227 | . "a linux kernel source tree.\n"; | 260 | . "a linux kernel source tree.\n"; |
228 | } | 261 | } |
229 | 262 | ||
230 | if ($email_git_all_signature_types) { | ||
231 | $signaturePattern = "(.+?)[Bb][Yy]:"; | ||
232 | } | ||
233 | |||
234 | |||
235 | |||
236 | ## Read MAINTAINERS for type/value pairs | 263 | ## Read MAINTAINERS for type/value pairs |
237 | 264 | ||
238 | my @typevalue = (); | 265 | my @typevalue = (); |
@@ -371,168 +398,193 @@ foreach my $file (@ARGV) { | |||
371 | 398 | ||
372 | @file_emails = uniq(@file_emails); | 399 | @file_emails = uniq(@file_emails); |
373 | 400 | ||
401 | my %email_hash_name; | ||
402 | my %email_hash_address; | ||
374 | my @email_to = (); | 403 | my @email_to = (); |
404 | my %hash_list_to; | ||
375 | my @list_to = (); | 405 | my @list_to = (); |
376 | my @scm = (); | 406 | my @scm = (); |
377 | my @web = (); | 407 | my @web = (); |
378 | my @subsystem = (); | 408 | my @subsystem = (); |
379 | my @status = (); | 409 | my @status = (); |
410 | my $signature_pattern; | ||
380 | 411 | ||
381 | # Find responsible parties | 412 | my @to = get_maintainer(); |
382 | 413 | ||
383 | foreach my $file (@files) { | 414 | @to = merge_email(@to); |
384 | 415 | ||
385 | my %hash; | 416 | output(@to) if (@to); |
386 | my $exact_pattern_match = 0; | 417 | |
387 | my $tvi = find_first_section(); | 418 | if ($scm) { |
388 | while ($tvi < @typevalue) { | 419 | @scm = uniq(@scm); |
389 | my $start = find_starting_index($tvi); | 420 | output(@scm); |
390 | my $end = find_ending_index($tvi); | 421 | } |
391 | my $exclude = 0; | 422 | |
392 | my $i; | 423 | if ($status) { |
393 | 424 | @status = uniq(@status); | |
394 | #Do not match excluded file patterns | 425 | output(@status); |
395 | 426 | } | |
396 | for ($i = $start; $i < $end; $i++) { | 427 | |
397 | my $line = $typevalue[$i]; | 428 | if ($subsystem) { |
398 | if ($line =~ m/^(\C):\s*(.*)/) { | 429 | @subsystem = uniq(@subsystem); |
399 | my $type = $1; | 430 | output(@subsystem); |
400 | my $value = $2; | 431 | } |
401 | if ($type eq 'X') { | 432 | |
402 | if (file_match_pattern($file, $value)) { | 433 | if ($web) { |
403 | $exclude = 1; | 434 | @web = uniq(@web); |
404 | last; | 435 | output(@web); |
405 | } | 436 | } |
406 | } | 437 | |
407 | } | 438 | exit($exit); |
408 | } | 439 | |
440 | sub get_maintainer { | ||
441 | %email_hash_name = (); | ||
442 | %email_hash_address = (); | ||
443 | %commit_author_hash = (); | ||
444 | %commit_signer_hash = (); | ||
445 | @email_to = (); | ||
446 | %hash_list_to = (); | ||
447 | @list_to = (); | ||
448 | @scm = (); | ||
449 | @web = (); | ||
450 | @subsystem = (); | ||
451 | @status = (); | ||
452 | |||
453 | if ($email_git_all_signature_types) { | ||
454 | $signature_pattern = "(.+?)[Bb][Yy]:"; | ||
455 | } else { | ||
456 | $signature_pattern = "\(" . join("|", @signature_tags) . "\)"; | ||
457 | } | ||
458 | |||
459 | # Find responsible parties | ||
460 | |||
461 | foreach my $file (@files) { | ||
462 | |||
463 | my %hash; | ||
464 | my $exact_pattern_match = 0; | ||
465 | my $tvi = find_first_section(); | ||
466 | while ($tvi < @typevalue) { | ||
467 | my $start = find_starting_index($tvi); | ||
468 | my $end = find_ending_index($tvi); | ||
469 | my $exclude = 0; | ||
470 | my $i; | ||
471 | |||
472 | #Do not match excluded file patterns | ||
409 | 473 | ||
410 | if (!$exclude) { | ||
411 | for ($i = $start; $i < $end; $i++) { | 474 | for ($i = $start; $i < $end; $i++) { |
412 | my $line = $typevalue[$i]; | 475 | my $line = $typevalue[$i]; |
413 | if ($line =~ m/^(\C):\s*(.*)/) { | 476 | if ($line =~ m/^(\C):\s*(.*)/) { |
414 | my $type = $1; | 477 | my $type = $1; |
415 | my $value = $2; | 478 | my $value = $2; |
416 | if ($type eq 'F') { | 479 | if ($type eq 'X') { |
417 | if (file_match_pattern($file, $value)) { | 480 | if (file_match_pattern($file, $value)) { |
418 | my $value_pd = ($value =~ tr@/@@); | 481 | $exclude = 1; |
419 | my $file_pd = ($file =~ tr@/@@); | 482 | last; |
420 | $value_pd++ if (substr($value,-1,1) ne "/"); | 483 | } |
421 | $value_pd = -1 if ($value =~ /^\.\*/); | 484 | } |
422 | $exact_pattern_match = 1 if ($value_pd >= $file_pd); | 485 | } |
423 | if ($pattern_depth == 0 || | 486 | } |
424 | (($file_pd - $value_pd) < $pattern_depth)) { | 487 | |
425 | $hash{$tvi} = $value_pd; | 488 | if (!$exclude) { |
489 | for ($i = $start; $i < $end; $i++) { | ||
490 | my $line = $typevalue[$i]; | ||
491 | if ($line =~ m/^(\C):\s*(.*)/) { | ||
492 | my $type = $1; | ||
493 | my $value = $2; | ||
494 | if ($type eq 'F') { | ||
495 | if (file_match_pattern($file, $value)) { | ||
496 | my $value_pd = ($value =~ tr@/@@); | ||
497 | my $file_pd = ($file =~ tr@/@@); | ||
498 | $value_pd++ if (substr($value,-1,1) ne "/"); | ||
499 | $value_pd = -1 if ($value =~ /^\.\*/); | ||
500 | $exact_pattern_match = 1 if ($value_pd >= $file_pd); | ||
501 | if ($pattern_depth == 0 || | ||
502 | (($file_pd - $value_pd) < $pattern_depth)) { | ||
503 | $hash{$tvi} = $value_pd; | ||
504 | } | ||
426 | } | 505 | } |
427 | } | 506 | } |
428 | } | 507 | } |
429 | } | 508 | } |
430 | } | 509 | } |
510 | $tvi = $end + 1; | ||
431 | } | 511 | } |
432 | 512 | ||
433 | $tvi = $end + 1; | 513 | foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) { |
434 | } | 514 | add_categories($line); |
435 | 515 | if ($sections) { | |
436 | foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) { | 516 | my $i; |
437 | add_categories($line); | 517 | my $start = find_starting_index($line); |
438 | if ($sections) { | 518 | my $end = find_ending_index($line); |
439 | my $i; | 519 | for ($i = $start; $i < $end; $i++) { |
440 | my $start = find_starting_index($line); | 520 | my $line = $typevalue[$i]; |
441 | my $end = find_ending_index($line); | 521 | if ($line =~ /^[FX]:/) { ##Restore file patterns |
442 | for ($i = $start; $i < $end; $i++) { | 522 | $line =~ s/([^\\])\.([^\*])/$1\?$2/g; |
443 | my $line = $typevalue[$i]; | 523 | $line =~ s/([^\\])\.$/$1\?/g; ##Convert . back to ? |
444 | if ($line =~ /^[FX]:/) { ##Restore file patterns | 524 | $line =~ s/\\\./\./g; ##Convert \. to . |
445 | $line =~ s/([^\\])\.([^\*])/$1\?$2/g; | 525 | $line =~ s/\.\*/\*/g; ##Convert .* to * |
446 | $line =~ s/([^\\])\.$/$1\?/g; ##Convert . back to ? | 526 | } |
447 | $line =~ s/\\\./\./g; ##Convert \. to . | 527 | $line =~ s/^([A-Z]):/$1:\t/g; |
448 | $line =~ s/\.\*/\*/g; ##Convert .* to * | 528 | print("$line\n"); |
449 | } | 529 | } |
450 | $line =~ s/^([A-Z]):/$1:\t/g; | 530 | print("\n"); |
451 | print("$line\n"); | ||
452 | } | 531 | } |
453 | print("\n"); | ||
454 | } | 532 | } |
455 | } | ||
456 | |||
457 | if ($email && | ||
458 | ($email_git || ($email_git_fallback && !$exact_pattern_match))) { | ||
459 | vcs_file_signoffs($file); | ||
460 | } | ||
461 | if ($email && $email_git_blame) { | ||
462 | vcs_file_blame($file); | ||
463 | } | ||
464 | if ($email && $interactive){ | ||
465 | vcs_file_shortlogs($file); | ||
466 | 533 | ||
534 | if ($email && ($email_git || | ||
535 | ($email_git_fallback && !$exact_pattern_match))) { | ||
536 | vcs_file_signoffs($file); | ||
537 | } | ||
538 | if ($email && $email_git_blame) { | ||
539 | vcs_file_blame($file); | ||
540 | } | ||
467 | } | 541 | } |
468 | } | ||
469 | 542 | ||
470 | if ($keywords) { | 543 | if ($keywords) { |
471 | @keyword_tvi = sort_and_uniq(@keyword_tvi); | 544 | @keyword_tvi = sort_and_uniq(@keyword_tvi); |
472 | foreach my $line (@keyword_tvi) { | 545 | foreach my $line (@keyword_tvi) { |
473 | add_categories($line); | 546 | add_categories($line); |
547 | } | ||
474 | } | 548 | } |
475 | } | ||
476 | 549 | ||
477 | if ($email) { | 550 | if ($email) { |
478 | foreach my $chief (@penguin_chief) { | 551 | foreach my $chief (@penguin_chief) { |
479 | if ($chief =~ m/^(.*):(.*)/) { | 552 | if ($chief =~ m/^(.*):(.*)/) { |
480 | my $email_address; | 553 | my $email_address; |
481 | 554 | ||
482 | $email_address = format_email($1, $2, $email_usename); | 555 | $email_address = format_email($1, $2, $email_usename); |
483 | if ($email_git_penguin_chiefs) { | 556 | if ($email_git_penguin_chiefs) { |
484 | push(@email_to, [$email_address, 'chief penguin']); | 557 | push(@email_to, [$email_address, 'chief penguin']); |
485 | } else { | 558 | } else { |
486 | @email_to = grep($_->[0] !~ /${email_address}/, @email_to); | 559 | @email_to = grep($_->[0] !~ /${email_address}/, @email_to); |
560 | } | ||
487 | } | 561 | } |
488 | } | 562 | } |
489 | } | ||
490 | 563 | ||
491 | foreach my $email (@file_emails) { | 564 | foreach my $email (@file_emails) { |
492 | my ($name, $address) = parse_email($email); | 565 | my ($name, $address) = parse_email($email); |
493 | 566 | ||
494 | my $tmp_email = format_email($name, $address, $email_usename); | 567 | my $tmp_email = format_email($name, $address, $email_usename); |
495 | push_email_address($tmp_email, ''); | 568 | push_email_address($tmp_email, ''); |
496 | add_role($tmp_email, 'in file'); | 569 | add_role($tmp_email, 'in file'); |
570 | } | ||
497 | } | 571 | } |
498 | } | ||
499 | |||
500 | 572 | ||
501 | if ($email || $email_list) { | ||
502 | my @to = (); | 573 | my @to = (); |
503 | if ($email) { | 574 | if ($email || $email_list) { |
504 | if ($interactive) { | 575 | if ($email) { |
505 | @email_to = @{vcs_interactive_menu(\@email_to)}; | 576 | @to = (@to, @email_to); |
577 | } | ||
578 | if ($email_list) { | ||
579 | @to = (@to, @list_to); | ||
506 | } | 580 | } |
507 | @to = (@to, @email_to); | ||
508 | } | ||
509 | if ($email_list) { | ||
510 | @to = (@to, @list_to); | ||
511 | } | 581 | } |
512 | output(merge_email(@to)); | ||
513 | } | ||
514 | 582 | ||
515 | if ($scm) { | 583 | @to = interactive_get_maintainer(\@to) if ($interactive); |
516 | @scm = uniq(@scm); | ||
517 | output(@scm); | ||
518 | } | ||
519 | if ($status) { | ||
520 | @status = uniq(@status); | ||
521 | output(@status); | ||
522 | } | ||
523 | 584 | ||
524 | if ($subsystem) { | 585 | return @to; |
525 | @subsystem = uniq(@subsystem); | ||
526 | output(@subsystem); | ||
527 | } | 586 | } |
528 | 587 | ||
529 | if ($web) { | ||
530 | @web = uniq(@web); | ||
531 | output(@web); | ||
532 | } | ||
533 | |||
534 | exit($exit); | ||
535 | |||
536 | sub file_match_pattern { | 588 | sub file_match_pattern { |
537 | my ($file, $pattern) = @_; | 589 | my ($file, $pattern) = @_; |
538 | if (substr($pattern, -1) eq "/") { | 590 | if (substr($pattern, -1) eq "/") { |
@@ -561,7 +613,7 @@ MAINTAINER field selection options: | |||
561 | --email => print email address(es) if any | 613 | --email => print email address(es) if any |
562 | --git => include recent git \*-by: signers | 614 | --git => include recent git \*-by: signers |
563 | --git-all-signature-types => include signers regardless of signature type | 615 | --git-all-signature-types => include signers regardless of signature type |
564 | or use only ${signaturePattern} signers (default: $email_git_all_signature_types) | 616 | or use only ${signature_pattern} signers (default: $email_git_all_signature_types) |
565 | --git-fallback => use git when no exact MAINTAINERS pattern (default: $email_git_fallback) | 617 | --git-fallback => use git when no exact MAINTAINERS pattern (default: $email_git_fallback) |
566 | --git-chief-penguins => include ${penguin_chiefs} | 618 | --git-chief-penguins => include ${penguin_chiefs} |
567 | --git-min-signatures => number of signatures required (default: $email_git_min_signatures) | 619 | --git-min-signatures => number of signatures required (default: $email_git_min_signatures) |
@@ -847,11 +899,19 @@ sub add_categories { | |||
847 | } | 899 | } |
848 | if ($list_additional =~ m/subscribers-only/) { | 900 | if ($list_additional =~ m/subscribers-only/) { |
849 | if ($email_subscriber_list) { | 901 | if ($email_subscriber_list) { |
850 | push(@list_to, [$list_address, "subscriber list${list_role}"]); | 902 | if (!$hash_list_to{$list_address}) { |
903 | $hash_list_to{$list_address} = 1; | ||
904 | push(@list_to, [$list_address, | ||
905 | "subscriber list${list_role}"]); | ||
906 | } | ||
851 | } | 907 | } |
852 | } else { | 908 | } else { |
853 | if ($email_list) { | 909 | if ($email_list) { |
854 | push(@list_to, [$list_address, "open list${list_role}"]); | 910 | if (!$hash_list_to{$list_address}) { |
911 | $hash_list_to{$list_address} = 1; | ||
912 | push(@list_to, [$list_address, | ||
913 | "open list${list_role}"]); | ||
914 | } | ||
855 | } | 915 | } |
856 | } | 916 | } |
857 | } elsif ($ptype eq "M") { | 917 | } elsif ($ptype eq "M") { |
@@ -882,9 +942,6 @@ sub add_categories { | |||
882 | } | 942 | } |
883 | } | 943 | } |
884 | 944 | ||
885 | my %email_hash_name; | ||
886 | my %email_hash_address; | ||
887 | |||
888 | sub email_inuse { | 945 | sub email_inuse { |
889 | my ($name, $address) = @_; | 946 | my ($name, $address) = @_; |
890 | 947 | ||
@@ -1037,10 +1094,31 @@ sub hg_execute_cmd { | |||
1037 | return @lines; | 1094 | return @lines; |
1038 | } | 1095 | } |
1039 | 1096 | ||
1097 | sub extract_formatted_signatures { | ||
1098 | my (@signature_lines) = @_; | ||
1099 | |||
1100 | my @type = @signature_lines; | ||
1101 | |||
1102 | s/\s*(.*):.*/$1/ for (@type); | ||
1103 | |||
1104 | # cut -f2- -d":" | ||
1105 | s/\s*.*:\s*(.+)\s*/$1/ for (@signature_lines); | ||
1106 | |||
1107 | ## Reformat email addresses (with names) to avoid badly written signatures | ||
1108 | |||
1109 | foreach my $signer (@signature_lines) { | ||
1110 | my ($name, $address) = parse_email($signer); | ||
1111 | $signer = format_email($name, $address, 1); | ||
1112 | } | ||
1113 | |||
1114 | return (\@type, \@signature_lines); | ||
1115 | } | ||
1116 | |||
1040 | sub vcs_find_signers { | 1117 | sub vcs_find_signers { |
1041 | my ($cmd) = @_; | 1118 | my ($cmd) = @_; |
1042 | my @lines = (); | ||
1043 | my $commits; | 1119 | my $commits; |
1120 | my @lines = (); | ||
1121 | my @signatures = (); | ||
1044 | 1122 | ||
1045 | @lines = &{$VCS_cmds{"execute_cmd"}}($cmd); | 1123 | @lines = &{$VCS_cmds{"execute_cmd"}}($cmd); |
1046 | 1124 | ||
@@ -1048,24 +1126,20 @@ sub vcs_find_signers { | |||
1048 | 1126 | ||
1049 | $commits = grep(/$pattern/, @lines); # of commits | 1127 | $commits = grep(/$pattern/, @lines); # of commits |
1050 | 1128 | ||
1051 | @lines = grep(/^[ \t]*${signaturePattern}.*\@.*$/, @lines); | 1129 | @signatures = grep(/^[ \t]*${signature_pattern}.*\@.*$/, @lines); |
1052 | if (!$email_git_penguin_chiefs) { | ||
1053 | @lines = grep(!/${penguin_chiefs}/i, @lines); | ||
1054 | } | ||
1055 | 1130 | ||
1056 | return (0, @lines) if !@lines; | 1131 | return (0, @signatures) if !@signatures; |
1057 | 1132 | ||
1058 | # cut -f2- -d":" | 1133 | save_commits_by_author(@lines) if ($interactive); |
1059 | s/.*:\s*(.+)\s*/$1/ for (@lines); | 1134 | save_commits_by_signer(@lines) if ($interactive); |
1060 | 1135 | ||
1061 | ## Reformat email addresses (with names) to avoid badly written signatures | 1136 | if (!$email_git_penguin_chiefs) { |
1062 | 1137 | @signatures = grep(!/${penguin_chiefs}/i, @signatures); | |
1063 | foreach my $line (@lines) { | ||
1064 | my ($name, $address) = parse_email($line); | ||
1065 | $line = format_email($name, $address, 1); | ||
1066 | } | 1138 | } |
1067 | 1139 | ||
1068 | return ($commits, @lines); | 1140 | my ($types_ref, $signers_ref) = extract_formatted_signatures(@signatures); |
1141 | |||
1142 | return ($commits, @$signers_ref); | ||
1069 | } | 1143 | } |
1070 | 1144 | ||
1071 | sub vcs_find_author { | 1145 | sub vcs_find_author { |
@@ -1080,14 +1154,20 @@ sub vcs_find_author { | |||
1080 | 1154 | ||
1081 | return @lines if !@lines; | 1155 | return @lines if !@lines; |
1082 | 1156 | ||
1083 | ## Reformat email addresses (with names) to avoid badly written signatures | 1157 | my @authors = (); |
1084 | |||
1085 | foreach my $line (@lines) { | 1158 | foreach my $line (@lines) { |
1086 | my ($name, $address) = parse_email($line); | 1159 | if ($line =~ m/$VCS_cmds{"author_pattern"}/) { |
1087 | $line = format_email($name, $address, 1); | 1160 | my $author = $1; |
1161 | my ($name, $address) = parse_email($author); | ||
1162 | $author = format_email($name, $address, 1); | ||
1163 | push(@authors, $author); | ||
1164 | } | ||
1088 | } | 1165 | } |
1089 | 1166 | ||
1090 | return @lines; | 1167 | save_commits_by_author(@lines) if ($interactive); |
1168 | save_commits_by_signer(@lines) if ($interactive); | ||
1169 | |||
1170 | return @authors; | ||
1091 | } | 1171 | } |
1092 | 1172 | ||
1093 | sub vcs_save_commits { | 1173 | sub vcs_save_commits { |
@@ -1159,7 +1239,7 @@ sub vcs_exists { | |||
1159 | %VCS_cmds = %VCS_cmds_git; | 1239 | %VCS_cmds = %VCS_cmds_git; |
1160 | return 1 if eval $VCS_cmds{"available"}; | 1240 | return 1 if eval $VCS_cmds{"available"}; |
1161 | %VCS_cmds = %VCS_cmds_hg; | 1241 | %VCS_cmds = %VCS_cmds_hg; |
1162 | return 1 if eval $VCS_cmds{"available"}; | 1242 | return 2 if eval $VCS_cmds{"available"}; |
1163 | %VCS_cmds = (); | 1243 | %VCS_cmds = (); |
1164 | if (!$printed_novcs) { | 1244 | if (!$printed_novcs) { |
1165 | warn("$P: No supported VCS found. Add --nogit to options?\n"); | 1245 | warn("$P: No supported VCS found. Add --nogit to options?\n"); |
@@ -1171,125 +1251,309 @@ sub vcs_exists { | |||
1171 | return 0; | 1251 | return 0; |
1172 | } | 1252 | } |
1173 | 1253 | ||
1174 | sub vcs_interactive_menu { | 1254 | sub vcs_is_git { |
1175 | my $list_ref = shift; | 1255 | return $vcs_used == 1; |
1256 | } | ||
1257 | |||
1258 | sub vcs_is_hg { | ||
1259 | return $vcs_used == 2; | ||
1260 | } | ||
1261 | |||
1262 | sub interactive_get_maintainer { | ||
1263 | my ($list_ref) = @_; | ||
1176 | my @list = @$list_ref; | 1264 | my @list = @$list_ref; |
1177 | 1265 | ||
1178 | return if (!vcs_exists()); | 1266 | vcs_exists(); |
1179 | 1267 | ||
1180 | my %selected; | 1268 | my %selected; |
1181 | my %shortlog; | 1269 | my %authored; |
1182 | my $input; | 1270 | my %signed; |
1183 | my $count = 0; | 1271 | my $count = 0; |
1184 | 1272 | ||
1185 | #select maintainers by default | 1273 | #select maintainers by default |
1186 | foreach my $entry (@list){ | 1274 | foreach my $entry (@list){ |
1187 | my $role = $entry->[1]; | 1275 | my $role = $entry->[1]; |
1188 | $selected{$count} = ($role =~ /maintainer:|supporter:/); | 1276 | $selected{$count} = ($role =~ /^(maintainer|supporter|open list)/); |
1189 | $count++; | 1277 | $authored{$count} = 0; |
1278 | $signed{$count} = 0; | ||
1279 | $count++; | ||
1190 | } | 1280 | } |
1191 | 1281 | ||
1192 | #menu loop | 1282 | #menu loop |
1193 | do { | 1283 | my $done = 0; |
1194 | my $count = 0; | 1284 | my $print_options = 0; |
1195 | foreach my $entry (@list){ | 1285 | my $redraw = 1; |
1196 | my $email = $entry->[0]; | 1286 | while (!$done) { |
1197 | my $role = $entry->[1]; | 1287 | $count = 0; |
1198 | if ($selected{$count}){ | 1288 | if ($redraw) { |
1199 | print STDERR "* "; | 1289 | printf STDERR "\n%1s %2s %-65sauth sign\n", |
1200 | } else { | 1290 | "*", "#", "email/list and role:stats"; |
1201 | print STDERR " "; | 1291 | foreach my $entry (@list) { |
1202 | } | 1292 | my $email = $entry->[0]; |
1203 | print STDERR "$count: $email,\t\t $role"; | 1293 | my $role = $entry->[1]; |
1204 | print STDERR "\n"; | 1294 | my $sel = ""; |
1205 | if ($shortlog{$count}){ | 1295 | $sel = "*" if ($selected{$count}); |
1206 | my $entries_ref = vcs_get_shortlog($email); | 1296 | my $commit_author = $commit_author_hash{$email}; |
1207 | foreach my $entry_ref (@{$entries_ref}){ | 1297 | my $commit_signer = $commit_signer_hash{$email}; |
1208 | my $filename = @{$entry_ref}[0]; | 1298 | my $authored = 0; |
1209 | my @shortlog = @{@{$entry_ref}[1]}; | 1299 | my $signed = 0; |
1210 | print STDERR "\tshortlog for $filename (authored commits: " . @shortlog . ").\n"; | 1300 | $authored++ for (@{$commit_author}); |
1211 | foreach my $commit (@shortlog){ | 1301 | $signed++ for (@{$commit_signer}); |
1212 | print STDERR "\t $commit\n"; | 1302 | printf STDERR "%1s %2d %-65s", $sel, $count + 1, $email; |
1303 | printf STDERR "%4d %4d", $authored, $signed | ||
1304 | if ($authored > 0 || $signed > 0); | ||
1305 | printf STDERR "\n %s\n", $role; | ||
1306 | if ($authored{$count}) { | ||
1307 | my $commit_author = $commit_author_hash{$email}; | ||
1308 | foreach my $ref (@{$commit_author}) { | ||
1309 | print STDERR " Author: @{$ref}[1]\n"; | ||
1213 | } | 1310 | } |
1214 | print STDERR "\n"; | ||
1215 | } | 1311 | } |
1312 | if ($signed{$count}) { | ||
1313 | my $commit_signer = $commit_signer_hash{$email}; | ||
1314 | foreach my $ref (@{$commit_signer}) { | ||
1315 | print STDERR " @{$ref}[2]: @{$ref}[1]\n"; | ||
1316 | } | ||
1317 | } | ||
1318 | |||
1319 | $count++; | ||
1216 | } | 1320 | } |
1217 | $count++; | ||
1218 | } | 1321 | } |
1219 | print STDERR "\n"; | 1322 | my $date_ref = \$email_git_since; |
1220 | print STDERR "Choose whom to cc by entering a commaseperated list of numbers and hitting enter.\n"; | 1323 | $date_ref = \$email_hg_since if (vcs_is_hg()); |
1221 | print STDERR "To show a short list of commits, precede the number by a '?',\n"; | 1324 | if ($print_options) { |
1222 | print STDERR "A blank line indicates that you are satisfied with your choice.\n"; | 1325 | $print_options = 0; |
1223 | $input = <STDIN>; | 1326 | if (vcs_exists()) { |
1327 | print STDERR | ||
1328 | "\nVersion Control options:\n" . | ||
1329 | "g use git history [$email_git]\n" . | ||
1330 | "gf use git-fallback [$email_git_fallback]\n" . | ||
1331 | "b use git blame [$email_git_blame]\n" . | ||
1332 | "bs use blame signatures [$email_git_blame_signatures]\n" . | ||
1333 | "c# minimum commits [$email_git_min_signatures]\n" . | ||
1334 | "%# min percent [$email_git_min_percent]\n" . | ||
1335 | "d# history to use [$$date_ref]\n" . | ||
1336 | "x# max maintainers [$email_git_max_maintainers]\n" . | ||
1337 | "t all signature types [$email_git_all_signature_types]\n"; | ||
1338 | } | ||
1339 | print STDERR "\nAdditional options:\n" . | ||
1340 | "0 toggle all\n" . | ||
1341 | "f emails in file [$file_emails]\n" . | ||
1342 | "k keywords in file [$keywords]\n" . | ||
1343 | "r remove duplicates [$email_remove_duplicates]\n" . | ||
1344 | "p# pattern match depth [$pattern_depth]\n"; | ||
1345 | } | ||
1346 | print STDERR | ||
1347 | "\n#(toggle), A#(author), S#(signed) *(all), ^(none), O(options), Y(approve): "; | ||
1348 | |||
1349 | my $input = <STDIN>; | ||
1224 | chomp($input); | 1350 | chomp($input); |
1225 | 1351 | ||
1226 | my @wish = split(/[, ]+/,$input); | 1352 | $redraw = 1; |
1227 | foreach my $nr (@wish){ | 1353 | my $rerun = 0; |
1228 | my $logtoggle = 0; | 1354 | my @wish = split(/[, ]+/, $input); |
1229 | if ($nr =~ /\?/){ | 1355 | foreach my $nr (@wish) { |
1230 | $nr =~ s/\?//; | 1356 | $nr = lc($nr); |
1231 | $logtoggle = 1; | 1357 | my $sel = substr($nr, 0, 1); |
1358 | my $str = substr($nr, 1); | ||
1359 | my $val = 0; | ||
1360 | $val = $1 if $str =~ /^(\d+)$/; | ||
1361 | |||
1362 | if ($sel eq "y") { | ||
1363 | $interactive = 0; | ||
1364 | $done = 1; | ||
1365 | $output_rolestats = 0; | ||
1366 | $output_roles = 0; | ||
1367 | last; | ||
1368 | } elsif ($nr =~ /^\d+$/ && $nr > 0 && $nr <= $count) { | ||
1369 | $selected{$nr - 1} = !$selected{$nr - 1}; | ||
1370 | } elsif ($sel eq "*" || $sel eq '^') { | ||
1371 | my $toggle = 0; | ||
1372 | $toggle = 1 if ($sel eq '*'); | ||
1373 | for (my $i = 0; $i < $count; $i++) { | ||
1374 | $selected{$i} = $toggle; | ||
1232 | } | 1375 | } |
1233 | 1376 | } elsif ($sel eq "0") { | |
1234 | #skip out of bounds numbers | 1377 | for (my $i = 0; $i < $count; $i++) { |
1235 | next unless ($nr <= $count && $nr >= 0); | 1378 | $selected{$i} = !$selected{$i}; |
1236 | 1379 | } | |
1237 | if ($logtoggle){ | 1380 | } elsif ($sel eq "a") { |
1238 | $shortlog{$nr} = !$shortlog{$nr}; | 1381 | if ($val > 0 && $val <= $count) { |
1382 | $authored{$val - 1} = !$authored{$val - 1}; | ||
1383 | } elsif ($str eq '*' || $str eq '^') { | ||
1384 | my $toggle = 0; | ||
1385 | $toggle = 1 if ($str eq '*'); | ||
1386 | for (my $i = 0; $i < $count; $i++) { | ||
1387 | $authored{$i} = $toggle; | ||
1388 | } | ||
1389 | } | ||
1390 | } elsif ($sel eq "s") { | ||
1391 | if ($val > 0 && $val <= $count) { | ||
1392 | $signed{$val - 1} = !$signed{$val - 1}; | ||
1393 | } elsif ($str eq '*' || $str eq '^') { | ||
1394 | my $toggle = 0; | ||
1395 | $toggle = 1 if ($str eq '*'); | ||
1396 | for (my $i = 0; $i < $count; $i++) { | ||
1397 | $signed{$i} = $toggle; | ||
1398 | } | ||
1399 | } | ||
1400 | } elsif ($sel eq "o") { | ||
1401 | $print_options = 1; | ||
1402 | $redraw = 1; | ||
1403 | } elsif ($sel eq "g") { | ||
1404 | if ($str eq "f") { | ||
1405 | bool_invert(\$email_git_fallback); | ||
1239 | } else { | 1406 | } else { |
1240 | $selected{$nr} = !$selected{$nr}; | 1407 | bool_invert(\$email_git); |
1241 | 1408 | } | |
1242 | #switch shortlog on if an entry get's selected | 1409 | $rerun = 1; |
1243 | if ($selected{$nr}){ | 1410 | } elsif ($sel eq "b") { |
1244 | $shortlog{$nr}=1; | 1411 | if ($str eq "s") { |
1245 | } | 1412 | bool_invert(\$email_git_blame_signatures); |
1413 | } else { | ||
1414 | bool_invert(\$email_git_blame); | ||
1415 | } | ||
1416 | $rerun = 1; | ||
1417 | } elsif ($sel eq "c") { | ||
1418 | if ($val > 0) { | ||
1419 | $email_git_min_signatures = $val; | ||
1420 | $rerun = 1; | ||
1421 | } | ||
1422 | } elsif ($sel eq "x") { | ||
1423 | if ($val > 0) { | ||
1424 | $email_git_max_maintainers = $val; | ||
1425 | $rerun = 1; | ||
1426 | } | ||
1427 | } elsif ($sel eq "%") { | ||
1428 | if ($str ne "" && $val >= 0) { | ||
1429 | $email_git_min_percent = $val; | ||
1430 | $rerun = 1; | ||
1246 | } | 1431 | } |
1247 | }; | 1432 | } elsif ($sel eq "d") { |
1248 | } while(length($input) > 0); | 1433 | if (vcs_is_git()) { |
1434 | $email_git_since = $str; | ||
1435 | } elsif (vcs_is_hg()) { | ||
1436 | $email_hg_since = $str; | ||
1437 | } | ||
1438 | $rerun = 1; | ||
1439 | } elsif ($sel eq "t") { | ||
1440 | bool_invert(\$email_git_all_signature_types); | ||
1441 | $rerun = 1; | ||
1442 | } elsif ($sel eq "f") { | ||
1443 | bool_invert(\$file_emails); | ||
1444 | $rerun = 1; | ||
1445 | } elsif ($sel eq "r") { | ||
1446 | bool_invert(\$email_remove_duplicates); | ||
1447 | $rerun = 1; | ||
1448 | } elsif ($sel eq "k") { | ||
1449 | bool_invert(\$keywords); | ||
1450 | $rerun = 1; | ||
1451 | } elsif ($sel eq "p") { | ||
1452 | if ($str ne "" && $val >= 0) { | ||
1453 | $pattern_depth = $val; | ||
1454 | $rerun = 1; | ||
1455 | } | ||
1456 | } else { | ||
1457 | print STDERR "invalid option: '$nr'\n"; | ||
1458 | $redraw = 0; | ||
1459 | } | ||
1460 | } | ||
1461 | if ($rerun) { | ||
1462 | print STDERR "git-blame can be very slow, please have patience..." | ||
1463 | if ($email_git_blame); | ||
1464 | goto &get_maintainer; | ||
1465 | } | ||
1466 | } | ||
1249 | 1467 | ||
1250 | #drop not selected entries | 1468 | #drop not selected entries |
1251 | $count = 0; | 1469 | $count = 0; |
1252 | my @new_emailto; | 1470 | my @new_emailto = (); |
1253 | foreach my $entry (@list){ | 1471 | foreach my $entry (@list) { |
1254 | if ($selected{$count}){ | 1472 | if ($selected{$count}) { |
1255 | push(@new_emailto,$list[$count]); | 1473 | push(@new_emailto, $list[$count]); |
1256 | print STDERR "$count: "; | ||
1257 | print STDERR $email_to[$count]->[0]; | ||
1258 | print STDERR ",\t\t "; | ||
1259 | print STDERR $email_to[$count]->[1]; | ||
1260 | print STDERR "\n"; | ||
1261 | } | 1474 | } |
1262 | $count++; | 1475 | $count++; |
1263 | } | 1476 | } |
1264 | return \@new_emailto; | 1477 | return @new_emailto; |
1265 | } | 1478 | } |
1266 | 1479 | ||
1267 | sub vcs_get_shortlog { | 1480 | sub bool_invert { |
1268 | my $arg = shift; | 1481 | my ($bool_ref) = @_; |
1269 | my ($name, $address) = parse_email($arg); | 1482 | |
1270 | return $shortlog_buffer{$address}; | 1483 | if ($$bool_ref) { |
1484 | $$bool_ref = 0; | ||
1485 | } else { | ||
1486 | $$bool_ref = 1; | ||
1487 | } | ||
1271 | } | 1488 | } |
1272 | 1489 | ||
1273 | sub vcs_file_shortlogs { | 1490 | sub save_commits_by_author { |
1274 | my ($file) = @_; | 1491 | my (@lines) = @_; |
1275 | print STDERR "shortlog processing $file:"; | 1492 | |
1276 | foreach my $entry (@email_to){ | 1493 | my @authors = (); |
1277 | my ($name, $address) = parse_email($entry->[0]); | 1494 | my @commits = (); |
1278 | print STDERR "."; | 1495 | my @subjects = (); |
1279 | my $commits_ref = vcs_email_shortlog($address, $file); | 1496 | |
1280 | push(@{$shortlog_buffer{$address}}, [ $file, $commits_ref ]); | 1497 | foreach my $line (@lines) { |
1498 | if ($line =~ m/$VCS_cmds{"author_pattern"}/) { | ||
1499 | my $author = $1; | ||
1500 | my ($name, $address) = parse_email($author); | ||
1501 | $author = format_email($name, $address, 1); | ||
1502 | push(@authors, $author); | ||
1503 | } | ||
1504 | push(@commits, $1) if ($line =~ m/$VCS_cmds{"commit_pattern"}/); | ||
1505 | push(@subjects, $1) if ($line =~ m/$VCS_cmds{"subject_pattern"}/); | ||
1506 | } | ||
1507 | |||
1508 | for (my $i = 0; $i < @authors; $i++) { | ||
1509 | my $exists = 0; | ||
1510 | foreach my $ref(@{$commit_author_hash{$authors[$i]}}) { | ||
1511 | if (@{$ref}[0] eq $commits[$i] && | ||
1512 | @{$ref}[1] eq $subjects[$i]) { | ||
1513 | $exists = 1; | ||
1514 | last; | ||
1515 | } | ||
1516 | } | ||
1517 | if (!$exists) { | ||
1518 | push(@{$commit_author_hash{$authors[$i]}}, | ||
1519 | [ ($commits[$i], $subjects[$i]) ]); | ||
1520 | } | ||
1281 | } | 1521 | } |
1282 | print STDERR "\n"; | ||
1283 | } | 1522 | } |
1284 | 1523 | ||
1285 | sub vcs_email_shortlog { | 1524 | sub save_commits_by_signer { |
1286 | my $email = shift; | 1525 | my (@lines) = @_; |
1287 | my ($file) = @_; | 1526 | |
1527 | my $commit = ""; | ||
1528 | my $subject = ""; | ||
1288 | 1529 | ||
1289 | my $cmd = $VCS_cmds{"shortlog_cmd"}; | 1530 | foreach my $line (@lines) { |
1290 | $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables | 1531 | $commit = $1 if ($line =~ m/$VCS_cmds{"commit_pattern"}/); |
1291 | my @lines = &{$VCS_cmds{"execute_cmd"}}($cmd); | 1532 | $subject = $1 if ($line =~ m/$VCS_cmds{"subject_pattern"}/); |
1292 | return \@lines; | 1533 | if ($line =~ /^[ \t]*${signature_pattern}.*\@.*$/) { |
1534 | my @signatures = ($line); | ||
1535 | my ($types_ref, $signers_ref) = extract_formatted_signatures(@signatures); | ||
1536 | my @types = @$types_ref; | ||
1537 | my @signers = @$signers_ref; | ||
1538 | |||
1539 | my $type = $types[0]; | ||
1540 | my $signer = $signers[0]; | ||
1541 | |||
1542 | my $exists = 0; | ||
1543 | foreach my $ref(@{$commit_signer_hash{$signer}}) { | ||
1544 | if (@{$ref}[0] eq $commit && | ||
1545 | @{$ref}[1] eq $subject && | ||
1546 | @{$ref}[2] eq $type) { | ||
1547 | $exists = 1; | ||
1548 | last; | ||
1549 | } | ||
1550 | } | ||
1551 | if (!$exists) { | ||
1552 | push(@{$commit_signer_hash{$signer}}, | ||
1553 | [ ($commit, $subject, $type) ]); | ||
1554 | } | ||
1555 | } | ||
1556 | } | ||
1293 | } | 1557 | } |
1294 | 1558 | ||
1295 | sub vcs_assign { | 1559 | sub vcs_assign { |
@@ -1342,7 +1606,8 @@ sub vcs_file_signoffs { | |||
1342 | my @signers = (); | 1606 | my @signers = (); |
1343 | my $commits; | 1607 | my $commits; |
1344 | 1608 | ||
1345 | return if (!vcs_exists()); | 1609 | $vcs_used = vcs_exists(); |
1610 | return if (!$vcs_used); | ||
1346 | 1611 | ||
1347 | my $cmd = $VCS_cmds{"find_signers_cmd"}; | 1612 | my $cmd = $VCS_cmds{"find_signers_cmd"}; |
1348 | $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd | 1613 | $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd |
@@ -1360,37 +1625,93 @@ sub vcs_file_blame { | |||
1360 | my $total_commits; | 1625 | my $total_commits; |
1361 | my $total_lines; | 1626 | my $total_lines; |
1362 | 1627 | ||
1363 | return if (!vcs_exists()); | 1628 | $vcs_used = vcs_exists(); |
1629 | return if (!$vcs_used); | ||
1364 | 1630 | ||
1365 | @all_commits = vcs_blame($file); | 1631 | @all_commits = vcs_blame($file); |
1366 | @commits = uniq(@all_commits); | 1632 | @commits = uniq(@all_commits); |
1367 | $total_commits = @commits; | 1633 | $total_commits = @commits; |
1368 | $total_lines = @all_commits; | 1634 | $total_lines = @all_commits; |
1369 | 1635 | ||
1370 | foreach my $commit (@commits) { | 1636 | if ($email_git_blame_signatures) { |
1371 | my $commit_count; | 1637 | if (vcs_is_hg()) { |
1372 | my @commit_signers = (); | 1638 | my $commit_count; |
1639 | my @commit_signers = (); | ||
1640 | my $commit = join(" -r ", @commits); | ||
1641 | my $cmd; | ||
1373 | 1642 | ||
1374 | my $cmd = $VCS_cmds{"find_commit_signers_cmd"}; | 1643 | $cmd = $VCS_cmds{"find_commit_signers_cmd"}; |
1375 | $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd | 1644 | $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd |
1376 | 1645 | ||
1377 | ($commit_count, @commit_signers) = vcs_find_signers($cmd); | 1646 | ($commit_count, @commit_signers) = vcs_find_signers($cmd); |
1378 | 1647 | ||
1379 | push(@signers, @commit_signers); | 1648 | push(@signers, @commit_signers); |
1649 | } else { | ||
1650 | foreach my $commit (@commits) { | ||
1651 | my $commit_count; | ||
1652 | my @commit_signers = (); | ||
1653 | my $cmd; | ||
1654 | |||
1655 | $cmd = $VCS_cmds{"find_commit_signers_cmd"}; | ||
1656 | $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd | ||
1657 | |||
1658 | ($commit_count, @commit_signers) = vcs_find_signers($cmd); | ||
1659 | |||
1660 | push(@signers, @commit_signers); | ||
1661 | } | ||
1662 | } | ||
1380 | } | 1663 | } |
1381 | 1664 | ||
1382 | if ($from_filename) { | 1665 | if ($from_filename) { |
1383 | if ($output_rolestats) { | 1666 | if ($output_rolestats) { |
1384 | my @blame_signers; | 1667 | my @blame_signers; |
1385 | foreach my $commit (@commits) { | 1668 | if (vcs_is_hg()) {{ # Double brace for last exit |
1386 | my $i; | 1669 | my $commit_count; |
1387 | my $cmd = $VCS_cmds{"find_commit_author_cmd"}; | 1670 | my @commit_signers = (); |
1388 | $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd | 1671 | @commits = uniq(@commits); |
1389 | my @author = vcs_find_author($cmd); | 1672 | @commits = sort(@commits); |
1390 | next if !@author; | 1673 | my $commit = join(" -r ", @commits); |
1391 | my $count = grep(/$commit/, @all_commits); | 1674 | my $cmd; |
1392 | for ($i = 0; $i < $count ; $i++) { | 1675 | |
1393 | push(@blame_signers, $author[0]); | 1676 | $cmd = $VCS_cmds{"find_commit_author_cmd"}; |
1677 | $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd | ||
1678 | |||
1679 | my @lines = (); | ||
1680 | |||
1681 | @lines = &{$VCS_cmds{"execute_cmd"}}($cmd); | ||
1682 | |||
1683 | if (!$email_git_penguin_chiefs) { | ||
1684 | @lines = grep(!/${penguin_chiefs}/i, @lines); | ||
1685 | } | ||
1686 | |||
1687 | last if !@lines; | ||
1688 | |||
1689 | my @authors = (); | ||
1690 | foreach my $line (@lines) { | ||
1691 | if ($line =~ m/$VCS_cmds{"author_pattern"}/) { | ||
1692 | my $author = $1; | ||
1693 | my ($name, $address) = parse_email($author); | ||
1694 | $author = format_email($name, $address, 1); | ||
1695 | push(@authors, $1); | ||
1696 | } | ||
1697 | } | ||
1698 | |||
1699 | save_commits_by_author(@lines) if ($interactive); | ||
1700 | save_commits_by_signer(@lines) if ($interactive); | ||
1701 | |||
1702 | push(@signers, @authors); | ||
1703 | }} | ||
1704 | else { | ||
1705 | foreach my $commit (@commits) { | ||
1706 | my $i; | ||
1707 | my $cmd = $VCS_cmds{"find_commit_author_cmd"}; | ||
1708 | $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd | ||
1709 | my @author = vcs_find_author($cmd); | ||
1710 | next if !@author; | ||
1711 | my $count = grep(/$commit/, @all_commits); | ||
1712 | for ($i = 0; $i < $count ; $i++) { | ||
1713 | push(@blame_signers, $author[0]); | ||
1714 | } | ||
1394 | } | 1715 | } |
1395 | } | 1716 | } |
1396 | if (@blame_signers) { | 1717 | if (@blame_signers) { |