Blame


1 15902770 2021-01-15 op /* -*- mode: fundamental; indent-tabs-mode: t; -*- */
2 15902770 2021-01-15 op %{
3 15902770 2021-01-15 op
4 15902770 2021-01-15 op /*
5 15902770 2021-01-15 op * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
6 15902770 2021-01-15 op *
7 15902770 2021-01-15 op * Permission to use, copy, modify, and distribute this software for any
8 15902770 2021-01-15 op * purpose with or without fee is hereby granted, provided that the above
9 15902770 2021-01-15 op * copyright notice and this permission notice appear in all copies.
10 15902770 2021-01-15 op *
11 15902770 2021-01-15 op * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
12 15902770 2021-01-15 op * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
13 15902770 2021-01-15 op * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
14 15902770 2021-01-15 op * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
15 15902770 2021-01-15 op * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
16 15902770 2021-01-15 op * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
17 15902770 2021-01-15 op * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18 15902770 2021-01-15 op */
19 15902770 2021-01-15 op
20 002a84a1 2021-02-10 op #include <errno.h>
21 6abda252 2021-02-06 op #include <stdarg.h>
22 15902770 2021-01-15 op #include <stdio.h>
23 32693ee6 2021-01-28 op #include <string.h>
24 15902770 2021-01-15 op
25 15902770 2021-01-15 op #include "gmid.h"
26 15902770 2021-01-15 op
27 15902770 2021-01-15 op /*
28 15902770 2021-01-15 op * #define YYDEBUG 1
29 15902770 2021-01-15 op * int yydebug = 1;
30 15902770 2021-01-15 op */
31 15902770 2021-01-15 op
32 ca21e100 2021-02-04 op struct vhost *host;
33 ca21e100 2021-02-04 op struct location *loc;
34 15902770 2021-01-15 op
35 13ed2fb6 2021-01-27 op int goterror = 0;
36 15902770 2021-01-15 op
37 b8e64ccd 2021-03-31 op static struct vhost *new_vhost(void);
38 b8e64ccd 2021-03-31 op static struct location *new_location(void);
39 b8e64ccd 2021-03-31 op
40 6abda252 2021-02-06 op void yyerror(const char*, ...);
41 13ed2fb6 2021-01-27 op int parse_portno(const char*);
42 13ed2fb6 2021-01-27 op void parse_conf(const char*);
43 e17642a7 2021-02-01 op char *ensure_absolute_path(char*);
44 6abda252 2021-02-06 op int check_block_code(int);
45 6abda252 2021-02-06 op char *check_block_fmt(char*);
46 6abda252 2021-02-06 op int check_strip_no(int);
47 a709ddf5 2021-02-07 op int check_prefork_num(int);
48 49b73ba1 2021-02-10 op void advance_loc(void);
49 13ed2fb6 2021-01-27 op
50 15902770 2021-01-15 op %}
51 15902770 2021-01-15 op
52 15902770 2021-01-15 op /* for bison: */
53 15902770 2021-01-15 op /* %define parse.error verbose */
54 15902770 2021-01-15 op
55 15902770 2021-01-15 op %union {
56 15902770 2021-01-15 op char *str;
57 15902770 2021-01-15 op int num;
58 15902770 2021-01-15 op }
59 15902770 2021-01-15 op
60 d06d6f4b 2021-04-29 op %token TIPV6 TPORT TPROTOCOLS TMIME TDEFAULT TTYPE TCHROOT TUSER TSERVER
61 d06d6f4b 2021-04-29 op %token TPREFORK TLOCATION TCERT TKEY TROOT TCGI TENV TLANG TLOG TINDEX TAUTO
62 cc8c2901 2021-04-29 op %token TSTRIP TBLOCK TRETURN TENTRYPOINT TREQUIRE TCLIENT TCA TALIAS
63 d06d6f4b 2021-04-29 op
64 15902770 2021-01-15 op %token TERR
65 15902770 2021-01-15 op
66 15902770 2021-01-15 op %token <str> TSTRING
67 15902770 2021-01-15 op %token <num> TNUM
68 15902770 2021-01-15 op %token <num> TBOOL
69 15902770 2021-01-15 op
70 15902770 2021-01-15 op %%
71 15902770 2021-01-15 op
72 15902770 2021-01-15 op conf : options vhosts ;
73 15902770 2021-01-15 op
74 15902770 2021-01-15 op options : /* empty */
75 15902770 2021-01-15 op | options option
76 15902770 2021-01-15 op ;
77 15902770 2021-01-15 op
78 eb59f87e 2021-02-09 op option : TCHROOT TSTRING { conf.chroot = $2; }
79 eb59f87e 2021-02-09 op | TIPV6 TBOOL { conf.ipv6 = $2; }
80 eb59f87e 2021-02-09 op | TMIME TSTRING TSTRING { add_mime(&conf.mime, $2, $3); }
81 15902770 2021-01-15 op | TPORT TNUM { conf.port = $2; }
82 eb59f87e 2021-02-09 op | TPREFORK TNUM { conf.prefork = check_prefork_num($2); }
83 5bc3c98e 2021-01-15 op | TPROTOCOLS TSTRING {
84 5bc3c98e 2021-01-15 op if (tls_config_parse_protocols(&conf.protos, $2) == -1)
85 002a84a1 2021-02-10 op yyerror("invalid protocols string \"%s\"", $2);
86 5bc3c98e 2021-01-15 op }
87 ae08ec7d 2021-01-25 op | TUSER TSTRING { conf.user = $2; }
88 15902770 2021-01-15 op ;
89 15902770 2021-01-15 op
90 15902770 2021-01-15 op vhosts : /* empty */
91 15902770 2021-01-15 op | vhosts vhost
92 15902770 2021-01-15 op ;
93 15902770 2021-01-15 op
94 b8e64ccd 2021-03-31 op vhost : TSERVER TSTRING {
95 b8e64ccd 2021-03-31 op host = new_vhost();
96 b8e64ccd 2021-03-31 op TAILQ_INSERT_HEAD(&hosts, host, vhosts);
97 b8e64ccd 2021-03-31 op
98 b8e64ccd 2021-03-31 op loc = new_location();
99 b8e64ccd 2021-03-31 op TAILQ_INSERT_HEAD(&host->locations, loc, locations);
100 b8e64ccd 2021-03-31 op
101 b8e64ccd 2021-03-31 op loc->match = xstrdup("*");
102 15902770 2021-01-15 op host->domain = $2;
103 15902770 2021-01-15 op
104 cbeee4ca 2021-01-28 op if (strstr($2, "xn--") != NULL) {
105 cbeee4ca 2021-01-28 op warnx("%s:%d \"%s\" looks like punycode: "
106 415ac7a2 2021-01-28 op "you should use the decoded hostname.",
107 415ac7a2 2021-01-28 op config_path, yylineno, $2);
108 cbeee4ca 2021-01-28 op }
109 b8e64ccd 2021-03-31 op } '{' servopts locations '}' {
110 cbeee4ca 2021-01-28 op
111 fdea6aa0 2021-04-30 op if (host->cert == NULL || host->key == NULL)
112 002a84a1 2021-02-10 op yyerror("invalid vhost definition: %s", $2);
113 15902770 2021-01-15 op }
114 15902770 2021-01-15 op | error '}' { yyerror("error in server directive"); }
115 15902770 2021-01-15 op ;
116 15902770 2021-01-15 op
117 15902770 2021-01-15 op servopts : /* empty */
118 15902770 2021-01-15 op | servopts servopt
119 15902770 2021-01-15 op ;
120 15902770 2021-01-15 op
121 cc8c2901 2021-04-29 op servopt : TALIAS TSTRING {
122 cc8c2901 2021-04-29 op struct alist *a;
123 cc8c2901 2021-04-29 op
124 cc8c2901 2021-04-29 op a = xcalloc(1, sizeof(*a));
125 cc8c2901 2021-04-29 op a->alias = $2;
126 cc8c2901 2021-04-29 op if (TAILQ_EMPTY(&host->aliases))
127 cc8c2901 2021-04-29 op TAILQ_INSERT_HEAD(&host->aliases, a, aliases);
128 cc8c2901 2021-04-29 op else
129 cc8c2901 2021-04-29 op TAILQ_INSERT_TAIL(&host->aliases, a, aliases);
130 cc8c2901 2021-04-29 op }
131 cc8c2901 2021-04-29 op | TCERT TSTRING { host->cert = ensure_absolute_path($2); }
132 15902770 2021-01-15 op | TCGI TSTRING {
133 15902770 2021-01-15 op /* drop the starting '/', if any */
134 709f4c94 2021-02-04 op if (*$2 == '/')
135 709f4c94 2021-02-04 op memmove($2, $2+1, strlen($2));
136 709f4c94 2021-02-04 op host->cgi = $2;
137 05c23a54 2021-01-19 op }
138 e3ddf390 2021-02-06 op | TENTRYPOINT TSTRING {
139 e3ddf390 2021-02-06 op if (host->entrypoint != NULL)
140 e3ddf390 2021-02-06 op yyerror("`entrypoint' specified more than once");
141 e3ddf390 2021-02-06 op while (*$2 == '/')
142 e3ddf390 2021-02-06 op memmove($2, $2+1, strlen($2));
143 e3ddf390 2021-02-06 op host->entrypoint = $2;
144 e3ddf390 2021-02-06 op }
145 9cc630aa 2021-04-28 op | TENV TSTRING TSTRING {
146 9cc630aa 2021-04-28 op struct envlist *e;
147 9cc630aa 2021-04-28 op
148 9cc630aa 2021-04-28 op e = xcalloc(1, sizeof(*e));
149 9cc630aa 2021-04-28 op e->name = $2;
150 9cc630aa 2021-04-28 op e->value = $3;
151 9cc630aa 2021-04-28 op if (TAILQ_EMPTY(&host->env))
152 9cc630aa 2021-04-28 op TAILQ_INSERT_HEAD(&host->env, e, envs);
153 9cc630aa 2021-04-28 op else
154 9cc630aa 2021-04-28 op TAILQ_INSERT_TAIL(&host->env, e, envs);
155 9cc630aa 2021-04-28 op }
156 eb59f87e 2021-02-09 op | TKEY TSTRING { host->key = ensure_absolute_path($2); }
157 c8b74339 2021-01-24 op | locopt
158 c8b74339 2021-01-24 op ;
159 c8b74339 2021-01-24 op
160 c8b74339 2021-01-24 op locations : /* empty */
161 c8b74339 2021-01-24 op | locations location
162 c8b74339 2021-01-24 op ;
163 c8b74339 2021-01-24 op
164 49b73ba1 2021-02-10 op location : TLOCATION { advance_loc(); } TSTRING '{' locopts '}' {
165 49b73ba1 2021-02-10 op /* drop the starting '/' if any */
166 49b73ba1 2021-02-10 op if (*$3 == '/')
167 49b73ba1 2021-02-10 op memmove($3, $3+1, strlen($3));
168 49b73ba1 2021-02-10 op loc->match = $3;
169 6119e13e 2021-01-19 op }
170 c8b74339 2021-01-24 op | error '}'
171 c8b74339 2021-01-24 op ;
172 c8b74339 2021-01-24 op
173 c8b74339 2021-01-24 op locopts : /* empty */
174 c8b74339 2021-01-24 op | locopts locopt
175 c8b74339 2021-01-24 op ;
176 c8b74339 2021-01-24 op
177 eb59f87e 2021-02-09 op locopt : TAUTO TINDEX TBOOL { loc->auto_index = $3 ? 1 : -1; }
178 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM TSTRING {
179 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
180 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
181 6abda252 2021-02-06 op loc->block_fmt = check_block_fmt($4);
182 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
183 6abda252 2021-02-06 op }
184 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM {
185 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
186 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
187 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
188 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
189 6abda252 2021-02-06 op if ($3 >= 30 && $3 < 40)
190 6abda252 2021-02-06 op yyerror("missing `meta' for block return %d", $3);
191 6abda252 2021-02-06 op }
192 6abda252 2021-02-06 op | TBLOCK {
193 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
194 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
195 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
196 6abda252 2021-02-06 op loc->block_code = 40;
197 6abda252 2021-02-06 op }
198 eb59f87e 2021-02-09 op | TDEFAULT TTYPE TSTRING {
199 eb59f87e 2021-02-09 op if (loc->default_mime != NULL)
200 eb59f87e 2021-02-09 op yyerror("`default type' specified more than once");
201 eb59f87e 2021-02-09 op loc->default_mime = $3;
202 eb59f87e 2021-02-09 op }
203 eb59f87e 2021-02-09 op | TINDEX TSTRING {
204 eb59f87e 2021-02-09 op if (loc->index != NULL)
205 eb59f87e 2021-02-09 op yyerror("`index' specified more than once");
206 eb59f87e 2021-02-09 op loc->index = $2;
207 eb59f87e 2021-02-09 op }
208 eb59f87e 2021-02-09 op | TLANG TSTRING {
209 eb59f87e 2021-02-09 op if (loc->lang != NULL)
210 eb59f87e 2021-02-09 op yyerror("`lang' specified more than once");
211 eb59f87e 2021-02-09 op loc->lang = $2;
212 eb59f87e 2021-02-09 op }
213 793835cb 2021-02-23 op | TLOG TBOOL { loc->disable_log = !$2; }
214 02be96c6 2021-02-09 op | TREQUIRE TCLIENT TCA TSTRING {
215 02be96c6 2021-02-09 op if (loc->reqca != NULL)
216 02be96c6 2021-02-09 op yyerror("`require client ca' specified more than once");
217 adbe6a64 2021-04-30 op
218 adbe6a64 2021-04-30 op ensure_absolute_path($4);
219 02be96c6 2021-02-09 op if ((loc->reqca = load_ca($4)) == NULL)
220 02be96c6 2021-02-09 op yyerror("couldn't load ca cert: %s", $4);
221 02be96c6 2021-02-09 op free($4);
222 fdea6aa0 2021-04-30 op }
223 fdea6aa0 2021-04-30 op | TROOT TSTRING {
224 fdea6aa0 2021-04-30 op if (loc->dir != NULL)
225 fdea6aa0 2021-04-30 op yyerror("`root' specified more than once");
226 fdea6aa0 2021-04-30 op
227 fdea6aa0 2021-04-30 op loc->dir = ensure_absolute_path($2);
228 02be96c6 2021-02-09 op }
229 eb59f87e 2021-02-09 op | TSTRIP TNUM { loc->strip = check_strip_no($2); }
230 15902770 2021-01-15 op ;
231 13ed2fb6 2021-01-27 op
232 13ed2fb6 2021-01-27 op %%
233 b8e64ccd 2021-03-31 op
234 b8e64ccd 2021-03-31 op static struct vhost *
235 b8e64ccd 2021-03-31 op new_vhost(void)
236 b8e64ccd 2021-03-31 op {
237 b8e64ccd 2021-03-31 op return xcalloc(1, sizeof(struct vhost));
238 b8e64ccd 2021-03-31 op }
239 13ed2fb6 2021-01-27 op
240 b8e64ccd 2021-03-31 op static struct location *
241 b8e64ccd 2021-03-31 op new_location(void)
242 b8e64ccd 2021-03-31 op {
243 fdea6aa0 2021-04-30 op struct location *l;
244 fdea6aa0 2021-04-30 op
245 fdea6aa0 2021-04-30 op l = xcalloc(1, sizeof(*l));
246 fdea6aa0 2021-04-30 op l->dirfd = -1;
247 fdea6aa0 2021-04-30 op return l;
248 b8e64ccd 2021-03-31 op }
249 b8e64ccd 2021-03-31 op
250 13ed2fb6 2021-01-27 op void
251 6abda252 2021-02-06 op yyerror(const char *msg, ...)
252 13ed2fb6 2021-01-27 op {
253 6abda252 2021-02-06 op va_list ap;
254 6abda252 2021-02-06 op
255 13ed2fb6 2021-01-27 op goterror = 1;
256 6abda252 2021-02-06 op
257 6abda252 2021-02-06 op va_start(ap, msg);
258 6abda252 2021-02-06 op fprintf(stderr, "%s:%d: ", config_path, yylineno);
259 6abda252 2021-02-06 op vfprintf(stderr, msg, ap);
260 a1373913 2021-02-07 op fprintf(stderr, "\n");
261 6abda252 2021-02-06 op va_end(ap);
262 13ed2fb6 2021-01-27 op }
263 13ed2fb6 2021-01-27 op
264 13ed2fb6 2021-01-27 op int
265 13ed2fb6 2021-01-27 op parse_portno(const char *p)
266 13ed2fb6 2021-01-27 op {
267 13ed2fb6 2021-01-27 op const char *errstr;
268 13ed2fb6 2021-01-27 op int n;
269 13ed2fb6 2021-01-27 op
270 13ed2fb6 2021-01-27 op n = strtonum(p, 0, UINT16_MAX, &errstr);
271 13ed2fb6 2021-01-27 op if (errstr != NULL)
272 2d34f732 2021-02-10 op yyerror("port number is %s: %s", errstr, p);
273 13ed2fb6 2021-01-27 op return n;
274 13ed2fb6 2021-01-27 op }
275 13ed2fb6 2021-01-27 op
276 13ed2fb6 2021-01-27 op void
277 13ed2fb6 2021-01-27 op parse_conf(const char *path)
278 13ed2fb6 2021-01-27 op {
279 13ed2fb6 2021-01-27 op config_path = path;
280 13ed2fb6 2021-01-27 op if ((yyin = fopen(path, "r")) == NULL)
281 48b69cb2 2021-04-28 op err(1, "cannot open config: %s", path);
282 13ed2fb6 2021-01-27 op yyparse();
283 13ed2fb6 2021-01-27 op fclose(yyin);
284 13ed2fb6 2021-01-27 op
285 13ed2fb6 2021-01-27 op if (goterror)
286 13ed2fb6 2021-01-27 op exit(1);
287 b8e64ccd 2021-03-31 op
288 b8e64ccd 2021-03-31 op if (TAILQ_FIRST(&hosts)->domain == NULL)
289 48b69cb2 2021-04-28 op errx(1, "no vhost defined in %s", path);
290 13ed2fb6 2021-01-27 op }
291 e17642a7 2021-02-01 op
292 e17642a7 2021-02-01 op char *
293 e17642a7 2021-02-01 op ensure_absolute_path(char *path)
294 e17642a7 2021-02-01 op {
295 e17642a7 2021-02-01 op if (path == NULL || *path != '/')
296 adbe6a64 2021-04-30 op yyerror("not an absolute path: %s", path);
297 e17642a7 2021-02-01 op return path;
298 e17642a7 2021-02-01 op }
299 6abda252 2021-02-06 op
300 6abda252 2021-02-06 op int
301 6abda252 2021-02-06 op check_block_code(int n)
302 6abda252 2021-02-06 op {
303 6abda252 2021-02-06 op if (n < 10 || n >= 70 || (n >= 20 && n <= 29))
304 6abda252 2021-02-06 op yyerror("invalid block code %d", n);
305 6abda252 2021-02-06 op return n;
306 6abda252 2021-02-06 op }
307 6abda252 2021-02-06 op
308 6abda252 2021-02-06 op char *
309 6abda252 2021-02-06 op check_block_fmt(char *fmt)
310 6abda252 2021-02-06 op {
311 6abda252 2021-02-06 op char *s;
312 6abda252 2021-02-06 op
313 6abda252 2021-02-06 op for (s = fmt; *s; ++s) {
314 6abda252 2021-02-06 op if (*s != '%')
315 6abda252 2021-02-06 op continue;
316 6abda252 2021-02-06 op switch (*++s) {
317 6abda252 2021-02-06 op case '%':
318 6abda252 2021-02-06 op case 'p':
319 6abda252 2021-02-06 op case 'q':
320 6abda252 2021-02-06 op case 'P':
321 6abda252 2021-02-06 op case 'N':
322 6abda252 2021-02-06 op break;
323 6abda252 2021-02-06 op default:
324 6abda252 2021-02-06 op yyerror("invalid format specifier %%%c", *s);
325 6abda252 2021-02-06 op }
326 6abda252 2021-02-06 op }
327 6abda252 2021-02-06 op
328 6abda252 2021-02-06 op return fmt;
329 6abda252 2021-02-06 op }
330 6abda252 2021-02-06 op
331 6abda252 2021-02-06 op int
332 6abda252 2021-02-06 op check_strip_no(int n)
333 6abda252 2021-02-06 op {
334 6abda252 2021-02-06 op if (n <= 0)
335 6abda252 2021-02-06 op yyerror("invalid strip number %d", n);
336 a709ddf5 2021-02-07 op return n;
337 a709ddf5 2021-02-07 op }
338 a709ddf5 2021-02-07 op
339 a709ddf5 2021-02-07 op int
340 a709ddf5 2021-02-07 op check_prefork_num(int n)
341 a709ddf5 2021-02-07 op {
342 2c3e53da 2021-03-03 op if (n <= 0 || n >= PROC_MAX)
343 a709ddf5 2021-02-07 op yyerror("invalid prefork number %d", n);
344 6abda252 2021-02-06 op return n;
345 6abda252 2021-02-06 op }
346 49b73ba1 2021-02-10 op
347 49b73ba1 2021-02-10 op void
348 49b73ba1 2021-02-10 op advance_loc(void)
349 49b73ba1 2021-02-10 op {
350 b8e64ccd 2021-03-31 op loc = new_location();
351 b8e64ccd 2021-03-31 op TAILQ_INSERT_TAIL(&host->locations, loc, locations);
352 49b73ba1 2021-02-10 op }