chiark / gitweb /
qmail-smtpd: Validation of recipient mailbox names.
[qmail] / addrcheck.c
1 #include "cdb.h"
2 #include "stralloc.h"
3 #include "byte.h"
4 #include "str.h"
5 #include "addrcheck.h"
6 #include <errno.h>
7
8 /* #define DEBUG */
9 #ifdef DEBUG
10 #  define D(x) x
11 #  include <stdio.h>
12 #  include <sys/types.h>
13 #  include <unistd.h>
14 #else
15 #  define D(x)
16 #endif
17
18 #define STRALLOC_INIT { 0 }
19
20 static int probe(int cdb, int prefix, const char *key, int len,
21                  const char *suffix, uint32 *dlen)
22 {
23   static stralloc k = STRALLOC_INIT;
24   char ch = prefix;
25   int rc;
26
27   k.len = 0;
28   if (!stralloc_append(&k, &ch) ||
29       !stralloc_catb(&k, key, len) ||
30       (suffix && !stralloc_cats(&k, suffix)))
31     return (-1);
32   D( fprintf(stderr, "*** `%.*s' -> ", k.len, k.s); )
33   rc = cdb_seek(cdb, k.s, k.len, dlen);
34   D( if (rc == -1)
35        fprintf(stderr, "error: %s\n", strerror(errno));
36      else if (rc == 0)
37        fprintf(stderr, "not found\n");
38      else if (!*dlen)
39        fprintf(stderr, "empty\n");
40      else {
41        int n = *dlen;
42        int nn;
43        char buf[256];
44        off_t pos = lseek(cdb, 0, SEEK_CUR);
45        fprintf(stderr, "`");
46        while (n) {
47          nn = sizeof(buf); if (nn > n) nn = n;
48          read(cdb, buf, nn);
49          fwrite(buf, 1, nn, stderr);
50          n -= nn;
51        }
52        fprintf(stderr, "'\n");
53        lseek(cdb, pos, SEEK_SET);
54      } )
55   return (rc);
56 }
57
58 static int localprobe(int cdb, const char *key, int len,
59                       const char *suffix, int *rc)
60 {
61   int err;
62   uint32 dlen;
63   char ch;
64
65   if ((err = probe(cdb, 'L', key, len, suffix, &dlen)) < 0)
66     return (-1);
67   if (!err) { *rc = 0; return (0); }
68   if (dlen != 1) { errno = EINVAL; return (-1); }
69   if (read(cdb, &ch, 1) != 1) { errno = EIO; return (-1); }
70   *rc = ch;
71   return (1);
72 }
73
74 static int local(int cdb, const char *l, int len, int *rc)
75 {
76   int code;
77   int err = 0;
78   int dash;
79
80   if ((err = localprobe(cdb, l, len, 0, &code)) != 0) goto done;
81
82   for (;;) {
83     dash = byte_rchr(l, len, '-');
84     if (dash == len) break;
85     if ((err = localprobe(cdb, l, dash, "-default", &code)) != 0) goto done;
86     len = dash;
87   }
88   *rc = 0;
89   return (0);
90   
91 done:
92   if (err >= 0) {
93     switch (code) {
94       case '+': *rc = 1; break;
95       case '-': *rc = 0; break;
96       default: errno = EINVAL; err = -1; break;
97     }
98   }
99   return (err);
100 }
101
102 static int virt(int cdb, const char *u, int ulen,
103                 const char *addr, int alen, int *rc)
104 {
105   static stralloc l = STRALLOC_INIT;
106   uint32 dlen;
107   int err;
108
109   if ((err = probe(cdb, 'V', addr, alen, 0, &dlen)) <= 0)
110     return (err);
111   if (!stralloc_ready(&l, dlen + 1)) return (-1);
112   if (read(cdb, l.s, dlen) != dlen) { errno = EIO; return (-1); }
113   l.s[dlen] = '-';
114   l.len = dlen + 1;
115   if (!stralloc_catb(&l, u, ulen)) return (-1);
116   D( printf("*** virtual map -> `%.*s'\n", l.len, l.s); )
117   if (local(cdb, l.s, l.len, rc) < 0) return (-1);
118   return (1);
119 }
120
121 int addrcheck(int cdb, const char *addr, int *rc)
122 {
123   int at, len, dot;
124   int err = 0;
125   uint32 dlen;
126
127   len = str_len(addr);
128   at = str_chr(addr, '@');
129   if (!addr[at])
130     return (local(cdb, addr, len, rc));
131
132   if ((err = virt(cdb, addr, at, addr, len, rc)) != 0)
133     return (err);
134   dot = at + 1;
135   while (addr[dot]) {
136     if ((err = virt(cdb, addr, at, addr + dot, len - dot, rc)) != 0)
137       return (err);
138     dot += byte_chr(addr + dot + 1, len - dot - 1, '.') + 1;
139   }
140
141   if ((err = probe(cdb, '@', addr + at + 1, len - at - 1, 0, &dlen)) < 0)
142     return (-1);
143   if (!err) { *rc = 1; return (0); }
144   if (dlen != 0) { errno = EINVAL; return (-1); }
145
146   return (local(cdb, addr, at, rc));
147 }
148
149 #ifdef TEST
150 #include <sys/types.h>
151 #include <unistd.h>
152 #include <fcntl.h>
153 #include <stdio.h>
154 #include <errno.h>
155 #include <unistd.h>
156
157 int main(int argc, char *argv[])
158 {
159   int fd;
160   int rc;
161   int i;
162
163   if (argc < 3) {
164     fprintf(stderr, "usage: addrcheck CDB ADDR...\n");
165     return (1);
166   }
167   if ((fd = open(argv[1], O_RDONLY)) < 0) {
168     perror(argv[1]);
169     return (1);
170   }
171   for (i = 2; i < argc; i++) {
172     if (addrcheck(fd, argv[i], &rc) < 0) {
173       perror("checking");
174       return (1);
175     }
176     printf("%s: %s\n", argv[i], rc ? "ok" : "bad");
177   }
178   return (0);
179 }
180
181 #endif