blob: 633f2dd3de279fda4799ca41ba88553e1c963961 [file] [log] [blame]
Joe Perchescb7301c2009-04-07 20:40:12 -07001#!/usr/bin/perl -w
2# (c) 2007, Joe Perches <joe@perches.com>
3# created from checkpatch.pl
4#
5# Print selected MAINTAINERS information for
6# the files modified in a patch or for a file
7#
Roel Kluin3bd7bf52009-11-11 14:26:13 -08008# usage: perl scripts/get_maintainer.pl [OPTIONS] <patch>
9# perl scripts/get_maintainer.pl [OPTIONS] -f <file>
Joe Perchescb7301c2009-04-07 20:40:12 -070010#
11# Licensed under the terms of the GNU GPL License version 2
12
13use strict;
14
15my $P = $0;
Joe Perches7e1863a2011-01-12 16:59:49 -080016my $V = '0.26';
Joe Perchescb7301c2009-04-07 20:40:12 -070017
18use Getopt::Long qw(:config no_auto_abbrev);
Joe Perchesbe17bdd2016-01-20 14:58:24 -080019use Cwd;
Joe Perchescb7301c2009-04-07 20:40:12 -070020
Joe Perchesbe17bdd2016-01-20 14:58:24 -080021my $cur_path = fastgetcwd() . '/';
Joe Perchescb7301c2009-04-07 20:40:12 -070022my $lk_path = "./";
23my $email = 1;
24my $email_usename = 1;
25my $email_maintainer = 1;
Joe Perchesc1c3f2c2014-06-02 12:05:17 -070026my $email_reviewer = 1;
Joe Perchescb7301c2009-04-07 20:40:12 -070027my $email_list = 1;
28my $email_subscriber_list = 0;
Joe Perchescb7301c2009-04-07 20:40:12 -070029my $email_git_penguin_chiefs = 0;
Joe Perchese3e9d112010-10-26 14:22:53 -070030my $email_git = 0;
Florian Mickler0fa05592010-05-24 14:33:20 -070031my $email_git_all_signature_types = 0;
Joe Perches60db31a2009-12-14 18:00:50 -080032my $email_git_blame = 0;
Joe Perches683c6f82010-10-26 14:22:55 -070033my $email_git_blame_signatures = 1;
Joe Perchese3e9d112010-10-26 14:22:53 -070034my $email_git_fallback = 1;
Joe Perchescb7301c2009-04-07 20:40:12 -070035my $email_git_min_signatures = 1;
36my $email_git_max_maintainers = 5;
Joe Perchesafa81ee2009-07-29 15:04:28 -070037my $email_git_min_percent = 5;
Joe Perchescb7301c2009-04-07 20:40:12 -070038my $email_git_since = "1-year-ago";
Joe Perches60db31a2009-12-14 18:00:50 -080039my $email_hg_since = "-365";
Florian Micklerdace8e32010-10-26 14:22:54 -070040my $interactive = 0;
Joe Perches11ecf532009-09-21 17:04:22 -070041my $email_remove_duplicates = 1;
Joe Perchesb9e23312010-10-26 14:22:58 -070042my $email_use_mailmap = 1;
Joe Perchescb7301c2009-04-07 20:40:12 -070043my $output_multiline = 1;
44my $output_separator = ", ";
Joe Perches3c7385b2009-12-14 18:00:46 -080045my $output_roles = 0;
Joe Perches7e1863a2011-01-12 16:59:49 -080046my $output_rolestats = 1;
Joe Perches364f68d2015-06-25 15:01:52 -070047my $output_section_maxlen = 50;
Joe Perchescb7301c2009-04-07 20:40:12 -070048my $scm = 0;
49my $web = 0;
50my $subsystem = 0;
51my $status = 0;
Joe Perches03aed212016-12-12 16:45:59 -080052my $letters = "";
Joe Perchesdcf36a92009-10-26 16:49:47 -070053my $keywords = 1;
Joe Perches4b76c9d2010-03-05 13:43:03 -080054my $sections = 0;
Joe Perches03372db2010-03-05 13:43:00 -080055my $file_emails = 0;
Joe Perches4a7fdb52009-04-10 12:28:57 -070056my $from_filename = 0;
Joe Perches3fb55652009-09-21 17:04:17 -070057my $pattern_depth = 0;
Joe Perchescb7301c2009-04-07 20:40:12 -070058my $version = 0;
59my $help = 0;
60
Joe Perches683c6f82010-10-26 14:22:55 -070061my $vcs_used = 0;
62
Joe Perchescb7301c2009-04-07 20:40:12 -070063my $exit = 0;
64
Joe Perches683c6f82010-10-26 14:22:55 -070065my %commit_author_hash;
66my %commit_signer_hash;
Florian Micklerdace8e32010-10-26 14:22:54 -070067
Joe Perchescb7301c2009-04-07 20:40:12 -070068my @penguin_chief = ();
Joe Perchese4d26b02010-05-24 14:33:17 -070069push(@penguin_chief, "Linus Torvalds:torvalds\@linux-foundation.org");
Joe Perchescb7301c2009-04-07 20:40:12 -070070#Andrew wants in on most everything - 2009/01/14
Joe Perchese4d26b02010-05-24 14:33:17 -070071#push(@penguin_chief, "Andrew Morton:akpm\@linux-foundation.org");
Joe Perchescb7301c2009-04-07 20:40:12 -070072
73my @penguin_chief_names = ();
74foreach my $chief (@penguin_chief) {
75 if ($chief =~ m/^(.*):(.*)/) {
76 my $chief_name = $1;
77 my $chief_addr = $2;
78 push(@penguin_chief_names, $chief_name);
79 }
80}
Joe Perchese4d26b02010-05-24 14:33:17 -070081my $penguin_chiefs = "\(" . join("|", @penguin_chief_names) . "\)";
82
83# Signature types of people who are either
84# a) responsible for the code in question, or
85# b) familiar enough with it to give relevant feedback
86my @signature_tags = ();
87push(@signature_tags, "Signed-off-by:");
88push(@signature_tags, "Reviewed-by:");
89push(@signature_tags, "Acked-by:");
Joe Perchescb7301c2009-04-07 20:40:12 -070090
Joe Perches7dea2682012-06-20 12:53:02 -070091my $signature_pattern = "\(" . join("|", @signature_tags) . "\)";
92
Joe Perches5f2441e2009-06-16 15:34:02 -070093# rfc822 email address - preloaded methods go here.
Joe Perches1b5e1cf2009-06-16 15:34:01 -070094my $rfc822_lwsp = "(?:(?:\\r\\n)?[ \\t])";
Joe Perchesdf4cc032009-06-16 15:34:04 -070095my $rfc822_char = '[\\000-\\377]';
Joe Perches1b5e1cf2009-06-16 15:34:01 -070096
Joe Perches60db31a2009-12-14 18:00:50 -080097# VCS command support: class-like functions and strings
98
99my %VCS_cmds;
100
101my %VCS_cmds_git = (
102 "execute_cmd" => \&git_execute_cmd,
Richard Genoudec83b612014-02-10 14:25:31 -0800103 "available" => '(which("git") ne "") && (-e ".git")',
Joe Perches683c6f82010-10-26 14:22:55 -0700104 "find_signers_cmd" =>
Ian Campbelled128fea2012-01-10 15:08:41 -0800105 "git log --no-color --follow --since=\$email_git_since " .
Joe Perchesc9ecefe2014-01-23 15:54:20 -0800106 '--numstat --no-merges ' .
Joe Perches683c6f82010-10-26 14:22:55 -0700107 '--format="GitCommit: %H%n' .
108 'GitAuthor: %an <%ae>%n' .
109 'GitDate: %aD%n' .
110 'GitSubject: %s%n' .
111 '%b%n"' .
112 " -- \$file",
113 "find_commit_signers_cmd" =>
114 "git log --no-color " .
Joe Perchesc9ecefe2014-01-23 15:54:20 -0800115 '--numstat ' .
Joe Perches683c6f82010-10-26 14:22:55 -0700116 '--format="GitCommit: %H%n' .
117 'GitAuthor: %an <%ae>%n' .
118 'GitDate: %aD%n' .
119 'GitSubject: %s%n' .
120 '%b%n"' .
121 " -1 \$commit",
122 "find_commit_author_cmd" =>
123 "git log --no-color " .
Joe Perchesc9ecefe2014-01-23 15:54:20 -0800124 '--numstat ' .
Joe Perches683c6f82010-10-26 14:22:55 -0700125 '--format="GitCommit: %H%n' .
126 'GitAuthor: %an <%ae>%n' .
127 'GitDate: %aD%n' .
128 'GitSubject: %s%n"' .
129 " -1 \$commit",
Joe Perches60db31a2009-12-14 18:00:50 -0800130 "blame_range_cmd" => "git blame -l -L \$diff_start,+\$diff_length \$file",
131 "blame_file_cmd" => "git blame -l \$file",
Joe Perches683c6f82010-10-26 14:22:55 -0700132 "commit_pattern" => "^GitCommit: ([0-9a-f]{40,40})",
Florian Micklerdace8e32010-10-26 14:22:54 -0700133 "blame_commit_pattern" => "^([0-9a-f]+) ",
Joe Perches683c6f82010-10-26 14:22:55 -0700134 "author_pattern" => "^GitAuthor: (.*)",
135 "subject_pattern" => "^GitSubject: (.*)",
Joe Perchesc9ecefe2014-01-23 15:54:20 -0800136 "stat_pattern" => "^(\\d+)\\t(\\d+)\\t\$file\$",
Joe Perches4cad35a2016-08-02 14:04:10 -0700137 "file_exists_cmd" => "git ls-files \$file",
Joe Perches60db31a2009-12-14 18:00:50 -0800138);
139
140my %VCS_cmds_hg = (
141 "execute_cmd" => \&hg_execute_cmd,
142 "available" => '(which("hg") ne "") && (-d ".hg")',
143 "find_signers_cmd" =>
Joe Perches683c6f82010-10-26 14:22:55 -0700144 "hg log --date=\$email_hg_since " .
145 "--template='HgCommit: {node}\\n" .
146 "HgAuthor: {author}\\n" .
147 "HgSubject: {desc}\\n'" .
148 " -- \$file",
149 "find_commit_signers_cmd" =>
150 "hg log " .
151 "--template='HgSubject: {desc}\\n'" .
152 " -r \$commit",
153 "find_commit_author_cmd" =>
154 "hg log " .
155 "--template='HgCommit: {node}\\n" .
156 "HgAuthor: {author}\\n" .
157 "HgSubject: {desc|firstline}\\n'" .
158 " -r \$commit",
Joe Perches60db31a2009-12-14 18:00:50 -0800159 "blame_range_cmd" => "", # not supported
Joe Perches683c6f82010-10-26 14:22:55 -0700160 "blame_file_cmd" => "hg blame -n \$file",
161 "commit_pattern" => "^HgCommit: ([0-9a-f]{40,40})",
162 "blame_commit_pattern" => "^([ 0-9a-f]+):",
163 "author_pattern" => "^HgAuthor: (.*)",
164 "subject_pattern" => "^HgSubject: (.*)",
Joe Perchesc9ecefe2014-01-23 15:54:20 -0800165 "stat_pattern" => "^(\\d+)\t(\\d+)\t\$file\$",
Joe Perches4cad35a2016-08-02 14:04:10 -0700166 "file_exists_cmd" => "hg files \$file",
Joe Perches60db31a2009-12-14 18:00:50 -0800167);
168
Joe Perchesbcde44e2010-10-26 14:22:53 -0700169my $conf = which_conf(".get_maintainer.conf");
170if (-f $conf) {
Joe Perches368669d2010-05-24 14:33:19 -0700171 my @conf_args;
Joe Perchesbcde44e2010-10-26 14:22:53 -0700172 open(my $conffile, '<', "$conf")
173 or warn "$P: Can't find a readable .get_maintainer.conf file $!\n";
174
Joe Perches368669d2010-05-24 14:33:19 -0700175 while (<$conffile>) {
176 my $line = $_;
177
178 $line =~ s/\s*\n?$//g;
179 $line =~ s/^\s*//g;
180 $line =~ s/\s+/ /g;
181
182 next if ($line =~ m/^\s*#/);
183 next if ($line =~ m/^\s*$/);
184
185 my @words = split(" ", $line);
186 foreach my $word (@words) {
187 last if ($word =~ m/^#/);
188 push (@conf_args, $word);
189 }
190 }
191 close($conffile);
192 unshift(@ARGV, @conf_args) if @conf_args;
193}
194
Joe Perches435de072015-06-25 15:01:50 -0700195my @ignore_emails = ();
196my $ignore_file = which_conf(".get_maintainer.ignore");
197if (-f $ignore_file) {
198 open(my $ignore, '<', "$ignore_file")
199 or warn "$P: Can't find a readable .get_maintainer.ignore file $!\n";
200 while (<$ignore>) {
201 my $line = $_;
202
203 $line =~ s/\s*\n?$//;
204 $line =~ s/^\s*//;
205 $line =~ s/\s+$//;
206 $line =~ s/#.*$//;
207
208 next if ($line =~ m/^\s*$/);
209 if (rfc822_valid($line)) {
210 push(@ignore_emails, $line);
211 }
212 }
213 close($ignore);
214}
215
Joe Perchescb7301c2009-04-07 20:40:12 -0700216if (!GetOptions(
217 'email!' => \$email,
218 'git!' => \$email_git,
Joe Perchese4d26b02010-05-24 14:33:17 -0700219 'git-all-signature-types!' => \$email_git_all_signature_types,
Joe Perches60db31a2009-12-14 18:00:50 -0800220 'git-blame!' => \$email_git_blame,
Joe Perches683c6f82010-10-26 14:22:55 -0700221 'git-blame-signatures!' => \$email_git_blame_signatures,
Joe Perchese3e9d112010-10-26 14:22:53 -0700222 'git-fallback!' => \$email_git_fallback,
Joe Perchescb7301c2009-04-07 20:40:12 -0700223 'git-chief-penguins!' => \$email_git_penguin_chiefs,
224 'git-min-signatures=i' => \$email_git_min_signatures,
225 'git-max-maintainers=i' => \$email_git_max_maintainers,
Joe Perchesafa81ee2009-07-29 15:04:28 -0700226 'git-min-percent=i' => \$email_git_min_percent,
Joe Perchescb7301c2009-04-07 20:40:12 -0700227 'git-since=s' => \$email_git_since,
Joe Perches60db31a2009-12-14 18:00:50 -0800228 'hg-since=s' => \$email_hg_since,
Florian Micklerdace8e32010-10-26 14:22:54 -0700229 'i|interactive!' => \$interactive,
Joe Perches11ecf532009-09-21 17:04:22 -0700230 'remove-duplicates!' => \$email_remove_duplicates,
Joe Perchesb9e23312010-10-26 14:22:58 -0700231 'mailmap!' => \$email_use_mailmap,
Joe Perchescb7301c2009-04-07 20:40:12 -0700232 'm!' => \$email_maintainer,
Joe Perchesc1c3f2c2014-06-02 12:05:17 -0700233 'r!' => \$email_reviewer,
Joe Perchescb7301c2009-04-07 20:40:12 -0700234 'n!' => \$email_usename,
235 'l!' => \$email_list,
236 's!' => \$email_subscriber_list,
237 'multiline!' => \$output_multiline,
Joe Perches3c7385b2009-12-14 18:00:46 -0800238 'roles!' => \$output_roles,
239 'rolestats!' => \$output_rolestats,
Joe Perchescb7301c2009-04-07 20:40:12 -0700240 'separator=s' => \$output_separator,
241 'subsystem!' => \$subsystem,
242 'status!' => \$status,
243 'scm!' => \$scm,
244 'web!' => \$web,
Joe Perches03aed212016-12-12 16:45:59 -0800245 'letters=s' => \$letters,
Joe Perches3fb55652009-09-21 17:04:17 -0700246 'pattern-depth=i' => \$pattern_depth,
Joe Perchesdcf36a92009-10-26 16:49:47 -0700247 'k|keywords!' => \$keywords,
Joe Perches4b76c9d2010-03-05 13:43:03 -0800248 'sections!' => \$sections,
Joe Perches03372db2010-03-05 13:43:00 -0800249 'fe|file-emails!' => \$file_emails,
Joe Perches4a7fdb52009-04-10 12:28:57 -0700250 'f|file' => \$from_filename,
Joe Perchescb7301c2009-04-07 20:40:12 -0700251 'v|version' => \$version,
Joe Perches64f77f32010-03-05 13:43:04 -0800252 'h|help|usage' => \$help,
Joe Perchescb7301c2009-04-07 20:40:12 -0700253 )) {
Joe Perches3c7385b2009-12-14 18:00:46 -0800254 die "$P: invalid argument - use --help if necessary\n";
Joe Perchescb7301c2009-04-07 20:40:12 -0700255}
256
257if ($help != 0) {
258 usage();
259 exit 0;
260}
261
262if ($version != 0) {
263 print("${P} ${V}\n");
264 exit 0;
265}
266
Joe Perches64f77f32010-03-05 13:43:04 -0800267if (-t STDIN && !@ARGV) {
268 # We're talking to a terminal, but have no command line arguments.
269 die "$P: missing patchfile or -f file - use --help if necessary\n";
Joe Perchescb7301c2009-04-07 20:40:12 -0700270}
271
Joe Perches683c6f82010-10-26 14:22:55 -0700272$output_multiline = 0 if ($output_separator ne ", ");
273$output_rolestats = 1 if ($interactive);
274$output_roles = 1 if ($output_rolestats);
Joe Perches3c7385b2009-12-14 18:00:46 -0800275
Joe Perches03aed212016-12-12 16:45:59 -0800276if ($sections || $letters ne "") {
277 $sections = 1;
Joe Perches4b76c9d2010-03-05 13:43:03 -0800278 $email = 0;
279 $email_list = 0;
280 $scm = 0;
281 $status = 0;
282 $subsystem = 0;
283 $web = 0;
284 $keywords = 0;
Joe Perches6ef1c522010-10-26 14:22:56 -0700285 $interactive = 0;
Joe Perches4b76c9d2010-03-05 13:43:03 -0800286} else {
287 my $selections = $email + $scm + $status + $subsystem + $web;
288 if ($selections == 0) {
Joe Perches4b76c9d2010-03-05 13:43:03 -0800289 die "$P: Missing required option: email, scm, status, subsystem or web\n";
290 }
Joe Perchescb7301c2009-04-07 20:40:12 -0700291}
292
Joe Perchesf5492662009-09-21 17:04:13 -0700293if ($email &&
Joe Perchesc1c3f2c2014-06-02 12:05:17 -0700294 ($email_maintainer + $email_reviewer +
295 $email_list + $email_subscriber_list +
Joe Perchesf5492662009-09-21 17:04:13 -0700296 $email_git + $email_git_penguin_chiefs + $email_git_blame) == 0) {
Joe Perchescb7301c2009-04-07 20:40:12 -0700297 die "$P: Please select at least 1 email option\n";
298}
299
300if (!top_of_kernel_tree($lk_path)) {
301 die "$P: The current directory does not appear to be "
302 . "a linux kernel source tree.\n";
303}
304
305## Read MAINTAINERS for type/value pairs
306
307my @typevalue = ();
Joe Perchesdcf36a92009-10-26 16:49:47 -0700308my %keyword_hash;
309
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800310open (my $maint, '<', "${lk_path}MAINTAINERS")
311 or die "$P: Can't open MAINTAINERS: $!\n";
312while (<$maint>) {
Joe Perchescb7301c2009-04-07 20:40:12 -0700313 my $line = $_;
314
Joe Perchesce8155f2015-06-25 15:01:55 -0700315 if ($line =~ m/^([A-Z]):\s*(.*)/) {
Joe Perchescb7301c2009-04-07 20:40:12 -0700316 my $type = $1;
317 my $value = $2;
318
319 ##Filename pattern matching
320 if ($type eq "F" || $type eq "X") {
321 $value =~ s@\.@\\\.@g; ##Convert . to \.
322 $value =~ s/\*/\.\*/g; ##Convert * to .*
323 $value =~ s/\?/\./g; ##Convert ? to .
Joe Perches870020f2009-07-29 15:04:28 -0700324 ##if pattern is a directory and it lacks a trailing slash, add one
325 if ((-d $value)) {
326 $value =~ s@([^/])$@$1/@;
327 }
Joe Perchesdcf36a92009-10-26 16:49:47 -0700328 } elsif ($type eq "K") {
329 $keyword_hash{@typevalue} = $value;
Joe Perchescb7301c2009-04-07 20:40:12 -0700330 }
331 push(@typevalue, "$type:$value");
332 } elsif (!/^(\s)*$/) {
333 $line =~ s/\n$//g;
334 push(@typevalue, $line);
335 }
336}
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800337close($maint);
Joe Perchescb7301c2009-04-07 20:40:12 -0700338
Joe Perches8cbb3a72009-09-21 17:04:21 -0700339
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700340#
341# Read mail address map
342#
343
Joe Perchesb9e23312010-10-26 14:22:58 -0700344my $mailmap;
345
346read_mailmap();
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700347
348sub read_mailmap {
Joe Perchesb9e23312010-10-26 14:22:58 -0700349 $mailmap = {
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700350 names => {},
351 addresses => {}
Joe Perches47abc722010-10-26 14:22:57 -0700352 };
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700353
Joe Perchesb9e23312010-10-26 14:22:58 -0700354 return if (!$email_use_mailmap || !(-f "${lk_path}.mailmap"));
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700355
356 open(my $mailmap_file, '<', "${lk_path}.mailmap")
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800357 or warn "$P: Can't open .mailmap: $!\n";
Joe Perches8cbb3a72009-09-21 17:04:21 -0700358
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700359 while (<$mailmap_file>) {
360 s/#.*$//; #strip comments
361 s/^\s+|\s+$//g; #trim
Joe Perches8cbb3a72009-09-21 17:04:21 -0700362
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700363 next if (/^\s*$/); #skip empty lines
364 #entries have one of the following formats:
365 # name1 <mail1>
366 # <mail1> <mail2>
367 # name1 <mail1> <mail2>
368 # name1 <mail1> name2 <mail2>
369 # (see man git-shortlog)
Joe Perches0334b382011-07-25 17:13:13 -0700370
371 if (/^([^<]+)<([^>]+)>$/) {
Joe Perches47abc722010-10-26 14:22:57 -0700372 my $real_name = $1;
373 my $address = $2;
Joe Perches8cbb3a72009-09-21 17:04:21 -0700374
Joe Perches47abc722010-10-26 14:22:57 -0700375 $real_name =~ s/\s+$//;
Joe Perchesb9e23312010-10-26 14:22:58 -0700376 ($real_name, $address) = parse_email("$real_name <$address>");
Joe Perches47abc722010-10-26 14:22:57 -0700377 $mailmap->{names}->{$address} = $real_name;
Joe Perches8cbb3a72009-09-21 17:04:21 -0700378
Joe Perches0334b382011-07-25 17:13:13 -0700379 } elsif (/^<([^>]+)>\s*<([^>]+)>$/) {
Joe Perches47abc722010-10-26 14:22:57 -0700380 my $real_address = $1;
381 my $wrong_address = $2;
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700382
Joe Perches47abc722010-10-26 14:22:57 -0700383 $mailmap->{addresses}->{$wrong_address} = $real_address;
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700384
Joe Perches0334b382011-07-25 17:13:13 -0700385 } elsif (/^(.+)<([^>]+)>\s*<([^>]+)>$/) {
Joe Perchesb9e23312010-10-26 14:22:58 -0700386 my $real_name = $1;
Joe Perches47abc722010-10-26 14:22:57 -0700387 my $real_address = $2;
388 my $wrong_address = $3;
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700389
Joe Perches47abc722010-10-26 14:22:57 -0700390 $real_name =~ s/\s+$//;
Joe Perchesb9e23312010-10-26 14:22:58 -0700391 ($real_name, $real_address) =
392 parse_email("$real_name <$real_address>");
Joe Perches47abc722010-10-26 14:22:57 -0700393 $mailmap->{names}->{$wrong_address} = $real_name;
394 $mailmap->{addresses}->{$wrong_address} = $real_address;
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700395
Joe Perches0334b382011-07-25 17:13:13 -0700396 } elsif (/^(.+)<([^>]+)>\s*(.+)\s*<([^>]+)>$/) {
Joe Perches47abc722010-10-26 14:22:57 -0700397 my $real_name = $1;
398 my $real_address = $2;
399 my $wrong_name = $3;
400 my $wrong_address = $4;
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700401
Joe Perches47abc722010-10-26 14:22:57 -0700402 $real_name =~ s/\s+$//;
Joe Perchesb9e23312010-10-26 14:22:58 -0700403 ($real_name, $real_address) =
404 parse_email("$real_name <$real_address>");
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700405
Joe Perchesb9e23312010-10-26 14:22:58 -0700406 $wrong_name =~ s/\s+$//;
407 ($wrong_name, $wrong_address) =
408 parse_email("$wrong_name <$wrong_address>");
409
410 my $wrong_email = format_email($wrong_name, $wrong_address, 1);
411 $mailmap->{names}->{$wrong_email} = $real_name;
412 $mailmap->{addresses}->{$wrong_email} = $real_address;
Joe Perches11ecf532009-09-21 17:04:22 -0700413 }
Joe Perches8cbb3a72009-09-21 17:04:21 -0700414 }
Florian Mickler7fa8ff22010-10-26 14:22:56 -0700415 close($mailmap_file);
Joe Perches8cbb3a72009-09-21 17:04:21 -0700416}
417
Joe Perches4a7fdb52009-04-10 12:28:57 -0700418## use the filenames on the command line or find the filenames in the patchfiles
Joe Perchescb7301c2009-04-07 20:40:12 -0700419
420my @files = ();
Joe Perchesf5492662009-09-21 17:04:13 -0700421my @range = ();
Joe Perchesdcf36a92009-10-26 16:49:47 -0700422my @keyword_tvi = ();
Joe Perches03372db2010-03-05 13:43:00 -0800423my @file_emails = ();
Joe Perchescb7301c2009-04-07 20:40:12 -0700424
Joe Perches64f77f32010-03-05 13:43:04 -0800425if (!@ARGV) {
426 push(@ARGV, "&STDIN");
427}
428
Joe Perches4a7fdb52009-04-10 12:28:57 -0700429foreach my $file (@ARGV) {
Joe Perches64f77f32010-03-05 13:43:04 -0800430 if ($file ne "&STDIN") {
431 ##if $file is a directory and it lacks a trailing slash, add one
432 if ((-d $file)) {
433 $file =~ s@([^/])$@$1/@;
434 } elsif (!(-f $file)) {
435 die "$P: file '${file}' not found\n";
436 }
Joe Perchescb7301c2009-04-07 20:40:12 -0700437 }
Joe Perchesaec742e2016-08-10 08:45:11 -0700438 if ($from_filename || ($file ne "&STDIN" && vcs_file_exists($file))) {
Joe Perchesbe17bdd2016-01-20 14:58:24 -0800439 $file =~ s/^\Q${cur_path}\E//; #strip any absolute path
440 $file =~ s/^\Q${lk_path}\E//; #or the path to the lk tree
Joe Perches4a7fdb52009-04-10 12:28:57 -0700441 push(@files, $file);
Joe Perchesfab9ed12010-10-26 14:22:52 -0700442 if ($file ne "MAINTAINERS" && -f $file && ($keywords || $file_emails)) {
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800443 open(my $f, '<', $file)
444 or die "$P: Can't open $file: $!\n";
445 my $text = do { local($/) ; <$f> };
446 close($f);
Joe Perches03372db2010-03-05 13:43:00 -0800447 if ($keywords) {
448 foreach my $line (keys %keyword_hash) {
449 if ($text =~ m/$keyword_hash{$line}/x) {
450 push(@keyword_tvi, $line);
451 }
Joe Perchesdcf36a92009-10-26 16:49:47 -0700452 }
453 }
Joe Perches03372db2010-03-05 13:43:00 -0800454 if ($file_emails) {
455 my @poss_addr = $text =~ m$[A-Za-zÀ-ÿ\"\' \,\.\+-]*\s*[\,]*\s*[\(\<\{]{0,1}[A-Za-z0-9_\.\+-]+\@[A-Za-z0-9\.-]+\.[A-Za-z0-9]+[\)\>\}]{0,1}$g;
456 push(@file_emails, clean_file_emails(@poss_addr));
457 }
Joe Perchesdcf36a92009-10-26 16:49:47 -0700458 }
Joe Perches4a7fdb52009-04-10 12:28:57 -0700459 } else {
460 my $file_cnt = @files;
Joe Perchesf5492662009-09-21 17:04:13 -0700461 my $lastfile;
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800462
Wolfram Sang3a4df132010-03-23 13:35:18 -0700463 open(my $patch, "< $file")
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800464 or die "$P: Can't open $file: $!\n";
Joe Perches7764dcb2011-03-22 16:34:24 -0700465
466 # We can check arbitrary information before the patch
467 # like the commit message, mail headers, etc...
468 # This allows us to match arbitrary keywords against any part
469 # of a git format-patch generated file (subject tags, etc...)
470
471 my $patch_prefix = ""; #Parsing the intro
472
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800473 while (<$patch>) {
Joe Perchesdcf36a92009-10-26 16:49:47 -0700474 my $patch_line = $_;
Geert Uytterhoeven6be07102013-02-21 16:43:12 -0800475 if (m/^\+\+\+\s+(\S+)/ or m/^---\s+(\S+)/) {
Joe Perches4a7fdb52009-04-10 12:28:57 -0700476 my $filename = $1;
477 $filename =~ s@^[^/]*/@@;
478 $filename =~ s@\n@@;
Joe Perchesf5492662009-09-21 17:04:13 -0700479 $lastfile = $filename;
Joe Perches4a7fdb52009-04-10 12:28:57 -0700480 push(@files, $filename);
Joe Perches7764dcb2011-03-22 16:34:24 -0700481 $patch_prefix = "^[+-].*"; #Now parsing the actual patch
Joe Perchesf5492662009-09-21 17:04:13 -0700482 } elsif (m/^\@\@ -(\d+),(\d+)/) {
483 if ($email_git_blame) {
484 push(@range, "$lastfile:$1:$2");
485 }
Joe Perchesdcf36a92009-10-26 16:49:47 -0700486 } elsif ($keywords) {
487 foreach my $line (keys %keyword_hash) {
Joe Perches7764dcb2011-03-22 16:34:24 -0700488 if ($patch_line =~ m/${patch_prefix}$keyword_hash{$line}/x) {
Joe Perchesdcf36a92009-10-26 16:49:47 -0700489 push(@keyword_tvi, $line);
490 }
491 }
Joe Perches4a7fdb52009-04-10 12:28:57 -0700492 }
Joe Perchescb7301c2009-04-07 20:40:12 -0700493 }
Stephen Hemminger22dd5b02010-03-05 13:43:06 -0800494 close($patch);
495
Joe Perches4a7fdb52009-04-10 12:28:57 -0700496 if ($file_cnt == @files) {
Joe Perches7f29fd272009-06-16 15:34:04 -0700497 warn "$P: file '${file}' doesn't appear to be a patch. "
Joe Perches4a7fdb52009-04-10 12:28:57 -0700498 . "Add -f to options?\n";
499 }
500 @files = sort_and_uniq(@files);
Joe Perchescb7301c2009-04-07 20:40:12 -0700501 }
Joe Perchescb7301c2009-04-07 20:40:12 -0700502}
503
Joe Perches03372db2010-03-05 13:43:00 -0800504@file_emails = uniq(@file_emails);
505
Joe Perches683c6f82010-10-26 14:22:55 -0700506my %email_hash_name;
507my %email_hash_address;
Joe Perchescb7301c2009-04-07 20:40:12 -0700508my @email_to = ();
Joe Perches683c6f82010-10-26 14:22:55 -0700509my %hash_list_to;
Joe Perches290603c2009-06-16 15:33:58 -0700510my @list_to = ();
Joe Perchescb7301c2009-04-07 20:40:12 -0700511my @scm = ();
512my @web = ();
513my @subsystem = ();
514my @status = ();
Joe Perchesb9e23312010-10-26 14:22:58 -0700515my %deduplicate_name_hash = ();
516my %deduplicate_address_hash = ();
Joe Perchescb7301c2009-04-07 20:40:12 -0700517
Joe Perches6ef1c522010-10-26 14:22:56 -0700518my @maintainers = get_maintainers();
Joe Perchescb7301c2009-04-07 20:40:12 -0700519
Joe Perches6ef1c522010-10-26 14:22:56 -0700520if (@maintainers) {
521 @maintainers = merge_email(@maintainers);
522 output(@maintainers);
523}
Joe Perchescb7301c2009-04-07 20:40:12 -0700524
525if ($scm) {
Joe Perchesb7816552009-09-21 17:04:24 -0700526 @scm = uniq(@scm);
Joe Perchescb7301c2009-04-07 20:40:12 -0700527 output(@scm);
528}
Joe Perches683c6f82010-10-26 14:22:55 -0700529
Joe Perchescb7301c2009-04-07 20:40:12 -0700530if ($status) {
Joe Perchesb7816552009-09-21 17:04:24 -0700531 @status = uniq(@status);
Joe Perchescb7301c2009-04-07 20:40:12 -0700532 output(@status);
533}
534
535if ($subsystem) {
Joe Perchesb7816552009-09-21 17:04:24 -0700536 @subsystem = uniq(@subsystem);
Joe Perchescb7301c2009-04-07 20:40:12 -0700537 output(@subsystem);
538}
539
540if ($web) {
Joe Perchesb7816552009-09-21 17:04:24 -0700541 @web = uniq(@web);
Joe Perchescb7301c2009-04-07 20:40:12 -0700542 output(@web);
543}
544
545exit($exit);
546
Joe Perches435de072015-06-25 15:01:50 -0700547sub ignore_email_address {
548 my ($address) = @_;
549
550 foreach my $ignore (@ignore_emails) {
551 return 1 if ($ignore eq $address);
552 }
553
554 return 0;
555}
556
Joe Perchesab6c9372011-01-12 16:59:50 -0800557sub range_is_maintained {
558 my ($start, $end) = @_;
559
560 for (my $i = $start; $i < $end; $i++) {
561 my $line = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -0700562 if ($line =~ m/^([A-Z]):\s*(.*)/) {
Joe Perchesab6c9372011-01-12 16:59:50 -0800563 my $type = $1;
564 my $value = $2;
565 if ($type eq 'S') {
566 if ($value =~ /(maintain|support)/i) {
567 return 1;
568 }
569 }
570 }
571 }
572 return 0;
573}
574
575sub range_has_maintainer {
576 my ($start, $end) = @_;
577
578 for (my $i = $start; $i < $end; $i++) {
579 my $line = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -0700580 if ($line =~ m/^([A-Z]):\s*(.*)/) {
Joe Perchesab6c9372011-01-12 16:59:50 -0800581 my $type = $1;
582 my $value = $2;
583 if ($type eq 'M') {
584 return 1;
585 }
586 }
587 }
588 return 0;
589}
590
Joe Perches6ef1c522010-10-26 14:22:56 -0700591sub get_maintainers {
Joe Perches683c6f82010-10-26 14:22:55 -0700592 %email_hash_name = ();
593 %email_hash_address = ();
594 %commit_author_hash = ();
595 %commit_signer_hash = ();
596 @email_to = ();
597 %hash_list_to = ();
598 @list_to = ();
599 @scm = ();
600 @web = ();
601 @subsystem = ();
602 @status = ();
Joe Perchesb9e23312010-10-26 14:22:58 -0700603 %deduplicate_name_hash = ();
604 %deduplicate_address_hash = ();
Joe Perches683c6f82010-10-26 14:22:55 -0700605 if ($email_git_all_signature_types) {
606 $signature_pattern = "(.+?)[Bb][Yy]:";
607 } else {
608 $signature_pattern = "\(" . join("|", @signature_tags) . "\)";
609 }
610
611 # Find responsible parties
612
Joe Perchesb9e23312010-10-26 14:22:58 -0700613 my %exact_pattern_match_hash = ();
Joe Perches6ef1c522010-10-26 14:22:56 -0700614
Joe Perches683c6f82010-10-26 14:22:55 -0700615 foreach my $file (@files) {
616
617 my %hash;
Joe Perches683c6f82010-10-26 14:22:55 -0700618 my $tvi = find_first_section();
619 while ($tvi < @typevalue) {
620 my $start = find_starting_index($tvi);
621 my $end = find_ending_index($tvi);
622 my $exclude = 0;
623 my $i;
624
625 #Do not match excluded file patterns
626
627 for ($i = $start; $i < $end; $i++) {
628 my $line = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -0700629 if ($line =~ m/^([A-Z]):\s*(.*)/) {
Joe Perches683c6f82010-10-26 14:22:55 -0700630 my $type = $1;
631 my $value = $2;
632 if ($type eq 'X') {
633 if (file_match_pattern($file, $value)) {
634 $exclude = 1;
635 last;
636 }
637 }
638 }
639 }
640
641 if (!$exclude) {
642 for ($i = $start; $i < $end; $i++) {
643 my $line = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -0700644 if ($line =~ m/^([A-Z]):\s*(.*)/) {
Joe Perches683c6f82010-10-26 14:22:55 -0700645 my $type = $1;
646 my $value = $2;
647 if ($type eq 'F') {
648 if (file_match_pattern($file, $value)) {
649 my $value_pd = ($value =~ tr@/@@);
650 my $file_pd = ($file =~ tr@/@@);
651 $value_pd++ if (substr($value,-1,1) ne "/");
652 $value_pd = -1 if ($value =~ /^\.\*/);
Joe Perchesab6c9372011-01-12 16:59:50 -0800653 if ($value_pd >= $file_pd &&
654 range_is_maintained($start, $end) &&
655 range_has_maintainer($start, $end)) {
Joe Perches6ef1c522010-10-26 14:22:56 -0700656 $exact_pattern_match_hash{$file} = 1;
657 }
Joe Perches683c6f82010-10-26 14:22:55 -0700658 if ($pattern_depth == 0 ||
659 (($file_pd - $value_pd) < $pattern_depth)) {
660 $hash{$tvi} = $value_pd;
661 }
662 }
Stephen Warrenbbbe96e2013-04-29 16:17:23 -0700663 } elsif ($type eq 'N') {
Stephen Warreneb90d082013-02-27 17:02:53 -0800664 if ($file =~ m/$value/x) {
665 $hash{$tvi} = 0;
666 }
Joe Perches683c6f82010-10-26 14:22:55 -0700667 }
668 }
669 }
670 }
671 $tvi = $end + 1;
672 }
673
674 foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) {
675 add_categories($line);
676 if ($sections) {
677 my $i;
678 my $start = find_starting_index($line);
679 my $end = find_ending_index($line);
680 for ($i = $start; $i < $end; $i++) {
681 my $line = $typevalue[$i];
682 if ($line =~ /^[FX]:/) { ##Restore file patterns
683 $line =~ s/([^\\])\.([^\*])/$1\?$2/g;
684 $line =~ s/([^\\])\.$/$1\?/g; ##Convert . back to ?
685 $line =~ s/\\\./\./g; ##Convert \. to .
686 $line =~ s/\.\*/\*/g; ##Convert .* to *
687 }
Joe Perches03aed212016-12-12 16:45:59 -0800688 my $count = $line =~ s/^([A-Z]):/$1:\t/g;
689 if ($letters eq "" || (!$count || $letters =~ /$1/i)) {
690 print("$line\n");
691 }
Joe Perches683c6f82010-10-26 14:22:55 -0700692 }
693 print("\n");
694 }
695 }
Joe Perches683c6f82010-10-26 14:22:55 -0700696 }
697
698 if ($keywords) {
699 @keyword_tvi = sort_and_uniq(@keyword_tvi);
700 foreach my $line (@keyword_tvi) {
701 add_categories($line);
702 }
703 }
704
Joe Perchesb9e23312010-10-26 14:22:58 -0700705 foreach my $email (@email_to, @list_to) {
706 $email->[0] = deduplicate_email($email->[0]);
707 }
Joe Perches6ef1c522010-10-26 14:22:56 -0700708
709 foreach my $file (@files) {
710 if ($email &&
711 ($email_git || ($email_git_fallback &&
712 !$exact_pattern_match_hash{$file}))) {
713 vcs_file_signoffs($file);
714 }
715 if ($email && $email_git_blame) {
716 vcs_file_blame($file);
717 }
718 }
719
Joe Perches683c6f82010-10-26 14:22:55 -0700720 if ($email) {
721 foreach my $chief (@penguin_chief) {
722 if ($chief =~ m/^(.*):(.*)/) {
723 my $email_address;
724
725 $email_address = format_email($1, $2, $email_usename);
726 if ($email_git_penguin_chiefs) {
727 push(@email_to, [$email_address, 'chief penguin']);
728 } else {
729 @email_to = grep($_->[0] !~ /${email_address}/, @email_to);
730 }
731 }
732 }
733
734 foreach my $email (@file_emails) {
735 my ($name, $address) = parse_email($email);
736
737 my $tmp_email = format_email($name, $address, $email_usename);
738 push_email_address($tmp_email, '');
739 add_role($tmp_email, 'in file');
740 }
741 }
742
743 my @to = ();
744 if ($email || $email_list) {
745 if ($email) {
746 @to = (@to, @email_to);
747 }
748 if ($email_list) {
749 @to = (@to, @list_to);
750 }
751 }
752
Joe Perches6ef1c522010-10-26 14:22:56 -0700753 if ($interactive) {
Joe Perchesb9e23312010-10-26 14:22:58 -0700754 @to = interactive_get_maintainers(\@to);
Joe Perches6ef1c522010-10-26 14:22:56 -0700755 }
Joe Perches683c6f82010-10-26 14:22:55 -0700756
757 return @to;
758}
759
Joe Perchescb7301c2009-04-07 20:40:12 -0700760sub file_match_pattern {
761 my ($file, $pattern) = @_;
762 if (substr($pattern, -1) eq "/") {
763 if ($file =~ m@^$pattern@) {
764 return 1;
765 }
766 } else {
767 if ($file =~ m@^$pattern@) {
768 my $s1 = ($file =~ tr@/@@);
769 my $s2 = ($pattern =~ tr@/@@);
770 if ($s1 == $s2) {
771 return 1;
772 }
773 }
774 }
775 return 0;
776}
777
778sub usage {
779 print <<EOT;
780usage: $P [options] patchfile
Joe Perches870020f2009-07-29 15:04:28 -0700781 $P [options] -f file|directory
Joe Perchescb7301c2009-04-07 20:40:12 -0700782version: $V
783
784MAINTAINER field selection options:
785 --email => print email address(es) if any
786 --git => include recent git \*-by: signers
Joe Perchese4d26b02010-05-24 14:33:17 -0700787 --git-all-signature-types => include signers regardless of signature type
Joe Perches683c6f82010-10-26 14:22:55 -0700788 or use only ${signature_pattern} signers (default: $email_git_all_signature_types)
Joe Perchese3e9d112010-10-26 14:22:53 -0700789 --git-fallback => use git when no exact MAINTAINERS pattern (default: $email_git_fallback)
Joe Perchescb7301c2009-04-07 20:40:12 -0700790 --git-chief-penguins => include ${penguin_chiefs}
Joe Perchese4d26b02010-05-24 14:33:17 -0700791 --git-min-signatures => number of signatures required (default: $email_git_min_signatures)
792 --git-max-maintainers => maximum maintainers to add (default: $email_git_max_maintainers)
793 --git-min-percent => minimum percentage of commits required (default: $email_git_min_percent)
Joe Perchesf5492662009-09-21 17:04:13 -0700794 --git-blame => use git blame to find modified commits for patch or file
Brian Norris3cbcca82015-11-06 16:30:41 -0800795 --git-blame-signatures => when used with --git-blame, also include all commit signers
Joe Perchese4d26b02010-05-24 14:33:17 -0700796 --git-since => git history to use (default: $email_git_since)
797 --hg-since => hg history to use (default: $email_hg_since)
Florian Micklerdace8e32010-10-26 14:22:54 -0700798 --interactive => display a menu (mostly useful if used with the --git option)
Joe Perchescb7301c2009-04-07 20:40:12 -0700799 --m => include maintainer(s) if any
Joe Perchesc1c3f2c2014-06-02 12:05:17 -0700800 --r => include reviewer(s) if any
Joe Perchescb7301c2009-04-07 20:40:12 -0700801 --n => include name 'Full Name <addr\@domain.tld>'
802 --l => include list(s) if any
803 --s => include subscriber only list(s) if any
Joe Perches11ecf532009-09-21 17:04:22 -0700804 --remove-duplicates => minimize duplicate email names/addresses
Joe Perches3c7385b2009-12-14 18:00:46 -0800805 --roles => show roles (status:subsystem, git-signer, list, etc...)
806 --rolestats => show roles and statistics (commits/total_commits, %)
Joe Perches03372db2010-03-05 13:43:00 -0800807 --file-emails => add email addresses found in -f file (default: 0 (off))
Joe Perchescb7301c2009-04-07 20:40:12 -0700808 --scm => print SCM tree(s) if any
809 --status => print status if any
810 --subsystem => print subsystem name if any
811 --web => print website(s) if any
812
813Output type options:
814 --separator [, ] => separator for multiple entries on 1 line
Joe Perches42498312009-09-21 17:04:21 -0700815 using --separator also sets --nomultiline if --separator is not [, ]
Joe Perchescb7301c2009-04-07 20:40:12 -0700816 --multiline => print 1 entry per line
817
Joe Perchescb7301c2009-04-07 20:40:12 -0700818Other options:
Joe Perches3fb55652009-09-21 17:04:17 -0700819 --pattern-depth => Number of pattern directory traversals (default: 0 (all))
Joe Perchesb9e23312010-10-26 14:22:58 -0700820 --keywords => scan patch for keywords (default: $keywords)
821 --sections => print all of the subsystem sections with pattern matches
Joe Perches03aed212016-12-12 16:45:59 -0800822 --letters => print all matching 'letter' types from all matching sections
Joe Perchesb9e23312010-10-26 14:22:58 -0700823 --mailmap => use .mailmap file (default: $email_use_mailmap)
Joe Perchesf5f5078d2009-06-16 15:34:00 -0700824 --version => show version
Joe Perchescb7301c2009-04-07 20:40:12 -0700825 --help => show this help information
826
Joe Perches3fb55652009-09-21 17:04:17 -0700827Default options:
Brian Norris4f075102015-11-06 16:30:49 -0800828 [--email --nogit --git-fallback --m --r --n --l --multiline --pattern-depth=0
Joe Perches7e1863a2011-01-12 16:59:49 -0800829 --remove-duplicates --rolestats]
Joe Perches3fb55652009-09-21 17:04:17 -0700830
Joe Perches870020f2009-07-29 15:04:28 -0700831Notes:
832 Using "-f directory" may give unexpected results:
Joe Perchesf5492662009-09-21 17:04:13 -0700833 Used with "--git", git signators for _all_ files in and below
834 directory are examined as git recurses directories.
835 Any specified X: (exclude) pattern matches are _not_ ignored.
836 Used with "--nogit", directory is used as a pattern match,
Joe Perches60db31a2009-12-14 18:00:50 -0800837 no individual file within the directory or subdirectory
838 is matched.
Joe Perchesf5492662009-09-21 17:04:13 -0700839 Used with "--git-blame", does not iterate all files in directory
840 Using "--git-blame" is slow and may add old committers and authors
841 that are no longer active maintainers to the output.
Joe Perches3c7385b2009-12-14 18:00:46 -0800842 Using "--roles" or "--rolestats" with git send-email --cc-cmd or any
843 other automated tools that expect only ["name"] <email address>
844 may not work because of additional output after <email address>.
845 Using "--rolestats" and "--git-blame" shows the #/total=% commits,
846 not the percentage of the entire file authored. # of commits is
847 not a good measure of amount of code authored. 1 major commit may
848 contain a thousand lines, 5 trivial commits may modify a single line.
Joe Perches60db31a2009-12-14 18:00:50 -0800849 If git is not installed, but mercurial (hg) is installed and an .hg
850 repository exists, the following options apply to mercurial:
851 --git,
852 --git-min-signatures, --git-max-maintainers, --git-min-percent, and
853 --git-blame
854 Use --hg-since not --git-since to control date selection
Joe Perches368669d2010-05-24 14:33:19 -0700855 File ".get_maintainer.conf", if it exists in the linux kernel source root
856 directory, can change whatever get_maintainer defaults are desired.
857 Entries in this file can be any command line argument.
858 This file is prepended to any additional command line arguments.
859 Multiple lines and # comments are allowed.
Brian Norrisb1312bf2015-11-06 16:30:46 -0800860 Most options have both positive and negative forms.
861 The negative forms for --<foo> are --no<foo> and --no-<foo>.
862
Joe Perchescb7301c2009-04-07 20:40:12 -0700863EOT
864}
865
866sub top_of_kernel_tree {
Joe Perches47abc722010-10-26 14:22:57 -0700867 my ($lk_path) = @_;
Joe Perchescb7301c2009-04-07 20:40:12 -0700868
Joe Perches47abc722010-10-26 14:22:57 -0700869 if ($lk_path ne "" && substr($lk_path,length($lk_path)-1,1) ne "/") {
870 $lk_path .= "/";
871 }
872 if ( (-f "${lk_path}COPYING")
873 && (-f "${lk_path}CREDITS")
874 && (-f "${lk_path}Kbuild")
875 && (-f "${lk_path}MAINTAINERS")
876 && (-f "${lk_path}Makefile")
877 && (-f "${lk_path}README")
878 && (-d "${lk_path}Documentation")
879 && (-d "${lk_path}arch")
880 && (-d "${lk_path}include")
881 && (-d "${lk_path}drivers")
882 && (-d "${lk_path}fs")
883 && (-d "${lk_path}init")
884 && (-d "${lk_path}ipc")
885 && (-d "${lk_path}kernel")
886 && (-d "${lk_path}lib")
887 && (-d "${lk_path}scripts")) {
888 return 1;
889 }
890 return 0;
Joe Perchescb7301c2009-04-07 20:40:12 -0700891}
892
Joe Perches0e70e832009-09-21 17:04:20 -0700893sub parse_email {
894 my ($formatted_email) = @_;
895
896 my $name = "";
897 my $address = "";
898
Joe Perches11ecf532009-09-21 17:04:22 -0700899 if ($formatted_email =~ /^([^<]+)<(.+\@.*)>.*$/) {
Joe Perches0e70e832009-09-21 17:04:20 -0700900 $name = $1;
901 $address = $2;
Joe Perches11ecf532009-09-21 17:04:22 -0700902 } elsif ($formatted_email =~ /^\s*<(.+\@\S*)>.*$/) {
Joe Perches0e70e832009-09-21 17:04:20 -0700903 $address = $1;
Joe Perchesb7816552009-09-21 17:04:24 -0700904 } elsif ($formatted_email =~ /^(.+\@\S*).*$/) {
Joe Perches0e70e832009-09-21 17:04:20 -0700905 $address = $1;
906 }
Joe Perchescb7301c2009-04-07 20:40:12 -0700907
908 $name =~ s/^\s+|\s+$//g;
Joe Perchesd7895042009-06-16 15:34:02 -0700909 $name =~ s/^\"|\"$//g;
Joe Perches0e70e832009-09-21 17:04:20 -0700910 $address =~ s/^\s+|\s+$//g;
Joe Perchescb7301c2009-04-07 20:40:12 -0700911
Stephen Hemmingera63ceb42010-03-05 13:43:06 -0800912 if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
Joe Perchescb7301c2009-04-07 20:40:12 -0700913 $name =~ s/(?<!\\)"/\\"/g; ##escape quotes
Joe Perches0e70e832009-09-21 17:04:20 -0700914 $name = "\"$name\"";
Joe Perchescb7301c2009-04-07 20:40:12 -0700915 }
Joe Perches0e70e832009-09-21 17:04:20 -0700916
917 return ($name, $address);
918}
919
920sub format_email {
Joe Perchesa8af2432009-12-14 18:00:49 -0800921 my ($name, $address, $usename) = @_;
Joe Perches0e70e832009-09-21 17:04:20 -0700922
923 my $formatted_email;
924
925 $name =~ s/^\s+|\s+$//g;
926 $name =~ s/^\"|\"$//g;
927 $address =~ s/^\s+|\s+$//g;
928
Stephen Hemmingera63ceb42010-03-05 13:43:06 -0800929 if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
Joe Perches0e70e832009-09-21 17:04:20 -0700930 $name =~ s/(?<!\\)"/\\"/g; ##escape quotes
931 $name = "\"$name\"";
932 }
933
Joe Perchesa8af2432009-12-14 18:00:49 -0800934 if ($usename) {
Joe Perches0e70e832009-09-21 17:04:20 -0700935 if ("$name" eq "") {
936 $formatted_email = "$address";
937 } else {
Joe Perchesa8af2432009-12-14 18:00:49 -0800938 $formatted_email = "$name <$address>";
Joe Perches0e70e832009-09-21 17:04:20 -0700939 }
940 } else {
941 $formatted_email = $address;
942 }
943
Joe Perchescb7301c2009-04-07 20:40:12 -0700944 return $formatted_email;
945}
946
Joe Perches272a8972010-01-08 14:42:48 -0800947sub find_first_section {
948 my $index = 0;
949
950 while ($index < @typevalue) {
951 my $tv = $typevalue[$index];
Joe Perchesce8155f2015-06-25 15:01:55 -0700952 if (($tv =~ m/^([A-Z]):\s*(.*)/)) {
Joe Perches272a8972010-01-08 14:42:48 -0800953 last;
954 }
955 $index++;
956 }
957
958 return $index;
959}
960
Joe Perchesb7816552009-09-21 17:04:24 -0700961sub find_starting_index {
Joe Perchesb7816552009-09-21 17:04:24 -0700962 my ($index) = @_;
963
964 while ($index > 0) {
965 my $tv = $typevalue[$index];
Joe Perchesce8155f2015-06-25 15:01:55 -0700966 if (!($tv =~ m/^([A-Z]):\s*(.*)/)) {
Joe Perchesb7816552009-09-21 17:04:24 -0700967 last;
968 }
969 $index--;
970 }
971
972 return $index;
973}
974
975sub find_ending_index {
976 my ($index) = @_;
977
978 while ($index < @typevalue) {
979 my $tv = $typevalue[$index];
Joe Perchesce8155f2015-06-25 15:01:55 -0700980 if (!($tv =~ m/^([A-Z]):\s*(.*)/)) {
Joe Perchesb7816552009-09-21 17:04:24 -0700981 last;
982 }
983 $index++;
984 }
985
986 return $index;
987}
988
Joe Perches2a7cb1d2015-11-06 16:30:52 -0800989sub get_subsystem_name {
990 my ($index) = @_;
991
992 my $start = find_starting_index($index);
993
994 my $subsystem = $typevalue[$start];
995 if ($output_section_maxlen && length($subsystem) > $output_section_maxlen) {
996 $subsystem = substr($subsystem, 0, $output_section_maxlen - 3);
997 $subsystem =~ s/\s*$//;
998 $subsystem = $subsystem . "...";
999 }
1000 return $subsystem;
1001}
1002
Joe Perches3c7385b2009-12-14 18:00:46 -08001003sub get_maintainer_role {
1004 my ($index) = @_;
1005
1006 my $i;
1007 my $start = find_starting_index($index);
1008 my $end = find_ending_index($index);
1009
Joe Perches0ede2742012-03-23 15:01:56 -07001010 my $role = "unknown";
Joe Perches2a7cb1d2015-11-06 16:30:52 -08001011 my $subsystem = get_subsystem_name($index);
Joe Perches3c7385b2009-12-14 18:00:46 -08001012
1013 for ($i = $start + 1; $i < $end; $i++) {
1014 my $tv = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -07001015 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001016 my $ptype = $1;
1017 my $pvalue = $2;
1018 if ($ptype eq "S") {
1019 $role = $pvalue;
1020 }
1021 }
1022 }
1023
1024 $role = lc($role);
1025 if ($role eq "supported") {
1026 $role = "supporter";
1027 } elsif ($role eq "maintained") {
1028 $role = "maintainer";
1029 } elsif ($role eq "odd fixes") {
1030 $role = "odd fixer";
1031 } elsif ($role eq "orphan") {
1032 $role = "orphan minder";
1033 } elsif ($role eq "obsolete") {
1034 $role = "obsolete minder";
1035 } elsif ($role eq "buried alive in reporters") {
1036 $role = "chief penguin";
1037 }
1038
1039 return $role . ":" . $subsystem;
1040}
1041
1042sub get_list_role {
1043 my ($index) = @_;
1044
Joe Perches2a7cb1d2015-11-06 16:30:52 -08001045 my $subsystem = get_subsystem_name($index);
Joe Perches3c7385b2009-12-14 18:00:46 -08001046
1047 if ($subsystem eq "THE REST") {
1048 $subsystem = "";
1049 }
1050
1051 return $subsystem;
1052}
1053
Joe Perchescb7301c2009-04-07 20:40:12 -07001054sub add_categories {
1055 my ($index) = @_;
1056
Joe Perchesb7816552009-09-21 17:04:24 -07001057 my $i;
1058 my $start = find_starting_index($index);
1059 my $end = find_ending_index($index);
1060
1061 push(@subsystem, $typevalue[$start]);
1062
1063 for ($i = $start + 1; $i < $end; $i++) {
1064 my $tv = $typevalue[$i];
Joe Perchesce8155f2015-06-25 15:01:55 -07001065 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
Joe Perchescb7301c2009-04-07 20:40:12 -07001066 my $ptype = $1;
1067 my $pvalue = $2;
1068 if ($ptype eq "L") {
Joe Perches290603c2009-06-16 15:33:58 -07001069 my $list_address = $pvalue;
1070 my $list_additional = "";
Joe Perches3c7385b2009-12-14 18:00:46 -08001071 my $list_role = get_list_role($i);
1072
1073 if ($list_role ne "") {
1074 $list_role = ":" . $list_role;
1075 }
Joe Perches290603c2009-06-16 15:33:58 -07001076 if ($list_address =~ m/([^\s]+)\s+(.*)$/) {
1077 $list_address = $1;
1078 $list_additional = $2;
1079 }
Joe Perchesbdf7c682009-06-16 15:33:59 -07001080 if ($list_additional =~ m/subscribers-only/) {
Joe Perchescb7301c2009-04-07 20:40:12 -07001081 if ($email_subscriber_list) {
Joe Perches6ef1c522010-10-26 14:22:56 -07001082 if (!$hash_list_to{lc($list_address)}) {
1083 $hash_list_to{lc($list_address)} = 1;
Joe Perches683c6f82010-10-26 14:22:55 -07001084 push(@list_to, [$list_address,
1085 "subscriber list${list_role}"]);
1086 }
Joe Perchescb7301c2009-04-07 20:40:12 -07001087 }
1088 } else {
1089 if ($email_list) {
Joe Perches6ef1c522010-10-26 14:22:56 -07001090 if (!$hash_list_to{lc($list_address)}) {
1091 $hash_list_to{lc($list_address)} = 1;
Richard Weinberger728f5a92012-03-23 15:01:56 -07001092 if ($list_additional =~ m/moderated/) {
1093 push(@list_to, [$list_address,
1094 "moderated list${list_role}"]);
1095 } else {
1096 push(@list_to, [$list_address,
1097 "open list${list_role}"]);
1098 }
Joe Perches683c6f82010-10-26 14:22:55 -07001099 }
Joe Perchescb7301c2009-04-07 20:40:12 -07001100 }
1101 }
1102 } elsif ($ptype eq "M") {
Joe Perches0e70e832009-09-21 17:04:20 -07001103 my ($name, $address) = parse_email($pvalue);
1104 if ($name eq "") {
Joe Perchesb7816552009-09-21 17:04:24 -07001105 if ($i > 0) {
1106 my $tv = $typevalue[$i - 1];
Joe Perchesce8155f2015-06-25 15:01:55 -07001107 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
Joe Perches0e70e832009-09-21 17:04:20 -07001108 if ($1 eq "P") {
1109 $name = $2;
Joe Perchesa8af2432009-12-14 18:00:49 -08001110 $pvalue = format_email($name, $address, $email_usename);
Joe Perches5f2441e2009-06-16 15:34:02 -07001111 }
1112 }
1113 }
1114 }
Joe Perches0e70e832009-09-21 17:04:20 -07001115 if ($email_maintainer) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001116 my $role = get_maintainer_role($i);
1117 push_email_addresses($pvalue, $role);
Joe Perchescb7301c2009-04-07 20:40:12 -07001118 }
Joe Perchesc1c3f2c2014-06-02 12:05:17 -07001119 } elsif ($ptype eq "R") {
1120 my ($name, $address) = parse_email($pvalue);
1121 if ($name eq "") {
1122 if ($i > 0) {
1123 my $tv = $typevalue[$i - 1];
Joe Perchesce8155f2015-06-25 15:01:55 -07001124 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
Joe Perchesc1c3f2c2014-06-02 12:05:17 -07001125 if ($1 eq "P") {
1126 $name = $2;
1127 $pvalue = format_email($name, $address, $email_usename);
1128 }
1129 }
1130 }
1131 }
1132 if ($email_reviewer) {
Joe Perches2a7cb1d2015-11-06 16:30:52 -08001133 my $subsystem = get_subsystem_name($i);
1134 push_email_addresses($pvalue, "reviewer:$subsystem");
Joe Perchesc1c3f2c2014-06-02 12:05:17 -07001135 }
Joe Perchescb7301c2009-04-07 20:40:12 -07001136 } elsif ($ptype eq "T") {
1137 push(@scm, $pvalue);
1138 } elsif ($ptype eq "W") {
1139 push(@web, $pvalue);
1140 } elsif ($ptype eq "S") {
1141 push(@status, $pvalue);
1142 }
Joe Perchescb7301c2009-04-07 20:40:12 -07001143 }
1144 }
1145}
1146
Joe Perches11ecf532009-09-21 17:04:22 -07001147sub email_inuse {
1148 my ($name, $address) = @_;
Joe Perches0e70e832009-09-21 17:04:20 -07001149
Joe Perches11ecf532009-09-21 17:04:22 -07001150 return 1 if (($name eq "") && ($address eq ""));
Joe Perches6ef1c522010-10-26 14:22:56 -07001151 return 1 if (($name ne "") && exists($email_hash_name{lc($name)}));
1152 return 1 if (($address ne "") && exists($email_hash_address{lc($address)}));
Joe Perches11ecf532009-09-21 17:04:22 -07001153
Joe Perches0e70e832009-09-21 17:04:20 -07001154 return 0;
1155}
1156
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001157sub push_email_address {
Joe Perches3c7385b2009-12-14 18:00:46 -08001158 my ($line, $role) = @_;
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001159
Joe Perches0e70e832009-09-21 17:04:20 -07001160 my ($name, $address) = parse_email($line);
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001161
Joe Perchesb7816552009-09-21 17:04:24 -07001162 if ($address eq "") {
1163 return 0;
1164 }
1165
Joe Perches11ecf532009-09-21 17:04:22 -07001166 if (!$email_remove_duplicates) {
Joe Perchesa8af2432009-12-14 18:00:49 -08001167 push(@email_to, [format_email($name, $address, $email_usename), $role]);
Joe Perches11ecf532009-09-21 17:04:22 -07001168 } elsif (!email_inuse($name, $address)) {
Joe Perchesa8af2432009-12-14 18:00:49 -08001169 push(@email_to, [format_email($name, $address, $email_usename), $role]);
Joe Perchesfae99202010-10-26 14:22:58 -07001170 $email_hash_name{lc($name)}++ if ($name ne "");
Joe Perches6ef1c522010-10-26 14:22:56 -07001171 $email_hash_address{lc($address)}++;
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001172 }
Joe Perchesb7816552009-09-21 17:04:24 -07001173
1174 return 1;
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001175}
1176
1177sub push_email_addresses {
Joe Perches3c7385b2009-12-14 18:00:46 -08001178 my ($address, $role) = @_;
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001179
1180 my @address_list = ();
1181
Joe Perches5f2441e2009-06-16 15:34:02 -07001182 if (rfc822_valid($address)) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001183 push_email_address($address, $role);
Joe Perches5f2441e2009-06-16 15:34:02 -07001184 } elsif (@address_list = rfc822_validlist($address)) {
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001185 my $array_count = shift(@address_list);
1186 while (my $entry = shift(@address_list)) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001187 push_email_address($entry, $role);
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001188 }
Joe Perches5f2441e2009-06-16 15:34:02 -07001189 } else {
Joe Perches3c7385b2009-12-14 18:00:46 -08001190 if (!push_email_address($address, $role)) {
Joe Perchesb7816552009-09-21 17:04:24 -07001191 warn("Invalid MAINTAINERS address: '" . $address . "'\n");
1192 }
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001193 }
Joe Perches1b5e1cf2009-06-16 15:34:01 -07001194}
1195
Joe Perches3c7385b2009-12-14 18:00:46 -08001196sub add_role {
1197 my ($line, $role) = @_;
1198
1199 my ($name, $address) = parse_email($line);
Joe Perchesa8af2432009-12-14 18:00:49 -08001200 my $email = format_email($name, $address, $email_usename);
Joe Perches3c7385b2009-12-14 18:00:46 -08001201
1202 foreach my $entry (@email_to) {
1203 if ($email_remove_duplicates) {
1204 my ($entry_name, $entry_address) = parse_email($entry->[0]);
Joe Perches03372db2010-03-05 13:43:00 -08001205 if (($name eq $entry_name || $address eq $entry_address)
1206 && ($role eq "" || !($entry->[1] =~ m/$role/))
1207 ) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001208 if ($entry->[1] eq "") {
1209 $entry->[1] = "$role";
1210 } else {
1211 $entry->[1] = "$entry->[1],$role";
1212 }
1213 }
1214 } else {
Joe Perches03372db2010-03-05 13:43:00 -08001215 if ($email eq $entry->[0]
1216 && ($role eq "" || !($entry->[1] =~ m/$role/))
1217 ) {
Joe Perches3c7385b2009-12-14 18:00:46 -08001218 if ($entry->[1] eq "") {
1219 $entry->[1] = "$role";
1220 } else {
1221 $entry->[1] = "$entry->[1],$role";
1222 }
1223 }
1224 }
1225 }
1226}
1227
Joe Perchescb7301c2009-04-07 20:40:12 -07001228sub which {
1229 my ($bin) = @_;
1230
Joe Perchesf5f5078d2009-06-16 15:34:00 -07001231 foreach my $path (split(/:/, $ENV{PATH})) {
Joe Perchescb7301c2009-04-07 20:40:12 -07001232 if (-e "$path/$bin") {
1233 return "$path/$bin";
1234 }
1235 }
1236
1237 return "";
1238}
1239
Joe Perchesbcde44e2010-10-26 14:22:53 -07001240sub which_conf {
1241 my ($conf) = @_;
1242
1243 foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) {
1244 if (-e "$path/$conf") {
1245 return "$path/$conf";
1246 }
1247 }
1248
1249 return "";
1250}
1251
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001252sub mailmap_email {
Joe Perchesb9e23312010-10-26 14:22:58 -07001253 my ($line) = @_;
Joe Perches8cbb3a72009-09-21 17:04:21 -07001254
Joe Perches47abc722010-10-26 14:22:57 -07001255 my ($name, $address) = parse_email($line);
1256 my $email = format_email($name, $address, 1);
1257 my $real_name = $name;
1258 my $real_address = $address;
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001259
Joe Perches47abc722010-10-26 14:22:57 -07001260 if (exists $mailmap->{names}->{$email} ||
1261 exists $mailmap->{addresses}->{$email}) {
1262 if (exists $mailmap->{names}->{$email}) {
1263 $real_name = $mailmap->{names}->{$email};
Joe Perches8cbb3a72009-09-21 17:04:21 -07001264 }
Joe Perches47abc722010-10-26 14:22:57 -07001265 if (exists $mailmap->{addresses}->{$email}) {
1266 $real_address = $mailmap->{addresses}->{$email};
1267 }
1268 } else {
1269 if (exists $mailmap->{names}->{$address}) {
1270 $real_name = $mailmap->{names}->{$address};
1271 }
1272 if (exists $mailmap->{addresses}->{$address}) {
1273 $real_address = $mailmap->{addresses}->{$address};
1274 }
1275 }
1276 return format_email($real_name, $real_address, 1);
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001277}
1278
1279sub mailmap {
1280 my (@addresses) = @_;
1281
Joe Perchesb9e23312010-10-26 14:22:58 -07001282 my @mapped_emails = ();
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001283 foreach my $line (@addresses) {
Joe Perchesb9e23312010-10-26 14:22:58 -07001284 push(@mapped_emails, mailmap_email($line));
Joe Perches8cbb3a72009-09-21 17:04:21 -07001285 }
Joe Perchesb9e23312010-10-26 14:22:58 -07001286 merge_by_realname(@mapped_emails) if ($email_use_mailmap);
1287 return @mapped_emails;
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001288}
1289
1290sub merge_by_realname {
Joe Perches47abc722010-10-26 14:22:57 -07001291 my %address_map;
1292 my (@emails) = @_;
Joe Perchesb9e23312010-10-26 14:22:58 -07001293
Joe Perches47abc722010-10-26 14:22:57 -07001294 foreach my $email (@emails) {
1295 my ($name, $address) = parse_email($email);
Joe Perchesb9e23312010-10-26 14:22:58 -07001296 if (exists $address_map{$name}) {
Joe Perches47abc722010-10-26 14:22:57 -07001297 $address = $address_map{$name};
Joe Perchesb9e23312010-10-26 14:22:58 -07001298 $email = format_email($name, $address, 1);
1299 } else {
1300 $address_map{$name} = $address;
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001301 }
Joe Perches47abc722010-10-26 14:22:57 -07001302 }
Joe Perches8cbb3a72009-09-21 17:04:21 -07001303}
1304
Joe Perches60db31a2009-12-14 18:00:50 -08001305sub git_execute_cmd {
1306 my ($cmd) = @_;
1307 my @lines = ();
Joe Perchescb7301c2009-04-07 20:40:12 -07001308
Joe Perches60db31a2009-12-14 18:00:50 -08001309 my $output = `$cmd`;
1310 $output =~ s/^\s*//gm;
1311 @lines = split("\n", $output);
1312
1313 return @lines;
Joe Perchesa8af2432009-12-14 18:00:49 -08001314}
1315
Joe Perches60db31a2009-12-14 18:00:50 -08001316sub hg_execute_cmd {
Joe Perchesa8af2432009-12-14 18:00:49 -08001317 my ($cmd) = @_;
Joe Perches60db31a2009-12-14 18:00:50 -08001318 my @lines = ();
Joe Perchesa8af2432009-12-14 18:00:49 -08001319
Joe Perches60db31a2009-12-14 18:00:50 -08001320 my $output = `$cmd`;
1321 @lines = split("\n", $output);
1322
1323 return @lines;
1324}
1325
Joe Perches683c6f82010-10-26 14:22:55 -07001326sub extract_formatted_signatures {
1327 my (@signature_lines) = @_;
1328
1329 my @type = @signature_lines;
1330
1331 s/\s*(.*):.*/$1/ for (@type);
1332
1333 # cut -f2- -d":"
1334 s/\s*.*:\s*(.+)\s*/$1/ for (@signature_lines);
1335
1336## Reformat email addresses (with names) to avoid badly written signatures
1337
1338 foreach my $signer (@signature_lines) {
Joe Perchesb9e23312010-10-26 14:22:58 -07001339 $signer = deduplicate_email($signer);
Joe Perches683c6f82010-10-26 14:22:55 -07001340 }
1341
1342 return (\@type, \@signature_lines);
1343}
1344
Joe Perches60db31a2009-12-14 18:00:50 -08001345sub vcs_find_signers {
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001346 my ($cmd, $file) = @_;
Joe Perchesa8af2432009-12-14 18:00:49 -08001347 my $commits;
Joe Perches683c6f82010-10-26 14:22:55 -07001348 my @lines = ();
1349 my @signatures = ();
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001350 my @authors = ();
1351 my @stats = ();
Joe Perchesa8af2432009-12-14 18:00:49 -08001352
Joe Perches60db31a2009-12-14 18:00:50 -08001353 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
Joe Perchescb7301c2009-04-07 20:40:12 -07001354
Joe Perches60db31a2009-12-14 18:00:50 -08001355 my $pattern = $VCS_cmds{"commit_pattern"};
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001356 my $author_pattern = $VCS_cmds{"author_pattern"};
1357 my $stat_pattern = $VCS_cmds{"stat_pattern"};
1358
1359 $stat_pattern =~ s/(\$\w+)/$1/eeg; #interpolate $stat_pattern
Joe Perchescb7301c2009-04-07 20:40:12 -07001360
Joe Perches60db31a2009-12-14 18:00:50 -08001361 $commits = grep(/$pattern/, @lines); # of commits
Joe Perchesafa81ee2009-07-29 15:04:28 -07001362
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001363 @authors = grep(/$author_pattern/, @lines);
Joe Perches683c6f82010-10-26 14:22:55 -07001364 @signatures = grep(/^[ \t]*${signature_pattern}.*\@.*$/, @lines);
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001365 @stats = grep(/$stat_pattern/, @lines);
Joe Perches683c6f82010-10-26 14:22:55 -07001366
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001367# print("stats: <@stats>\n");
1368
1369 return (0, \@signatures, \@authors, \@stats) if !@signatures;
Joe Perches683c6f82010-10-26 14:22:55 -07001370
1371 save_commits_by_author(@lines) if ($interactive);
1372 save_commits_by_signer(@lines) if ($interactive);
1373
Joe Perches0e70e832009-09-21 17:04:20 -07001374 if (!$email_git_penguin_chiefs) {
Joe Perches683c6f82010-10-26 14:22:55 -07001375 @signatures = grep(!/${penguin_chiefs}/i, @signatures);
Joe Perches0e70e832009-09-21 17:04:20 -07001376 }
Joe Perches63ab52d2010-10-26 14:22:51 -07001377
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001378 my ($author_ref, $authors_ref) = extract_formatted_signatures(@authors);
Joe Perches683c6f82010-10-26 14:22:55 -07001379 my ($types_ref, $signers_ref) = extract_formatted_signatures(@signatures);
Joe Perches63ab52d2010-10-26 14:22:51 -07001380
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001381 return ($commits, $signers_ref, $authors_ref, \@stats);
Joe Perchesa8af2432009-12-14 18:00:49 -08001382}
1383
Joe Perches63ab52d2010-10-26 14:22:51 -07001384sub vcs_find_author {
1385 my ($cmd) = @_;
1386 my @lines = ();
1387
1388 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
1389
1390 if (!$email_git_penguin_chiefs) {
1391 @lines = grep(!/${penguin_chiefs}/i, @lines);
1392 }
1393
1394 return @lines if !@lines;
1395
Joe Perches683c6f82010-10-26 14:22:55 -07001396 my @authors = ();
Joe Perches63ab52d2010-10-26 14:22:51 -07001397 foreach my $line (@lines) {
Joe Perches683c6f82010-10-26 14:22:55 -07001398 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
1399 my $author = $1;
1400 my ($name, $address) = parse_email($author);
1401 $author = format_email($name, $address, 1);
1402 push(@authors, $author);
1403 }
Joe Perches63ab52d2010-10-26 14:22:51 -07001404 }
1405
Joe Perches683c6f82010-10-26 14:22:55 -07001406 save_commits_by_author(@lines) if ($interactive);
1407 save_commits_by_signer(@lines) if ($interactive);
1408
1409 return @authors;
Joe Perches63ab52d2010-10-26 14:22:51 -07001410}
1411
Joe Perches60db31a2009-12-14 18:00:50 -08001412sub vcs_save_commits {
1413 my ($cmd) = @_;
1414 my @lines = ();
1415 my @commits = ();
1416
1417 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
1418
1419 foreach my $line (@lines) {
1420 if ($line =~ m/$VCS_cmds{"blame_commit_pattern"}/) {
1421 push(@commits, $1);
1422 }
1423 }
1424
1425 return @commits;
1426}
1427
1428sub vcs_blame {
1429 my ($file) = @_;
1430 my $cmd;
1431 my @commits = ();
1432
1433 return @commits if (!(-f $file));
1434
1435 if (@range && $VCS_cmds{"blame_range_cmd"} eq "") {
1436 my @all_commits = ();
1437
1438 $cmd = $VCS_cmds{"blame_file_cmd"};
1439 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1440 @all_commits = vcs_save_commits($cmd);
1441
1442 foreach my $file_range_diff (@range) {
1443 next if (!($file_range_diff =~ m/(.+):(.+):(.+)/));
1444 my $diff_file = $1;
1445 my $diff_start = $2;
1446 my $diff_length = $3;
1447 next if ("$file" ne "$diff_file");
1448 for (my $i = $diff_start; $i < $diff_start + $diff_length; $i++) {
1449 push(@commits, $all_commits[$i]);
1450 }
1451 }
1452 } elsif (@range) {
1453 foreach my $file_range_diff (@range) {
1454 next if (!($file_range_diff =~ m/(.+):(.+):(.+)/));
1455 my $diff_file = $1;
1456 my $diff_start = $2;
1457 my $diff_length = $3;
1458 next if ("$file" ne "$diff_file");
1459 $cmd = $VCS_cmds{"blame_range_cmd"};
1460 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1461 push(@commits, vcs_save_commits($cmd));
1462 }
1463 } else {
1464 $cmd = $VCS_cmds{"blame_file_cmd"};
1465 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1466 @commits = vcs_save_commits($cmd);
1467 }
1468
Joe Perches63ab52d2010-10-26 14:22:51 -07001469 foreach my $commit (@commits) {
1470 $commit =~ s/^\^//g;
1471 }
1472
Joe Perches60db31a2009-12-14 18:00:50 -08001473 return @commits;
1474}
1475
1476my $printed_novcs = 0;
1477sub vcs_exists {
1478 %VCS_cmds = %VCS_cmds_git;
1479 return 1 if eval $VCS_cmds{"available"};
1480 %VCS_cmds = %VCS_cmds_hg;
Joe Perches683c6f82010-10-26 14:22:55 -07001481 return 2 if eval $VCS_cmds{"available"};
Joe Perches60db31a2009-12-14 18:00:50 -08001482 %VCS_cmds = ();
1483 if (!$printed_novcs) {
1484 warn("$P: No supported VCS found. Add --nogit to options?\n");
1485 warn("Using a git repository produces better results.\n");
1486 warn("Try Linus Torvalds' latest git repository using:\n");
Ralf Thielow3d1c2f72011-08-25 15:59:07 -07001487 warn("git clone git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git\n");
Joe Perches60db31a2009-12-14 18:00:50 -08001488 $printed_novcs = 1;
1489 }
1490 return 0;
1491}
1492
Joe Perches683c6f82010-10-26 14:22:55 -07001493sub vcs_is_git {
Joe Perchesb9e23312010-10-26 14:22:58 -07001494 vcs_exists();
Joe Perches683c6f82010-10-26 14:22:55 -07001495 return $vcs_used == 1;
1496}
1497
1498sub vcs_is_hg {
1499 return $vcs_used == 2;
1500}
1501
Joe Perches6ef1c522010-10-26 14:22:56 -07001502sub interactive_get_maintainers {
Joe Perches683c6f82010-10-26 14:22:55 -07001503 my ($list_ref) = @_;
Florian Micklerdace8e32010-10-26 14:22:54 -07001504 my @list = @$list_ref;
1505
Joe Perches683c6f82010-10-26 14:22:55 -07001506 vcs_exists();
Florian Micklerdace8e32010-10-26 14:22:54 -07001507
1508 my %selected;
Joe Perches683c6f82010-10-26 14:22:55 -07001509 my %authored;
1510 my %signed;
Florian Micklerdace8e32010-10-26 14:22:54 -07001511 my $count = 0;
Joe Perches6ef1c522010-10-26 14:22:56 -07001512 my $maintained = 0;
Joe Perches6ef1c522010-10-26 14:22:56 -07001513 foreach my $entry (@list) {
Joe Perchesb9e23312010-10-26 14:22:58 -07001514 $maintained = 1 if ($entry->[1] =~ /^(maintainer|supporter)/i);
1515 $selected{$count} = 1;
Joe Perches683c6f82010-10-26 14:22:55 -07001516 $authored{$count} = 0;
1517 $signed{$count} = 0;
1518 $count++;
Florian Micklerdace8e32010-10-26 14:22:54 -07001519 }
1520
1521 #menu loop
Joe Perches683c6f82010-10-26 14:22:55 -07001522 my $done = 0;
1523 my $print_options = 0;
1524 my $redraw = 1;
1525 while (!$done) {
1526 $count = 0;
1527 if ($redraw) {
Joe Perches6ef1c522010-10-26 14:22:56 -07001528 printf STDERR "\n%1s %2s %-65s",
1529 "*", "#", "email/list and role:stats";
1530 if ($email_git ||
1531 ($email_git_fallback && !$maintained) ||
1532 $email_git_blame) {
1533 print STDERR "auth sign";
1534 }
1535 print STDERR "\n";
Joe Perches683c6f82010-10-26 14:22:55 -07001536 foreach my $entry (@list) {
1537 my $email = $entry->[0];
1538 my $role = $entry->[1];
1539 my $sel = "";
1540 $sel = "*" if ($selected{$count});
1541 my $commit_author = $commit_author_hash{$email};
1542 my $commit_signer = $commit_signer_hash{$email};
1543 my $authored = 0;
1544 my $signed = 0;
1545 $authored++ for (@{$commit_author});
1546 $signed++ for (@{$commit_signer});
1547 printf STDERR "%1s %2d %-65s", $sel, $count + 1, $email;
1548 printf STDERR "%4d %4d", $authored, $signed
1549 if ($authored > 0 || $signed > 0);
1550 printf STDERR "\n %s\n", $role;
1551 if ($authored{$count}) {
1552 my $commit_author = $commit_author_hash{$email};
1553 foreach my $ref (@{$commit_author}) {
1554 print STDERR " Author: @{$ref}[1]\n";
Florian Micklerdace8e32010-10-26 14:22:54 -07001555 }
Florian Micklerdace8e32010-10-26 14:22:54 -07001556 }
Joe Perches683c6f82010-10-26 14:22:55 -07001557 if ($signed{$count}) {
1558 my $commit_signer = $commit_signer_hash{$email};
1559 foreach my $ref (@{$commit_signer}) {
1560 print STDERR " @{$ref}[2]: @{$ref}[1]\n";
1561 }
1562 }
1563
1564 $count++;
Florian Micklerdace8e32010-10-26 14:22:54 -07001565 }
Florian Micklerdace8e32010-10-26 14:22:54 -07001566 }
Joe Perches683c6f82010-10-26 14:22:55 -07001567 my $date_ref = \$email_git_since;
1568 $date_ref = \$email_hg_since if (vcs_is_hg());
1569 if ($print_options) {
1570 $print_options = 0;
1571 if (vcs_exists()) {
Joe Perchesb9e23312010-10-26 14:22:58 -07001572 print STDERR <<EOT
1573
1574Version Control options:
1575g use git history [$email_git]
1576gf use git-fallback [$email_git_fallback]
1577b use git blame [$email_git_blame]
1578bs use blame signatures [$email_git_blame_signatures]
1579c# minimum commits [$email_git_min_signatures]
1580%# min percent [$email_git_min_percent]
1581d# history to use [$$date_ref]
1582x# max maintainers [$email_git_max_maintainers]
1583t all signature types [$email_git_all_signature_types]
1584m use .mailmap [$email_use_mailmap]
1585EOT
Joe Perches683c6f82010-10-26 14:22:55 -07001586 }
Joe Perchesb9e23312010-10-26 14:22:58 -07001587 print STDERR <<EOT
1588
1589Additional options:
15900 toggle all
1591tm toggle maintainers
1592tg toggle git entries
1593tl toggle open list entries
1594ts toggle subscriber list entries
1595f emails in file [$file_emails]
1596k keywords in file [$keywords]
1597r remove duplicates [$email_remove_duplicates]
1598p# pattern match depth [$pattern_depth]
1599EOT
Joe Perches683c6f82010-10-26 14:22:55 -07001600 }
1601 print STDERR
1602"\n#(toggle), A#(author), S#(signed) *(all), ^(none), O(options), Y(approve): ";
1603
1604 my $input = <STDIN>;
Florian Micklerdace8e32010-10-26 14:22:54 -07001605 chomp($input);
1606
Joe Perches683c6f82010-10-26 14:22:55 -07001607 $redraw = 1;
1608 my $rerun = 0;
1609 my @wish = split(/[, ]+/, $input);
1610 foreach my $nr (@wish) {
1611 $nr = lc($nr);
1612 my $sel = substr($nr, 0, 1);
1613 my $str = substr($nr, 1);
1614 my $val = 0;
1615 $val = $1 if $str =~ /^(\d+)$/;
1616
1617 if ($sel eq "y") {
1618 $interactive = 0;
1619 $done = 1;
1620 $output_rolestats = 0;
1621 $output_roles = 0;
1622 last;
1623 } elsif ($nr =~ /^\d+$/ && $nr > 0 && $nr <= $count) {
1624 $selected{$nr - 1} = !$selected{$nr - 1};
1625 } elsif ($sel eq "*" || $sel eq '^') {
1626 my $toggle = 0;
1627 $toggle = 1 if ($sel eq '*');
1628 for (my $i = 0; $i < $count; $i++) {
1629 $selected{$i} = $toggle;
Florian Micklerdace8e32010-10-26 14:22:54 -07001630 }
Joe Perches683c6f82010-10-26 14:22:55 -07001631 } elsif ($sel eq "0") {
1632 for (my $i = 0; $i < $count; $i++) {
1633 $selected{$i} = !$selected{$i};
1634 }
Joe Perchesb9e23312010-10-26 14:22:58 -07001635 } elsif ($sel eq "t") {
1636 if (lc($str) eq "m") {
1637 for (my $i = 0; $i < $count; $i++) {
1638 $selected{$i} = !$selected{$i}
1639 if ($list[$i]->[1] =~ /^(maintainer|supporter)/i);
1640 }
1641 } elsif (lc($str) eq "g") {
1642 for (my $i = 0; $i < $count; $i++) {
1643 $selected{$i} = !$selected{$i}
1644 if ($list[$i]->[1] =~ /^(author|commit|signer)/i);
1645 }
1646 } elsif (lc($str) eq "l") {
1647 for (my $i = 0; $i < $count; $i++) {
1648 $selected{$i} = !$selected{$i}
1649 if ($list[$i]->[1] =~ /^(open list)/i);
1650 }
1651 } elsif (lc($str) eq "s") {
1652 for (my $i = 0; $i < $count; $i++) {
1653 $selected{$i} = !$selected{$i}
1654 if ($list[$i]->[1] =~ /^(subscriber list)/i);
1655 }
1656 }
Joe Perches683c6f82010-10-26 14:22:55 -07001657 } elsif ($sel eq "a") {
1658 if ($val > 0 && $val <= $count) {
1659 $authored{$val - 1} = !$authored{$val - 1};
1660 } elsif ($str eq '*' || $str eq '^') {
1661 my $toggle = 0;
1662 $toggle = 1 if ($str eq '*');
1663 for (my $i = 0; $i < $count; $i++) {
1664 $authored{$i} = $toggle;
1665 }
1666 }
1667 } elsif ($sel eq "s") {
1668 if ($val > 0 && $val <= $count) {
1669 $signed{$val - 1} = !$signed{$val - 1};
1670 } elsif ($str eq '*' || $str eq '^') {
1671 my $toggle = 0;
1672 $toggle = 1 if ($str eq '*');
1673 for (my $i = 0; $i < $count; $i++) {
1674 $signed{$i} = $toggle;
1675 }
1676 }
1677 } elsif ($sel eq "o") {
1678 $print_options = 1;
1679 $redraw = 1;
1680 } elsif ($sel eq "g") {
1681 if ($str eq "f") {
1682 bool_invert(\$email_git_fallback);
Florian Micklerdace8e32010-10-26 14:22:54 -07001683 } else {
Joe Perches683c6f82010-10-26 14:22:55 -07001684 bool_invert(\$email_git);
Florian Micklerdace8e32010-10-26 14:22:54 -07001685 }
Joe Perches683c6f82010-10-26 14:22:55 -07001686 $rerun = 1;
1687 } elsif ($sel eq "b") {
1688 if ($str eq "s") {
1689 bool_invert(\$email_git_blame_signatures);
1690 } else {
1691 bool_invert(\$email_git_blame);
1692 }
1693 $rerun = 1;
1694 } elsif ($sel eq "c") {
1695 if ($val > 0) {
1696 $email_git_min_signatures = $val;
1697 $rerun = 1;
1698 }
1699 } elsif ($sel eq "x") {
1700 if ($val > 0) {
1701 $email_git_max_maintainers = $val;
1702 $rerun = 1;
1703 }
1704 } elsif ($sel eq "%") {
1705 if ($str ne "" && $val >= 0) {
1706 $email_git_min_percent = $val;
1707 $rerun = 1;
1708 }
1709 } elsif ($sel eq "d") {
1710 if (vcs_is_git()) {
1711 $email_git_since = $str;
1712 } elsif (vcs_is_hg()) {
1713 $email_hg_since = $str;
1714 }
1715 $rerun = 1;
1716 } elsif ($sel eq "t") {
1717 bool_invert(\$email_git_all_signature_types);
1718 $rerun = 1;
1719 } elsif ($sel eq "f") {
1720 bool_invert(\$file_emails);
1721 $rerun = 1;
1722 } elsif ($sel eq "r") {
1723 bool_invert(\$email_remove_duplicates);
1724 $rerun = 1;
Joe Perchesb9e23312010-10-26 14:22:58 -07001725 } elsif ($sel eq "m") {
1726 bool_invert(\$email_use_mailmap);
1727 read_mailmap();
1728 $rerun = 1;
Joe Perches683c6f82010-10-26 14:22:55 -07001729 } elsif ($sel eq "k") {
1730 bool_invert(\$keywords);
1731 $rerun = 1;
1732 } elsif ($sel eq "p") {
1733 if ($str ne "" && $val >= 0) {
1734 $pattern_depth = $val;
1735 $rerun = 1;
1736 }
Joe Perches6ef1c522010-10-26 14:22:56 -07001737 } elsif ($sel eq "h" || $sel eq "?") {
1738 print STDERR <<EOT
1739
1740Interactive mode allows you to select the various maintainers, submitters,
1741commit signers and mailing lists that could be CC'd on a patch.
1742
1743Any *'d entry is selected.
1744
Joe Perches47abc722010-10-26 14:22:57 -07001745If you have git or hg installed, you can choose to summarize the commit
Joe Perches6ef1c522010-10-26 14:22:56 -07001746history of files in the patch. Also, each line of the current file can
1747be matched to its commit author and that commits signers with blame.
1748
1749Various knobs exist to control the length of time for active commit
1750tracking, the maximum number of commit authors and signers to add,
1751and such.
1752
1753Enter selections at the prompt until you are satisfied that the selected
1754maintainers are appropriate. You may enter multiple selections separated
1755by either commas or spaces.
1756
1757EOT
Joe Perches683c6f82010-10-26 14:22:55 -07001758 } else {
1759 print STDERR "invalid option: '$nr'\n";
1760 $redraw = 0;
1761 }
1762 }
1763 if ($rerun) {
1764 print STDERR "git-blame can be very slow, please have patience..."
1765 if ($email_git_blame);
Joe Perches6ef1c522010-10-26 14:22:56 -07001766 goto &get_maintainers;
Joe Perches683c6f82010-10-26 14:22:55 -07001767 }
1768 }
Florian Micklerdace8e32010-10-26 14:22:54 -07001769
1770 #drop not selected entries
1771 $count = 0;
Joe Perches683c6f82010-10-26 14:22:55 -07001772 my @new_emailto = ();
1773 foreach my $entry (@list) {
1774 if ($selected{$count}) {
1775 push(@new_emailto, $list[$count]);
Florian Micklerdace8e32010-10-26 14:22:54 -07001776 }
1777 $count++;
1778 }
Joe Perches683c6f82010-10-26 14:22:55 -07001779 return @new_emailto;
Florian Micklerdace8e32010-10-26 14:22:54 -07001780}
1781
Joe Perches683c6f82010-10-26 14:22:55 -07001782sub bool_invert {
1783 my ($bool_ref) = @_;
Florian Micklerdace8e32010-10-26 14:22:54 -07001784
Joe Perches683c6f82010-10-26 14:22:55 -07001785 if ($$bool_ref) {
1786 $$bool_ref = 0;
1787 } else {
1788 $$bool_ref = 1;
Florian Micklerdace8e32010-10-26 14:22:54 -07001789 }
Florian Micklerdace8e32010-10-26 14:22:54 -07001790}
1791
Joe Perchesb9e23312010-10-26 14:22:58 -07001792sub deduplicate_email {
1793 my ($email) = @_;
1794
1795 my $matched = 0;
1796 my ($name, $address) = parse_email($email);
1797 $email = format_email($name, $address, 1);
1798 $email = mailmap_email($email);
1799
1800 return $email if (!$email_remove_duplicates);
1801
1802 ($name, $address) = parse_email($email);
1803
Joe Perchesfae99202010-10-26 14:22:58 -07001804 if ($name ne "" && $deduplicate_name_hash{lc($name)}) {
Joe Perchesb9e23312010-10-26 14:22:58 -07001805 $name = $deduplicate_name_hash{lc($name)}->[0];
1806 $address = $deduplicate_name_hash{lc($name)}->[1];
1807 $matched = 1;
1808 } elsif ($deduplicate_address_hash{lc($address)}) {
1809 $name = $deduplicate_address_hash{lc($address)}->[0];
1810 $address = $deduplicate_address_hash{lc($address)}->[1];
1811 $matched = 1;
1812 }
1813 if (!$matched) {
1814 $deduplicate_name_hash{lc($name)} = [ $name, $address ];
1815 $deduplicate_address_hash{lc($address)} = [ $name, $address ];
1816 }
1817 $email = format_email($name, $address, 1);
1818 $email = mailmap_email($email);
1819 return $email;
1820}
1821
Joe Perches683c6f82010-10-26 14:22:55 -07001822sub save_commits_by_author {
1823 my (@lines) = @_;
Florian Micklerdace8e32010-10-26 14:22:54 -07001824
Joe Perches683c6f82010-10-26 14:22:55 -07001825 my @authors = ();
1826 my @commits = ();
1827 my @subjects = ();
1828
1829 foreach my $line (@lines) {
1830 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
1831 my $author = $1;
Joe Perchesb9e23312010-10-26 14:22:58 -07001832 $author = deduplicate_email($author);
Joe Perches683c6f82010-10-26 14:22:55 -07001833 push(@authors, $author);
1834 }
1835 push(@commits, $1) if ($line =~ m/$VCS_cmds{"commit_pattern"}/);
1836 push(@subjects, $1) if ($line =~ m/$VCS_cmds{"subject_pattern"}/);
1837 }
1838
1839 for (my $i = 0; $i < @authors; $i++) {
1840 my $exists = 0;
1841 foreach my $ref(@{$commit_author_hash{$authors[$i]}}) {
1842 if (@{$ref}[0] eq $commits[$i] &&
1843 @{$ref}[1] eq $subjects[$i]) {
1844 $exists = 1;
1845 last;
1846 }
1847 }
1848 if (!$exists) {
1849 push(@{$commit_author_hash{$authors[$i]}},
1850 [ ($commits[$i], $subjects[$i]) ]);
1851 }
1852 }
1853}
1854
1855sub save_commits_by_signer {
1856 my (@lines) = @_;
1857
1858 my $commit = "";
1859 my $subject = "";
1860
1861 foreach my $line (@lines) {
1862 $commit = $1 if ($line =~ m/$VCS_cmds{"commit_pattern"}/);
1863 $subject = $1 if ($line =~ m/$VCS_cmds{"subject_pattern"}/);
1864 if ($line =~ /^[ \t]*${signature_pattern}.*\@.*$/) {
1865 my @signatures = ($line);
1866 my ($types_ref, $signers_ref) = extract_formatted_signatures(@signatures);
1867 my @types = @$types_ref;
1868 my @signers = @$signers_ref;
1869
1870 my $type = $types[0];
1871 my $signer = $signers[0];
1872
Joe Perchesb9e23312010-10-26 14:22:58 -07001873 $signer = deduplicate_email($signer);
Joe Perches6ef1c522010-10-26 14:22:56 -07001874
Joe Perches683c6f82010-10-26 14:22:55 -07001875 my $exists = 0;
1876 foreach my $ref(@{$commit_signer_hash{$signer}}) {
1877 if (@{$ref}[0] eq $commit &&
1878 @{$ref}[1] eq $subject &&
1879 @{$ref}[2] eq $type) {
1880 $exists = 1;
1881 last;
1882 }
1883 }
1884 if (!$exists) {
1885 push(@{$commit_signer_hash{$signer}},
1886 [ ($commit, $subject, $type) ]);
1887 }
1888 }
1889 }
Florian Micklerdace8e32010-10-26 14:22:54 -07001890}
1891
Joe Perches60db31a2009-12-14 18:00:50 -08001892sub vcs_assign {
Joe Perchesa8af2432009-12-14 18:00:49 -08001893 my ($role, $divisor, @lines) = @_;
1894
1895 my %hash;
1896 my $count = 0;
1897
Joe Perchesa8af2432009-12-14 18:00:49 -08001898 return if (@lines <= 0);
1899
1900 if ($divisor <= 0) {
Joe Perches60db31a2009-12-14 18:00:50 -08001901 warn("Bad divisor in " . (caller(0))[3] . ": $divisor\n");
Joe Perchesa8af2432009-12-14 18:00:49 -08001902 $divisor = 1;
Joe Perches3c7385b2009-12-14 18:00:46 -08001903 }
Joe Perches8cbb3a72009-09-21 17:04:21 -07001904
Florian Mickler7fa8ff22010-10-26 14:22:56 -07001905 @lines = mailmap(@lines);
Joe Perches0e70e832009-09-21 17:04:20 -07001906
Joe Perches63ab52d2010-10-26 14:22:51 -07001907 return if (@lines <= 0);
1908
Joe Perches0e70e832009-09-21 17:04:20 -07001909 @lines = sort(@lines);
Joe Perchesafa81ee2009-07-29 15:04:28 -07001910
Joe Perches11ecf532009-09-21 17:04:22 -07001911 # uniq -c
1912 $hash{$_}++ for @lines;
1913
1914 # sort -rn
1915 foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) {
1916 my $sign_offs = $hash{$line};
Joe Perchesa8af2432009-12-14 18:00:49 -08001917 my $percent = $sign_offs * 100 / $divisor;
Joe Perches3c7385b2009-12-14 18:00:46 -08001918
Joe Perchesa8af2432009-12-14 18:00:49 -08001919 $percent = 100 if ($percent > 100);
Joe Perches435de072015-06-25 15:01:50 -07001920 next if (ignore_email_address($line));
Joe Perches11ecf532009-09-21 17:04:22 -07001921 $count++;
1922 last if ($sign_offs < $email_git_min_signatures ||
1923 $count > $email_git_max_maintainers ||
Joe Perchesa8af2432009-12-14 18:00:49 -08001924 $percent < $email_git_min_percent);
Joe Perches3c7385b2009-12-14 18:00:46 -08001925 push_email_address($line, '');
Joe Perches3c7385b2009-12-14 18:00:46 -08001926 if ($output_rolestats) {
Joe Perchesa8af2432009-12-14 18:00:49 -08001927 my $fmt_percent = sprintf("%.0f", $percent);
1928 add_role($line, "$role:$sign_offs/$divisor=$fmt_percent%");
1929 } else {
1930 add_role($line, $role);
Joe Perches3c7385b2009-12-14 18:00:46 -08001931 }
Joe Perchesf5492662009-09-21 17:04:13 -07001932 }
1933}
1934
Joe Perches60db31a2009-12-14 18:00:50 -08001935sub vcs_file_signoffs {
Joe Perchesa8af2432009-12-14 18:00:49 -08001936 my ($file) = @_;
1937
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001938 my $authors_ref;
1939 my $signers_ref;
1940 my $stats_ref;
1941 my @authors = ();
Joe Perchesa8af2432009-12-14 18:00:49 -08001942 my @signers = ();
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001943 my @stats = ();
Joe Perches60db31a2009-12-14 18:00:50 -08001944 my $commits;
Joe Perchesa8af2432009-12-14 18:00:49 -08001945
Joe Perches683c6f82010-10-26 14:22:55 -07001946 $vcs_used = vcs_exists();
1947 return if (!$vcs_used);
Joe Perchesa8af2432009-12-14 18:00:49 -08001948
Joe Perches60db31a2009-12-14 18:00:50 -08001949 my $cmd = $VCS_cmds{"find_signers_cmd"};
1950 $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd
1951
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001952 ($commits, $signers_ref, $authors_ref, $stats_ref) = vcs_find_signers($cmd, $file);
1953
1954 @signers = @{$signers_ref} if defined $signers_ref;
1955 @authors = @{$authors_ref} if defined $authors_ref;
1956 @stats = @{$stats_ref} if defined $stats_ref;
1957
1958# print("commits: <$commits>\nsigners:<@signers>\nauthors: <@authors>\nstats: <@stats>\n");
Joe Perchesb9e23312010-10-26 14:22:58 -07001959
1960 foreach my $signer (@signers) {
1961 $signer = deduplicate_email($signer);
1962 }
1963
Joe Perches60db31a2009-12-14 18:00:50 -08001964 vcs_assign("commit_signer", $commits, @signers);
Joe Perchesc9ecefe2014-01-23 15:54:20 -08001965 vcs_assign("authored", $commits, @authors);
1966 if ($#authors == $#stats) {
1967 my $stat_pattern = $VCS_cmds{"stat_pattern"};
1968 $stat_pattern =~ s/(\$\w+)/$1/eeg; #interpolate $stat_pattern
1969
1970 my $added = 0;
1971 my $deleted = 0;
1972 for (my $i = 0; $i <= $#stats; $i++) {
1973 if ($stats[$i] =~ /$stat_pattern/) {
1974 $added += $1;
1975 $deleted += $2;
1976 }
1977 }
1978 my @tmp_authors = uniq(@authors);
1979 foreach my $author (@tmp_authors) {
1980 $author = deduplicate_email($author);
1981 }
1982 @tmp_authors = uniq(@tmp_authors);
1983 my @list_added = ();
1984 my @list_deleted = ();
1985 foreach my $author (@tmp_authors) {
1986 my $auth_added = 0;
1987 my $auth_deleted = 0;
1988 for (my $i = 0; $i <= $#stats; $i++) {
1989 if ($author eq deduplicate_email($authors[$i]) &&
1990 $stats[$i] =~ /$stat_pattern/) {
1991 $auth_added += $1;
1992 $auth_deleted += $2;
1993 }
1994 }
1995 for (my $i = 0; $i < $auth_added; $i++) {
1996 push(@list_added, $author);
1997 }
1998 for (my $i = 0; $i < $auth_deleted; $i++) {
1999 push(@list_deleted, $author);
2000 }
2001 }
2002 vcs_assign("added_lines", $added, @list_added);
2003 vcs_assign("removed_lines", $deleted, @list_deleted);
2004 }
Joe Perchesa8af2432009-12-14 18:00:49 -08002005}
2006
Joe Perches60db31a2009-12-14 18:00:50 -08002007sub vcs_file_blame {
Joe Perchesf5492662009-09-21 17:04:13 -07002008 my ($file) = @_;
2009
Joe Perches60db31a2009-12-14 18:00:50 -08002010 my @signers = ();
Joe Perches63ab52d2010-10-26 14:22:51 -07002011 my @all_commits = ();
Joe Perchesa8af2432009-12-14 18:00:49 -08002012 my @commits = ();
Joe Perchesa8af2432009-12-14 18:00:49 -08002013 my $total_commits;
Joe Perches63ab52d2010-10-26 14:22:51 -07002014 my $total_lines;
Joe Perchesf5492662009-09-21 17:04:13 -07002015
Joe Perches683c6f82010-10-26 14:22:55 -07002016 $vcs_used = vcs_exists();
2017 return if (!$vcs_used);
Joe Perchesf5492662009-09-21 17:04:13 -07002018
Joe Perches63ab52d2010-10-26 14:22:51 -07002019 @all_commits = vcs_blame($file);
2020 @commits = uniq(@all_commits);
Joe Perchesa8af2432009-12-14 18:00:49 -08002021 $total_commits = @commits;
Joe Perches63ab52d2010-10-26 14:22:51 -07002022 $total_lines = @all_commits;
Joe Perchesa8af2432009-12-14 18:00:49 -08002023
Joe Perches683c6f82010-10-26 14:22:55 -07002024 if ($email_git_blame_signatures) {
2025 if (vcs_is_hg()) {
2026 my $commit_count;
Joe Perchesc9ecefe2014-01-23 15:54:20 -08002027 my $commit_authors_ref;
2028 my $commit_signers_ref;
2029 my $stats_ref;
2030 my @commit_authors = ();
Joe Perches683c6f82010-10-26 14:22:55 -07002031 my @commit_signers = ();
2032 my $commit = join(" -r ", @commits);
2033 my $cmd;
Joe Perchesf5492662009-09-21 17:04:13 -07002034
Joe Perches683c6f82010-10-26 14:22:55 -07002035 $cmd = $VCS_cmds{"find_commit_signers_cmd"};
2036 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
Joe Perches60db31a2009-12-14 18:00:50 -08002037
Joe Perchesc9ecefe2014-01-23 15:54:20 -08002038 ($commit_count, $commit_signers_ref, $commit_authors_ref, $stats_ref) = vcs_find_signers($cmd, $file);
2039 @commit_authors = @{$commit_authors_ref} if defined $commit_authors_ref;
2040 @commit_signers = @{$commit_signers_ref} if defined $commit_signers_ref;
Joe Perches63ab52d2010-10-26 14:22:51 -07002041
Joe Perches683c6f82010-10-26 14:22:55 -07002042 push(@signers, @commit_signers);
2043 } else {
2044 foreach my $commit (@commits) {
2045 my $commit_count;
Joe Perchesc9ecefe2014-01-23 15:54:20 -08002046 my $commit_authors_ref;
2047 my $commit_signers_ref;
2048 my $stats_ref;
2049 my @commit_authors = ();
Joe Perches683c6f82010-10-26 14:22:55 -07002050 my @commit_signers = ();
2051 my $cmd;
2052
2053 $cmd = $VCS_cmds{"find_commit_signers_cmd"};
2054 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
2055
Joe Perchesc9ecefe2014-01-23 15:54:20 -08002056 ($commit_count, $commit_signers_ref, $commit_authors_ref, $stats_ref) = vcs_find_signers($cmd, $file);
2057 @commit_authors = @{$commit_authors_ref} if defined $commit_authors_ref;
2058 @commit_signers = @{$commit_signers_ref} if defined $commit_signers_ref;
Joe Perches683c6f82010-10-26 14:22:55 -07002059
2060 push(@signers, @commit_signers);
2061 }
2062 }
Joe Perchesf5492662009-09-21 17:04:13 -07002063 }
2064
Joe Perchesa8af2432009-12-14 18:00:49 -08002065 if ($from_filename) {
Joe Perches63ab52d2010-10-26 14:22:51 -07002066 if ($output_rolestats) {
2067 my @blame_signers;
Joe Perches683c6f82010-10-26 14:22:55 -07002068 if (vcs_is_hg()) {{ # Double brace for last exit
2069 my $commit_count;
2070 my @commit_signers = ();
2071 @commits = uniq(@commits);
2072 @commits = sort(@commits);
2073 my $commit = join(" -r ", @commits);
2074 my $cmd;
2075
2076 $cmd = $VCS_cmds{"find_commit_author_cmd"};
2077 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
2078
2079 my @lines = ();
2080
2081 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
2082
2083 if (!$email_git_penguin_chiefs) {
2084 @lines = grep(!/${penguin_chiefs}/i, @lines);
2085 }
2086
2087 last if !@lines;
2088
2089 my @authors = ();
2090 foreach my $line (@lines) {
2091 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
2092 my $author = $1;
Joe Perchesb9e23312010-10-26 14:22:58 -07002093 $author = deduplicate_email($author);
2094 push(@authors, $author);
Joe Perches683c6f82010-10-26 14:22:55 -07002095 }
2096 }
2097
2098 save_commits_by_author(@lines) if ($interactive);
2099 save_commits_by_signer(@lines) if ($interactive);
2100
2101 push(@signers, @authors);
2102 }}
2103 else {
2104 foreach my $commit (@commits) {
2105 my $i;
2106 my $cmd = $VCS_cmds{"find_commit_author_cmd"};
2107 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
2108 my @author = vcs_find_author($cmd);
2109 next if !@author;
Joe Perchesb9e23312010-10-26 14:22:58 -07002110
2111 my $formatted_author = deduplicate_email($author[0]);
2112
Joe Perches683c6f82010-10-26 14:22:55 -07002113 my $count = grep(/$commit/, @all_commits);
2114 for ($i = 0; $i < $count ; $i++) {
Joe Perchesb9e23312010-10-26 14:22:58 -07002115 push(@blame_signers, $formatted_author);
Joe Perches683c6f82010-10-26 14:22:55 -07002116 }
Joe Perches63ab52d2010-10-26 14:22:51 -07002117 }
2118 }
2119 if (@blame_signers) {
2120 vcs_assign("authored lines", $total_lines, @blame_signers);
2121 }
2122 }
Joe Perchesb9e23312010-10-26 14:22:58 -07002123 foreach my $signer (@signers) {
2124 $signer = deduplicate_email($signer);
2125 }
Joe Perches60db31a2009-12-14 18:00:50 -08002126 vcs_assign("commits", $total_commits, @signers);
Joe Perchesa8af2432009-12-14 18:00:49 -08002127 } else {
Joe Perchesb9e23312010-10-26 14:22:58 -07002128 foreach my $signer (@signers) {
2129 $signer = deduplicate_email($signer);
2130 }
Joe Perches60db31a2009-12-14 18:00:50 -08002131 vcs_assign("modified commits", $total_commits, @signers);
Joe Perchesf5492662009-09-21 17:04:13 -07002132 }
Joe Perchescb7301c2009-04-07 20:40:12 -07002133}
2134
Joe Perches4cad35a2016-08-02 14:04:10 -07002135sub vcs_file_exists {
2136 my ($file) = @_;
2137
2138 my $exists;
2139
2140 my $vcs_used = vcs_exists();
2141 return 0 if (!$vcs_used);
2142
2143 my $cmd = $VCS_cmds{"file_exists_cmd"};
2144 $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd
Joe Perches8582fb52016-08-25 15:16:48 -07002145 $cmd .= " 2>&1";
Joe Perches4cad35a2016-08-02 14:04:10 -07002146 $exists = &{$VCS_cmds{"execute_cmd"}}($cmd);
2147
Joe Perches8582fb52016-08-25 15:16:48 -07002148 return 0 if ($? != 0);
2149
Joe Perches4cad35a2016-08-02 14:04:10 -07002150 return $exists;
2151}
2152
Joe Perchescb7301c2009-04-07 20:40:12 -07002153sub uniq {
Joe Perchesa8af2432009-12-14 18:00:49 -08002154 my (@parms) = @_;
Joe Perchescb7301c2009-04-07 20:40:12 -07002155
2156 my %saw;
2157 @parms = grep(!$saw{$_}++, @parms);
2158 return @parms;
2159}
2160
2161sub sort_and_uniq {
Joe Perchesa8af2432009-12-14 18:00:49 -08002162 my (@parms) = @_;
Joe Perchescb7301c2009-04-07 20:40:12 -07002163
2164 my %saw;
2165 @parms = sort @parms;
2166 @parms = grep(!$saw{$_}++, @parms);
2167 return @parms;
2168}
2169
Joe Perches03372db2010-03-05 13:43:00 -08002170sub clean_file_emails {
2171 my (@file_emails) = @_;
2172 my @fmt_emails = ();
2173
2174 foreach my $email (@file_emails) {
2175 $email =~ s/[\(\<\{]{0,1}([A-Za-z0-9_\.\+-]+\@[A-Za-z0-9\.-]+)[\)\>\}]{0,1}/\<$1\>/g;
2176 my ($name, $address) = parse_email($email);
2177 if ($name eq '"[,\.]"') {
2178 $name = "";
2179 }
2180
2181 my @nw = split(/[^A-Za-zÀ-ÿ\'\,\.\+-]/, $name);
2182 if (@nw > 2) {
2183 my $first = $nw[@nw - 3];
2184 my $middle = $nw[@nw - 2];
2185 my $last = $nw[@nw - 1];
2186
2187 if (((length($first) == 1 && $first =~ m/[A-Za-z]/) ||
2188 (length($first) == 2 && substr($first, -1) eq ".")) ||
2189 (length($middle) == 1 ||
2190 (length($middle) == 2 && substr($middle, -1) eq "."))) {
2191 $name = "$first $middle $last";
2192 } else {
2193 $name = "$middle $last";
2194 }
2195 }
2196
2197 if (substr($name, -1) =~ /[,\.]/) {
2198 $name = substr($name, 0, length($name) - 1);
2199 } elsif (substr($name, -2) =~ /[,\.]"/) {
2200 $name = substr($name, 0, length($name) - 2) . '"';
2201 }
2202
2203 if (substr($name, 0, 1) =~ /[,\.]/) {
2204 $name = substr($name, 1, length($name) - 1);
2205 } elsif (substr($name, 0, 2) =~ /"[,\.]/) {
2206 $name = '"' . substr($name, 2, length($name) - 2);
2207 }
2208
2209 my $fmt_email = format_email($name, $address, $email_usename);
2210 push(@fmt_emails, $fmt_email);
2211 }
2212 return @fmt_emails;
2213}
2214
Joe Perches3c7385b2009-12-14 18:00:46 -08002215sub merge_email {
2216 my @lines;
2217 my %saw;
2218
2219 for (@_) {
2220 my ($address, $role) = @$_;
2221 if (!$saw{$address}) {
2222 if ($output_roles) {
Joe Perches60db31a2009-12-14 18:00:50 -08002223 push(@lines, "$address ($role)");
Joe Perches3c7385b2009-12-14 18:00:46 -08002224 } else {
Joe Perches60db31a2009-12-14 18:00:50 -08002225 push(@lines, $address);
Joe Perches3c7385b2009-12-14 18:00:46 -08002226 }
2227 $saw{$address} = 1;
2228 }
2229 }
2230
2231 return @lines;
2232}
2233
Joe Perchescb7301c2009-04-07 20:40:12 -07002234sub output {
Joe Perchesa8af2432009-12-14 18:00:49 -08002235 my (@parms) = @_;
Joe Perchescb7301c2009-04-07 20:40:12 -07002236
2237 if ($output_multiline) {
2238 foreach my $line (@parms) {
2239 print("${line}\n");
2240 }
2241 } else {
2242 print(join($output_separator, @parms));
2243 print("\n");
2244 }
2245}
Joe Perches1b5e1cf2009-06-16 15:34:01 -07002246
2247my $rfc822re;
2248
2249sub make_rfc822re {
2250# Basic lexical tokens are specials, domain_literal, quoted_string, atom, and
2251# comment. We must allow for rfc822_lwsp (or comments) after each of these.
2252# This regexp will only work on addresses which have had comments stripped
2253# and replaced with rfc822_lwsp.
2254
2255 my $specials = '()<>@,;:\\\\".\\[\\]';
2256 my $controls = '\\000-\\037\\177';
2257
2258 my $dtext = "[^\\[\\]\\r\\\\]";
2259 my $domain_literal = "\\[(?:$dtext|\\\\.)*\\]$rfc822_lwsp*";
2260
2261 my $quoted_string = "\"(?:[^\\\"\\r\\\\]|\\\\.|$rfc822_lwsp)*\"$rfc822_lwsp*";
2262
2263# Use zero-width assertion to spot the limit of an atom. A simple
2264# $rfc822_lwsp* causes the regexp engine to hang occasionally.
2265 my $atom = "[^$specials $controls]+(?:$rfc822_lwsp+|\\Z|(?=[\\[\"$specials]))";
2266 my $word = "(?:$atom|$quoted_string)";
2267 my $localpart = "$word(?:\\.$rfc822_lwsp*$word)*";
2268
2269 my $sub_domain = "(?:$atom|$domain_literal)";
2270 my $domain = "$sub_domain(?:\\.$rfc822_lwsp*$sub_domain)*";
2271
2272 my $addr_spec = "$localpart\@$rfc822_lwsp*$domain";
2273
2274 my $phrase = "$word*";
2275 my $route = "(?:\@$domain(?:,\@$rfc822_lwsp*$domain)*:$rfc822_lwsp*)";
2276 my $route_addr = "\\<$rfc822_lwsp*$route?$addr_spec\\>$rfc822_lwsp*";
2277 my $mailbox = "(?:$addr_spec|$phrase$route_addr)";
2278
2279 my $group = "$phrase:$rfc822_lwsp*(?:$mailbox(?:,\\s*$mailbox)*)?;\\s*";
2280 my $address = "(?:$mailbox|$group)";
2281
2282 return "$rfc822_lwsp*$address";
2283}
2284
2285sub rfc822_strip_comments {
2286 my $s = shift;
2287# Recursively remove comments, and replace with a single space. The simpler
2288# regexps in the Email Addressing FAQ are imperfect - they will miss escaped
2289# chars in atoms, for example.
2290
2291 while ($s =~ s/^((?:[^"\\]|\\.)*
2292 (?:"(?:[^"\\]|\\.)*"(?:[^"\\]|\\.)*)*)
2293 \((?:[^()\\]|\\.)*\)/$1 /osx) {}
2294 return $s;
2295}
2296
2297# valid: returns true if the parameter is an RFC822 valid address
2298#
Stephen Hemminger22dd5b02010-03-05 13:43:06 -08002299sub rfc822_valid {
Joe Perches1b5e1cf2009-06-16 15:34:01 -07002300 my $s = rfc822_strip_comments(shift);
2301
2302 if (!$rfc822re) {
2303 $rfc822re = make_rfc822re();
2304 }
2305
2306 return $s =~ m/^$rfc822re$/so && $s =~ m/^$rfc822_char*$/;
2307}
2308
2309# validlist: In scalar context, returns true if the parameter is an RFC822
2310# valid list of addresses.
2311#
2312# In list context, returns an empty list on failure (an invalid
2313# address was found); otherwise a list whose first element is the
2314# number of addresses found and whose remaining elements are the
2315# addresses. This is needed to disambiguate failure (invalid)
2316# from success with no addresses found, because an empty string is
2317# a valid list.
2318
Stephen Hemminger22dd5b02010-03-05 13:43:06 -08002319sub rfc822_validlist {
Joe Perches1b5e1cf2009-06-16 15:34:01 -07002320 my $s = rfc822_strip_comments(shift);
2321
2322 if (!$rfc822re) {
2323 $rfc822re = make_rfc822re();
2324 }
2325 # * null list items are valid according to the RFC
2326 # * the '1' business is to aid in distinguishing failure from no results
2327
2328 my @r;
2329 if ($s =~ m/^(?:$rfc822re)?(?:,(?:$rfc822re)?)*$/so &&
2330 $s =~ m/^$rfc822_char*$/) {
Joe Perches5f2441e2009-06-16 15:34:02 -07002331 while ($s =~ m/(?:^|,$rfc822_lwsp*)($rfc822re)/gos) {
Joe Perches60db31a2009-12-14 18:00:50 -08002332 push(@r, $1);
Joe Perches1b5e1cf2009-06-16 15:34:01 -07002333 }
2334 return wantarray ? (scalar(@r), @r) : 1;
2335 }
Joe Perches60db31a2009-12-14 18:00:50 -08002336 return wantarray ? () : 0;
Joe Perches1b5e1cf2009-06-16 15:34:01 -07002337}