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 15902770 2021-01-15 op #include <err.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 size_t ihost;
34 ca21e100 2021-02-04 op struct location *loc;
35 ca21e100 2021-02-04 op size_t iloc;
36 15902770 2021-01-15 op
37 13ed2fb6 2021-01-27 op int goterror = 0;
38 13ed2fb6 2021-01-27 op const char *config_path;
39 15902770 2021-01-15 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 13ed2fb6 2021-01-27 op
49 15902770 2021-01-15 op %}
50 15902770 2021-01-15 op
51 15902770 2021-01-15 op /* for bison: */
52 15902770 2021-01-15 op /* %define parse.error verbose */
53 15902770 2021-01-15 op
54 15902770 2021-01-15 op %union {
55 15902770 2021-01-15 op char *str;
56 15902770 2021-01-15 op int num;
57 15902770 2021-01-15 op }
58 15902770 2021-01-15 op
59 46af8c6c 2021-01-27 op %token TIPV6 TPORT TPROTOCOLS TMIME TDEFAULT TTYPE
60 a709ddf5 2021-02-07 op %token TCHROOT TUSER TSERVER TPREFORK
61 252908e6 2021-01-24 op %token TLOCATION TCERT TKEY TROOT TCGI TLANG TINDEX TAUTO
62 eef0492e 2021-02-07 op %token TSTRIP TBLOCK TRETURN TENTRYPOINT
63 15902770 2021-01-15 op %token TERR
64 15902770 2021-01-15 op
65 15902770 2021-01-15 op %token <str> TSTRING
66 15902770 2021-01-15 op %token <num> TNUM
67 15902770 2021-01-15 op %token <num> TBOOL
68 15902770 2021-01-15 op
69 15902770 2021-01-15 op %%
70 15902770 2021-01-15 op
71 15902770 2021-01-15 op conf : options vhosts ;
72 15902770 2021-01-15 op
73 15902770 2021-01-15 op options : /* empty */
74 15902770 2021-01-15 op | options option
75 15902770 2021-01-15 op ;
76 15902770 2021-01-15 op
77 46af8c6c 2021-01-27 op option : TIPV6 TBOOL { conf.ipv6 = $2; }
78 15902770 2021-01-15 op | TPORT TNUM { conf.port = $2; }
79 5bc3c98e 2021-01-15 op | TPROTOCOLS TSTRING {
80 5bc3c98e 2021-01-15 op if (tls_config_parse_protocols(&conf.protos, $2) == -1)
81 5bc3c98e 2021-01-15 op errx(1, "invalid protocols string \"%s\"", $2);
82 5bc3c98e 2021-01-15 op }
83 b2a6b613 2021-01-21 op | TMIME TSTRING TSTRING { add_mime(&conf.mime, $2, $3); }
84 ae08ec7d 2021-01-25 op | TCHROOT TSTRING { conf.chroot = $2; }
85 ae08ec7d 2021-01-25 op | TUSER TSTRING { conf.user = $2; }
86 a709ddf5 2021-02-07 op | TPREFORK TNUM { conf.prefork = check_prefork_num($2); }
87 15902770 2021-01-15 op ;
88 15902770 2021-01-15 op
89 15902770 2021-01-15 op vhosts : /* empty */
90 15902770 2021-01-15 op | vhosts vhost
91 15902770 2021-01-15 op ;
92 15902770 2021-01-15 op
93 c8b74339 2021-01-24 op vhost : TSERVER TSTRING '{' servopts locations '}' {
94 ca21e100 2021-02-04 op host->locations[0].match = xstrdup("*");
95 15902770 2021-01-15 op host->domain = $2;
96 15902770 2021-01-15 op
97 cbeee4ca 2021-01-28 op if (strstr($2, "xn--") != NULL) {
98 cbeee4ca 2021-01-28 op warnx("%s:%d \"%s\" looks like punycode: "
99 415ac7a2 2021-01-28 op "you should use the decoded hostname.",
100 415ac7a2 2021-01-28 op config_path, yylineno, $2);
101 cbeee4ca 2021-01-28 op }
102 cbeee4ca 2021-01-28 op
103 15902770 2021-01-15 op if (host->cert == NULL || host->key == NULL ||
104 15902770 2021-01-15 op host->dir == NULL)
105 15902770 2021-01-15 op errx(1, "invalid vhost definition: %s", $2);
106 c8b74339 2021-01-24 op
107 15902770 2021-01-15 op if (++ihost == HOSTSLEN)
108 15902770 2021-01-15 op errx(1, "too much vhosts defined");
109 c8b74339 2021-01-24 op
110 c8b74339 2021-01-24 op host++;
111 c8b74339 2021-01-24 op loc = &host->locations[0];
112 c8b74339 2021-01-24 op iloc = 0;
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 e17642a7 2021-02-01 op servopt : TCERT TSTRING { host->cert = ensure_absolute_path($2); }
122 e17642a7 2021-02-01 op | TKEY TSTRING { host->key = ensure_absolute_path($2); }
123 e17642a7 2021-02-01 op | TROOT TSTRING { host->dir = ensure_absolute_path($2); }
124 15902770 2021-01-15 op | TCGI TSTRING {
125 15902770 2021-01-15 op /* drop the starting '/', if any */
126 709f4c94 2021-02-04 op if (*$2 == '/')
127 709f4c94 2021-02-04 op memmove($2, $2+1, strlen($2));
128 709f4c94 2021-02-04 op host->cgi = $2;
129 05c23a54 2021-01-19 op }
130 e3ddf390 2021-02-06 op | TENTRYPOINT TSTRING {
131 e3ddf390 2021-02-06 op if (host->entrypoint != NULL)
132 e3ddf390 2021-02-06 op yyerror("`entrypoint' specified more than once");
133 e3ddf390 2021-02-06 op while (*$2 == '/')
134 e3ddf390 2021-02-06 op memmove($2, $2+1, strlen($2));
135 e3ddf390 2021-02-06 op host->entrypoint = $2;
136 e3ddf390 2021-02-06 op }
137 c8b74339 2021-01-24 op | locopt
138 c8b74339 2021-01-24 op ;
139 c8b74339 2021-01-24 op
140 c8b74339 2021-01-24 op locations : /* empty */
141 c8b74339 2021-01-24 op | locations location
142 c8b74339 2021-01-24 op ;
143 c8b74339 2021-01-24 op
144 c8b74339 2021-01-24 op location : TLOCATION TSTRING '{' locopts '}' {
145 c8b74339 2021-01-24 op loc->match = $2;
146 c8b74339 2021-01-24 op if (++iloc == LOCLEN)
147 c8b74339 2021-01-24 op errx(1, "too much location rules defined");
148 c8b74339 2021-01-24 op loc++;
149 6119e13e 2021-01-19 op }
150 c8b74339 2021-01-24 op | error '}'
151 c8b74339 2021-01-24 op ;
152 c8b74339 2021-01-24 op
153 c8b74339 2021-01-24 op locopts : /* empty */
154 c8b74339 2021-01-24 op | locopts locopt
155 c8b74339 2021-01-24 op ;
156 c8b74339 2021-01-24 op
157 c8b74339 2021-01-24 op locopt : TDEFAULT TTYPE TSTRING {
158 fe5967cd 2021-01-27 op if (loc->default_mime != NULL)
159 fe5967cd 2021-01-27 op yyerror("`default type' specified more than once");
160 c8b74339 2021-01-24 op loc->default_mime = $3;
161 c8b74339 2021-01-24 op }
162 05c23a54 2021-01-19 op | TLANG TSTRING {
163 fe5967cd 2021-01-27 op if (loc->lang != NULL)
164 fe5967cd 2021-01-27 op yyerror("`lang' specified more than once");
165 c8b74339 2021-01-24 op loc->lang = $2;
166 05c23a54 2021-01-19 op }
167 e7a2a99b 2021-01-24 op | TINDEX TSTRING {
168 fe5967cd 2021-01-27 op if (loc->index != NULL)
169 fe5967cd 2021-01-27 op yyerror("`index' specified more than once");
170 c8b74339 2021-01-24 op loc->index = $2;
171 e7a2a99b 2021-01-24 op }
172 252908e6 2021-01-24 op | TAUTO TINDEX TBOOL { loc->auto_index = $3 ? 1 : -1; }
173 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM TSTRING {
174 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
175 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
176 6abda252 2021-02-06 op loc->block_fmt = check_block_fmt($4);
177 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
178 6abda252 2021-02-06 op }
179 6abda252 2021-02-06 op | TBLOCK TRETURN TNUM {
180 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
181 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
182 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
183 6abda252 2021-02-06 op loc->block_code = check_block_code($3);
184 6abda252 2021-02-06 op if ($3 >= 30 && $3 < 40)
185 6abda252 2021-02-06 op yyerror("missing `meta' for block return %d", $3);
186 6abda252 2021-02-06 op }
187 6abda252 2021-02-06 op | TBLOCK {
188 6abda252 2021-02-06 op if (loc->block_fmt != NULL)
189 6abda252 2021-02-06 op yyerror("`block' rule specified more than once");
190 6abda252 2021-02-06 op loc->block_fmt = xstrdup("temporary failure");
191 6abda252 2021-02-06 op loc->block_code = 40;
192 6abda252 2021-02-06 op }
193 6abda252 2021-02-06 op | TSTRIP TNUM { loc->strip = check_strip_no($2); }
194 15902770 2021-01-15 op ;
195 13ed2fb6 2021-01-27 op
196 13ed2fb6 2021-01-27 op %%
197 13ed2fb6 2021-01-27 op
198 13ed2fb6 2021-01-27 op void
199 6abda252 2021-02-06 op yyerror(const char *msg, ...)
200 13ed2fb6 2021-01-27 op {
201 6abda252 2021-02-06 op va_list ap;
202 6abda252 2021-02-06 op
203 13ed2fb6 2021-01-27 op goterror = 1;
204 6abda252 2021-02-06 op
205 6abda252 2021-02-06 op va_start(ap, msg);
206 6abda252 2021-02-06 op fprintf(stderr, "%s:%d: ", config_path, yylineno);
207 6abda252 2021-02-06 op vfprintf(stderr, msg, ap);
208 6abda252 2021-02-06 op va_end(ap);
209 13ed2fb6 2021-01-27 op }
210 13ed2fb6 2021-01-27 op
211 13ed2fb6 2021-01-27 op int
212 13ed2fb6 2021-01-27 op parse_portno(const char *p)
213 13ed2fb6 2021-01-27 op {
214 13ed2fb6 2021-01-27 op const char *errstr;
215 13ed2fb6 2021-01-27 op int n;
216 13ed2fb6 2021-01-27 op
217 13ed2fb6 2021-01-27 op n = strtonum(p, 0, UINT16_MAX, &errstr);
218 13ed2fb6 2021-01-27 op if (errstr != NULL)
219 13ed2fb6 2021-01-27 op errx(1, "port number is %s: %s", errstr, p);
220 13ed2fb6 2021-01-27 op return n;
221 13ed2fb6 2021-01-27 op }
222 13ed2fb6 2021-01-27 op
223 13ed2fb6 2021-01-27 op void
224 13ed2fb6 2021-01-27 op parse_conf(const char *path)
225 13ed2fb6 2021-01-27 op {
226 ca21e100 2021-02-04 op host = &hosts[0];
227 ca21e100 2021-02-04 op ihost = 0;
228 ca21e100 2021-02-04 op loc = &hosts[0].locations[0];
229 ca21e100 2021-02-04 op iloc = 0;
230 ca21e100 2021-02-04 op
231 13ed2fb6 2021-01-27 op config_path = path;
232 13ed2fb6 2021-01-27 op if ((yyin = fopen(path, "r")) == NULL)
233 13ed2fb6 2021-01-27 op fatal("cannot open config %s", path);
234 13ed2fb6 2021-01-27 op yyparse();
235 13ed2fb6 2021-01-27 op fclose(yyin);
236 13ed2fb6 2021-01-27 op
237 13ed2fb6 2021-01-27 op if (goterror)
238 13ed2fb6 2021-01-27 op exit(1);
239 13ed2fb6 2021-01-27 op }
240 e17642a7 2021-02-01 op
241 e17642a7 2021-02-01 op char *
242 e17642a7 2021-02-01 op ensure_absolute_path(char *path)
243 e17642a7 2021-02-01 op {
244 e17642a7 2021-02-01 op if (path == NULL || *path != '/')
245 e17642a7 2021-02-01 op yyerror("not an absolute path");
246 e17642a7 2021-02-01 op return path;
247 e17642a7 2021-02-01 op }
248 6abda252 2021-02-06 op
249 6abda252 2021-02-06 op int
250 6abda252 2021-02-06 op check_block_code(int n)
251 6abda252 2021-02-06 op {
252 6abda252 2021-02-06 op if (n < 10 || n >= 70 || (n >= 20 && n <= 29))
253 6abda252 2021-02-06 op yyerror("invalid block code %d", n);
254 6abda252 2021-02-06 op return n;
255 6abda252 2021-02-06 op }
256 6abda252 2021-02-06 op
257 6abda252 2021-02-06 op char *
258 6abda252 2021-02-06 op check_block_fmt(char *fmt)
259 6abda252 2021-02-06 op {
260 6abda252 2021-02-06 op char *s;
261 6abda252 2021-02-06 op
262 6abda252 2021-02-06 op for (s = fmt; *s; ++s) {
263 6abda252 2021-02-06 op if (*s != '%')
264 6abda252 2021-02-06 op continue;
265 6abda252 2021-02-06 op switch (*++s) {
266 6abda252 2021-02-06 op case '%':
267 6abda252 2021-02-06 op case 'p':
268 6abda252 2021-02-06 op case 'q':
269 6abda252 2021-02-06 op case 'P':
270 6abda252 2021-02-06 op case 'N':
271 6abda252 2021-02-06 op break;
272 6abda252 2021-02-06 op default:
273 6abda252 2021-02-06 op yyerror("invalid format specifier %%%c", *s);
274 6abda252 2021-02-06 op }
275 6abda252 2021-02-06 op }
276 6abda252 2021-02-06 op
277 6abda252 2021-02-06 op return fmt;
278 6abda252 2021-02-06 op }
279 6abda252 2021-02-06 op
280 6abda252 2021-02-06 op int
281 6abda252 2021-02-06 op check_strip_no(int n)
282 6abda252 2021-02-06 op {
283 6abda252 2021-02-06 op if (n <= 0)
284 6abda252 2021-02-06 op yyerror("invalid strip number %d", n);
285 a709ddf5 2021-02-07 op return n;
286 a709ddf5 2021-02-07 op }
287 a709ddf5 2021-02-07 op
288 a709ddf5 2021-02-07 op int
289 a709ddf5 2021-02-07 op check_prefork_num(int n)
290 a709ddf5 2021-02-07 op {
291 a709ddf5 2021-02-07 op if (n <= 0)
292 a709ddf5 2021-02-07 op yyerror("invalid prefork number %d", n);
293 6abda252 2021-02-06 op return n;
294 6abda252 2021-02-06 op }