masqmail

annotate src/header.c @ 371:f122535c589e

Refactoring: early failure exit.
author markus schnalke <meillo@marmaro.de>
date Tue, 25 Oct 2011 13:51:43 +0200
parents 41958685480d
children 028bc124d744
rev   line source
meillo@367 1 /*
meillo@367 2 ** MasqMail
meillo@367 3 ** Copyright (C) 2000 Oliver Kurth
meillo@367 4 **
meillo@367 5 ** This program is free software; you can redistribute it and/or modify
meillo@367 6 ** it under the terms of the GNU General Public License as published by
meillo@367 7 ** the Free Software Foundation; either version 2 of the License, or
meillo@367 8 ** (at your option) any later version.
meillo@367 9 **
meillo@367 10 ** This program is distributed in the hope that it will be useful,
meillo@367 11 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
meillo@367 12 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
meillo@367 13 ** GNU General Public License for more details.
meillo@367 14 **
meillo@367 15 ** You should have received a copy of the GNU General Public License
meillo@367 16 ** along with this program; if not, write to the Free Software
meillo@367 17 ** Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
meillo@0 18 */
meillo@0 19 #include "masqmail.h"
meillo@0 20
meillo@10 21 header_name header_names[] = {
meillo@15 22 {"From", HEAD_FROM,},
meillo@15 23 {"Sender", HEAD_SENDER,},
meillo@15 24 {"To", HEAD_TO,},
meillo@15 25 {"Cc", HEAD_CC,},
meillo@15 26 {"Bcc", HEAD_BCC,},
meillo@15 27 {"Date", HEAD_DATE,},
meillo@15 28 {"Message-Id", HEAD_MESSAGE_ID,},
meillo@15 29 {"Reply-To", HEAD_REPLY_TO,},
meillo@15 30 {"Subject", HEAD_SUBJECT,},
meillo@15 31 {"Return-Path", HEAD_RETURN_PATH,},
meillo@15 32 {"Envelope-To", HEAD_ENVELOPE_TO,},
meillo@15 33 {"Received", HEAD_RECEIVED},
meillo@0 34 };
meillo@0 35
meillo@0 36 /* this was borrowed from exim and slightly changed */
meillo@10 37 gchar*
meillo@10 38 rec_timestamp()
meillo@0 39 {
meillo@10 40 static gchar buf[64];
meillo@10 41 int len;
meillo@0 42
meillo@10 43 time_t now = time(NULL);
meillo@10 44 struct tm *t = localtime(&now);
meillo@0 45
meillo@10 46 int diff_hour, diff_min;
meillo@10 47 struct tm local;
meillo@10 48 struct tm *gmt;
meillo@0 49
meillo@10 50 memcpy(&local, t, sizeof(struct tm));
meillo@10 51 gmt = gmtime(&now);
meillo@10 52 diff_min = 60 * (local.tm_hour - gmt->tm_hour) + local.tm_min - gmt->tm_min;
meillo@301 53 if (local.tm_year != gmt->tm_year) {
meillo@10 54 diff_min += (local.tm_year > gmt->tm_year) ? 1440 : -1440;
meillo@301 55 } else if (local.tm_yday != gmt->tm_yday) {
meillo@10 56 diff_min += (local.tm_yday > gmt->tm_yday) ? 1440 : -1440;
meillo@301 57 }
meillo@10 58 diff_hour = diff_min / 60;
meillo@10 59 diff_min = abs(diff_min - diff_hour * 60);
meillo@0 60
meillo@10 61 len = strftime(buf, sizeof(buf), "%a, ", &local);
meillo@10 62 g_snprintf(buf + len, sizeof(buf) - len, "%02d ", local.tm_mday);
meillo@10 63 len += strlen(buf + len);
meillo@10 64 len += strftime(buf + len, sizeof(buf) - len, "%b %Y %H:%M:%S", &local);
meillo@10 65 g_snprintf(buf + len, sizeof(buf) - len, " %+03d%02d", diff_hour, diff_min);
meillo@10 66
meillo@10 67 return buf;
meillo@0 68 }
meillo@0 69
meillo@367 70 /*
meillo@367 71 ** finds list of headers matching id
meillo@367 72 ** if id == HEAD_UNKNOWN and header == NULL finds all unknown headers
meillo@367 73 ** else finds all headers matching header
meillo@0 74 */
meillo@10 75 GList*
meillo@366 76 find_header(GList *hdr_list, header_id id, gchar *hdr_str)
meillo@0 77 {
meillo@10 78 GList *found_list = NULL;
meillo@10 79 GList *node;
meillo@0 80
meillo@301 81 if ((id != HEAD_UNKNOWN) || !hdr_str) {
meillo@10 82 foreach(hdr_list, node) {
meillo@10 83 header *hdr = (header *) (node->data);
meillo@301 84 if (hdr->id == id) {
meillo@10 85 found_list = g_list_append(found_list, hdr);
meillo@301 86 }
meillo@10 87 }
meillo@301 88 return found_list;
meillo@301 89 }
meillo@10 90
meillo@301 91 foreach(hdr_list, node) {
meillo@301 92 header *hdr = (header *) (node->data);
meillo@301 93 gchar buf[64], *q = buf, *p = hdr->header;
meillo@10 94
meillo@301 95 while (*p != ':' && q < buf+sizeof(buf)-1 && *p) {
meillo@301 96 *(q++) = *(p++);
meillo@301 97 }
meillo@301 98 *q = '\0';
meillo@301 99
meillo@301 100 if (strcasecmp(buf, hdr_str) == 0) {
meillo@301 101 found_list = g_list_append(found_list, hdr);
meillo@10 102 }
meillo@10 103 }
meillo@10 104 return found_list;
meillo@0 105 }
meillo@0 106
meillo@10 107 void
meillo@366 108 header_unfold(header *hdr)
meillo@0 109 {
meillo@302 110 char *src = hdr->header;
meillo@302 111 char *dest = src;
meillo@302 112 char *p;
meillo@0 113
meillo@302 114 p = strchr(src, '\n');
meillo@302 115 if (!p || !p[1]) {
meillo@302 116 /* no folded header */
meillo@302 117 return;
meillo@302 118 }
meillo@0 119
meillo@302 120 while (*src) {
meillo@302 121 if (*src == '\n') {
meillo@302 122 /* ignore */
meillo@302 123 src++;
meillo@302 124 } else {
meillo@302 125 /* copy */
meillo@302 126 *(dest++) = *(src++);
meillo@302 127 }
meillo@302 128 }
meillo@302 129 *(dest++) = '\n';
meillo@302 130 *(dest++) = '\0';
meillo@0 131 }
meillo@0 132
meillo@303 133 /*
meillo@367 134 ** fold the header at maxlen chars (newline excluded)
meillo@367 135 ** (We exclude the newline because the RFCs deal with it this way)
meillo@303 136 */
meillo@10 137 void
meillo@366 138 header_fold(header *hdr, unsigned int maxlen)
meillo@0 139 {
meillo@303 140 int len = strlen(hdr->header);
meillo@366 141 char *src = hdr->header;
meillo@366 142 char *dest;
meillo@366 143 char *tmp;
meillo@366 144 char *p;
meillo@301 145 int valueoffset;
meillo@301 146
meillo@303 147 if (len <= maxlen) {
meillo@301 148 /* we don't need to do anything */
meillo@301 149 return;
meillo@301 150 }
meillo@301 151
meillo@303 152 /* strip trailing whitespace */
meillo@303 153 for (p=src+len-1; *p==' '||*p=='\t'||*p=='\n'; p--) {
meillo@303 154 *p = '\0';
meillo@303 155 len--;
meillo@303 156 printf(" trailing whitespace\n");
meillo@303 157 }
meillo@303 158 printf("stripped len: %d\n", len);
meillo@303 159
meillo@367 160 /*
meillo@367 161 ** FIXME: would be nice to have a better size calculation
meillo@367 162 ** (the current size + what we insert as break, twice as often as
meillo@367 163 ** we have breaks in the optimal case)
meillo@367 164 */
meillo@303 165 tmp = malloc(len + 2 * (len/maxlen) * strlen("\n\t"));
meillo@303 166 dest = tmp;
meillo@303 167
meillo@303 168 /* the position in hdr->header where the value part start */
meillo@301 169 valueoffset = hdr->value - hdr->header;
meillo@301 170
meillo@303 171 while (strlen(src) > maxlen) {
meillo@303 172 int i, l;
meillo@303 173 char *pp;
meillo@0 174
meillo@303 175 for (pp=src+maxlen; pp>src; pp--) {
meillo@303 176 if (*pp==' ' || *pp=='\t' || *p=='\n') {
meillo@303 177 break;
meillo@303 178 }
meillo@303 179 }
meillo@303 180
meillo@303 181 if (src == pp) {
meillo@303 182 /* no potential break point was found within
meillo@303 183 maxlen so advance further until the next */
meillo@303 184 for (pp=src+maxlen; *pp; pp++) {
meillo@303 185 if (*pp==' ' || *pp=='\t' || *p=='\n') {
meillo@303 186 break;
meillo@303 187 }
meillo@303 188 }
meillo@303 189 }
meillo@303 190 if (!*pp) {
meillo@303 191 break;
meillo@303 192 }
meillo@0 193
meillo@303 194 memcpy(dest, src, pp-src);
meillo@303 195 dest += pp-src;
meillo@303 196 *(dest++) = '\n';
meillo@303 197 *(dest++) = '\t';
meillo@303 198 while (*pp == ' ' || *pp == '\t' || *p=='\n') {
meillo@303 199 pp++;
meillo@303 200 }
meillo@303 201 src = pp;
meillo@303 202 }
meillo@303 203 memcpy(dest, src, strlen(src));
meillo@303 204 dest += strlen(src);
meillo@303 205
meillo@303 206 if (*(dest-1) != '\n') {
meillo@303 207 *dest = '\n';
meillo@303 208 *(dest+1) = '\0';
meillo@301 209 }
meillo@0 210
meillo@303 211 hdr->header = tmp;
meillo@301 212 hdr->value = hdr->header + valueoffset;
meillo@0 213 }
meillo@0 214
meillo@10 215 header*
meillo@366 216 create_header(header_id id, gchar *fmt, ...)
meillo@0 217 {
meillo@10 218 gchar *p;
meillo@10 219 header *hdr;
meillo@10 220 va_list args;
meillo@10 221 va_start(args, fmt);
meillo@0 222
meillo@301 223 /* g_malloc() calls exit on failure */
meillo@301 224 hdr = g_malloc(sizeof(header));
meillo@0 225
meillo@301 226 hdr->id = id;
meillo@301 227 hdr->header = g_strdup_vprintf(fmt, args);
meillo@301 228 hdr->value = NULL;
meillo@0 229
meillo@367 230 /*
meillo@367 231 ** value shall point to the first non-whitespace char in the
meillo@367 232 ** value part of the header line (i.e. after the first colon)
meillo@367 233 */
meillo@301 234 p = strchr(hdr->header, ':');
meillo@301 235 if (p) {
meillo@301 236 p++;
meillo@301 237 while (*p == ' ' || *p == '\t' || *p == '\n') {
meillo@10 238 p++;
meillo@301 239 }
meillo@301 240 hdr->value = (*p) ? p : NULL;
meillo@10 241 }
meillo@0 242
meillo@301 243 DEBUG(3) debugf("create_header(): hdr: `%s'\n", hdr->header);
meillo@301 244 DEBUG(3) debugf("create_header(): val: `%s'\n", hdr->value);
meillo@301 245
meillo@10 246 va_end(args);
meillo@10 247 return hdr;
meillo@0 248 }
meillo@0 249
meillo@10 250 void
meillo@366 251 destroy_header(header *hdr)
meillo@0 252 {
meillo@10 253 if (hdr) {
meillo@301 254 if (hdr->header) {
meillo@10 255 g_free(hdr->header);
meillo@301 256 }
meillo@10 257 g_free(hdr);
meillo@10 258 }
meillo@0 259 }
meillo@0 260
meillo@10 261 header*
meillo@366 262 copy_header(header *hdr)
meillo@0 263 {
meillo@10 264 header *new_hdr = NULL;
meillo@0 265
meillo@10 266 if (hdr) {
meillo@301 267 new_hdr = g_malloc(sizeof(header));
meillo@301 268 new_hdr->id = hdr->id;
meillo@301 269 new_hdr->header = g_strdup(hdr->header);
meillo@301 270 new_hdr->value = new_hdr->header + (hdr->value - hdr->header);
meillo@10 271 }
meillo@10 272 return new_hdr;
meillo@0 273 }
meillo@0 274
meillo@10 275 header*
meillo@366 276 get_header(gchar *line)
meillo@0 277 {
meillo@10 278 gchar *p = line;
meillo@10 279 gchar buf[64], *q = buf;
meillo@10 280 gint i;
meillo@10 281 header *hdr;
meillo@0 282
meillo@301 283 while (*p && (*p != ':') && (q < buf+sizeof(buf)-1)) {
meillo@10 284 *(q++) = *(p++);
meillo@301 285 }
meillo@15 286 *q = '\0';
meillo@0 287
meillo@301 288 if (*p != ':') {
meillo@10 289 return NULL;
meillo@301 290 }
meillo@0 291
meillo@10 292 hdr = g_malloc(sizeof(header));
meillo@0 293
meillo@10 294 hdr->value = NULL;
meillo@10 295 p++;
meillo@0 296
meillo@301 297 while (*p && (*p == ' ' || *p == '\t')) {
meillo@10 298 p++;
meillo@301 299 }
meillo@10 300 hdr->value = p;
meillo@367 301 /*
meillo@367 302 ** Note: an empty value can also mean that it's only the first part
meillo@367 303 ** of a folded header line
meillo@367 304 */
meillo@0 305
meillo@10 306 for (i = 0; i < HEAD_NUM_IDS; i++) {
meillo@301 307 if (strcasecmp(header_names[i].header, buf) == 0) {
meillo@10 308 break;
meillo@301 309 }
meillo@10 310 }
meillo@10 311 hdr->id = (header_id) i;
meillo@10 312 hdr->header = g_strdup(line);
meillo@10 313 hdr->value = hdr->header + (hdr->value - line);
meillo@10 314
meillo@322 315 DEBUG(4) debugf("header: %d = %s[...]", hdr->id, hdr->header);
meillo@301 316 /* Note: This only outputs the first line if the header is folded */
meillo@10 317
meillo@10 318 return hdr;
meillo@0 319 }