2 a596b957 2022-07-14 tracey * Copyright (c) 2016-2019, 2020-2021 Tracey Emery <tracey@traceyemery.net>
3 a596b957 2022-07-14 tracey * Copyright (c) 2004, 2005 Esben Norby <norby@openbsd.org>
4 a596b957 2022-07-14 tracey * Copyright (c) 2004 Ryan McBride <mcbride@openbsd.org>
5 a596b957 2022-07-14 tracey * Copyright (c) 2002, 2003, 2004 Henning Brauer <henning@openbsd.org>
6 a596b957 2022-07-14 tracey * Copyright (c) 2001 Markus Friedl. All rights reserved.
7 a596b957 2022-07-14 tracey * Copyright (c) 2001 Daniel Hartmeier. All rights reserved.
8 a596b957 2022-07-14 tracey * Copyright (c) 2001 Theo de Raadt. All rights reserved.
10 a596b957 2022-07-14 tracey * Permission to use, copy, modify, and distribute this software for any
11 a596b957 2022-07-14 tracey * purpose with or without fee is hereby granted, provided that the above
12 a596b957 2022-07-14 tracey * copyright notice and this permission notice appear in all copies.
14 a596b957 2022-07-14 tracey * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
15 a596b957 2022-07-14 tracey * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
16 a596b957 2022-07-14 tracey * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
17 a596b957 2022-07-14 tracey * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
18 a596b957 2022-07-14 tracey * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
19 a596b957 2022-07-14 tracey * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
20 a596b957 2022-07-14 tracey * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
24 a596b957 2022-07-14 tracey #include <sys/ioctl.h>
25 a596b957 2022-07-14 tracey #include <sys/types.h>
26 a596b957 2022-07-14 tracey #include <sys/queue.h>
27 a596b957 2022-07-14 tracey #include <sys/socket.h>
28 a596b957 2022-07-14 tracey #include <sys/stat.h>
30 a596b957 2022-07-14 tracey #include <net/if.h>
31 a596b957 2022-07-14 tracey #include <netinet/in.h>
33 a596b957 2022-07-14 tracey #include <arpa/inet.h>
35 a596b957 2022-07-14 tracey #include <ctype.h>
36 a596b957 2022-07-14 tracey #include <err.h>
37 a596b957 2022-07-14 tracey #include <errno.h>
38 a596b957 2022-07-14 tracey #include <event.h>
39 a596b957 2022-07-14 tracey #include <ifaddrs.h>
40 a596b957 2022-07-14 tracey #include <imsg.h>
41 a596b957 2022-07-14 tracey #include <limits.h>
42 a596b957 2022-07-14 tracey #include <netdb.h>
43 a596b957 2022-07-14 tracey #include <stdarg.h>
44 a596b957 2022-07-14 tracey #include <stdlib.h>
45 a596b957 2022-07-14 tracey #include <stdio.h>
46 a596b957 2022-07-14 tracey #include <string.h>
47 a596b957 2022-07-14 tracey #include <syslog.h>
48 a596b957 2022-07-14 tracey #include <unistd.h>
50 df2d3cd2 2023-03-11 op #include "got_reference.h"
52 a596b957 2022-07-14 tracey #include "proc.h"
53 a596b957 2022-07-14 tracey #include "gotwebd.h"
55 a596b957 2022-07-14 tracey TAILQ_HEAD(files, file) files = TAILQ_HEAD_INITIALIZER(files);
56 a596b957 2022-07-14 tracey static struct file {
57 a596b957 2022-07-14 tracey TAILQ_ENTRY(file) entry;
58 a596b957 2022-07-14 tracey FILE *stream;
59 a596b957 2022-07-14 tracey char *name;
60 a596b957 2022-07-14 tracey int lineno;
61 a596b957 2022-07-14 tracey int errors;
63 a596b957 2022-07-14 tracey struct file *newfile(const char *, int);
64 a596b957 2022-07-14 tracey static void closefile(struct file *);
65 a596b957 2022-07-14 tracey int check_file_secrecy(int, const char *);
66 a596b957 2022-07-14 tracey int yyparse(void);
67 a596b957 2022-07-14 tracey int yylex(void);
68 a596b957 2022-07-14 tracey int yyerror(const char *, ...)
69 a596b957 2022-07-14 tracey __attribute__((__format__ (printf, 1, 2)))
70 a596b957 2022-07-14 tracey __attribute__((__nonnull__ (1)));
71 a596b957 2022-07-14 tracey int kw_cmp(const void *, const void *);
72 a596b957 2022-07-14 tracey int lookup(char *);
73 a596b957 2022-07-14 tracey int lgetc(int);
74 a596b957 2022-07-14 tracey int lungetc(int);
75 a596b957 2022-07-14 tracey int findeol(void);
77 a596b957 2022-07-14 tracey TAILQ_HEAD(symhead, sym) symhead = TAILQ_HEAD_INITIALIZER(symhead);
78 a596b957 2022-07-14 tracey struct sym {
79 a596b957 2022-07-14 tracey TAILQ_ENTRY(sym) entry;
81 a596b957 2022-07-14 tracey int persist;
82 a596b957 2022-07-14 tracey char *nam;
83 a596b957 2022-07-14 tracey char *val;
86 a596b957 2022-07-14 tracey int symset(const char *, const char *, int);
87 a596b957 2022-07-14 tracey char *symget(const char *);
89 a596b957 2022-07-14 tracey static int errors;
91 a596b957 2022-07-14 tracey static struct gotwebd *gotwebd;
92 a596b957 2022-07-14 tracey static struct server *new_srv;
93 a596b957 2022-07-14 tracey static struct server *conf_new_server(const char *);
94 a596b957 2022-07-14 tracey int getservice(const char *);
97 89cfaaa7 2023-11-15 op int get_addrs(const char *, const char *, struct server *);
98 67d8de2a 2022-08-29 stsp int addr_dup_check(struct addresslist *, struct address *,
99 67d8de2a 2022-08-29 stsp const char *, const char *);
100 67d8de2a 2022-08-29 stsp int add_addr(struct server *, struct address *);
102 a596b957 2022-07-14 tracey typedef struct {
104 a596b957 2022-07-14 tracey long long number;
105 a596b957 2022-07-14 tracey char *string;
107 a596b957 2022-07-14 tracey int lineno;
108 a596b957 2022-07-14 tracey } YYSTYPE;
112 d8473d93 2022-08-11 stsp %token LISTEN WWW_PATH MAX_REPOS SITE_NAME SITE_OWNER SITE_LINK LOGO
113 a596b957 2022-07-14 tracey %token LOGO_URL SHOW_REPO_OWNER SHOW_REPO_AGE SHOW_REPO_DESCRIPTION
114 a596b957 2022-07-14 tracey %token MAX_REPOS_DISPLAY REPOS_PATH MAX_COMMITS_DISPLAY ON ERROR
115 d5996b9e 2022-10-31 landry %token SHOW_SITE_OWNER SHOW_REPO_CLONEURL PORT PREFORK RESPECT_EXPORTOK
116 3a1c1a1b 2023-01-04 op %token UNIX_SOCKET UNIX_SOCKET_NAME SERVER CHROOT CUSTOM_CSS SOCKET
118 a596b957 2022-07-14 tracey %token <v.string> STRING
119 a596b957 2022-07-14 tracey %token <v.number> NUMBER
120 a596b957 2022-07-14 tracey %type <v.number> boolean
121 c5e111b9 2023-11-15 op %type <v.string> listen_addr
125 47b307cd 2022-10-02 op grammar : /* empty */
126 a596b957 2022-07-14 tracey | grammar '\n'
127 47b307cd 2022-10-02 op | grammar varset '\n'
128 a596b957 2022-07-14 tracey | grammar main '\n'
129 a596b957 2022-07-14 tracey | grammar server '\n'
130 47b307cd 2022-10-02 op | grammar error '\n' { file->errors++; }
133 47b307cd 2022-10-02 op varset : STRING '=' STRING {
134 47b307cd 2022-10-02 op char *s = $1;
135 47b307cd 2022-10-02 op while (*s++) {
136 47b307cd 2022-10-02 op if (isspace((unsigned char)*s)) {
137 47b307cd 2022-10-02 op yyerror("macro name cannot contain "
138 47b307cd 2022-10-02 op "whitespace");
144 47b307cd 2022-10-02 op if (symset($1, $3, 0) == -1)
145 47b307cd 2022-10-02 op fatal("cannot store variable");
151 a596b957 2022-07-14 tracey boolean : STRING {
152 a596b957 2022-07-14 tracey if (strcasecmp($1, "1") == 0 ||
153 a596b957 2022-07-14 tracey strcasecmp($1, "on") == 0)
155 a596b957 2022-07-14 tracey else if (strcasecmp($1, "0") == 0 ||
156 031687ba 2023-06-15 op strcasecmp($1, "off") == 0)
159 a596b957 2022-07-14 tracey yyerror("invalid boolean value '%s'", $1);
160 a596b957 2022-07-14 tracey free($1);
163 a596b957 2022-07-14 tracey free($1);
165 a596b957 2022-07-14 tracey | ON { $$ = 1; }
167 1a0c81fb 2023-06-15 op if ($1 != 0 && $1 != 1) {
168 1a0c81fb 2023-06-15 op yyerror("invalid boolean value '%lld'", $1);
175 c5e111b9 2023-11-15 op listen_addr : '*' { $$ = NULL; }
179 a596b957 2022-07-14 tracey main : PREFORK NUMBER {
180 1a0c81fb 2023-06-15 op if ($2 <= 0 || $2 > PROC_MAX_INSTANCES) {
181 1a0c81fb 2023-06-15 op yyerror("prefork is %s: %lld",
182 1a0c81fb 2023-06-15 op $2 <= 0 ? "too small" : "too large", $2);
185 a596b957 2022-07-14 tracey gotwebd->prefork_gotwebd = $2;
187 a596b957 2022-07-14 tracey | CHROOT STRING {
188 a678036d 2023-06-15 op if (*$2 == '\0') {
189 a678036d 2023-06-15 op yyerror("chroot path can't be an empty"
195 a596b957 2022-07-14 tracey n = strlcpy(gotwebd->httpd_chroot, $2,
196 a596b957 2022-07-14 tracey sizeof(gotwebd->httpd_chroot));
197 a596b957 2022-07-14 tracey if (n >= sizeof(gotwebd->httpd_chroot)) {
198 a596b957 2022-07-14 tracey yyerror("%s: httpd_chroot truncated", __func__);
199 a596b957 2022-07-14 tracey free($2);
202 a596b957 2022-07-14 tracey free($2);
204 a596b957 2022-07-14 tracey | UNIX_SOCKET boolean {
205 a596b957 2022-07-14 tracey gotwebd->unix_socket = $2;
207 a596b957 2022-07-14 tracey | UNIX_SOCKET_NAME STRING {
208 a596b957 2022-07-14 tracey n = snprintf(gotwebd->unix_socket_name,
209 a596b957 2022-07-14 tracey sizeof(gotwebd->unix_socket_name), "%s%s",
210 f4a5cef1 2023-06-15 op gotwebd->httpd_chroot, $2);
212 438d0cc3 2022-08-16 op (size_t)n >= sizeof(gotwebd->unix_socket_name)) {
213 a596b957 2022-07-14 tracey yyerror("%s: unix_socket_name truncated",
214 a596b957 2022-07-14 tracey __func__);
215 a596b957 2022-07-14 tracey free($2);
218 a596b957 2022-07-14 tracey free($2);
222 a596b957 2022-07-14 tracey server : SERVER STRING {
223 a596b957 2022-07-14 tracey struct server *srv;
225 2ad48e9a 2022-08-16 stsp TAILQ_FOREACH(srv, &gotwebd->servers, entry) {
226 a596b957 2022-07-14 tracey if (strcmp(srv->name, $2) == 0) {
227 a596b957 2022-07-14 tracey yyerror("server name exists '%s'", $2);
228 a596b957 2022-07-14 tracey free($2);
233 a596b957 2022-07-14 tracey new_srv = conf_new_server($2);
234 a596b957 2022-07-14 tracey log_debug("adding server %s", $2);
235 a596b957 2022-07-14 tracey free($2);
237 a596b957 2022-07-14 tracey | SERVER STRING {
238 a596b957 2022-07-14 tracey struct server *srv;
240 2ad48e9a 2022-08-16 stsp TAILQ_FOREACH(srv, &gotwebd->servers, entry) {
241 a596b957 2022-07-14 tracey if (strcmp(srv->name, $2) == 0) {
242 a596b957 2022-07-14 tracey yyerror("server name exists '%s'", $2);
243 a596b957 2022-07-14 tracey free($2);
248 a596b957 2022-07-14 tracey new_srv = conf_new_server($2);
249 a596b957 2022-07-14 tracey log_debug("adding server %s", $2);
250 a596b957 2022-07-14 tracey free($2);
251 a596b957 2022-07-14 tracey } '{' optnl serveropts2 '}' {
255 a596b957 2022-07-14 tracey serveropts1 : REPOS_PATH STRING {
256 a596b957 2022-07-14 tracey n = strlcpy(new_srv->repos_path, $2,
257 a596b957 2022-07-14 tracey sizeof(new_srv->repos_path));
258 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->repos_path)) {
259 a596b957 2022-07-14 tracey yyerror("%s: repos_path truncated", __func__);
260 a596b957 2022-07-14 tracey free($2);
263 a596b957 2022-07-14 tracey free($2);
265 a596b957 2022-07-14 tracey | SITE_NAME STRING {
266 a596b957 2022-07-14 tracey n = strlcpy(new_srv->site_name, $2,
267 a596b957 2022-07-14 tracey sizeof(new_srv->site_name));
268 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->site_name)) {
269 a596b957 2022-07-14 tracey yyerror("%s: site_name truncated", __func__);
270 a596b957 2022-07-14 tracey free($2);
273 a596b957 2022-07-14 tracey free($2);
275 a596b957 2022-07-14 tracey | SITE_OWNER STRING {
276 a596b957 2022-07-14 tracey n = strlcpy(new_srv->site_owner, $2,
277 a596b957 2022-07-14 tracey sizeof(new_srv->site_owner));
278 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->site_owner)) {
279 a596b957 2022-07-14 tracey yyerror("%s: site_owner truncated", __func__);
280 a596b957 2022-07-14 tracey free($2);
283 a596b957 2022-07-14 tracey free($2);
285 a596b957 2022-07-14 tracey | SITE_LINK STRING {
286 a596b957 2022-07-14 tracey n = strlcpy(new_srv->site_link, $2,
287 a596b957 2022-07-14 tracey sizeof(new_srv->site_link));
288 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->site_link)) {
289 a596b957 2022-07-14 tracey yyerror("%s: site_link truncated", __func__);
290 a596b957 2022-07-14 tracey free($2);
293 a596b957 2022-07-14 tracey free($2);
295 a596b957 2022-07-14 tracey | LOGO STRING {
296 a596b957 2022-07-14 tracey n = strlcpy(new_srv->logo, $2, sizeof(new_srv->logo));
297 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->logo)) {
298 a596b957 2022-07-14 tracey yyerror("%s: logo truncated", __func__);
299 a596b957 2022-07-14 tracey free($2);
302 a596b957 2022-07-14 tracey free($2);
304 a596b957 2022-07-14 tracey | LOGO_URL STRING {
305 a596b957 2022-07-14 tracey n = strlcpy(new_srv->logo_url, $2,
306 a596b957 2022-07-14 tracey sizeof(new_srv->logo_url));
307 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->logo_url)) {
308 a596b957 2022-07-14 tracey yyerror("%s: logo_url truncated", __func__);
309 a596b957 2022-07-14 tracey free($2);
312 a596b957 2022-07-14 tracey free($2);
314 a596b957 2022-07-14 tracey | CUSTOM_CSS STRING {
315 a596b957 2022-07-14 tracey n = strlcpy(new_srv->custom_css, $2,
316 a596b957 2022-07-14 tracey sizeof(new_srv->custom_css));
317 a596b957 2022-07-14 tracey if (n >= sizeof(new_srv->custom_css)) {
318 a596b957 2022-07-14 tracey yyerror("%s: custom_css truncated", __func__);
319 a596b957 2022-07-14 tracey free($2);
322 a596b957 2022-07-14 tracey free($2);
324 89cfaaa7 2023-11-15 op | LISTEN ON listen_addr PORT STRING {
325 89cfaaa7 2023-11-15 op if (get_addrs($3, $5, new_srv) == -1) {
326 67d8de2a 2022-08-29 stsp yyerror("could not get addrs");
331 6c8aa58f 2022-08-30 stsp new_srv->fcgi_socket = 1;
333 89cfaaa7 2023-11-15 op | LISTEN ON listen_addr PORT NUMBER {
334 89cfaaa7 2023-11-15 op char portno[32];
337 89cfaaa7 2023-11-15 op n = snprintf(portno, sizeof(portno), "%lld",
338 89cfaaa7 2023-11-15 op (long long)$5);
339 89cfaaa7 2023-11-15 op if (n < 0 || (size_t)n >= sizeof(portno))
340 89cfaaa7 2023-11-15 op fatalx("port number too long: %lld",
341 89cfaaa7 2023-11-15 op (long long)$5);
343 89cfaaa7 2023-11-15 op if (get_addrs($3, portno, new_srv) == -1) {
344 89cfaaa7 2023-11-15 op yyerror("could not get addrs");
348 89cfaaa7 2023-11-15 op new_srv->fcgi_socket = 1;
350 3a1c1a1b 2023-01-04 op | LISTEN ON SOCKET STRING {
351 031687ba 2023-06-15 op if (strcasecmp($4, "off") == 0) {
352 3a1c1a1b 2023-01-04 op new_srv->unix_socket = 0;
357 3a1c1a1b 2023-01-04 op new_srv->unix_socket = 1;
359 3a1c1a1b 2023-01-04 op n = snprintf(new_srv->unix_socket_name,
360 3a1c1a1b 2023-01-04 op sizeof(new_srv->unix_socket_name), "%s%s",
361 f4a5cef1 2023-06-15 op gotwebd->httpd_chroot, $4);
363 3a1c1a1b 2023-01-04 op (size_t)n >= sizeof(new_srv->unix_socket_name)) {
364 3a1c1a1b 2023-01-04 op yyerror("%s: unix_socket_name truncated",
371 a596b957 2022-07-14 tracey | MAX_REPOS NUMBER {
372 1a0c81fb 2023-06-15 op if ($2 <= 0) {
373 1a0c81fb 2023-06-15 op yyerror("max_repos is too small: %lld", $2);
376 1a0c81fb 2023-06-15 op new_srv->max_repos = $2;
378 a596b957 2022-07-14 tracey | SHOW_SITE_OWNER boolean {
379 a596b957 2022-07-14 tracey new_srv->show_site_owner = $2;
381 a596b957 2022-07-14 tracey | SHOW_REPO_OWNER boolean {
382 a596b957 2022-07-14 tracey new_srv->show_repo_owner = $2;
384 a596b957 2022-07-14 tracey | SHOW_REPO_AGE boolean {
385 a596b957 2022-07-14 tracey new_srv->show_repo_age = $2;
387 a596b957 2022-07-14 tracey | SHOW_REPO_DESCRIPTION boolean {
388 a596b957 2022-07-14 tracey new_srv->show_repo_description = $2;
390 a596b957 2022-07-14 tracey | SHOW_REPO_CLONEURL boolean {
391 a596b957 2022-07-14 tracey new_srv->show_repo_cloneurl = $2;
393 d5996b9e 2022-10-31 landry | RESPECT_EXPORTOK boolean {
394 d5996b9e 2022-10-31 landry new_srv->respect_exportok = $2;
396 a596b957 2022-07-14 tracey | MAX_REPOS_DISPLAY NUMBER {
397 20f27972 2023-06-19 op if ($2 < 0) {
398 1a0c81fb 2023-06-15 op yyerror("max_repos_display is too small: %lld",
402 1a0c81fb 2023-06-15 op new_srv->max_repos_display = $2;
404 a596b957 2022-07-14 tracey | MAX_COMMITS_DISPLAY NUMBER {
405 f4425f95 2023-06-14 op if ($2 <= 1) {
406 f4425f95 2023-06-14 op yyerror("max_commits_display is too small:"
407 f4425f95 2023-06-14 op " %lld", $2);
410 f4425f95 2023-06-14 op new_srv->max_commits_display = $2;
414 a596b957 2022-07-14 tracey serveropts2 : serveropts2 serveropts1 nl
415 a596b957 2022-07-14 tracey | serveropts1 optnl
418 a596b957 2022-07-14 tracey nl : '\n' optnl
421 a596b957 2022-07-14 tracey optnl : '\n' optnl /* zero or more newlines */
422 a596b957 2022-07-14 tracey | /* empty */
427 a596b957 2022-07-14 tracey struct keywords {
428 a596b957 2022-07-14 tracey const char *k_name;
429 a596b957 2022-07-14 tracey int k_val;
433 a596b957 2022-07-14 tracey yyerror(const char *fmt, ...)
435 a596b957 2022-07-14 tracey va_list ap;
436 a596b957 2022-07-14 tracey char *msg;
438 a596b957 2022-07-14 tracey file->errors++;
439 a596b957 2022-07-14 tracey va_start(ap, fmt);
440 a596b957 2022-07-14 tracey if (vasprintf(&msg, fmt, ap) == -1)
441 a596b957 2022-07-14 tracey fatalx("yyerror vasprintf");
442 a596b957 2022-07-14 tracey va_end(ap);
443 a596b957 2022-07-14 tracey logit(LOG_CRIT, "%s:%d: %s", file->name, yylval.lineno, msg);
444 a596b957 2022-07-14 tracey free(msg);
445 a596b957 2022-07-14 tracey return (0);
449 a596b957 2022-07-14 tracey kw_cmp(const void *k, const void *e)
451 a596b957 2022-07-14 tracey return (strcmp(k, ((const struct keywords *)e)->k_name));
455 a596b957 2022-07-14 tracey lookup(char *s)
457 a596b957 2022-07-14 tracey /* This has to be sorted always. */
458 a596b957 2022-07-14 tracey static const struct keywords keywords[] = {
459 a596b957 2022-07-14 tracey { "chroot", CHROOT },
460 a596b957 2022-07-14 tracey { "custom_css", CUSTOM_CSS },
461 d8473d93 2022-08-11 stsp { "listen", LISTEN },
462 a596b957 2022-07-14 tracey { "logo", LOGO },
463 8556b86b 2023-01-02 op { "logo_url", LOGO_URL },
464 a596b957 2022-07-14 tracey { "max_commits_display", MAX_COMMITS_DISPLAY },
465 a596b957 2022-07-14 tracey { "max_repos", MAX_REPOS },
466 a596b957 2022-07-14 tracey { "max_repos_display", MAX_REPOS_DISPLAY },
467 d8473d93 2022-08-11 stsp { "on", ON },
468 a596b957 2022-07-14 tracey { "port", PORT },
469 a596b957 2022-07-14 tracey { "prefork", PREFORK },
470 a596b957 2022-07-14 tracey { "repos_path", REPOS_PATH },
471 d5996b9e 2022-10-31 landry { "respect_exportok", RESPECT_EXPORTOK },
472 a596b957 2022-07-14 tracey { "server", SERVER },
473 a596b957 2022-07-14 tracey { "show_repo_age", SHOW_REPO_AGE },
474 a596b957 2022-07-14 tracey { "show_repo_cloneurl", SHOW_REPO_CLONEURL },
475 a596b957 2022-07-14 tracey { "show_repo_description", SHOW_REPO_DESCRIPTION },
476 a596b957 2022-07-14 tracey { "show_repo_owner", SHOW_REPO_OWNER },
477 a596b957 2022-07-14 tracey { "show_site_owner", SHOW_SITE_OWNER },
478 a596b957 2022-07-14 tracey { "site_link", SITE_LINK },
479 a596b957 2022-07-14 tracey { "site_name", SITE_NAME },
480 a596b957 2022-07-14 tracey { "site_owner", SITE_OWNER },
481 3a1c1a1b 2023-01-04 op { "socket", SOCKET },
482 a596b957 2022-07-14 tracey { "unix_socket", UNIX_SOCKET },
483 a596b957 2022-07-14 tracey { "unix_socket_name", UNIX_SOCKET_NAME },
485 a596b957 2022-07-14 tracey const struct keywords *p;
487 a596b957 2022-07-14 tracey p = bsearch(s, keywords, sizeof(keywords)/sizeof(keywords[0]),
488 a596b957 2022-07-14 tracey sizeof(keywords[0]), kw_cmp);
491 a596b957 2022-07-14 tracey return (p->k_val);
493 a596b957 2022-07-14 tracey return (STRING);
496 a596b957 2022-07-14 tracey #define MAXPUSHBACK 128
498 a596b957 2022-07-14 tracey unsigned char *parsebuf;
499 a596b957 2022-07-14 tracey int parseindex;
500 a596b957 2022-07-14 tracey unsigned char pushback_buffer[MAXPUSHBACK];
501 a596b957 2022-07-14 tracey int pushback_index = 0;
504 a596b957 2022-07-14 tracey lgetc(int quotec)
506 a596b957 2022-07-14 tracey int c, next;
508 a596b957 2022-07-14 tracey if (parsebuf) {
509 a596b957 2022-07-14 tracey /* Read character from the parsebuffer instead of input. */
510 a596b957 2022-07-14 tracey if (parseindex >= 0) {
511 a596b957 2022-07-14 tracey c = parsebuf[parseindex++];
512 a596b957 2022-07-14 tracey if (c != '\0')
513 a596b957 2022-07-14 tracey return (c);
514 a596b957 2022-07-14 tracey parsebuf = NULL;
516 a596b957 2022-07-14 tracey parseindex++;
519 a596b957 2022-07-14 tracey if (pushback_index)
520 a596b957 2022-07-14 tracey return (pushback_buffer[--pushback_index]);
522 a596b957 2022-07-14 tracey if (quotec) {
523 a596b957 2022-07-14 tracey c = getc(file->stream);
524 a596b957 2022-07-14 tracey if (c == EOF)
525 a596b957 2022-07-14 tracey yyerror("reached end of file while parsing "
526 a596b957 2022-07-14 tracey "quoted string");
527 a596b957 2022-07-14 tracey return (c);
530 a596b957 2022-07-14 tracey c = getc(file->stream);
531 a596b957 2022-07-14 tracey while (c == '\\') {
532 a596b957 2022-07-14 tracey next = getc(file->stream);
533 a596b957 2022-07-14 tracey if (next != '\n') {
534 a596b957 2022-07-14 tracey c = next;
537 a596b957 2022-07-14 tracey yylval.lineno = file->lineno;
538 a596b957 2022-07-14 tracey file->lineno++;
539 a596b957 2022-07-14 tracey c = getc(file->stream);
542 a596b957 2022-07-14 tracey return (c);
546 a596b957 2022-07-14 tracey lungetc(int c)
548 a596b957 2022-07-14 tracey if (c == EOF)
549 a596b957 2022-07-14 tracey return (EOF);
550 a596b957 2022-07-14 tracey if (parsebuf) {
551 a596b957 2022-07-14 tracey parseindex--;
552 a596b957 2022-07-14 tracey if (parseindex >= 0)
553 a596b957 2022-07-14 tracey return (c);
555 a596b957 2022-07-14 tracey if (pushback_index < MAXPUSHBACK-1)
556 a596b957 2022-07-14 tracey return (pushback_buffer[pushback_index++] = c);
558 a596b957 2022-07-14 tracey return (EOF);
562 a596b957 2022-07-14 tracey findeol(void)
566 a596b957 2022-07-14 tracey parsebuf = NULL;
568 a596b957 2022-07-14 tracey /* Skip to either EOF or the first real EOL. */
569 a596b957 2022-07-14 tracey while (1) {
570 a596b957 2022-07-14 tracey if (pushback_index)
571 a596b957 2022-07-14 tracey c = pushback_buffer[--pushback_index];
573 a596b957 2022-07-14 tracey c = lgetc(0);
574 a596b957 2022-07-14 tracey if (c == '\n') {
575 a596b957 2022-07-14 tracey file->lineno++;
578 a596b957 2022-07-14 tracey if (c == EOF)
581 a596b957 2022-07-14 tracey return (ERROR);
585 a596b957 2022-07-14 tracey yylex(void)
587 a596b957 2022-07-14 tracey unsigned char buf[8096];
588 a596b957 2022-07-14 tracey unsigned char *p, *val;
589 a596b957 2022-07-14 tracey int quotec, next, c;
590 a596b957 2022-07-14 tracey int token;
594 a596b957 2022-07-14 tracey c = lgetc(0);
595 a596b957 2022-07-14 tracey while (c == ' ' || c == '\t')
596 a596b957 2022-07-14 tracey c = lgetc(0); /* nothing */
598 a596b957 2022-07-14 tracey yylval.lineno = file->lineno;
599 a596b957 2022-07-14 tracey if (c == '#') {
600 a596b957 2022-07-14 tracey c = lgetc(0);
601 a596b957 2022-07-14 tracey while (c != '\n' && c != EOF)
602 a596b957 2022-07-14 tracey c = lgetc(0); /* nothing */
604 a596b957 2022-07-14 tracey if (c == '$' && parsebuf == NULL) {
605 a596b957 2022-07-14 tracey while (1) {
606 a596b957 2022-07-14 tracey c = lgetc(0);
607 a596b957 2022-07-14 tracey if (c == EOF)
608 a596b957 2022-07-14 tracey return (0);
610 a596b957 2022-07-14 tracey if (p + 1 >= buf + sizeof(buf) - 1) {
611 a596b957 2022-07-14 tracey yyerror("string too long");
612 a596b957 2022-07-14 tracey return (findeol());
614 a596b957 2022-07-14 tracey if (isalnum(c) || c == '_') {
615 a596b957 2022-07-14 tracey *p++ = c;
616 a596b957 2022-07-14 tracey continue;
618 a596b957 2022-07-14 tracey *p = '\0';
619 a596b957 2022-07-14 tracey lungetc(c);
622 a596b957 2022-07-14 tracey val = symget(buf);
623 a596b957 2022-07-14 tracey if (val == NULL) {
624 a596b957 2022-07-14 tracey yyerror("macro '%s' not defined", buf);
625 a596b957 2022-07-14 tracey return (findeol());
627 a596b957 2022-07-14 tracey parsebuf = val;
628 a596b957 2022-07-14 tracey parseindex = 0;
629 a596b957 2022-07-14 tracey goto top;
632 a596b957 2022-07-14 tracey switch (c) {
633 a596b957 2022-07-14 tracey case '\'':
634 a596b957 2022-07-14 tracey case '"':
635 a596b957 2022-07-14 tracey quotec = c;
636 a596b957 2022-07-14 tracey while (1) {
637 a596b957 2022-07-14 tracey c = lgetc(quotec);
638 a596b957 2022-07-14 tracey if (c == EOF)
639 a596b957 2022-07-14 tracey return (0);
640 a596b957 2022-07-14 tracey if (c == '\n') {
641 a596b957 2022-07-14 tracey file->lineno++;
642 a596b957 2022-07-14 tracey continue;
643 a596b957 2022-07-14 tracey } else if (c == '\\') {
644 a596b957 2022-07-14 tracey next = lgetc(quotec);
645 a596b957 2022-07-14 tracey if (next == EOF)
646 a596b957 2022-07-14 tracey return (0);
647 a596b957 2022-07-14 tracey if (next == quotec || c == ' ' || c == '\t')
648 a596b957 2022-07-14 tracey c = next;
649 a596b957 2022-07-14 tracey else if (next == '\n') {
650 a596b957 2022-07-14 tracey file->lineno++;
651 a596b957 2022-07-14 tracey continue;
653 a596b957 2022-07-14 tracey lungetc(next);
654 a596b957 2022-07-14 tracey } else if (c == quotec) {
655 a596b957 2022-07-14 tracey *p = '\0';
657 a596b957 2022-07-14 tracey } else if (c == '\0') {
658 a596b957 2022-07-14 tracey yyerror("syntax error");
659 a596b957 2022-07-14 tracey return (findeol());
661 a596b957 2022-07-14 tracey if (p + 1 >= buf + sizeof(buf) - 1) {
662 a596b957 2022-07-14 tracey yyerror("string too long");
663 a596b957 2022-07-14 tracey return (findeol());
665 a596b957 2022-07-14 tracey *p++ = c;
667 a596b957 2022-07-14 tracey yylval.v.string = strdup(buf);
668 a596b957 2022-07-14 tracey if (yylval.v.string == NULL)
669 a596b957 2022-07-14 tracey err(1, "yylex: strdup");
670 a596b957 2022-07-14 tracey return (STRING);
673 a596b957 2022-07-14 tracey #define allowed_to_end_number(x) \
674 a596b957 2022-07-14 tracey (isspace(x) || x == ')' || x ==',' || x == '/' || x == '}' || x == '=')
676 a596b957 2022-07-14 tracey if (c == '-' || isdigit(c)) {
678 a596b957 2022-07-14 tracey *p++ = c;
679 a596b957 2022-07-14 tracey if ((unsigned)(p-buf) >= sizeof(buf)) {
680 a596b957 2022-07-14 tracey yyerror("string too long");
681 a596b957 2022-07-14 tracey return (findeol());
683 a596b957 2022-07-14 tracey c = lgetc(0);
684 a596b957 2022-07-14 tracey } while (c != EOF && isdigit(c));
685 a596b957 2022-07-14 tracey lungetc(c);
686 a596b957 2022-07-14 tracey if (p == buf + 1 && buf[0] == '-')
687 a596b957 2022-07-14 tracey goto nodigits;
688 a596b957 2022-07-14 tracey if (c == EOF || allowed_to_end_number(c)) {
689 a596b957 2022-07-14 tracey const char *errstr = NULL;
691 a596b957 2022-07-14 tracey *p = '\0';
692 a596b957 2022-07-14 tracey yylval.v.number = strtonum(buf, LLONG_MIN,
693 a596b957 2022-07-14 tracey LLONG_MAX, &errstr);
694 a596b957 2022-07-14 tracey if (errstr) {
695 a596b957 2022-07-14 tracey yyerror("\"%s\" invalid number: %s",
696 a596b957 2022-07-14 tracey buf, errstr);
697 a596b957 2022-07-14 tracey return (findeol());
699 a596b957 2022-07-14 tracey return (NUMBER);
701 a596b957 2022-07-14 tracey nodigits:
702 a596b957 2022-07-14 tracey while (p > buf + 1)
703 a596b957 2022-07-14 tracey lungetc(*--p);
704 a596b957 2022-07-14 tracey c = *--p;
705 a596b957 2022-07-14 tracey if (c == '-')
706 a596b957 2022-07-14 tracey return (c);
710 a596b957 2022-07-14 tracey #define allowed_in_string(x) \
711 a596b957 2022-07-14 tracey (isalnum(x) || (ispunct(x) && x != '(' && x != ')' && \
712 a596b957 2022-07-14 tracey x != '{' && x != '}' && \
713 a596b957 2022-07-14 tracey x != '!' && x != '=' && x != '#' && \
714 a596b957 2022-07-14 tracey x != ','))
716 a596b957 2022-07-14 tracey if (isalnum(c) || c == ':' || c == '_') {
718 a596b957 2022-07-14 tracey *p++ = c;
719 a596b957 2022-07-14 tracey if ((unsigned)(p-buf) >= sizeof(buf)) {
720 a596b957 2022-07-14 tracey yyerror("string too long");
721 a596b957 2022-07-14 tracey return (findeol());
723 a596b957 2022-07-14 tracey c = lgetc(0);
724 a596b957 2022-07-14 tracey } while (c != EOF && (allowed_in_string(c)));
725 a596b957 2022-07-14 tracey lungetc(c);
726 a596b957 2022-07-14 tracey *p = '\0';
727 a596b957 2022-07-14 tracey token = lookup(buf);
728 a596b957 2022-07-14 tracey if (token == STRING) {
729 a596b957 2022-07-14 tracey yylval.v.string = strdup(buf);
730 a596b957 2022-07-14 tracey if (yylval.v.string == NULL)
731 a596b957 2022-07-14 tracey err(1, "yylex: strdup");
733 a596b957 2022-07-14 tracey return (token);
735 a596b957 2022-07-14 tracey if (c == '\n') {
736 a596b957 2022-07-14 tracey yylval.lineno = file->lineno;
737 a596b957 2022-07-14 tracey file->lineno++;
739 a596b957 2022-07-14 tracey if (c == EOF)
740 a596b957 2022-07-14 tracey return (0);
741 a596b957 2022-07-14 tracey return (c);
745 a596b957 2022-07-14 tracey check_file_secrecy(int fd, const char *fname)
747 a596b957 2022-07-14 tracey struct stat st;
749 a596b957 2022-07-14 tracey if (fstat(fd, &st)) {
750 a596b957 2022-07-14 tracey log_warn("cannot stat %s", fname);
751 a596b957 2022-07-14 tracey return (-1);
753 a596b957 2022-07-14 tracey if (st.st_uid != 0 && st.st_uid != getuid()) {
754 a596b957 2022-07-14 tracey log_warnx("%s: owner not root or current user", fname);
755 a596b957 2022-07-14 tracey return (-1);
757 a596b957 2022-07-14 tracey if (st.st_mode & (S_IWGRP | S_IXGRP | S_IRWXO)) {
758 a596b957 2022-07-14 tracey log_warnx("%s: group writable or world read/writable", fname);
759 a596b957 2022-07-14 tracey return (-1);
761 a596b957 2022-07-14 tracey return (0);
764 a596b957 2022-07-14 tracey struct file *
765 a596b957 2022-07-14 tracey newfile(const char *name, int secret)
767 a596b957 2022-07-14 tracey struct file *nfile;
769 a596b957 2022-07-14 tracey nfile = calloc(1, sizeof(struct file));
770 a596b957 2022-07-14 tracey if (nfile == NULL) {
771 a596b957 2022-07-14 tracey log_warn("calloc");
772 a596b957 2022-07-14 tracey return (NULL);
774 a596b957 2022-07-14 tracey nfile->name = strdup(name);
775 a596b957 2022-07-14 tracey if (nfile->name == NULL) {
776 a596b957 2022-07-14 tracey log_warn("strdup");
777 a596b957 2022-07-14 tracey free(nfile);
778 a596b957 2022-07-14 tracey return (NULL);
780 a596b957 2022-07-14 tracey nfile->stream = fopen(nfile->name, "r");
781 a596b957 2022-07-14 tracey if (nfile->stream == NULL) {
782 a596b957 2022-07-14 tracey /* no warning, we don't require a conf file */
783 a596b957 2022-07-14 tracey free(nfile->name);
784 a596b957 2022-07-14 tracey free(nfile);
785 a596b957 2022-07-14 tracey return (NULL);
786 a596b957 2022-07-14 tracey } else if (secret &&
787 a596b957 2022-07-14 tracey check_file_secrecy(fileno(nfile->stream), nfile->name)) {
788 a596b957 2022-07-14 tracey fclose(nfile->stream);
789 a596b957 2022-07-14 tracey free(nfile->name);
790 a596b957 2022-07-14 tracey free(nfile);
791 a596b957 2022-07-14 tracey return (NULL);
793 a596b957 2022-07-14 tracey nfile->lineno = 1;
794 a596b957 2022-07-14 tracey return (nfile);
797 a596b957 2022-07-14 tracey static void
798 a596b957 2022-07-14 tracey closefile(struct file *xfile)
800 a596b957 2022-07-14 tracey fclose(xfile->stream);
801 a596b957 2022-07-14 tracey free(xfile->name);
802 a596b957 2022-07-14 tracey free(xfile);
805 a0037b73 2022-08-03 stsp static void
806 a0037b73 2022-08-03 stsp add_default_server(void)
808 a0037b73 2022-08-03 stsp new_srv = conf_new_server(D_SITENAME);
809 a0037b73 2022-08-03 stsp log_debug("%s: adding default server %s", __func__, D_SITENAME);
813 a596b957 2022-07-14 tracey parse_config(const char *filename, struct gotwebd *env)
815 a596b957 2022-07-14 tracey struct sym *sym, *next;
817 a596b957 2022-07-14 tracey if (config_init(env) == -1)
818 a596b957 2022-07-14 tracey fatalx("failed to initialize configuration");
820 a596b957 2022-07-14 tracey gotwebd = env;
822 a0037b73 2022-08-03 stsp file = newfile(filename, 0);
823 a0037b73 2022-08-03 stsp if (file == NULL) {
824 a0037b73 2022-08-03 stsp add_default_server();
825 a0037b73 2022-08-03 stsp sockets_parse_sockets(env);
826 a0037b73 2022-08-03 stsp /* just return, as we don't require a conf file */
827 a0037b73 2022-08-03 stsp return (0);
830 a596b957 2022-07-14 tracey yyparse();
831 a596b957 2022-07-14 tracey errors = file->errors;
832 a596b957 2022-07-14 tracey closefile(file);
834 a596b957 2022-07-14 tracey /* Free macros and check which have not been used. */
835 a596b957 2022-07-14 tracey TAILQ_FOREACH_SAFE(sym, &symhead, entry, next) {
836 a596b957 2022-07-14 tracey if ((gotwebd->gotwebd_verbose > 1) && !sym->used)
837 a596b957 2022-07-14 tracey fprintf(stderr, "warning: macro '%s' not used\n",
838 a596b957 2022-07-14 tracey sym->nam);
839 a596b957 2022-07-14 tracey if (!sym->persist) {
840 a596b957 2022-07-14 tracey free(sym->nam);
841 a596b957 2022-07-14 tracey free(sym->val);
842 a596b957 2022-07-14 tracey TAILQ_REMOVE(&symhead, sym, entry);
843 a596b957 2022-07-14 tracey free(sym);
847 a596b957 2022-07-14 tracey if (errors)
848 a596b957 2022-07-14 tracey return (-1);
850 a596b957 2022-07-14 tracey /* just add default server if no config specified */
851 a0037b73 2022-08-03 stsp if (gotwebd->server_cnt == 0)
852 a0037b73 2022-08-03 stsp add_default_server();
854 a596b957 2022-07-14 tracey /* setup our listening sockets */
855 a596b957 2022-07-14 tracey sockets_parse_sockets(env);
857 a596b957 2022-07-14 tracey return (0);
860 a596b957 2022-07-14 tracey struct server *
861 a596b957 2022-07-14 tracey conf_new_server(const char *name)
863 a596b957 2022-07-14 tracey struct server *srv = NULL;
865 a596b957 2022-07-14 tracey srv = calloc(1, sizeof(*srv));
866 a596b957 2022-07-14 tracey if (srv == NULL)
867 a596b957 2022-07-14 tracey fatalx("%s: calloc", __func__);
869 a596b957 2022-07-14 tracey n = strlcpy(srv->name, name, sizeof(srv->name));
870 a596b957 2022-07-14 tracey if (n >= sizeof(srv->name))
871 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
872 a596b957 2022-07-14 tracey n = snprintf(srv->unix_socket_name,
873 a596b957 2022-07-14 tracey sizeof(srv->unix_socket_name), "%s%s", D_HTTPD_CHROOT,
874 a596b957 2022-07-14 tracey D_UNIX_SOCKET);
875 438d0cc3 2022-08-16 op if (n < 0 || (size_t)n >= sizeof(srv->unix_socket_name))
876 a596b957 2022-07-14 tracey fatalx("%s: snprintf", __func__);
877 a596b957 2022-07-14 tracey n = strlcpy(srv->repos_path, D_GOTPATH,
878 a596b957 2022-07-14 tracey sizeof(srv->repos_path));
879 a596b957 2022-07-14 tracey if (n >= sizeof(srv->repos_path))
880 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
881 a596b957 2022-07-14 tracey n = strlcpy(srv->site_name, D_SITENAME,
882 a596b957 2022-07-14 tracey sizeof(srv->site_name));
883 a596b957 2022-07-14 tracey if (n >= sizeof(srv->site_name))
884 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
885 a596b957 2022-07-14 tracey n = strlcpy(srv->site_owner, D_SITEOWNER,
886 a596b957 2022-07-14 tracey sizeof(srv->site_owner));
887 a596b957 2022-07-14 tracey if (n >= sizeof(srv->site_owner))
888 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
889 a596b957 2022-07-14 tracey n = strlcpy(srv->site_link, D_SITELINK,
890 a596b957 2022-07-14 tracey sizeof(srv->site_link));
891 a596b957 2022-07-14 tracey if (n >= sizeof(srv->site_link))
892 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
893 a596b957 2022-07-14 tracey n = strlcpy(srv->logo, D_GOTLOGO,
894 a596b957 2022-07-14 tracey sizeof(srv->logo));
895 a596b957 2022-07-14 tracey if (n >= sizeof(srv->logo))
896 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
897 a596b957 2022-07-14 tracey n = strlcpy(srv->logo_url, D_GOTURL, sizeof(srv->logo_url));
898 a596b957 2022-07-14 tracey if (n >= sizeof(srv->logo_url))
899 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
900 a596b957 2022-07-14 tracey n = strlcpy(srv->custom_css, D_GOTWEBCSS, sizeof(srv->custom_css));
901 a596b957 2022-07-14 tracey if (n >= sizeof(srv->custom_css))
902 a596b957 2022-07-14 tracey fatalx("%s: strlcpy", __func__);
904 a596b957 2022-07-14 tracey srv->show_site_owner = D_SHOWSOWNER;
905 a596b957 2022-07-14 tracey srv->show_repo_owner = D_SHOWROWNER;
906 a596b957 2022-07-14 tracey srv->show_repo_age = D_SHOWAGE;
907 a596b957 2022-07-14 tracey srv->show_repo_description = D_SHOWDESC;
908 a596b957 2022-07-14 tracey srv->show_repo_cloneurl = D_SHOWURL;
909 d5996b9e 2022-10-31 landry srv->respect_exportok = D_RESPECTEXPORTOK;
911 a596b957 2022-07-14 tracey srv->max_repos_display = D_MAXREPODISP;
912 a596b957 2022-07-14 tracey srv->max_commits_display = D_MAXCOMMITDISP;
913 a596b957 2022-07-14 tracey srv->max_repos = D_MAXREPO;
915 a596b957 2022-07-14 tracey srv->unix_socket = 1;
916 6c8aa58f 2022-08-30 stsp srv->fcgi_socket = 0;
918 e087e1f6 2022-08-16 stsp TAILQ_INIT(&srv->al);
919 2ad48e9a 2022-08-16 stsp TAILQ_INSERT_TAIL(&gotwebd->servers, srv, entry);
920 a596b957 2022-07-14 tracey gotwebd->server_cnt++;
922 a596b957 2022-07-14 tracey return srv;
926 a596b957 2022-07-14 tracey symset(const char *nam, const char *val, int persist)
928 a596b957 2022-07-14 tracey struct sym *sym;
930 a596b957 2022-07-14 tracey TAILQ_FOREACH(sym, &symhead, entry) {
931 a596b957 2022-07-14 tracey if (strcmp(nam, sym->nam) == 0)
935 a596b957 2022-07-14 tracey if (sym != NULL) {
936 a596b957 2022-07-14 tracey if (sym->persist == 1)
937 a596b957 2022-07-14 tracey return (0);
939 a596b957 2022-07-14 tracey free(sym->nam);
940 a596b957 2022-07-14 tracey free(sym->val);
941 a596b957 2022-07-14 tracey TAILQ_REMOVE(&symhead, sym, entry);
942 a596b957 2022-07-14 tracey free(sym);
945 a596b957 2022-07-14 tracey sym = calloc(1, sizeof(*sym));
946 a596b957 2022-07-14 tracey if (sym == NULL)
947 a596b957 2022-07-14 tracey return (-1);
949 a596b957 2022-07-14 tracey sym->nam = strdup(nam);
950 a596b957 2022-07-14 tracey if (sym->nam == NULL) {
951 a596b957 2022-07-14 tracey free(sym);
952 a596b957 2022-07-14 tracey return (-1);
954 a596b957 2022-07-14 tracey sym->val = strdup(val);
955 a596b957 2022-07-14 tracey if (sym->val == NULL) {
956 a596b957 2022-07-14 tracey free(sym->nam);
957 a596b957 2022-07-14 tracey free(sym);
958 a596b957 2022-07-14 tracey return (-1);
960 a596b957 2022-07-14 tracey sym->used = 0;
961 a596b957 2022-07-14 tracey sym->persist = persist;
962 a596b957 2022-07-14 tracey TAILQ_INSERT_TAIL(&symhead, sym, entry);
963 a596b957 2022-07-14 tracey return (0);
967 a596b957 2022-07-14 tracey cmdline_symset(char *s)
969 a596b957 2022-07-14 tracey char *sym, *val;
972 a596b957 2022-07-14 tracey val = strrchr(s, '=');
973 a596b957 2022-07-14 tracey if (val == NULL)
974 a596b957 2022-07-14 tracey return (-1);
976 4cdd299d 2022-09-05 op sym = strndup(s, val - s);
977 a596b957 2022-07-14 tracey if (sym == NULL)
978 4cdd299d 2022-09-05 op fatal("%s: strndup", __func__);
980 a596b957 2022-07-14 tracey ret = symset(sym, val + 1, 1);
981 a596b957 2022-07-14 tracey free(sym);
983 a596b957 2022-07-14 tracey return (ret);
987 a596b957 2022-07-14 tracey symget(const char *nam)
989 a596b957 2022-07-14 tracey struct sym *sym;
991 a596b957 2022-07-14 tracey TAILQ_FOREACH(sym, &symhead, entry) {
992 a596b957 2022-07-14 tracey if (strcmp(nam, sym->nam) == 0) {
993 a596b957 2022-07-14 tracey sym->used = 1;
994 a596b957 2022-07-14 tracey return (sym->val);
997 a596b957 2022-07-14 tracey return (NULL);
1001 89cfaaa7 2023-11-15 op get_addrs(const char *hostname, const char *servname, struct server *new_srv)
1003 a596b957 2022-07-14 tracey struct addrinfo hints, *res0, *res;
1005 cdfd248a 2023-11-15 op struct sockaddr_in *sin;
1006 cdfd248a 2023-11-15 op struct sockaddr_in6 *sin6;
1007 a596b957 2022-07-14 tracey struct address *h;
1009 a596b957 2022-07-14 tracey memset(&hints, 0, sizeof(hints));
1010 fb307946 2023-05-29 op hints.ai_family = AF_UNSPEC;
1011 89cfaaa7 2023-11-15 op hints.ai_socktype = SOCK_STREAM;
1012 c5e111b9 2023-11-15 op hints.ai_flags = AI_PASSIVE | AI_ADDRCONFIG;
1013 89cfaaa7 2023-11-15 op error = getaddrinfo(hostname, servname, &hints, &res0);
1014 a596b957 2022-07-14 tracey if (error) {
1015 89cfaaa7 2023-11-15 op log_warnx("%s: could not parse \"%s:%s\": %s", __func__,
1016 89cfaaa7 2023-11-15 op hostname, servname, gai_strerror(error));
1017 a596b957 2022-07-14 tracey return (-1);
1020 c5e111b9 2023-11-15 op for (res = res0; res; res = res->ai_next) {
1021 a596b957 2022-07-14 tracey if ((h = calloc(1, sizeof(*h))) == NULL)
1022 a596b957 2022-07-14 tracey fatal(__func__);
1024 89cfaaa7 2023-11-15 op if (hostname == NULL) {
1025 c5e111b9 2023-11-15 op strlcpy(h->ifname, "*", sizeof(h->ifname));
1027 89cfaaa7 2023-11-15 op if (strlcpy(h->ifname, hostname, sizeof(h->ifname)) >=
1028 a596b957 2022-07-14 tracey sizeof(h->ifname)) {
1029 b8b20b3c 2023-11-15 op log_warnx("%s: address truncated: %s",
1030 b8b20b3c 2023-11-15 op __func__, hostname);
1031 a596b957 2022-07-14 tracey freeaddrinfo(res0);
1032 a596b957 2022-07-14 tracey free(h);
1033 a596b957 2022-07-14 tracey return (-1);
1036 a596b957 2022-07-14 tracey h->ss.ss_family = res->ai_family;
1038 cdfd248a 2023-11-15 op memcpy(&h->ss, res->ai_addr, res->ai_addrlen);
1039 cdfd248a 2023-11-15 op h->slen = res->ai_addrlen;
1041 89cfaaa7 2023-11-15 op switch (res->ai_family) {
1042 89cfaaa7 2023-11-15 op case AF_INET:
1043 cdfd248a 2023-11-15 op sin = (struct sockaddr_in *)res->ai_addr;
1044 cdfd248a 2023-11-15 op h->port = ntohs(sin->sin_port);
1046 89cfaaa7 2023-11-15 op case AF_INET6:
1047 cdfd248a 2023-11-15 op sin6 = (struct sockaddr_in6 *)res->ai_addr;
1048 cdfd248a 2023-11-15 op h->port = ntohs(sin6->sin6_port);
1051 89cfaaa7 2023-11-15 op fatalx("unknown address family %d", res->ai_family);
1054 67d8de2a 2022-08-29 stsp if (add_addr(new_srv, h))
1055 67d8de2a 2022-08-29 stsp return -1;
1057 a596b957 2022-07-14 tracey freeaddrinfo(res0);
1058 a596b957 2022-07-14 tracey return (0);
1062 67d8de2a 2022-08-29 stsp addr_dup_check(struct addresslist *al, struct address *h, const char *new_srv,
1063 67d8de2a 2022-08-29 stsp const char *other_srv)
1065 67d8de2a 2022-08-29 stsp struct address *a;
1067 67d8de2a 2022-08-29 stsp char buf[INET6_ADDRSTRLEN];
1068 67d8de2a 2022-08-29 stsp const char *addrstr;
1070 67d8de2a 2022-08-29 stsp TAILQ_FOREACH(a, al, entry) {
1071 cdfd248a 2023-11-15 op if (a->slen != h->slen ||
1072 cdfd248a 2023-11-15 op memcmp(&a->ss, &h->ss, a->slen) != 0)
1075 67d8de2a 2022-08-29 stsp switch (h->ss.ss_family) {
1076 67d8de2a 2022-08-29 stsp case AF_INET:
1077 67d8de2a 2022-08-29 stsp ia = &((struct sockaddr_in *)(&h->ss))->sin_addr;
1079 67d8de2a 2022-08-29 stsp case AF_INET6:
1080 67d8de2a 2022-08-29 stsp ia = &((struct sockaddr_in6 *)(&h->ss))->sin6_addr;
1083 67d8de2a 2022-08-29 stsp yyerror("unknown address family: %d", h->ss.ss_family);
1084 67d8de2a 2022-08-29 stsp return -1;
1086 67d8de2a 2022-08-29 stsp addrstr = inet_ntop(h->ss.ss_family, ia, buf, sizeof(buf));
1087 67d8de2a 2022-08-29 stsp if (addrstr) {
1088 67d8de2a 2022-08-29 stsp if (other_srv) {
1089 67d8de2a 2022-08-29 stsp yyerror("server %s: duplicate fcgi listen "
1090 67d8de2a 2022-08-29 stsp "address %s:%d, already used by server %s",
1091 67d8de2a 2022-08-29 stsp new_srv, addrstr, h->port, other_srv);
1093 67d8de2a 2022-08-29 stsp log_warnx("server: %s: duplicate fcgi listen "
1094 67d8de2a 2022-08-29 stsp "address %s:%d", new_srv, addrstr, h->port);
1097 67d8de2a 2022-08-29 stsp if (other_srv) {
1098 67d8de2a 2022-08-29 stsp yyerror("server: %s: duplicate fcgi listen "
1099 67d8de2a 2022-08-29 stsp "address, already used by server %s",
1100 67d8de2a 2022-08-29 stsp new_srv, other_srv);
1102 67d8de2a 2022-08-29 stsp log_warnx("server %s: duplicate fcgi listen "
1103 67d8de2a 2022-08-29 stsp "address", new_srv);
1107 67d8de2a 2022-08-29 stsp return -1;
1114 67d8de2a 2022-08-29 stsp add_addr(struct server *new_srv, struct address *h)
1116 67d8de2a 2022-08-29 stsp struct server *srv;
1118 67d8de2a 2022-08-29 stsp /* Address cannot be shared between different servers. */
1119 67d8de2a 2022-08-29 stsp TAILQ_FOREACH(srv, &gotwebd->servers, entry) {
1120 67d8de2a 2022-08-29 stsp if (srv == new_srv)
1122 67d8de2a 2022-08-29 stsp if (addr_dup_check(&srv->al, h, new_srv->name, srv->name))
1123 67d8de2a 2022-08-29 stsp return -1;
1126 67d8de2a 2022-08-29 stsp /* Tolerate duplicate address lines within the scope of a server. */
1127 67d8de2a 2022-08-29 stsp if (addr_dup_check(&new_srv->al, h, NULL, NULL) == 0)
1128 67d8de2a 2022-08-29 stsp TAILQ_INSERT_TAIL(&new_srv->al, h, entry);