dma: rework config parsing
[dragonfly.git] / libexec / dma / conf.c
blobb676f9ce9209dc9e48eca57364a2e09fa7fc7667
1 /*
2 * Copyright (c) 2008 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Matthias Schmidt <matthias@dragonflybsd.org>, University of Marburg,
6 * Germany.
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in
16 * the documentation and/or other materials provided with the
17 * distribution.
18 * 3. Neither the name of The DragonFly Project nor the names of its
19 * contributors may be used to endorse or promote products derived
20 * from this software without specific, prior written permission.
22 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
23 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
24 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
25 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
26 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
27 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
28 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
30 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
31 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
32 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 * SUCH DAMAGE.
36 #include <err.h>
37 #include <errno.h>
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <string.h>
41 #include <syslog.h>
42 #include <stdarg.h>
44 #include "dma.h"
46 #define DP ": \t"
47 #define EQS " \t"
51 * Remove trailing \n's
53 void
54 trim_line(char *line)
56 size_t linelen;
57 char *p;
59 p = line;
61 if ((p = strchr(line, '\n')))
62 *p = (char)0;
64 /* Escape leading dot in every case */
65 linelen = strlen(line);
66 if (line[0] == '.') {
67 if ((linelen + 2) > 1000) {
68 syslog(LOG_CRIT, "Cannot escape leading dot. Buffer overflow");
69 exit(1);
71 memmove((line + 1), line, (linelen + 1));
72 line[0] = '.';
76 static void
77 chomp(char *str)
79 size_t len = strlen(str);
81 if (len == 0)
82 return;
83 if (str[len - 1] == '\n')
84 str[len - 1] = 0;
88 * Read the virtual user table
90 void
91 parse_virtuser(const char *path)
93 char line[2048];
94 FILE *v;
95 char *data;
96 struct virtuser *vu;
97 int lineno = 0;
99 v = fopen(path, "r");
100 if (v == NULL) {
101 errlog(1, "can not open virtuser file `%s'", path);
102 /* NOTREACHED */
105 while (!feof(v)) {
106 if (fgets(line, sizeof(line), v) == NULL)
107 break;
108 lineno++;
110 chomp(line);
112 /* We hit a comment */
113 if (*line == '#')
114 continue;
115 /* Ignore empty lines */
116 if (*line == 0)
117 continue;
119 vu = calloc(1, sizeof(*vu));
120 if (vu == NULL)
121 errlog(1, NULL);
123 data = strdup(line);
124 vu->login = strsep(&data, DP);
125 vu->address = data;
127 if (vu->login == NULL ||
128 vu->address == NULL) {
129 errlogx(1, "syntax error in virtuser file %s:%d",
130 path, lineno);
131 /* NOTREACHED */
134 SLIST_INSERT_HEAD(&virtusers, vu, next);
137 fclose(v);
141 * Read the SMTP authentication config file
143 * file format is:
144 * user|host:password
146 * A line starting with # is treated as comment and ignored.
148 void
149 parse_authfile(const char *path)
151 char line[2048];
152 struct authuser *au;
153 FILE *a;
154 char *data;
155 int lineno = 0;
157 a = fopen(path, "r");
158 if (a == NULL) {
159 errlog(1, "can not open auth file `%s'", path);
160 /* NOTREACHED */
163 while (!feof(a)) {
164 if (fgets(line, sizeof(line), a) == NULL)
165 break;
166 lineno++;
168 chomp(line);
170 /* We hit a comment */
171 if (*line == '#')
172 continue;
173 /* Ignore empty lines */
174 if (*line == 0)
175 continue;
177 au = calloc(1, sizeof(*au));
178 if (au == NULL)
179 errlog(1, NULL);
181 data = strdup(line);
182 au->login = strsep(&data, "|");
183 au->host = strsep(&data, DP);
184 au->password = data;
186 if (au->login == NULL ||
187 au->host == NULL ||
188 au->password == NULL) {
189 errlogx(1, "syntax error in authfile %s:%d",
190 path, lineno);
191 /* NOTREACHED */
194 SLIST_INSERT_HEAD(&authusers, au, next);
197 fclose(a);
201 * XXX TODO
202 * Check for bad things[TM]
204 void
205 parse_conf(const char *config_path)
207 char *word;
208 char *data;
209 FILE *conf;
210 char line[2048];
211 int lineno = 0;
213 conf = fopen(config_path, "r");
214 if (conf == NULL) {
215 /* Don't treat a non-existing config file as error */
216 if (errno == ENOENT)
217 return;
218 errlog(1, "can not open config `%s'", config_path);
219 /* NOTREACHED */
222 while (!feof(conf)) {
223 if (fgets(line, sizeof(line), conf) == NULL)
224 break;
225 lineno++;
227 chomp(line);
229 /* We hit a comment */
230 if (strchr(line, '#'))
231 *strchr(line, '#') = 0;
233 data = line;
234 word = strsep(&data, EQS);
236 /* Ignore empty lines */
237 if (word == NULL || *word == 0)
238 continue;
240 if (data != NULL && *data != 0)
241 data = strdup(data);
242 else
243 data = NULL;
245 if (strcmp(word, "SMARTHOST") == 0 && data != NULL)
246 config.smarthost = data;
247 else if (strcmp(word, "PORT") == 0 && data != NULL)
248 config.port = atoi(data);
249 else if (strcmp(word, "ALIASES") == 0 && data != NULL)
250 config.aliases = data;
251 else if (strcmp(word, "SPOOLDIR") == 0 && data != NULL)
252 config.spooldir = data;
253 else if (strcmp(word, "VIRTPATH") == 0 && data != NULL)
254 config.virtualpath = data;
255 else if (strcmp(word, "AUTHPATH") == 0 && data != NULL)
256 config.authpath= data;
257 else if (strcmp(word, "CERTFILE") == 0 && data != NULL)
258 config.certfile = data;
259 else if (strcmp(word, "MAILNAME") == 0 && data != NULL)
260 config.mailname = data;
261 else if (strcmp(word, "MAILNAMEFILE") == 0 && data != NULL)
262 config.mailnamefile = data;
263 else if (strcmp(word, "VIRTUAL") == 0 && data == NULL)
264 config.features |= VIRTUAL;
265 else if (strcmp(word, "STARTTLS") == 0 && data == NULL)
266 config.features |= STARTTLS;
267 else if (strcmp(word, "SECURETRANSFER") == 0 && data == NULL)
268 config.features |= SECURETRANS;
269 else if (strcmp(word, "DEFER") == 0 && data == NULL)
270 config.features |= DEFER;
271 else if (strcmp(word, "INSECURE") == 0 && data == NULL)
272 config.features |= INSECURE;
273 else if (strcmp(word, "FULLBOUNCE") == 0 && data == NULL)
274 config.features |= FULLBOUNCE;
275 else {
276 errlogx(1, "syntax error in %s:%d", config_path, lineno);
277 /* NOTREACHED */
281 fclose(conf);