1use crate::config::ConfigSet;
8use crate::error::Error as GustError;
9use crate::objects::ObjectKind;
10use crate::repo::Repository;
11use crate::rev_parse::resolve_revision;
12use std::collections::BTreeMap;
13use std::fs;
14use std::io::Read;
15use std::path::{Path, PathBuf};
16
17type Result<T> = std::result::Result<T, GustError>;
18
19#[derive(Debug, Clone)]
21pub struct MailmapEntry {
22 pub canonical_name: Option<String>,
24 pub canonical_email: Option<String>,
26 pub match_name: Option<String>,
28 pub match_email: String,
30}
31
32#[derive(Debug, Default, Clone)]
33struct MailmapInfo {
34 name: Option<String>,
35 email: Option<String>,
36}
37
38#[derive(Debug, Default, Clone)]
39struct MailmapBucket {
40 simple: MailmapInfo,
42 by_name: BTreeMap<String, MailmapInfo>,
44}
45
46#[derive(Debug, Default, Clone)]
48pub struct MailmapTable {
49 buckets: BTreeMap<String, MailmapBucket>,
51}
52
53impl MailmapTable {
54 #[must_use]
56 pub fn is_empty(&self) -> bool {
57 self.buckets.is_empty()
58 }
59
60 #[must_use]
64 pub fn map_user(&self, mut name: String, mut email: String) -> (String, String) {
65 let key = email.to_ascii_lowercase();
66 let Some(bucket) = self.buckets.get(&key) else {
67 return (name, email);
68 };
69
70 let info = if !bucket.by_name.is_empty() {
71 let nk = name.to_ascii_lowercase();
72 bucket.by_name.get(&nk).or_else(|| {
73 if bucket.simple.name.is_some() || bucket.simple.email.is_some() {
74 Some(&bucket.simple)
75 } else {
76 None
77 }
78 })
79 } else if bucket.simple.name.is_some() || bucket.simple.email.is_some() {
80 Some(&bucket.simple)
81 } else {
82 None
83 };
84
85 let Some(info) = info else {
86 return (name, email);
87 };
88 if info.name.is_none() && info.email.is_none() {
89 return (name, email);
90 }
91 if let Some(ref e) = info.email {
92 email.clone_from(e);
93 }
94 if let Some(ref n) = info.name {
95 name.clone_from(n);
96 }
97 (name, email)
98 }
99}
100
101fn ascii_lowercase_owned(s: &str) -> String {
102 s.chars().map(|c| c.to_ascii_lowercase()).collect()
103}
104
105fn add_mapping(
106 table: &mut MailmapTable,
107 new_name: Option<String>,
108 new_email: Option<String>,
109 old_name: Option<String>,
110 old_email: Option<String>,
111) {
112 let (old_email, new_email) = match (old_email, new_email) {
115 (None, Some(e)) => (e, None),
116 (Some(old), new) => (old, new),
117 (None, None) => return,
118 };
119
120 let key = ascii_lowercase_owned(&old_email);
121 let bucket = table.buckets.entry(key).or_default();
122
123 if let Some(old_n) = old_name {
124 let nk = ascii_lowercase_owned(&old_n);
125 let mut mi = MailmapInfo::default();
126 mi.name = new_name;
127 mi.email = new_email;
128 bucket.by_name.insert(nk, mi);
129 } else {
130 if let Some(n) = new_name {
131 bucket.simple.name = Some(n);
132 }
133 if let Some(e) = new_email {
134 bucket.simple.email = Some(e);
135 }
136 }
137}
138
139fn parse_name_and_email(
141 buffer: &str,
142 allow_empty_email: bool,
143) -> Option<(Option<String>, Option<String>, &str)> {
144 let left = buffer.find('<')?;
145 let rest = &buffer[left + 1..];
146 let right_rel = rest.find('>')?;
147 if !allow_empty_email && right_rel == 0 {
148 return None;
149 }
150 let email = rest[..right_rel].to_string();
153 let right = left + 1 + right_rel;
154 let name_part = buffer[..left].trim_end_matches(|c: char| c.is_ascii_whitespace());
155 let name = if name_part.is_empty() {
156 None
157 } else {
158 Some(name_part.to_string())
159 };
160 let after = buffer.get(right + 1..).unwrap_or("");
161 Some((name, Some(email), after))
162}
163
164fn read_mailmap_line_into(table: &mut MailmapTable, line: &str) {
165 let line = line.trim_end_matches(['\r', '\n']);
166 let line = line.trim_start();
167 if line.is_empty() || line.starts_with('#') {
168 return;
169 }
170
171 let (name1, email1, rest1) = match parse_name_and_email(line, false) {
175 Some(x) => x,
176 None => return,
177 };
178
179 let (name2, email2) = if rest1.trim().is_empty() {
180 (None, None)
181 } else {
182 match parse_name_and_email(rest1.trim_start(), true) {
183 Some((n, e, tail)) if tail.trim().is_empty() => (n, e),
184 _ => return,
185 }
186 };
187
188 add_mapping(table, name1, email1, name2, email2);
189}
190
191pub fn read_mailmap_string(table: &mut MailmapTable, buf: &str) {
193 let mut start = 0usize;
194 for (i, ch) in buf.char_indices() {
195 if ch == '\n' {
196 read_mailmap_line_into(table, &buf[start..i]);
197 start = i + 1;
198 }
199 }
200 if start < buf.len() {
201 read_mailmap_line_into(table, &buf[start..]);
202 }
203}
204
205#[must_use]
207pub fn table_from_entries(entries: &[MailmapEntry]) -> MailmapTable {
208 let mut table = MailmapTable::default();
209 for e in entries {
210 add_mapping(
211 &mut table,
212 e.canonical_name.clone(),
213 e.canonical_email.clone(),
214 e.match_name.clone(),
215 Some(e.match_email.clone()),
216 );
217 }
218 table
219}
220
221#[must_use]
223pub fn parse_mailmap(content: &str) -> Vec<MailmapEntry> {
224 table_to_entries(&build_mailmap_table_from_str(content))
225}
226
227fn build_mailmap_table_from_str(content: &str) -> MailmapTable {
228 let mut table = MailmapTable::default();
229 read_mailmap_string(&mut table, content);
230 table
231}
232
233fn table_to_entries(table: &MailmapTable) -> Vec<MailmapEntry> {
234 let mut out = Vec::new();
235 for (email_lc, bucket) in &table.buckets {
236 if bucket.simple.name.is_some() || bucket.simple.email.is_some() {
237 out.push(MailmapEntry {
238 canonical_name: bucket.simple.name.clone(),
239 canonical_email: bucket.simple.email.clone(),
240 match_name: None,
241 match_email: email_lc.clone(),
242 });
243 }
244 for (name_lc, mi) in &bucket.by_name {
245 out.push(MailmapEntry {
246 canonical_name: mi.name.clone(),
247 canonical_email: mi.email.clone(),
248 match_name: Some(name_lc.clone()),
249 match_email: email_lc.clone(),
250 });
251 }
252 }
253 out
254}
255
256#[must_use]
258pub fn parse_contact(contact: &str) -> (Option<String>, Option<String>) {
259 let contact = contact.trim();
260 if let Some(lt) = contact.find('<') {
261 if let Some(gt) = contact.find('>') {
262 let name = contact[..lt].trim();
263 let email = contact[lt + 1..gt].trim();
264 return (
265 if name.is_empty() {
266 None
267 } else {
268 Some(name.to_string())
269 },
270 if email.is_empty() {
271 None
272 } else {
273 Some(email.to_string())
274 },
275 );
276 }
277 }
278 if contact.contains('@') && !contact.chars().any(char::is_whitespace) {
279 return (None, Some(contact.to_string()));
280 }
281
282 (Some(contact.to_string()), None)
283}
284
285#[must_use]
287pub fn map_contact(
288 name: Option<&str>,
289 email: Option<&str>,
290 mailmap: &[MailmapEntry],
291) -> (String, String) {
292 let mut table = MailmapTable::default();
293 for e in mailmap {
294 add_mapping(
295 &mut table,
296 e.canonical_name.clone(),
297 e.canonical_email.clone(),
298 e.match_name.clone(),
299 Some(e.match_email.clone()),
300 );
301 }
302 let n = name.unwrap_or("").to_string();
303 let e = email.unwrap_or("").to_string();
304 table.map_user(n, e)
305}
306
307#[must_use]
309pub fn map_contact_table(
310 name: Option<&str>,
311 email: Option<&str>,
312 table: &MailmapTable,
313) -> (String, String) {
314 let n = name.unwrap_or("").to_string();
315 let e = email.unwrap_or("").to_string();
316 table.map_user(n, e)
317}
318
319#[must_use]
321pub fn render_contact(name: &str, email: &str) -> String {
322 if email.is_empty() {
323 return name.to_string();
324 }
325 if name.is_empty() {
326 return format!("<{email}>");
327 }
328 format!("{name} <{email}>")
329}
330
331fn resolve_mailmap_path(base: &Path, value: &str) -> PathBuf {
332 let candidate = Path::new(value);
333 if candidate.is_absolute() {
334 candidate.to_path_buf()
335 } else {
336 base.join(candidate)
337 }
338}
339
340fn read_mailmap_file_nofollow(path: &Path) -> Result<String> {
341 #[cfg(unix)]
342 {
343 use std::os::unix::fs::OpenOptionsExt;
344
345 let mut file = fs::OpenOptions::new()
347 .read(true)
348 .custom_flags(libc::O_NOFOLLOW)
349 .open(path)
350 .map_err(|_| {
351 GustError::PathError(format!("unable to open mailmap at {}", path.display()))
352 })?;
353 let mut s = String::new();
354 file.read_to_string(&mut s)
355 .map_err(|e| GustError::PathError(format!("reading {}: {e}", path.display())))?;
356 Ok(s)
357 }
358 #[cfg(not(unix))]
359 {
360 fs::read_to_string(path)
361 .map_err(|e| GustError::PathError(format!("reading {}: {e}", path.display())))
362 }
363}
364
365fn read_optional_mailmap_file(path: &Path, nofollow: bool) -> Result<String> {
366 if !path.exists() {
367 return Ok(String::new());
368 }
369 if nofollow {
370 read_mailmap_file_nofollow(path)
371 } else {
372 fs::read_to_string(path)
373 .map_err(|e| GustError::PathError(format!("reading {}: {e}", path.display())))
374 }
375}
376
377pub fn read_mailmap_blob(repo: &Repository, spec: &str) -> Result<String> {
379 let oid = resolve_revision(repo, spec)
380 .map_err(|e| GustError::PathError(format!("resolving mailmap blob '{spec}': {e}")))?;
381 let obj = repo
382 .odb
383 .read(&oid)
384 .map_err(|e| GustError::PathError(format!("reading mailmap blob '{spec}': {e}")))?;
385 if obj.kind != ObjectKind::Blob {
386 return Err(GustError::PathError(format!(
387 "mailmap is not a blob: {spec}"
388 )));
389 }
390 Ok(String::from_utf8_lossy(&obj.data).into_owned())
391}
392
393fn try_read_mailmap_blob(repo: &Repository, spec: &str) -> Result<Option<String>> {
394 let oid = match resolve_revision(repo, spec) {
395 Ok(o) => o,
396 Err(_) => return Ok(None),
397 };
398 let obj = repo
399 .odb
400 .read(&oid)
401 .map_err(|e| GustError::PathError(format!("reading mailmap blob '{spec}': {e}")))?;
402 if obj.kind != ObjectKind::Blob {
403 return Err(GustError::PathError(format!(
404 "mailmap is not a blob: {spec}"
405 )));
406 }
407 Ok(Some(String::from_utf8_lossy(&obj.data).into_owned()))
408}
409
410pub fn load_mailmap_table(repo: &Repository) -> Result<MailmapTable> {
412 let mut table = MailmapTable::default();
413 load_mailmap_into(repo, &mut table)?;
414 Ok(table)
415}
416
417pub fn load_mailmap_into(repo: &Repository, table: &mut MailmapTable) -> Result<()> {
419 let config = ConfigSet::load(Some(&repo.git_dir), true)?;
420 let mut mailmap_blob = config.get("mailmap.blob");
421 let is_bare = repo.work_tree.is_none();
422 if mailmap_blob.is_none() && is_bare {
423 mailmap_blob = Some("HEAD:.mailmap".to_string());
424 }
425
426 let base_dir = repo
427 .work_tree
428 .as_deref()
429 .unwrap_or(repo.git_dir.as_path())
430 .to_path_buf();
431
432 if let Some(ref wt) = repo.work_tree {
433 let in_tree = wt.join(".mailmap");
434 let body = read_optional_mailmap_file(&in_tree, true)?;
435 read_mailmap_string(table, &body);
436 }
437
438 if let Some(ref blob) = mailmap_blob {
439 match try_read_mailmap_blob(repo, blob) {
440 Ok(Some(content)) => read_mailmap_string(table, &content),
441 Ok(None) => {}
442 Err(e) => {
443 let msg = e.to_string();
446 if msg.contains("mailmap is not a blob") {
447 eprintln!("{msg}");
448 } else {
449 return Err(e);
450 }
451 }
452 }
453 }
454
455 if let Some(file) = config.get("mailmap.file") {
456 read_mailmap_string(
457 table,
458 &read_optional_mailmap_file(&resolve_mailmap_path(&base_dir, &file), false)?,
459 );
460 }
461
462 Ok(())
463}
464
465pub fn load_mailmap_raw(repo: &Repository) -> Result<String> {
467 let config = ConfigSet::load(Some(&repo.git_dir), true)?;
468 let mut mailmap_blob = config.get("mailmap.blob");
469 let is_bare = repo.work_tree.is_none();
470 if mailmap_blob.is_none() && is_bare {
471 mailmap_blob = Some("HEAD:.mailmap".to_string());
472 }
473
474 let base_dir = repo
475 .work_tree
476 .as_deref()
477 .unwrap_or(repo.git_dir.as_path())
478 .to_path_buf();
479
480 let mut out = String::new();
481
482 if let Some(ref wt) = repo.work_tree {
483 let body = read_optional_mailmap_file(&wt.join(".mailmap"), true)?;
484 if !body.is_empty() {
485 out.push_str(&body);
486 if !out.ends_with('\n') {
487 out.push('\n');
488 }
489 }
490 }
491
492 if let Some(ref blob) = mailmap_blob {
493 match try_read_mailmap_blob(repo, blob) {
494 Ok(Some(content)) => {
495 if !content.is_empty() {
496 out.push_str(&content);
497 if !out.ends_with('\n') {
498 out.push('\n');
499 }
500 }
501 }
502 Ok(None) => {}
503 Err(e) => {
504 let msg = e.to_string();
505 if msg.contains("mailmap is not a blob") {
506 eprintln!("{msg}");
507 } else {
508 return Err(e);
509 }
510 }
511 }
512 }
513
514 if let Some(file) = config.get("mailmap.file") {
515 let body = read_optional_mailmap_file(&resolve_mailmap_path(&base_dir, &file), false)?;
516 if !body.is_empty() {
517 out.push_str(&body);
518 if !out.ends_with('\n') {
519 out.push('\n');
520 }
521 }
522 }
523
524 Ok(out)
525}
526
527pub fn load_mailmap(repo: &Repository) -> Result<Vec<MailmapEntry>> {
529 let table = load_mailmap_table(repo)?;
530 Ok(table_to_entries(&table))
531}
532
533#[must_use]
537pub fn apply_mailmap_to_commit_or_tag_bytes(data: &[u8], mailmap: &MailmapTable) -> Vec<u8> {
538 if mailmap.is_empty() {
539 return data.to_vec();
540 }
541 let Some(pos) = data.windows(2).position(|w| w == b"\n\n") else {
542 return data.to_vec();
543 };
544 let (headers, rest) = data.split_at(pos + 1);
545 let header_text = String::from_utf8_lossy(headers);
546 let mut out = String::with_capacity(data.len() + 64);
547 for line in header_text.lines() {
548 let rewritten = rewrite_identity_header_line(line, mailmap);
549 out.push_str(&rewritten);
550 out.push('\n');
551 }
552 out.push('\n');
553 out.push_str(&String::from_utf8_lossy(&rest[1..]));
554 out.into_bytes()
555}
556
557fn rewrite_identity_header_line(line: &str, mailmap: &MailmapTable) -> String {
558 for pref in ["author ", "committer ", "tagger "] {
559 if let Some(rest) = line.strip_prefix(pref) {
560 let rest = rest.trim_end_matches('\r');
561 let Some(gt) = rest.rfind('>') else {
562 return line.to_string();
563 };
564 let ident = &rest[..=gt];
565 let tail = rest[gt + 1..].trim_start();
566 let (name, email) = parse_contact(ident);
567 let (n, e) = map_contact_table(name.as_deref(), email.as_deref(), mailmap);
568 let new_ident = render_contact(&n, &e);
569 if tail.is_empty() {
570 return format!("{pref}{new_ident}");
571 }
572 return format!("{pref}{new_ident} {tail}");
573 }
574 }
575 line.to_string()
576}
577
578#[cfg(test)]
579mod tests {
580 use super::*;
581
582 #[test]
583 fn name_entry_after_email_merges() {
584 let mut t = MailmapTable::default();
585 read_mailmap_string(
586 &mut t,
587 "<bugs@company.xy> <bugs@company.xx>\nInternal Guy <bugs@company.xx>\n",
588 );
589 let (n, e) = t.map_user("nick1".into(), "bugs@company.xx".into());
590 assert_eq!(n, "Internal Guy");
591 assert_eq!(e, "bugs@company.xy");
592 }
593
594 #[test]
595 fn single_pair_line_maps_name_only() {
596 let mut t = MailmapTable::default();
597 read_mailmap_string(&mut t, "Committed <committer@example.com>\n");
598 let (n, e) = t.map_user("C O Mitter".into(), "committer@example.com".into());
599 assert_eq!(n, "Committed");
600 assert_eq!(e, "committer@example.com");
601 }
602
603 #[test]
604 fn whitespace_inside_angle_brackets_is_part_of_map_key() {
605 let mut t = MailmapTable::default();
606 read_mailmap_string(&mut t, "Ah <ah@example.com> < a@example.com >\n");
607 let (n, e) = t.map_user("A".into(), "a@example.com".into());
608 assert_eq!(n, "A");
609 assert_eq!(e, "a@example.com");
610 let (n2, e2) = t.map_user("A".into(), " a@example.com ".into());
611 assert_eq!(n2, "Ah");
612 assert_eq!(e2, "ah@example.com");
613 }
614}