Patrick Welche <prlw1@cam.ac.uk>
[netbsd-mini2440.git] / crypto / dist / heimdal / appl / popper / pop_dropinfo.c
blobff74ccf17f1e3af32b385c67dc77e477353a3a20
1 /*
2 * Copyright (c) 1989 Regents of the University of California.
3 * All rights reserved. The Berkeley software License Agreement
4 * specifies the terms and conditions for redistribution.
5 */
7 #include <popper.h>
8 __RCSID("$Heimdal: pop_dropinfo.c 16226 2005-10-22 21:13:53Z lha $"
9 "$NetBSD$");
11 #if defined(UIDL) || defined(XOVER)
14 * Copy the string found after after : into a malloced buffer. Stop
15 * copying at end of string or end of line. End of line delimiter is
16 * not part of the resulting copy.
18 static
19 char *
20 find_value_after_colon(char *p)
22 char *t, *tmp;
24 for (; *p != 0 && *p != ':'; p++) /* Find : */
27 if (*p == 0)
28 goto error;
30 p++; /* Skip over : */
32 for(; *p == ' ' || *p == '\t'; p++) /* Remove white space */
35 for (t = p; *t != 0 && *t != '\n' && *t != '\r'; t++) /* Find end of str */
38 tmp = t = malloc(t - p + 1);
39 if (tmp == 0)
40 goto error;
42 for (; *p != 0 && *p != '\n' && *p != '\r'; p++, t++) /* Copy characters */
43 *t = *p;
44 *t = 0; /* Terminate string */
45 return tmp;
47 error:
48 return "ErrorUIDL";
50 #endif
52 void
53 parse_header(MsgInfoList *mp, char *buffer)
55 #if defined(UIDL) || defined(XOVER)
56 if (strncasecmp("Message-Id:",buffer, 11) == 0) {
57 if (mp->msg_id == NULL)
58 mp->msg_id = find_value_after_colon(buffer);
60 #ifdef UIDL
61 else if (strncasecmp(buffer, "X-UIDL:", 7) == 0) {
62 /* Courtesy to Qualcomm, there really is no such
63 thing as X-UIDL */
64 mp->msg_id = find_value_after_colon(buffer);
66 #endif
67 #endif
68 #ifdef XOVER
69 else if (strncasecmp("Subject:", buffer, 8) == 0) {
70 if(mp->subject == NULL){
71 char *p;
72 mp->subject = find_value_after_colon(buffer);
73 for(p = mp->subject; *p; p++)
74 if(*p == '\t') *p = ' ';
77 else if (strncasecmp("From:", buffer, 5) == 0) {
78 if(mp->from == NULL){
79 char *p;
80 mp->from = find_value_after_colon(buffer);
81 for(p = mp->from; *p; p++)
82 if(*p == '\t') *p = ' ';
85 else if (strncasecmp("Date:", buffer, 5) == 0) {
86 if(mp->date == NULL){
87 char *p;
88 mp->date = find_value_after_colon(buffer);
89 for(p = mp->date; *p; p++)
90 if(*p == '\t') *p = ' ';
93 #endif
96 int
97 add_missing_headers(POP *p, MsgInfoList *mp)
99 #if defined(UIDL) || defined(XOVER)
100 if (mp->msg_id == NULL) {
101 if (asprintf(&mp->msg_id, "no-message-id-%d", mp->number) == -1) {
102 fclose (p->drop);
103 p->msg_count = 0;
104 return pop_msg (p,POP_FAILURE,
105 "Can't build message list for '%s': Out of memory",
106 p->user);
109 #endif
110 #ifdef XOVER
111 if (mp->subject == NULL)
112 mp->subject = "<none>";
113 if (mp->from == NULL)
114 mp->from = "<unknown>";
115 if (mp->date == NULL)
116 mp->date = "<unknown>";
117 #endif
118 return POP_SUCCESS;
122 * dropinfo: Extract information about the POP maildrop and store
123 * it for use by the other POP routines.
127 pop_dropinfo(POP *p)
129 char buffer[BUFSIZ]; /* Read buffer */
130 MsgInfoList * mp; /* Pointer to message
131 info list */
132 int msg_num; /* Current message
133 counter */
134 int nchar; /* Bytes written/read */
135 int blank_line = 1; /* previous line was blank */
136 int in_header = 0; /* if we are in a header block */
138 /* Initialize maildrop status variables in the POP parameter block */
139 p->msg_count = 0;
140 p->msgs_deleted = 0;
141 p->last_msg = 0;
142 p->bytes_deleted = 0;
143 p->drop_size = 0;
145 /* Allocate memory for message information structures */
146 p->msg_count = ALLOC_MSGS;
147 p->mlp = (MsgInfoList *)calloc((unsigned)p->msg_count,sizeof(MsgInfoList));
148 if (p->mlp == NULL){
149 fclose (p->drop);
150 p->msg_count = 0;
151 return pop_msg (p,POP_FAILURE,
152 "Can't build message list for '%s': Out of memory", p->user);
155 rewind (p->drop);
157 /* Scan the file, loading the message information list with
158 information about each message */
160 for (msg_num = p->drop_size = 0, mp = p->mlp - 1;
161 fgets(buffer,MAXMSGLINELEN,p->drop);) {
163 nchar = strlen(buffer);
165 if (blank_line && strncmp(buffer,"From ",5) == 0) {
166 in_header = 1;
167 if (++msg_num > p->msg_count) {
168 p->mlp=(MsgInfoList *) realloc(p->mlp,
169 (p->msg_count+=ALLOC_MSGS)*sizeof(MsgInfoList));
170 if (p->mlp == NULL){
171 fclose (p->drop);
172 p->msg_count = 0;
173 return pop_msg (p,POP_FAILURE,
174 "Can't build message list for '%s': Out of memory",
175 p->user);
177 mp = p->mlp + msg_num - 2;
179 ++mp;
180 mp->number = msg_num;
181 mp->length = 0;
182 mp->lines = 0;
183 mp->offset = ftell(p->drop) - nchar;
184 mp->flags = 0;
185 #if defined(UIDL) || defined(XOVER)
186 mp->msg_id = 0;
187 #endif
188 #ifdef XOVER
189 mp->subject = 0;
190 mp->from = 0;
191 mp->date = 0;
192 #endif
193 #ifdef DEBUG
194 if(p->debug)
195 pop_log(p, POP_DEBUG,
196 "Msg %d at offset %ld being added to list",
197 mp->number, mp->offset);
198 #endif /* DEBUG */
199 } else if(in_header)
200 parse_header(mp, buffer);
201 blank_line = (strncmp(buffer, "\n", nchar) == 0);
202 if(blank_line) {
203 int e;
204 in_header = 0;
205 e = add_missing_headers(p, mp);
206 if(e != POP_SUCCESS)
207 return e;
209 mp->length += nchar;
210 p->drop_size += nchar;
211 mp->lines++;
213 p->msg_count = msg_num;
215 #ifdef DEBUG
216 if(p->debug && msg_num > 0) {
217 int i;
218 for (i = 0, mp = p->mlp; i < p->msg_count; i++, mp++)
219 #ifdef UIDL
220 pop_log(p,POP_DEBUG,
221 "Msg %d at offset %ld is %ld octets long and has %u lines and id %s.",
222 mp->number,mp->offset,mp->length,mp->lines, mp->msg_id);
223 #else
224 pop_log(p,POP_DEBUG,
225 "Msg %d at offset %d is %d octets long and has %u lines.",
226 mp->number,mp->offset,mp->length,mp->lines);
227 #endif
229 #endif /* DEBUG */
231 return(POP_SUCCESS);