Line | Count | Source (jump to first uncovered line) |
1 | | #define USE_THE_REPOSITORY_VARIABLE |
2 | | |
3 | | #include "git-compat-util.h" |
4 | | #include "environment.h" |
5 | | #include "string-list.h" |
6 | | #include "mailmap.h" |
7 | | #include "object-name.h" |
8 | | #include "object-store-ll.h" |
9 | | #include "setup.h" |
10 | | |
11 | | char *git_mailmap_file; |
12 | | char *git_mailmap_blob; |
13 | | |
14 | | struct mailmap_info { |
15 | | char *name; |
16 | | char *email; |
17 | | }; |
18 | | |
19 | | struct mailmap_entry { |
20 | | /* name and email for the simple mail-only case */ |
21 | | char *name; |
22 | | char *email; |
23 | | |
24 | | /* name and email for the complex mail and name matching case */ |
25 | | struct string_list namemap; |
26 | | }; |
27 | | |
28 | | static void free_mailmap_info(void *p, const char *s UNUSED) |
29 | 0 | { |
30 | 0 | struct mailmap_info *mi = (struct mailmap_info *)p; |
31 | 0 | free(mi->name); |
32 | 0 | free(mi->email); |
33 | 0 | free(mi); |
34 | 0 | } |
35 | | |
36 | | static void free_mailmap_entry(void *p, const char *s UNUSED) |
37 | 0 | { |
38 | 0 | struct mailmap_entry *me = (struct mailmap_entry *)p; |
39 | |
|
40 | 0 | free(me->name); |
41 | 0 | free(me->email); |
42 | |
|
43 | 0 | me->namemap.strdup_strings = 1; |
44 | 0 | string_list_clear_func(&me->namemap, free_mailmap_info); |
45 | 0 | free(me); |
46 | 0 | } |
47 | | |
48 | | /* |
49 | | * On some systems (e.g. MinGW 4.0), string.h has _only_ inline |
50 | | * definition of strcasecmp and no non-inline implementation is |
51 | | * supplied anywhere, which is, eh, "unusual"; we cannot take an |
52 | | * address of such a function to store it in namemap.cmp. This is |
53 | | * here as a workaround---do not assign strcasecmp directly to |
54 | | * namemap.cmp until we know no systems that matter have such an |
55 | | * "unusual" string.h. |
56 | | */ |
57 | | static int namemap_cmp(const char *a, const char *b) |
58 | 0 | { |
59 | 0 | return strcasecmp(a, b); |
60 | 0 | } |
61 | | |
62 | | static void add_mapping(struct string_list *map, |
63 | | char *new_name, char *new_email, |
64 | | char *old_name, char *old_email) |
65 | 0 | { |
66 | 0 | struct mailmap_entry *me; |
67 | 0 | struct string_list_item *item; |
68 | |
|
69 | 0 | if (!old_email) { |
70 | 0 | old_email = new_email; |
71 | 0 | new_email = NULL; |
72 | 0 | } |
73 | |
|
74 | 0 | item = string_list_insert(map, old_email); |
75 | 0 | if (item->util) { |
76 | 0 | me = (struct mailmap_entry *)item->util; |
77 | 0 | } else { |
78 | 0 | CALLOC_ARRAY(me, 1); |
79 | 0 | me->namemap.strdup_strings = 1; |
80 | 0 | me->namemap.cmp = namemap_cmp; |
81 | 0 | item->util = me; |
82 | 0 | } |
83 | |
|
84 | 0 | if (!old_name) { |
85 | | /* Replace current name and new email for simple entry */ |
86 | 0 | if (new_name) { |
87 | 0 | free(me->name); |
88 | 0 | me->name = xstrdup(new_name); |
89 | 0 | } |
90 | 0 | if (new_email) { |
91 | 0 | free(me->email); |
92 | 0 | me->email = xstrdup(new_email); |
93 | 0 | } |
94 | 0 | } else { |
95 | 0 | struct mailmap_info *mi = xcalloc(1, sizeof(struct mailmap_info)); |
96 | 0 | mi->name = xstrdup_or_null(new_name); |
97 | 0 | mi->email = xstrdup_or_null(new_email); |
98 | 0 | string_list_insert(&me->namemap, old_name)->util = mi; |
99 | 0 | } |
100 | 0 | } |
101 | | |
102 | | static char *parse_name_and_email(char *buffer, char **name, |
103 | | char **email, int allow_empty_email) |
104 | 0 | { |
105 | 0 | char *left, *right, *nstart, *nend; |
106 | 0 | *name = *email = NULL; |
107 | |
|
108 | 0 | if (!(left = strchr(buffer, '<'))) |
109 | 0 | return NULL; |
110 | 0 | if (!(right = strchr(left + 1, '>'))) |
111 | 0 | return NULL; |
112 | 0 | if (!allow_empty_email && (left+1 == right)) |
113 | 0 | return NULL; |
114 | | |
115 | | /* remove whitespace from beginning and end of name */ |
116 | 0 | nstart = buffer; |
117 | 0 | while (isspace(*nstart) && nstart < left) |
118 | 0 | ++nstart; |
119 | 0 | nend = left-1; |
120 | 0 | while (nend > nstart && isspace(*nend)) |
121 | 0 | --nend; |
122 | |
|
123 | 0 | *name = (nstart <= nend ? nstart : NULL); |
124 | 0 | *email = left+1; |
125 | 0 | *(nend+1) = '\0'; |
126 | 0 | *right++ = '\0'; |
127 | |
|
128 | 0 | return (*right == '\0' ? NULL : right); |
129 | 0 | } |
130 | | |
131 | | static void read_mailmap_line(struct string_list *map, char *buffer) |
132 | 0 | { |
133 | 0 | char *name1 = NULL, *email1 = NULL, *name2 = NULL, *email2 = NULL; |
134 | |
|
135 | 0 | if (buffer[0] == '#') |
136 | 0 | return; |
137 | | |
138 | 0 | if ((name2 = parse_name_and_email(buffer, &name1, &email1, 0))) |
139 | 0 | parse_name_and_email(name2, &name2, &email2, 1); |
140 | |
|
141 | 0 | if (email1) |
142 | 0 | add_mapping(map, name1, email1, name2, email2); |
143 | 0 | } |
144 | | |
145 | | int read_mailmap_file(struct string_list *map, const char *filename, |
146 | | unsigned flags) |
147 | 0 | { |
148 | 0 | char buffer[1024]; |
149 | 0 | FILE *f; |
150 | 0 | int fd; |
151 | |
|
152 | 0 | if (!filename) |
153 | 0 | return 0; |
154 | | |
155 | 0 | if (flags & MAILMAP_NOFOLLOW) |
156 | 0 | fd = open_nofollow(filename, O_RDONLY); |
157 | 0 | else |
158 | 0 | fd = open(filename, O_RDONLY); |
159 | |
|
160 | 0 | if (fd < 0) { |
161 | 0 | if (errno == ENOENT) |
162 | 0 | return 0; |
163 | 0 | return error_errno("unable to open mailmap at %s", filename); |
164 | 0 | } |
165 | 0 | f = xfdopen(fd, "r"); |
166 | |
|
167 | 0 | while (fgets(buffer, sizeof(buffer), f) != NULL) |
168 | 0 | read_mailmap_line(map, buffer); |
169 | 0 | fclose(f); |
170 | 0 | return 0; |
171 | 0 | } |
172 | | |
173 | | static void read_mailmap_string(struct string_list *map, char *buf) |
174 | 0 | { |
175 | 0 | while (*buf) { |
176 | 0 | char *end = strchrnul(buf, '\n'); |
177 | |
|
178 | 0 | if (*end) |
179 | 0 | *end++ = '\0'; |
180 | |
|
181 | 0 | read_mailmap_line(map, buf); |
182 | 0 | buf = end; |
183 | 0 | } |
184 | 0 | } |
185 | | |
186 | | int read_mailmap_blob(struct string_list *map, const char *name) |
187 | 0 | { |
188 | 0 | struct object_id oid; |
189 | 0 | char *buf; |
190 | 0 | unsigned long size; |
191 | 0 | enum object_type type; |
192 | |
|
193 | 0 | if (!name) |
194 | 0 | return 0; |
195 | 0 | if (repo_get_oid(the_repository, name, &oid) < 0) |
196 | 0 | return 0; |
197 | | |
198 | 0 | buf = repo_read_object_file(the_repository, &oid, &type, &size); |
199 | 0 | if (!buf) |
200 | 0 | return error("unable to read mailmap object at %s", name); |
201 | 0 | if (type != OBJ_BLOB) { |
202 | 0 | free(buf); |
203 | 0 | return error("mailmap is not a blob: %s", name); |
204 | 0 | } |
205 | | |
206 | 0 | read_mailmap_string(map, buf); |
207 | |
|
208 | 0 | free(buf); |
209 | 0 | return 0; |
210 | 0 | } |
211 | | |
212 | | int read_mailmap(struct string_list *map) |
213 | 0 | { |
214 | 0 | int err = 0; |
215 | |
|
216 | 0 | map->strdup_strings = 1; |
217 | 0 | map->cmp = namemap_cmp; |
218 | |
|
219 | 0 | if (!git_mailmap_blob && is_bare_repository()) |
220 | 0 | git_mailmap_blob = xstrdup("HEAD:.mailmap"); |
221 | |
|
222 | 0 | if (!startup_info->have_repository || !is_bare_repository()) |
223 | 0 | err |= read_mailmap_file(map, ".mailmap", |
224 | 0 | startup_info->have_repository ? |
225 | 0 | MAILMAP_NOFOLLOW : 0); |
226 | 0 | if (startup_info->have_repository) |
227 | 0 | err |= read_mailmap_blob(map, git_mailmap_blob); |
228 | 0 | err |= read_mailmap_file(map, git_mailmap_file, 0); |
229 | 0 | return err; |
230 | 0 | } |
231 | | |
232 | | void clear_mailmap(struct string_list *map) |
233 | 0 | { |
234 | 0 | map->strdup_strings = 1; |
235 | 0 | string_list_clear_func(map, free_mailmap_entry); |
236 | 0 | } |
237 | | |
238 | | /* |
239 | | * Look for an entry in map that match string[0:len]; string[len] |
240 | | * does not have to be NUL (but it could be). |
241 | | */ |
242 | | static struct string_list_item *lookup_prefix(struct string_list *map, |
243 | | const char *string, size_t len) |
244 | 0 | { |
245 | 0 | int i = string_list_find_insert_index(map, string, 1); |
246 | 0 | if (i < 0) { |
247 | | /* exact match */ |
248 | 0 | i = -1 - i; |
249 | 0 | if (!string[len]) |
250 | 0 | return &map->items[i]; |
251 | | /* |
252 | | * that map entry matches exactly to the string, including |
253 | | * the cruft at the end beyond "len". That is not a match |
254 | | * with string[0:len] that we are looking for. |
255 | | */ |
256 | 0 | } else if (!string[len]) { |
257 | | /* |
258 | | * asked with the whole string, and got nothing. No |
259 | | * matching entry can exist in the map. |
260 | | */ |
261 | 0 | return NULL; |
262 | 0 | } |
263 | | |
264 | | /* |
265 | | * i is at the exact match to an overlong key, or location the |
266 | | * overlong key would be inserted, which must come after the |
267 | | * real location of the key if one exists. |
268 | | */ |
269 | 0 | while (0 <= --i && i < map->nr) { |
270 | 0 | int cmp = strncasecmp(map->items[i].string, string, len); |
271 | 0 | if (cmp < 0) |
272 | | /* |
273 | | * "i" points at a key definitely below the prefix; |
274 | | * the map does not have string[0:len] in it. |
275 | | */ |
276 | 0 | break; |
277 | 0 | else if (!cmp && !map->items[i].string[len]) |
278 | | /* found it */ |
279 | 0 | return &map->items[i]; |
280 | | /* |
281 | | * otherwise, the string at "i" may be string[0:len] |
282 | | * followed by a string that sorts later than string[len:]; |
283 | | * keep trying. |
284 | | */ |
285 | 0 | } |
286 | 0 | return NULL; |
287 | 0 | } |
288 | | |
289 | | int map_user(struct string_list *map, |
290 | | const char **email, size_t *emaillen, |
291 | | const char **name, size_t *namelen) |
292 | 0 | { |
293 | 0 | struct string_list_item *item; |
294 | 0 | struct mailmap_entry *me; |
295 | |
|
296 | 0 | item = lookup_prefix(map, *email, *emaillen); |
297 | 0 | if (item) { |
298 | 0 | me = (struct mailmap_entry *)item->util; |
299 | 0 | if (me->namemap.nr) { |
300 | | /* |
301 | | * The item has multiple items, so we'll look up on |
302 | | * name too. If the name is not found, we choose the |
303 | | * simple entry. |
304 | | */ |
305 | 0 | struct string_list_item *subitem; |
306 | 0 | subitem = lookup_prefix(&me->namemap, *name, *namelen); |
307 | 0 | if (subitem) |
308 | 0 | item = subitem; |
309 | 0 | } |
310 | 0 | } |
311 | 0 | if (item) { |
312 | 0 | struct mailmap_info *mi = (struct mailmap_info *)item->util; |
313 | 0 | if (mi->name == NULL && mi->email == NULL) |
314 | 0 | return 0; |
315 | 0 | if (mi->email) { |
316 | 0 | *email = mi->email; |
317 | 0 | *emaillen = strlen(*email); |
318 | 0 | } |
319 | 0 | if (mi->name) { |
320 | 0 | *name = mi->name; |
321 | 0 | *namelen = strlen(*name); |
322 | 0 | } |
323 | 0 | return 1; |
324 | 0 | } |
325 | 0 | return 0; |
326 | 0 | } |