oopsie
[apps/madmutt.git] / from.c
1 /*
2  * Copyright notice from original mutt:
3  * Copyright (C) 1996-2000 Michael R. Elkins <me@mutt.org>
4  *
5  * This file is part of mutt-ng, see http://www.muttng.org/.
6  * It's licensed under the GNU General Public License,
7  * please see the file GPL in the top level source directory.
8  */
9
10 #if HAVE_CONFIG_H
11 # include "config.h"
12 #endif
13
14 #include <ctype.h>
15 #include <string.h>
16
17 #include <lib-lib/mem.h>
18 #include <lib-lib/str.h>
19 #include <lib-lib/macros.h>
20
21 #include "mutt.h"
22
23 int mutt_check_month (const char *s)
24 {
25   int i;
26
27   for (i = 0; i < 12; i++)
28     if (m_strncasecmp(s, Months[i], 3) == 0)
29       return (i);
30   return (-1);                  /* error */
31 }
32
33 static int is_day_name (const char *s)
34 {
35   int i;
36
37   if ((m_strlen(s) < 3) || !*(s + 3) || !ISSPACE (*(s + 3)))
38     return 0;
39   for (i = 0; i < 7; i++)
40     if (m_strncasecmp(s, Weekdays[i], 3) == 0)
41       return 1;
42   return 0;
43 }
44
45 /*
46  * A valid message separator looks like:
47  *
48  * From [ <return-path> ] <weekday> <month> <day> <time> [ <timezone> ] <year>
49  */
50
51 int is_from (const char *s, char *path, ssize_t pathlen, time_t * tp)
52 {
53   struct tm tm;
54   int yr;
55
56   if (path)
57     *path = 0;
58
59   if (m_strncmp("From ", s, 5) != 0)
60     return 0;
61
62   s = m_strnextsp(s);            /* skip over the From part. */
63   s = skipspaces(s);
64   if (!*s)
65     return 0;
66
67   if (!is_day_name (s)) {
68     const char *p;
69     ssize_t len;
70     short q = 0;
71
72     for (p = s; *p && (q || !ISSPACE (*p)); p++) {
73       if (*p == '\\') {
74         if (*++p == '\0')
75           return 0;
76       }
77       else if (*p == '"') {
78         q = !q;
79       }
80     }
81
82     if (q || !*p)
83       return 0;
84
85     if (path) {
86       len = p - s;
87       if (len + 1 > pathlen)
88         len = pathlen - 1;
89       memcpy (path, s, len);
90       path[len] = 0;
91     }
92
93     s = vskipspaces(p + 1);
94     if (!*s)
95       return 0;
96
97     if (!is_day_name (s)) {
98       return 0;
99     }
100   }
101
102   s = m_strnextsp(s);
103   s = skipspaces(s);
104   if (!*s)
105     return 0;
106
107   /* do a quick check to make sure that this isn't really the day of the week.
108    * this could happen when receiving mail from a local user whose login name
109    * is the same as a three-letter abbreviation of the day of the week.
110    */
111   if (is_day_name (s)) {
112     s = m_strnextsp(s);
113     s = skipspaces(s);
114     if (!*s)
115       return 0;
116   }
117
118   /* now we should be on the month. */
119   if ((tm.tm_mon = mutt_check_month (s)) < 0)
120     return 0;
121
122   /* day */
123   s = m_strnextsp(s);
124   s = skipspaces(s);
125   if (!*s)
126     return 0;
127   if (sscanf (s, "%d", &tm.tm_mday) != 1)
128     return 0;
129
130   /* time */
131   s = m_strnextsp(s);
132   s = skipspaces(s);
133   if (!*s)
134     return 0;
135
136   /* Accept either HH:MM or HH:MM:SS */
137   if (sscanf (s, "%d:%d:%d", &tm.tm_hour, &tm.tm_min, &tm.tm_sec) == 3);
138   else if (sscanf (s, "%d:%d", &tm.tm_hour, &tm.tm_min) == 2)
139     tm.tm_sec = 0;
140   else
141     return 0;
142
143   s = m_strnextsp(s);
144   s = skipspaces(s);
145   if (!*s)
146     return 0;
147
148   /* timezone? */
149   if (isalpha ((unsigned char) *s) || *s == '+' || *s == '-') {
150     s = m_strnextsp(s);
151     s = skipspaces(s);
152     if (!*s)
153       return 0;
154
155     /*
156      * some places have two timezone fields after the time, e.g.
157      *      From xxxx@yyyyyyy.fr Wed Aug  2 00:39:12 MET DST 1995
158      */
159     if (isalpha ((unsigned char) *s)) {
160       s = m_strnextsp(s);
161       s = skipspaces(s);
162       if (!*s)
163         return 0;
164     }
165   }
166
167   /* year */
168   if (sscanf (s, "%d", &yr) != 1)
169     return 0;
170   tm.tm_year = yr > 1900 ? yr - 1900 : (yr < 70 ? yr + 100 : yr);
171
172   tm.tm_isdst = -1;
173
174   if (tp)
175     *tp = mutt_mktime (&tm, 0);
176   return 1;
177 }