masqmail

annotate src/header.c @ 10:26e34ae9a3e3

changed indention and line wrapping to a more consistent style
author meillo@marmaro.de
date Mon, 27 Oct 2008 16:23:10 +0100
parents 08114f7dcc23
children f671821d8222
rev   line source
meillo@0 1 /* MasqMail
meillo@0 2 Copyright (C) 2000 Oliver Kurth
meillo@0 3
meillo@0 4 This program is free software; you can redistribute it and/or modify
meillo@0 5 it under the terms of the GNU General Public License as published by
meillo@0 6 the Free Software Foundation; either version 2 of the License, or
meillo@0 7 (at your option) any later version.
meillo@0 8
meillo@0 9 This program is distributed in the hope that it will be useful,
meillo@0 10 but WITHOUT ANY WARRANTY; without even the implied warranty of
meillo@0 11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
meillo@0 12 GNU General Public License for more details.
meillo@0 13
meillo@0 14 You should have received a copy of the GNU General Public License
meillo@0 15 along with this program; if not, write to the Free Software
meillo@0 16 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
meillo@0 17 */
meillo@0 18 #include "masqmail.h"
meillo@0 19
meillo@10 20 header_name header_names[] = {
meillo@10 21 {"From", HEAD_FROM,}
meillo@10 22 ,
meillo@10 23 {"Sender", HEAD_SENDER,}
meillo@10 24 ,
meillo@10 25 {"To", HEAD_TO,}
meillo@10 26 ,
meillo@10 27 {"Cc", HEAD_CC,}
meillo@10 28 ,
meillo@10 29 {"Bcc", HEAD_BCC,}
meillo@10 30 ,
meillo@10 31 {"Date", HEAD_DATE,}
meillo@10 32 ,
meillo@10 33 {"Message-Id", HEAD_MESSAGE_ID,}
meillo@10 34 ,
meillo@10 35 {"Reply-To", HEAD_REPLY_TO,}
meillo@10 36 ,
meillo@10 37 {"Subject", HEAD_SUBJECT,}
meillo@10 38 ,
meillo@10 39 {"Return-Path", HEAD_RETURN_PATH,}
meillo@10 40 ,
meillo@10 41 {"Envelope-To", HEAD_ENVELOPE_TO,}
meillo@10 42 ,
meillo@10 43 {"Received", HEAD_RECEIVED}
meillo@10 44 ,
meillo@0 45 };
meillo@0 46
meillo@0 47 /* this was borrowed from exim and slightly changed */
meillo@10 48 gchar*
meillo@10 49 rec_timestamp()
meillo@0 50 {
meillo@10 51 static gchar buf[64];
meillo@10 52 int len;
meillo@0 53
meillo@10 54 time_t now = time(NULL);
meillo@10 55 struct tm *t = localtime(&now);
meillo@0 56
meillo@10 57 int diff_hour, diff_min;
meillo@10 58 struct tm local;
meillo@10 59 struct tm *gmt;
meillo@0 60
meillo@10 61 memcpy(&local, t, sizeof(struct tm));
meillo@10 62 gmt = gmtime(&now);
meillo@10 63 diff_min = 60 * (local.tm_hour - gmt->tm_hour) + local.tm_min - gmt->tm_min;
meillo@10 64 if (local.tm_year != gmt->tm_year)
meillo@10 65 diff_min += (local.tm_year > gmt->tm_year) ? 1440 : -1440;
meillo@10 66 else if (local.tm_yday != gmt->tm_yday)
meillo@10 67 diff_min += (local.tm_yday > gmt->tm_yday) ? 1440 : -1440;
meillo@10 68 diff_hour = diff_min / 60;
meillo@10 69 diff_min = abs(diff_min - diff_hour * 60);
meillo@0 70
meillo@10 71 len = strftime(buf, sizeof(buf), "%a, ", &local);
meillo@10 72 g_snprintf(buf + len, sizeof(buf) - len, "%02d ", local.tm_mday);
meillo@10 73 len += strlen(buf + len);
meillo@10 74 len += strftime(buf + len, sizeof(buf) - len, "%b %Y %H:%M:%S", &local);
meillo@10 75 g_snprintf(buf + len, sizeof(buf) - len, " %+03d%02d", diff_hour, diff_min);
meillo@10 76
meillo@10 77 return buf;
meillo@0 78 }
meillo@0 79
meillo@0 80 /* finds list of headers matching id
meillo@0 81 if id == HEAD_UNKNOWN and header == NULL finds all unknown headers
meillo@0 82 else finds all headers matching header
meillo@0 83 */
meillo@10 84 GList*
meillo@10 85 find_header(GList * hdr_list, header_id id, gchar * hdr_str)
meillo@0 86 {
meillo@10 87 GList *found_list = NULL;
meillo@10 88 GList *node;
meillo@0 89
meillo@10 90 if ((id != HEAD_UNKNOWN) || (hdr_str == NULL)) {
meillo@10 91 foreach(hdr_list, node) {
meillo@10 92 header *hdr = (header *) (node->data);
meillo@10 93 if (hdr->id == id)
meillo@10 94 found_list = g_list_append(found_list, hdr);
meillo@10 95 }
meillo@10 96 } else {
meillo@10 97 foreach(hdr_list, node) {
meillo@10 98 header *hdr = (header *) (node->data);
meillo@10 99 gchar buf[64], *q = buf, *p = hdr->header;
meillo@10 100
meillo@10 101 while (*p != ':' && q < buf + 63 && *p)
meillo@10 102 *(q++) = *(p++);
meillo@10 103 *q = 0;
meillo@10 104
meillo@10 105 if (strcasecmp(buf, hdr_str) == 0)
meillo@10 106 found_list = g_list_append(found_list, hdr);
meillo@10 107 }
meillo@10 108 }
meillo@10 109 return found_list;
meillo@0 110 }
meillo@0 111
meillo@10 112 void
meillo@10 113 header_unfold(header * hdr)
meillo@0 114 {
meillo@10 115 gchar *tmp_hdr = g_malloc(strlen(hdr->header));
meillo@10 116 gchar *p = hdr->header, *q = tmp_hdr;
meillo@10 117 gboolean flag = FALSE;
meillo@0 118
meillo@10 119 while (*p) {
meillo@10 120 if (*p != '\n')
meillo@10 121 *(q++) = *p;
meillo@10 122 else
meillo@10 123 flag = TRUE;
meillo@10 124 p++;
meillo@10 125 }
meillo@10 126 *(q++) = '\n';
meillo@0 127
meillo@10 128 if (flag) {
meillo@10 129 gchar *new_hdr;
meillo@0 130
meillo@10 131 g_free(hdr->header);
meillo@10 132 new_hdr = g_strdup(tmp_hdr);
meillo@10 133 g_free(tmp_hdr);
meillo@10 134 hdr->value = new_hdr + (hdr->value - hdr->header);
meillo@10 135 hdr->header = new_hdr;
meillo@10 136 }
meillo@0 137 }
meillo@0 138
meillo@0 139 #define MAX_HDR_LEN 72
meillo@10 140 void
meillo@10 141 header_fold(header * hdr)
meillo@0 142 {
meillo@10 143 gint len = strlen(hdr->header);
meillo@10 144 gchar *p, *q;
meillo@10 145 /* size is probably overestimated, but so we are on the safe side */
meillo@10 146 gchar *tmp_hdr = g_malloc(len + 2 * len / MAX_HDR_LEN);
meillo@0 147
meillo@10 148 p = hdr->header;
meillo@10 149 q = tmp_hdr;
meillo@0 150
meillo@10 151 if (p[len - 1] == '\n')
meillo@10 152 p[len - 1] = 0;
meillo@0 153
meillo@10 154 while (*p) {
meillo@10 155 gint i, l;
meillo@10 156 gchar *pp;
meillo@0 157
meillo@10 158 /* look forward and find potential break points */
meillo@10 159 i = 0;
meillo@10 160 l = -1;
meillo@10 161 pp = p;
meillo@10 162 while (*pp && (i < MAX_HDR_LEN)) {
meillo@10 163 if ((*pp == ' ') || (*pp == '\t'))
meillo@10 164 l = i;
meillo@10 165 pp++;
meillo@10 166 i++;
meillo@10 167 }
meillo@10 168 if (!*pp)
meillo@10 169 l = pp - p; /* take rest, if EOS found */
meillo@0 170
meillo@10 171 if (l == -1) {
meillo@10 172 /* no potential break point was found within MAX_HDR_LEN so advance further until the next */
meillo@10 173 while (*pp && *pp != ' ' && *pp != '\t') {
meillo@10 174 pp++;
meillo@10 175 i++;
meillo@10 176 }
meillo@10 177 l = i;
meillo@10 178 }
meillo@10 179
meillo@10 180 /* copy */
meillo@10 181 i = 0;
meillo@10 182 while (i < l) {
meillo@10 183 *(q++) = *(p++);
meillo@10 184 i++;
meillo@10 185 }
meillo@10 186 *(q++) = '\n';
meillo@10 187 *(q++) = *(p++); /* this is either space, tab or 0 */
meillo@10 188 }
meillo@10 189 {
meillo@10 190 gchar *new_hdr;
meillo@10 191
meillo@10 192 g_free(hdr->header);
meillo@10 193 new_hdr = g_strdup(tmp_hdr);
meillo@10 194 g_free(tmp_hdr);
meillo@10 195 hdr->value = new_hdr + (hdr->value - hdr->header);
meillo@10 196 hdr->header = new_hdr;
meillo@10 197 }
meillo@0 198 }
meillo@0 199
meillo@10 200 header*
meillo@10 201 create_header(header_id id, gchar * fmt, ...)
meillo@0 202 {
meillo@10 203 gchar *p;
meillo@10 204 header *hdr;
meillo@10 205 va_list args;
meillo@10 206 va_start(args, fmt);
meillo@0 207
meillo@10 208 if ((hdr = g_malloc(sizeof(header)))) {
meillo@0 209
meillo@10 210 hdr->id = id;
meillo@10 211 hdr->header = g_strdup_vprintf(fmt, args);
meillo@10 212 hdr->value = NULL;
meillo@0 213
meillo@10 214 p = hdr->header;
meillo@10 215 while (*p && *p != ':')
meillo@10 216 p++;
meillo@10 217 if (*p)
meillo@10 218 hdr->value = p + 1;
meillo@10 219 }
meillo@0 220
meillo@10 221 va_end(args);
meillo@10 222 return hdr;
meillo@0 223 }
meillo@0 224
meillo@10 225 void
meillo@10 226 destroy_header(header * hdr)
meillo@0 227 {
meillo@10 228 if (hdr) {
meillo@10 229 if (hdr->header)
meillo@10 230 g_free(hdr->header);
meillo@10 231 g_free(hdr);
meillo@10 232 }
meillo@0 233 }
meillo@0 234
meillo@10 235 header*
meillo@10 236 copy_header(header * hdr)
meillo@0 237 {
meillo@10 238 header *new_hdr = NULL;
meillo@0 239
meillo@10 240 if (hdr) {
meillo@10 241 if ((new_hdr = g_malloc(sizeof(header)))) {
meillo@10 242 new_hdr->id = hdr->id;
meillo@10 243 new_hdr->header = g_strdup(hdr->header);
meillo@10 244 new_hdr->value = new_hdr->header + (hdr->value - hdr->header);
meillo@10 245 }
meillo@10 246 }
meillo@10 247 return new_hdr;
meillo@0 248 }
meillo@0 249
meillo@10 250 header*
meillo@10 251 get_header(gchar * line)
meillo@0 252 {
meillo@10 253 gchar *p = line;
meillo@10 254 gchar buf[64], *q = buf;
meillo@10 255 gint i;
meillo@10 256 header *hdr;
meillo@0 257
meillo@10 258 while (*p && (*p != ':') && (q < buf + 63))
meillo@10 259 *(q++) = *(p++);
meillo@10 260 *q = 0;
meillo@0 261
meillo@10 262 if (*p != ':')
meillo@10 263 return NULL;
meillo@0 264
meillo@10 265 hdr = g_malloc(sizeof(header));
meillo@0 266
meillo@10 267 hdr->value = NULL;
meillo@10 268 p++;
meillo@0 269
meillo@10 270 while (*p && (*p == ' ' || *p == '\t'))
meillo@10 271 p++;
meillo@10 272 hdr->value = p;
meillo@0 273
meillo@10 274 for (i = 0; i < HEAD_NUM_IDS; i++) {
meillo@10 275 if (strcasecmp(header_names[i].header, buf) == 0)
meillo@10 276 break;
meillo@10 277 }
meillo@10 278 hdr->id = (header_id) i;
meillo@10 279 hdr->header = g_strdup(line);
meillo@10 280 hdr->value = hdr->header + (hdr->value - line);
meillo@10 281
meillo@10 282 DEBUG(4) debugf("header: %d = %s", hdr->id, hdr->header);
meillo@10 283
meillo@10 284 return hdr;
meillo@0 285 }