1/*
2 * ident.c
3 *
4 * create git identifier lines of the form "name <email> date"
5 *
6 * Copyright (C) 2005 Linus Torvalds
7 */
8#include "cache.h"
9
10#define MAX_GITNAME (1000)
11static char git_default_name[MAX_GITNAME];
12static char git_default_email[MAX_GITNAME];
13static char git_default_date[50];
14int user_ident_explicitly_given;
15
16#ifdef NO_GECOS_IN_PWENT
17#define get_gecos(ignored) "&"
18#else
19#define get_gecos(struct_passwd) ((struct_passwd)->pw_gecos)
20#endif
21
22static void copy_gecos(const struct passwd *w, char *name, size_t sz)
23{
24 char *src, *dst;
25 size_t len, nlen;
26
27 nlen = strlen(w->pw_name);
28
29 /* Traditionally GECOS field had office phone numbers etc, separated
30 * with commas. Also & stands for capitalized form of the login name.
31 */
32
33 for (len = 0, dst = name, src = get_gecos(w); len < sz; src++) {
34 int ch = *src;
35 if (ch != '&') {
36 *dst++ = ch;
37 if (ch == 0 || ch == ',')
38 break;
39 len++;
40 continue;
41 }
42 if (len + nlen < sz) {
43 /* Sorry, Mr. McDonald... */
44 *dst++ = toupper(*w->pw_name);
45 memcpy(dst, w->pw_name + 1, nlen - 1);
46 dst += nlen - 1;
47 len += nlen;
48 }
49 }
50 if (len < sz)
51 name[len] = 0;
52 else
53 die("Your parents must have hated you!");
54
55}
56
57static int add_mailname_host(char *buf, size_t len)
58{
59 FILE *mailname;
60
61 mailname = fopen("/etc/mailname", "r");
62 if (!mailname) {
63 if (errno != ENOENT)
64 warning("cannot open /etc/mailname: %s",
65 strerror(errno));
66 return -1;
67 }
68 if (!fgets(buf, len, mailname)) {
69 if (ferror(mailname))
70 warning("cannot read /etc/mailname: %s",
71 strerror(errno));
72 fclose(mailname);
73 return -1;
74 }
75 /* success! */
76 fclose(mailname);
77
78 len = strlen(buf);
79 if (len && buf[len-1] == '\n')
80 buf[len-1] = '\0';
81 return 0;
82}
83
84static void add_domainname(char *buf, size_t len)
85{
86 struct hostent *he;
87 size_t namelen;
88 const char *domainname;
89
90 if (gethostname(buf, len)) {
91 warning("cannot get host name: %s", strerror(errno));
92 strlcpy(buf, "(none)", len);
93 return;
94 }
95 namelen = strlen(buf);
96 if (memchr(buf, '.', namelen))
97 return;
98
99 he = gethostbyname(buf);
100 buf[namelen++] = '.';
101 buf += namelen;
102 len -= namelen;
103 if (he && (domainname = strchr(he->h_name, '.')))
104 strlcpy(buf, domainname + 1, len);
105 else
106 strlcpy(buf, "(none)", len);
107}
108
109static void copy_email(const struct passwd *pw)
110{
111 /*
112 * Make up a fake email address
113 * (name + '@' + hostname [+ '.' + domainname])
114 */
115 size_t len = strlen(pw->pw_name);
116 if (len > sizeof(git_default_email)/2)
117 die("Your sysadmin must hate you!");
118 memcpy(git_default_email, pw->pw_name, len);
119 git_default_email[len++] = '@';
120
121 if (!add_mailname_host(git_default_email + len,
122 sizeof(git_default_email) - len))
123 return; /* read from "/etc/mailname" (Debian) */
124 add_domainname(git_default_email + len,
125 sizeof(git_default_email) - len);
126}
127
128const char *ident_default_name(void)
129{
130 if (!git_default_name[0]) {
131 struct passwd *pw = getpwuid(getuid());
132 if (!pw)
133 die("You don't exist. Go away!");
134 copy_gecos(pw, git_default_name, sizeof(git_default_name));
135 }
136 return git_default_name;
137}
138
139const char *ident_default_email(void)
140{
141 if (!git_default_email[0]) {
142 const char *email = getenv("EMAIL");
143
144 if (email && email[0]) {
145 strlcpy(git_default_email, email,
146 sizeof(git_default_email));
147 user_ident_explicitly_given |= IDENT_MAIL_GIVEN;
148 } else {
149 struct passwd *pw = getpwuid(getuid());
150 if (!pw)
151 die("You don't exist. Go away!");
152 copy_email(pw);
153 }
154 }
155 return git_default_email;
156}
157
158const char *ident_default_date(void)
159{
160 if (!git_default_date[0])
161 datestamp(git_default_date, sizeof(git_default_date));
162 return git_default_date;
163}
164
165static int add_raw(char *buf, size_t size, int offset, const char *str)
166{
167 size_t len = strlen(str);
168 if (offset + len > size)
169 return size;
170 memcpy(buf + offset, str, len);
171 return offset + len;
172}
173
174static int crud(unsigned char c)
175{
176 return c <= 32 ||
177 c == '.' ||
178 c == ',' ||
179 c == ':' ||
180 c == ';' ||
181 c == '<' ||
182 c == '>' ||
183 c == '"' ||
184 c == '\\' ||
185 c == '\'';
186}
187
188/*
189 * Copy over a string to the destination, but avoid special
190 * characters ('\n', '<' and '>') and remove crud at the end
191 */
192static int copy(char *buf, size_t size, int offset, const char *src)
193{
194 size_t i, len;
195 unsigned char c;
196
197 /* Remove crud from the beginning.. */
198 while ((c = *src) != 0) {
199 if (!crud(c))
200 break;
201 src++;
202 }
203
204 /* Remove crud from the end.. */
205 len = strlen(src);
206 while (len > 0) {
207 c = src[len-1];
208 if (!crud(c))
209 break;
210 --len;
211 }
212
213 /*
214 * Copy the rest to the buffer, but avoid the special
215 * characters '\n' '<' and '>' that act as delimiters on
216 * an identification line
217 */
218 for (i = 0; i < len; i++) {
219 c = *src++;
220 switch (c) {
221 case '\n': case '<': case '>':
222 continue;
223 }
224 if (offset >= size)
225 return size;
226 buf[offset++] = c;
227 }
228 return offset;
229}
230
231/*
232 * Reverse of fmt_ident(); given an ident line, split the fields
233 * to allow the caller to parse it.
234 * Signal a success by returning 0, but date/tz fields of the result
235 * can still be NULL if the input line only has the name/email part
236 * (e.g. reading from a reflog entry).
237 */
238int split_ident_line(struct ident_split *split, const char *line, int len)
239{
240 const char *cp;
241 size_t span;
242 int status = -1;
243
244 memset(split, 0, sizeof(*split));
245
246 split->name_begin = line;
247 for (cp = line; *cp && cp < line + len; cp++)
248 if (*cp == '<') {
249 split->mail_begin = cp + 1;
250 break;
251 }
252 if (!split->mail_begin)
253 return status;
254
255 for (cp = split->mail_begin - 2; line < cp; cp--)
256 if (!isspace(*cp)) {
257 split->name_end = cp + 1;
258 break;
259 }
260 if (!split->name_end)
261 return status;
262
263 for (cp = split->mail_begin; cp < line + len; cp++)
264 if (*cp == '>') {
265 split->mail_end = cp;
266 break;
267 }
268 if (!split->mail_end)
269 return status;
270
271 for (cp = split->mail_end + 1; cp < line + len && isspace(*cp); cp++)
272 ;
273 if (line + len <= cp)
274 goto person_only;
275 split->date_begin = cp;
276 span = strspn(cp, "0123456789");
277 if (!span)
278 goto person_only;
279 split->date_end = split->date_begin + span;
280 for (cp = split->date_end; cp < line + len && isspace(*cp); cp++)
281 ;
282 if (line + len <= cp || (*cp != '+' && *cp != '-'))
283 goto person_only;
284 split->tz_begin = cp;
285 span = strspn(cp + 1, "0123456789");
286 if (!span)
287 goto person_only;
288 split->tz_end = split->tz_begin + 1 + span;
289 return 0;
290
291person_only:
292 split->date_begin = NULL;
293 split->date_end = NULL;
294 split->tz_begin = NULL;
295 split->tz_end = NULL;
296 return 0;
297}
298
299static const char *env_hint =
300"\n"
301"*** Please tell me who you are.\n"
302"\n"
303"Run\n"
304"\n"
305" git config --global user.email \"you@example.com\"\n"
306" git config --global user.name \"Your Name\"\n"
307"\n"
308"to set your account\'s default identity.\n"
309"Omit --global to set the identity only in this repository.\n"
310"\n";
311
312const char *fmt_ident(const char *name, const char *email,
313 const char *date_str, int flag)
314{
315 static char buffer[1000];
316 char date[50];
317 int i;
318 int error_on_no_name = (flag & IDENT_ERROR_ON_NO_NAME);
319 int name_addr_only = (flag & IDENT_NO_DATE);
320
321 if (!name)
322 name = ident_default_name();
323 if (!email)
324 email = ident_default_email();
325
326 if (!*name) {
327 struct passwd *pw;
328
329 if (error_on_no_name) {
330 if (name == git_default_name)
331 fputs(env_hint, stderr);
332 die("empty ident %s <%s> not allowed", name, email);
333 }
334 pw = getpwuid(getuid());
335 if (!pw)
336 die("You don't exist. Go away!");
337 name = pw->pw_name;
338 }
339
340 strcpy(date, ident_default_date());
341 if (!name_addr_only && date_str && date_str[0]) {
342 if (parse_date(date_str, date, sizeof(date)) < 0)
343 die("invalid date format: %s", date_str);
344 }
345
346 i = copy(buffer, sizeof(buffer), 0, name);
347 i = add_raw(buffer, sizeof(buffer), i, " <");
348 i = copy(buffer, sizeof(buffer), i, email);
349 if (!name_addr_only) {
350 i = add_raw(buffer, sizeof(buffer), i, "> ");
351 i = copy(buffer, sizeof(buffer), i, date);
352 } else {
353 i = add_raw(buffer, sizeof(buffer), i, ">");
354 }
355 if (i >= sizeof(buffer))
356 die("Impossibly long personal identifier");
357 buffer[i] = 0;
358 return buffer;
359}
360
361const char *fmt_name(const char *name, const char *email)
362{
363 return fmt_ident(name, email, NULL, IDENT_ERROR_ON_NO_NAME | IDENT_NO_DATE);
364}
365
366const char *git_author_info(int flag)
367{
368 return fmt_ident(getenv("GIT_AUTHOR_NAME"),
369 getenv("GIT_AUTHOR_EMAIL"),
370 getenv("GIT_AUTHOR_DATE"),
371 flag);
372}
373
374const char *git_committer_info(int flag)
375{
376 if (getenv("GIT_COMMITTER_NAME"))
377 user_ident_explicitly_given |= IDENT_NAME_GIVEN;
378 if (getenv("GIT_COMMITTER_EMAIL"))
379 user_ident_explicitly_given |= IDENT_MAIL_GIVEN;
380 return fmt_ident(getenv("GIT_COMMITTER_NAME"),
381 getenv("GIT_COMMITTER_EMAIL"),
382 getenv("GIT_COMMITTER_DATE"),
383 flag);
384}
385
386int user_ident_sufficiently_given(void)
387{
388#ifndef WINDOWS
389 return (user_ident_explicitly_given & IDENT_MAIL_GIVEN);
390#else
391 return (user_ident_explicitly_given == IDENT_ALL_GIVEN);
392#endif
393}
394
395int git_ident_config(const char *var, const char *value, void *data)
396{
397 if (!strcmp(var, "user.name")) {
398 if (!value)
399 return config_error_nonbool(var);
400 strlcpy(git_default_name, value, sizeof(git_default_name));
401 user_ident_explicitly_given |= IDENT_NAME_GIVEN;
402 return 0;
403 }
404
405 if (!strcmp(var, "user.email")) {
406 if (!value)
407 return config_error_nonbool(var);
408 strlcpy(git_default_email, value, sizeof(git_default_email));
409 user_ident_explicitly_given |= IDENT_MAIL_GIVEN;
410 return 0;
411 }
412
413 return 0;
414}