commit 9539ae1b7a52273c7ea7155b89792c64be2ac234 from: Stefan Sperling via: Thomas Adam date: Sun May 26 14:04:47 2024 UTC fix Date header generated by got-notify-email asctime_r() is not a source of valid date strings for email. Borrow code from OpenSMTPD instead for this purpose. Problem reported by naddy, who also found the smtpd code to use. commit - 8b8e9d495501b7322133eb2a18782683cd532950 commit + 9539ae1b7a52273c7ea7155b89792c64be2ac234 blob - 5ddab275c66e03bc5ba192903150a4c188f511ba blob + 2fe1cd472af58848cffb1c7c0163ebb3c88c08f3 --- gotd/libexec/got-notify-email/got-notify-email.c +++ gotd/libexec/got-notify-email/got-notify-email.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2024 Stefan Sperling + * Copyright (c) 2008, 2012 Gilles Chehade * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above @@ -198,24 +199,6 @@ send_smtp_msg(int s, const char *fmt, ...) return 0; } -static char * -get_datestr(time_t *time, char *datebuf) -{ - struct tm mytm, *tm; - char *p, *s; - - tm = gmtime_r(time, &mytm); - if (tm == NULL) - return NULL; - s = asctime_r(tm, datebuf); - if (s == NULL) - return NULL; - p = strchr(s, '\n'); - if (p) - *p = '\0'; - return s; -} - static const struct got_error * print_date(int s, char *date, int shortfmt) { @@ -247,6 +230,55 @@ print_date(int s, char *date, int shortfmt) if (strftime(datebuf, sizeof(datebuf), "%F ", &tm) == 0) return got_error_set_errno(EINVAL, "invalid timestamp"); return got_poll_write_full(s, datebuf, strlen(datebuf)); +} + +/* from usr.sbin/smtpd/util.c */ +static int +bsnprintf(char *str, size_t size, const char *format, ...) +{ + int ret; + va_list ap; + + va_start(ap, format); + ret = vsnprintf(str, size, format, ap); + va_end(ap); + if (ret < 0 || (size_t)ret >= size) + return 0; + + return 1; +} + +/* based on usr.sbin/smtpd/to.c */ +static const char * +time_to_text(time_t when, char *buf, size_t buf_size) +{ + struct tm *lt; + const char *day[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; + const char *month[] = {"Jan","Feb","Mar","Apr","May","Jun", + "Jul","Aug","Sep","Oct","Nov","Dec"}; + const char *tz; + long offset; + + lt = gmtime(&when); + if (lt == NULL || when == 0) + fatalx("time_to_text: localtime"); + + offset = lt->tm_gmtoff; + tz = lt->tm_zone; + + /* We do not use strftime because it is subject to locale substitution*/ + if (!bsnprintf(buf, buf_size, + "%s, %d %s %d %02d:%02d:%02d %c%02d%02d (%s)", + day[lt->tm_wday], lt->tm_mday, month[lt->tm_mon], + lt->tm_year + 1900, + lt->tm_hour, lt->tm_min, lt->tm_sec, + offset >= 0 ? '+' : '-', + abs((int)offset / 3600), + abs((int)offset % 3600) / 60, + tz)) + fatalx("time_to_text: bsnprintf"); + + return buf; } static void @@ -258,13 +290,11 @@ send_email(int s, const char *myfromaddr, const char * char *line = NULL; size_t linesize = 0; ssize_t linelen; - time_t now; int firstline = 1, shortfmt = 0; - char datebuf[26]; - char *datestr; + char datebuf[40]; + const char *datestr; - now = time(NULL); - datestr = get_datestr(&now, datebuf); + datestr = time_to_text(time(NULL), datebuf, sizeof(datebuf)); if (read_smtp_code(s, "220")) fatalx("unexpected SMTP greeting received"); @@ -297,7 +327,7 @@ send_email(int s, const char *myfromaddr, const char * if (send_smtp_msg(s, "Reply-To: %s\r\n", replytoaddr)) fatalx("could not send Reply-To header"); } - if (send_smtp_msg(s, "Date: %s +0000 (UTC)\r\n", datestr)) + if (send_smtp_msg(s, "Date: %s\r\n", datestr)) fatalx("could not send Date header"); if (send_smtp_msg(s, "Subject: %s\r\n", subject))