1 15902770 2021-01-15 op /* -*- mode: fundamental; indent-tabs-mode: t; -*- */
5 15902770 2021-01-15 op * Copyright (c) 2021 Omar Polo <op@omarpolo.com>
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.
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.
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>
25 15902770 2021-01-15 op #include "gmid.h"
28 15902770 2021-01-15 op * #define YYDEBUG 1
29 15902770 2021-01-15 op * int yydebug = 1;
32 ca21e100 2021-02-04 op struct vhost *host;
33 ca21e100 2021-02-04 op struct location *loc;
35 13ed2fb6 2021-01-27 op int goterror = 0;
37 b8e64ccd 2021-03-31 op static struct vhost *new_vhost(void);
38 b8e64ccd 2021-03-31 op static struct location *new_location(void);
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 c705ecb1 2021-05-03 op void only_once(const void*, const char*);
50 8ad1c570 2021-05-09 op void only_oncei(int, const char*);
51 8ad1c570 2021-05-09 op int fastcgi_conf(char *, char *, char *);
55 15902770 2021-01-15 op /* for bison: */
56 15902770 2021-01-15 op /* %define parse.error verbose */
63 d06d6f4b 2021-04-29 op %token TIPV6 TPORT TPROTOCOLS TMIME TDEFAULT TTYPE TCHROOT TUSER TSERVER
64 d06d6f4b 2021-04-29 op %token TPREFORK TLOCATION TCERT TKEY TROOT TCGI TENV TLANG TLOG TINDEX TAUTO
65 8ad1c570 2021-05-09 op %token TSTRIP TBLOCK TRETURN TENTRYPOINT TREQUIRE TCLIENT TCA TALIAS TTCP
66 8ad1c570 2021-05-09 op %token TFASTCGI TSPAWN
70 15902770 2021-01-15 op %token <str> TSTRING
71 15902770 2021-01-15 op %token <num> TNUM
72 15902770 2021-01-15 op %token <num> TBOOL
76 15902770 2021-01-15 op conf : options vhosts ;
78 15902770 2021-01-15 op options : /* empty */
79 15902770 2021-01-15 op | options option
82 eb59f87e 2021-02-09 op option : TCHROOT TSTRING { conf.chroot = $2; }
83 eb59f87e 2021-02-09 op | TIPV6 TBOOL { conf.ipv6 = $2; }
84 eb59f87e 2021-02-09 op | TMIME TSTRING TSTRING { add_mime(&conf.mime, $2, $3); }
85 15902770 2021-01-15 op | TPORT TNUM { conf.port = $2; }
86 eb59f87e 2021-02-09 op | TPREFORK TNUM { conf.prefork = check_prefork_num($2); }
87 5bc3c98e 2021-01-15 op | TPROTOCOLS TSTRING {
88 5bc3c98e 2021-01-15 op if (tls_config_parse_protocols(&conf.protos, $2) == -1)
89 002a84a1 2021-02-10 op yyerror("invalid protocols string \"%s\"", $2);
91 ae08ec7d 2021-01-25 op | TUSER TSTRING { conf.user = $2; }
94 15902770 2021-01-15 op vhosts : /* empty */
95 15902770 2021-01-15 op | vhosts vhost
98 b8e64ccd 2021-03-31 op vhost : TSERVER TSTRING {
99 b8e64ccd 2021-03-31 op host = new_vhost();
100 b8e64ccd 2021-03-31 op TAILQ_INSERT_HEAD(&hosts, host, vhosts);
102 b8e64ccd 2021-03-31 op loc = new_location();
103 b8e64ccd 2021-03-31 op TAILQ_INSERT_HEAD(&host->locations, loc, locations);
105 b8e64ccd 2021-03-31 op loc->match = xstrdup("*");
106 15902770 2021-01-15 op host->domain = $2;
108 cbeee4ca 2021-01-28 op if (strstr($2, "xn--") != NULL) {
109 cbeee4ca 2021-01-28 op warnx("%s:%d \"%s\" looks like punycode: "
110 415ac7a2 2021-01-28 op "you should use the decoded hostname.",
111 415ac7a2 2021-01-28 op config_path, yylineno, $2);
113 b8e64ccd 2021-03-31 op } '{' servopts locations '}' {
115 fdea6aa0 2021-04-30 op if (host->cert == NULL || host->key == NULL)
116 002a84a1 2021-02-10 op yyerror("invalid vhost definition: %s", $2);
118 15902770 2021-01-15 op | error '}' { yyerror("error in server directive"); }
121 15902770 2021-01-15 op servopts : /* empty */
122 15902770 2021-01-15 op | servopts servopt
125 cc8c2901 2021-04-29 op servopt : TALIAS TSTRING {
126 cc8c2901 2021-04-29 op struct alist *a;
128 cc8c2901 2021-04-29 op a = xcalloc(1, sizeof(*a));
129 cc8c2901 2021-04-29 op a->alias = $2;
130 cc8c2901 2021-04-29 op if (TAILQ_EMPTY(&host->aliases))
131 cc8c2901 2021-04-29 op TAILQ_INSERT_HEAD(&host->aliases, a, aliases);
133 cc8c2901 2021-04-29 op TAILQ_INSERT_TAIL(&host->aliases, a, aliases);
135 c705ecb1 2021-05-03 op | TCERT TSTRING {
136 c705ecb1 2021-05-03 op only_once(host->cert, "cert");
137 c705ecb1 2021-05-03 op host->cert = ensure_absolute_path($2);
139 15902770 2021-01-15 op | TCGI TSTRING {
140 c705ecb1 2021-05-03 op only_once(host->cgi, "cgi");
141 15902770 2021-01-15 op /* drop the starting '/', if any */
142 709f4c94 2021-02-04 op if (*$2 == '/')
143 709f4c94 2021-02-04 op memmove($2, $2+1, strlen($2));
144 709f4c94 2021-02-04 op host->cgi = $2;
146 e3ddf390 2021-02-06 op | TENTRYPOINT TSTRING {
147 c705ecb1 2021-05-03 op only_once(host->entrypoint, "entrypoint");
148 e3ddf390 2021-02-06 op while (*$2 == '/')
149 e3ddf390 2021-02-06 op memmove($2, $2+1, strlen($2));
150 e3ddf390 2021-02-06 op host->entrypoint = $2;
152 9cc630aa 2021-04-28 op | TENV TSTRING TSTRING {
153 9cc630aa 2021-04-28 op struct envlist *e;
155 9cc630aa 2021-04-28 op e = xcalloc(1, sizeof(*e));
156 9cc630aa 2021-04-28 op e->name = $2;
157 9cc630aa 2021-04-28 op e->value = $3;
158 9cc630aa 2021-04-28 op if (TAILQ_EMPTY(&host->env))
159 9cc630aa 2021-04-28 op TAILQ_INSERT_HEAD(&host->env, e, envs);
161 9cc630aa 2021-04-28 op TAILQ_INSERT_TAIL(&host->env, e, envs);
163 c705ecb1 2021-05-03 op | TKEY TSTRING {
164 c705ecb1 2021-05-03 op only_once(host->key, "key");
165 c705ecb1 2021-05-03 op host->key = ensure_absolute_path($2);
170 c8b74339 2021-01-24 op locations : /* empty */
171 c8b74339 2021-01-24 op | locations location
174 49b73ba1 2021-02-10 op location : TLOCATION { advance_loc(); } TSTRING '{' locopts '}' {
175 49b73ba1 2021-02-10 op /* drop the starting '/' if any */
176 49b73ba1 2021-02-10 op if (*$3 == '/')
177 49b73ba1 2021-02-10 op memmove($3, $3+1, strlen($3));
178 49b73ba1 2021-02-10 op loc->match = $3;
183 c8b74339 2021-01-24 op locopts : /* empty */
184 c8b74339 2021-01-24 op | locopts locopt
187 eb59f87e 2021-02-09 op locopt : TAUTO TINDEX TBOOL { loc->auto_index = $3 ? 1 : -1; }
188 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM TSTRING {
189 c705ecb1 2021-05-03 op only_once(loc->block_fmt, "block");
190 6abda252 2021-02-06 op loc->block_fmt = check_block_fmt($4);
191 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
193 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM {
194 c705ecb1 2021-05-03 op only_once(loc->block_fmt, "block");
195 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
196 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
197 6abda252 2021-02-06 op if ($3 >= 30 && $3 < 40)
198 6abda252 2021-02-06 op yyerror("missing `meta' for block return %d", $3);
201 c705ecb1 2021-05-03 op only_once(loc->block_fmt, "block");
202 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
203 6abda252 2021-02-06 op loc->block_code = 40;
205 eb59f87e 2021-02-09 op | TDEFAULT TTYPE TSTRING {
206 c705ecb1 2021-05-03 op only_once(loc->default_mime, "default type");
207 eb59f87e 2021-02-09 op loc->default_mime = $3;
209 8ad1c570 2021-05-09 op | TFASTCGI TSPAWN TSTRING {
210 8ad1c570 2021-05-09 op only_oncei(loc->fcgi, "fastcgi");
211 8ad1c570 2021-05-09 op loc->fcgi = fastcgi_conf(NULL, NULL, $3);
213 8ad1c570 2021-05-09 op | TFASTCGI TSTRING {
214 8ad1c570 2021-05-09 op only_oncei(loc->fcgi, "fastcgi");
215 8ad1c570 2021-05-09 op loc->fcgi = fastcgi_conf($2, NULL, NULL);
217 8ad1c570 2021-05-09 op | TFASTCGI TTCP TSTRING TNUM {
219 8ad1c570 2021-05-09 op if (asprintf(&c, "%d", $4) == -1)
220 8ad1c570 2021-05-09 op err(1, "asprintf");
221 8ad1c570 2021-05-09 op only_oncei(loc->fcgi, "fastcgi");
222 8ad1c570 2021-05-09 op loc->fcgi = fastcgi_conf($3, c, NULL);
224 8ad1c570 2021-05-09 op | TFASTCGI TTCP TSTRING {
225 8ad1c570 2021-05-09 op only_oncei(loc->fcgi, "fastcgi");
226 8ad1c570 2021-05-09 op loc->fcgi = fastcgi_conf($3, xstrdup("9000"), NULL);
228 8ad1c570 2021-05-09 op | TFASTCGI TTCP TSTRING TSTRING {
229 8ad1c570 2021-05-09 op only_oncei(loc->fcgi, "fastcgi");
230 8ad1c570 2021-05-09 op loc->fcgi = fastcgi_conf($3, $4, NULL);
232 eb59f87e 2021-02-09 op | TINDEX TSTRING {
233 c705ecb1 2021-05-03 op only_once(loc->index, "index");
234 eb59f87e 2021-02-09 op loc->index = $2;
236 eb59f87e 2021-02-09 op | TLANG TSTRING {
237 c705ecb1 2021-05-03 op only_once(loc->lang, "lang");
238 eb59f87e 2021-02-09 op loc->lang = $2;
240 793835cb 2021-02-23 op | TLOG TBOOL { loc->disable_log = !$2; }
241 02be96c6 2021-02-09 op | TREQUIRE TCLIENT TCA TSTRING {
242 c705ecb1 2021-05-03 op only_once(loc->reqca, "require client ca");
243 adbe6a64 2021-04-30 op ensure_absolute_path($4);
244 02be96c6 2021-02-09 op if ((loc->reqca = load_ca($4)) == NULL)
245 02be96c6 2021-02-09 op yyerror("couldn't load ca cert: %s", $4);
248 fdea6aa0 2021-04-30 op | TROOT TSTRING {
249 c705ecb1 2021-05-03 op only_once(loc->dir, "root");
250 fdea6aa0 2021-04-30 op loc->dir = ensure_absolute_path($2);
252 eb59f87e 2021-02-09 op | TSTRIP TNUM { loc->strip = check_strip_no($2); }
257 b8e64ccd 2021-03-31 op static struct vhost *
258 b8e64ccd 2021-03-31 op new_vhost(void)
260 b8e64ccd 2021-03-31 op return xcalloc(1, sizeof(struct vhost));
263 b8e64ccd 2021-03-31 op static struct location *
264 b8e64ccd 2021-03-31 op new_location(void)
266 fdea6aa0 2021-04-30 op struct location *l;
268 fdea6aa0 2021-04-30 op l = xcalloc(1, sizeof(*l));
269 fdea6aa0 2021-04-30 op l->dirfd = -1;
270 8ad1c570 2021-05-09 op l->fcgi = -1;
275 6abda252 2021-02-06 op yyerror(const char *msg, ...)
279 13ed2fb6 2021-01-27 op goterror = 1;
281 6abda252 2021-02-06 op va_start(ap, msg);
282 6abda252 2021-02-06 op fprintf(stderr, "%s:%d: ", config_path, yylineno);
283 6abda252 2021-02-06 op vfprintf(stderr, msg, ap);
284 a1373913 2021-02-07 op fprintf(stderr, "\n");
289 13ed2fb6 2021-01-27 op parse_portno(const char *p)
291 13ed2fb6 2021-01-27 op const char *errstr;
294 13ed2fb6 2021-01-27 op n = strtonum(p, 0, UINT16_MAX, &errstr);
295 13ed2fb6 2021-01-27 op if (errstr != NULL)
296 2d34f732 2021-02-10 op yyerror("port number is %s: %s", errstr, p);
301 13ed2fb6 2021-01-27 op parse_conf(const char *path)
303 13ed2fb6 2021-01-27 op config_path = path;
304 13ed2fb6 2021-01-27 op if ((yyin = fopen(path, "r")) == NULL)
305 48b69cb2 2021-04-28 op err(1, "cannot open config: %s", path);
307 13ed2fb6 2021-01-27 op fclose(yyin);
309 13ed2fb6 2021-01-27 op if (goterror)
312 b8e64ccd 2021-03-31 op if (TAILQ_FIRST(&hosts)->domain == NULL)
313 48b69cb2 2021-04-28 op errx(1, "no vhost defined in %s", path);
317 e17642a7 2021-02-01 op ensure_absolute_path(char *path)
319 e17642a7 2021-02-01 op if (path == NULL || *path != '/')
320 adbe6a64 2021-04-30 op yyerror("not an absolute path: %s", path);
325 6abda252 2021-02-06 op check_block_code(int n)
327 6abda252 2021-02-06 op if (n < 10 || n >= 70 || (n >= 20 && n <= 29))
328 6abda252 2021-02-06 op yyerror("invalid block code %d", n);
333 6abda252 2021-02-06 op check_block_fmt(char *fmt)
337 6abda252 2021-02-06 op for (s = fmt; *s; ++s) {
338 6abda252 2021-02-06 op if (*s != '%')
340 6abda252 2021-02-06 op switch (*++s) {
348 6abda252 2021-02-06 op yyerror("invalid format specifier %%%c", *s);
356 6abda252 2021-02-06 op check_strip_no(int n)
359 6abda252 2021-02-06 op yyerror("invalid strip number %d", n);
364 a709ddf5 2021-02-07 op check_prefork_num(int n)
366 2c3e53da 2021-03-03 op if (n <= 0 || n >= PROC_MAX)
367 a709ddf5 2021-02-07 op yyerror("invalid prefork number %d", n);
372 49b73ba1 2021-02-10 op advance_loc(void)
374 b8e64ccd 2021-03-31 op loc = new_location();
375 b8e64ccd 2021-03-31 op TAILQ_INSERT_TAIL(&host->locations, loc, locations);
379 c705ecb1 2021-05-03 op only_once(const void *ptr, const char *name)
381 c705ecb1 2021-05-03 op if (ptr != NULL)
382 c705ecb1 2021-05-03 op yyerror("`%s' specified more than once", name);
386 8ad1c570 2021-05-09 op only_oncei(int i, const char *name)
389 8ad1c570 2021-05-09 op yyerror("`%s' specified more than once", name);
393 8ad1c570 2021-05-09 op fastcgi_conf(char *path, char *port, char *prog)
395 8ad1c570 2021-05-09 op struct fcgi *f;
398 8ad1c570 2021-05-09 op for (i = 0; i < FCGI_MAX; ++i) {
399 8ad1c570 2021-05-09 op f = &fcgi[i];
401 8ad1c570 2021-05-09 op if (f->path == NULL) {
403 8ad1c570 2021-05-09 op f->path = path;
404 8ad1c570 2021-05-09 op f->port = port;
405 8ad1c570 2021-05-09 op f->prog = prog;
409 8ad1c570 2021-05-09 op /* XXX: what to do with prog? */
410 8ad1c570 2021-05-09 op if (!strcmp(f->path, path) &&
411 8ad1c570 2021-05-09 op ((port == NULL && f->port == NULL) ||
412 8ad1c570 2021-05-09 op !strcmp(f->port, port))) {
419 8ad1c570 2021-05-09 op yyerror("too much `fastcgi' rules defined.");