-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgit-commit-as.pl
executable file
·121 lines (98 loc) · 3.37 KB
/
git-commit-as.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#!/usr/bin/perl
use v5.26;
our $VERSION = v0.1.1;
# Return codes:
# * 1: Missing or invalid argument or data
# * 2: External tool error
my $USAGE = <<~END;
Usage: git commit-as <as-user> <arguments>
git commit-as -h | --help
git commit-as --man | --manual
git commit-as --version
END
my $MAN = <<~END;
$USAGE
This script overrides the GIT_AUTHOR_* and GIT_COMMITTER_* environment variables
with git config values users.<as-user>.name and users.<as-user>.email.
For a user named John, you would add this to your git config:
```
[users "John"]
name = "John Doe"
email = "john.doe\@example.com"
```
And call it as so: git commit-as John <arguments>
Note that <as-user> is case-sensitive and can not start with a hyphen.
Dependencies: perl, sh, env, git
Licensed by Tilwa Qendov under The Artistic 2.0 license
END
# Check first argument
my $useras = shift @ARGV;
unless ($useras) {
say STDERR "Error: missing as-user";
print STDERR $USAGE;
exit 1;
}
# Handle help, manual, version, and unknown flags
# NB: $useras is the first argument, which can be a flag
if ($useras eq '--help' || $useras eq '-h') {
print $USAGE;
exit 0;
} elsif ($useras eq '--man' || $useras eq '--manual') {
print $MAN;
exit 0;
elsif ($useras eq '--version') {
say "git-commit-as v$VERSION";
exit 0;
} elsif ($useras =~ /^-/) {
say STDERR "Error: unknown flag '$useras'";
print STDERR $USAGE;
exit 1;
}
# Filter invalid characters
# NB: command line can't contain the null byte anyways
my $invalid = $useras =~ s/[\n\0]//g;
if ($invalid) {
say STDERR "Error: invalid as-user '$useras'. Newlines and null characters aren't allowed.";
exit 1;
}
# Quote for shell
my $useras_e = $useras =~ s/'/'\\''/gr;
# Query git config
my ($ret, $namemissing, $emailmissing);
my $name = qx[ git config --get users.'$useras_e'.name ];
chomp $name;
$ret = $? >> 8;
$namemissing = ($ret == 1);
if ($ret > 1) { say STDERR "git errored with return code $ret while querying config."; exit 2 }
my $email = qx[ git config --get users.'$useras_e'.email ];
chomp $email;
$ret = $? >> 8;
$emailmissing = ($ret == 1);
if ($ret > 1) { say STDERR "git errored with return code $ret while querying config."; exit 2 }
# Handle errors in a more user-friendly way
if ($namemissing && $emailmissing) {
# Check if other keys exist
if (qx[ git config --get-regexp users\\."$useras"\\..* ]) {
say STDERR qq(Found unexpected entries under section [users "$useras"] in git config. Did you misspell the keys?\nSee git commit-as --manual for more information.);
} else {
say STDERR qq(Missing entries under section [users "$useras"] in git config. Did you forget to add it?\nSee git commit-as --manual for more information.);
}
} elsif ($namemissing) {
say STDERR qq(Missing 'name' field under section [users "$useras"] in git config. Did you forget to add it?);
} elsif ($emailmissing) {
say STDERR qq(Missing 'email' field under section [users "$useras"] in git config. Did you forget to add it?);
}
if($namemissing || $emailmissing) {
exit 2;
}
say qq(Committing as "$name" <$email>.);
my @command = @ARGV;
# Add our command
unshift @command, "git", "commit";
# Add our environment variables
unshift @command, "GIT_AUTHOR_NAME=$name";
unshift @command, "GIT_AUTHOR_EMAIL=$email";
unshift @command, "GIT_COMMITTER_NAME=$name";
unshift @command, "GIT_COMMITTER_EMAIL=$email";
unshift @command, "env";
exec { $command[0] } @command;