Blame


1 0126d91d 2022-09-07 op /*
2 0126d91d 2022-09-07 op * Copyright (c) 2022 Omar Polo <op@omarpolo.com>
3 0126d91d 2022-09-07 op *
4 0126d91d 2022-09-07 op * Permission to use, copy, modify, and distribute this software for any
5 0126d91d 2022-09-07 op * purpose with or without fee is hereby granted, provided that the above
6 0126d91d 2022-09-07 op * copyright notice and this permission notice appear in all copies.
7 0126d91d 2022-09-07 op *
8 0126d91d 2022-09-07 op * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 0126d91d 2022-09-07 op * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 0126d91d 2022-09-07 op * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 0126d91d 2022-09-07 op * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 0126d91d 2022-09-07 op * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 0126d91d 2022-09-07 op * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 0126d91d 2022-09-07 op * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 0126d91d 2022-09-07 op */
16 0126d91d 2022-09-07 op
17 0126d91d 2022-09-07 op #include "gmid.h"
18 0126d91d 2022-09-07 op
19 0126d91d 2022-09-07 op #include <sys/stat.h>
20 0126d91d 2022-09-07 op #include <sys/wait.h>
21 0126d91d 2022-09-07 op
22 0126d91d 2022-09-07 op #include <errno.h>
23 0126d91d 2022-09-07 op #include <fcntl.h>
24 0126d91d 2022-09-07 op #include <getopt.h>
25 0046c1fe 2023-06-06 op #include <locale.h>
26 0126d91d 2022-09-07 op #include <libgen.h>
27 0126d91d 2022-09-07 op #include <signal.h>
28 0126d91d 2022-09-07 op #include <string.h>
29 eae52ad4 2023-06-06 op #include <syslog.h>
30 0126d91d 2022-09-07 op #include <unistd.h>
31 0126d91d 2022-09-07 op
32 eae52ad4 2023-06-06 op #include "log.h"
33 df5058c9 2023-06-05 op
34 797c4609 2023-06-08 op int privsep_process;
35 0126d91d 2022-09-07 op
36 0126d91d 2022-09-07 op static const struct option opts[] = {
37 0126d91d 2022-09-07 op {"help", no_argument, NULL, 'h'},
38 0126d91d 2022-09-07 op {"version", no_argument, NULL, 'V'},
39 0126d91d 2022-09-07 op {NULL, 0, NULL, 0},
40 0126d91d 2022-09-07 op };
41 797c4609 2023-06-08 op
42 797c4609 2023-06-08 op void
43 797c4609 2023-06-08 op log_request(struct client *c, char *meta, size_t l)
44 797c4609 2023-06-08 op {
45 797c4609 2023-06-08 op char hbuf[NI_MAXHOST], sbuf[NI_MAXSERV], b[GEMINI_URL_LEN];
46 797c4609 2023-06-08 op const char *t;
47 797c4609 2023-06-08 op size_t len;
48 797c4609 2023-06-08 op int ec;
49 797c4609 2023-06-08 op
50 797c4609 2023-06-08 op len = sizeof(c->addr);
51 797c4609 2023-06-08 op ec = getnameinfo((struct sockaddr*)&c->addr, len,
52 797c4609 2023-06-08 op hbuf, sizeof(hbuf),
53 797c4609 2023-06-08 op sbuf, sizeof(sbuf),
54 797c4609 2023-06-08 op NI_NUMERICHOST | NI_NUMERICSERV);
55 797c4609 2023-06-08 op if (ec != 0)
56 797c4609 2023-06-08 op fatalx("getnameinfo: %s", gai_strerror(ec));
57 797c4609 2023-06-08 op
58 797c4609 2023-06-08 op if (c->iri.schema != NULL) {
59 797c4609 2023-06-08 op /* serialize the IRI */
60 797c4609 2023-06-08 op strlcpy(b, c->iri.schema, sizeof(b));
61 797c4609 2023-06-08 op strlcat(b, "://", sizeof(b));
62 797c4609 2023-06-08 op
63 797c4609 2023-06-08 op /* log the decoded host name, but if it was invalid
64 797c4609 2023-06-08 op * use the raw one. */
65 797c4609 2023-06-08 op if (*c->domain != '\0')
66 797c4609 2023-06-08 op strlcat(b, c->domain, sizeof(b));
67 797c4609 2023-06-08 op else
68 797c4609 2023-06-08 op strlcat(b, c->iri.host, sizeof(b));
69 797c4609 2023-06-08 op
70 797c4609 2023-06-08 op if (*c->iri.path != '/')
71 797c4609 2023-06-08 op strlcat(b, "/", sizeof(b));
72 797c4609 2023-06-08 op strlcat(b, c->iri.path, sizeof(b)); /* TODO: sanitize UTF8 */
73 797c4609 2023-06-08 op if (*c->iri.query != '\0') { /* TODO: sanitize UTF8 */
74 797c4609 2023-06-08 op strlcat(b, "?", sizeof(b));
75 797c4609 2023-06-08 op strlcat(b, c->iri.query, sizeof(b));
76 797c4609 2023-06-08 op }
77 797c4609 2023-06-08 op } else {
78 797c4609 2023-06-08 op if ((t = c->req) == NULL)
79 797c4609 2023-06-08 op t = "";
80 797c4609 2023-06-08 op strlcpy(b, t, sizeof(b));
81 797c4609 2023-06-08 op }
82 0126d91d 2022-09-07 op
83 797c4609 2023-06-08 op if ((t = memchr(meta, '\r', l)) == NULL)
84 797c4609 2023-06-08 op t = meta + len;
85 797c4609 2023-06-08 op
86 797c4609 2023-06-08 op fprintf(stderr, "%s:%s GET %s %.*s\n", hbuf, sbuf, b,
87 797c4609 2023-06-08 op (int)(t-meta), meta);
88 797c4609 2023-06-08 op }
89 797c4609 2023-06-08 op
90 0126d91d 2022-09-07 op void
91 0126d91d 2022-09-07 op load_local_cert(struct vhost *h, const char *hostname, const char *dir)
92 0126d91d 2022-09-07 op {
93 0126d91d 2022-09-07 op char *cert, *key;
94 0126d91d 2022-09-07 op
95 0126d91d 2022-09-07 op if (asprintf(&cert, "%s/%s.cert.pem", dir, hostname) == -1)
96 2dd5994a 2023-06-06 op fatal("asprintf");
97 0126d91d 2022-09-07 op if (asprintf(&key, "%s/%s.key.pem", dir, hostname) == -1)
98 2dd5994a 2023-06-06 op fatal("asprintf");
99 0126d91d 2022-09-07 op
100 0126d91d 2022-09-07 op if (access(cert, R_OK) == -1 || access(key, R_OK) == -1)
101 0126d91d 2022-09-07 op gen_certificate(hostname, cert, key);
102 0126d91d 2022-09-07 op
103 797c4609 2023-06-08 op h->cert = tls_load_file(cert, &h->certlen, NULL);
104 797c4609 2023-06-08 op if (h->cert == NULL)
105 797c4609 2023-06-08 op fatal("can't load %s", cert);
106 797c4609 2023-06-08 op
107 797c4609 2023-06-08 op h->key = tls_load_file(key, &h->keylen, NULL);
108 797c4609 2023-06-08 op if (h->key == NULL)
109 797c4609 2023-06-08 op fatal("can't load %s", key);
110 797c4609 2023-06-08 op
111 534afd0d 2022-10-05 op strlcpy(h->domain, hostname, sizeof(h->domain));
112 0126d91d 2022-09-07 op }
113 0126d91d 2022-09-07 op
114 0126d91d 2022-09-07 op /* wrapper around dirname(3). dn must be PATH_MAX+1 at least. */
115 0126d91d 2022-09-07 op static void
116 0126d91d 2022-09-07 op pdirname(const char *path, char *dn)
117 0126d91d 2022-09-07 op {
118 0126d91d 2022-09-07 op char p[PATH_MAX+1];
119 0126d91d 2022-09-07 op char *t;
120 0126d91d 2022-09-07 op
121 0126d91d 2022-09-07 op strlcpy(p, path, sizeof(p));
122 0126d91d 2022-09-07 op t = dirname(p);
123 0126d91d 2022-09-07 op memmove(dn, t, strlen(t)+1);
124 0126d91d 2022-09-07 op }
125 0126d91d 2022-09-07 op
126 0126d91d 2022-09-07 op static void
127 0126d91d 2022-09-07 op mkdirs(const char *path, mode_t mode)
128 0126d91d 2022-09-07 op {
129 0126d91d 2022-09-07 op char dname[PATH_MAX+1];
130 0126d91d 2022-09-07 op
131 0126d91d 2022-09-07 op pdirname(path, dname);
132 0126d91d 2022-09-07 op if (!strcmp(dname, "/"))
133 0126d91d 2022-09-07 op return;
134 0126d91d 2022-09-07 op mkdirs(dname, mode);
135 0126d91d 2022-09-07 op if (mkdir(path, mode) != 0 && errno != EEXIST)
136 df5058c9 2023-06-05 op fatal("can't mkdir %s", path);
137 0126d91d 2022-09-07 op }
138 0126d91d 2022-09-07 op
139 0126d91d 2022-09-07 op /* $XDG_DATA_HOME/gmid */
140 0126d91d 2022-09-07 op char *
141 0126d91d 2022-09-07 op data_dir(void)
142 0126d91d 2022-09-07 op {
143 0126d91d 2022-09-07 op const char *home, *xdg;
144 0126d91d 2022-09-07 op char *t;
145 0126d91d 2022-09-07 op
146 0126d91d 2022-09-07 op if ((xdg = getenv("XDG_DATA_HOME")) == NULL) {
147 0126d91d 2022-09-07 op if ((home = getenv("HOME")) == NULL)
148 2dd5994a 2023-06-06 op fatalx("XDG_DATA_HOME and HOME both empty");
149 0126d91d 2022-09-07 op if (asprintf(&t, "%s/.local/share/gmid", home) == -1)
150 2dd5994a 2023-06-06 op fatalx("asprintf");
151 0126d91d 2022-09-07 op } else {
152 0126d91d 2022-09-07 op if (asprintf(&t, "%s/gmid", xdg) == -1)
153 2dd5994a 2023-06-06 op fatal("asprintf");
154 0126d91d 2022-09-07 op }
155 0126d91d 2022-09-07 op
156 0126d91d 2022-09-07 op mkdirs(t, 0755);
157 0126d91d 2022-09-07 op return t;
158 0126d91d 2022-09-07 op }
159 0126d91d 2022-09-07 op
160 0126d91d 2022-09-07 op static int
161 af1dab18 2023-06-09 op serve(struct conf *conf, const char *host, int port, const char *dir)
162 0126d91d 2022-09-07 op {
163 0126d91d 2022-09-07 op struct addrinfo hints, *res, *res0;
164 4267093e 2023-06-06 op int r, error, saved_errno, sock = -1;
165 0126d91d 2022-09-07 op const char *cause = NULL;
166 0126d91d 2022-09-07 op char service[32];
167 0126d91d 2022-09-07 op
168 4267093e 2023-06-06 op r = snprintf(service, sizeof(service), "%d", port);
169 4267093e 2023-06-06 op if (r < 0 || (size_t)r >= sizeof(service))
170 0126d91d 2022-09-07 op fatal("snprintf");
171 0126d91d 2022-09-07 op
172 0126d91d 2022-09-07 op memset(&hints, 0, sizeof(hints));
173 0126d91d 2022-09-07 op hints.ai_family = AF_UNSPEC;
174 0126d91d 2022-09-07 op hints.ai_socktype = SOCK_STREAM;
175 0126d91d 2022-09-07 op hints.ai_flags = AI_PASSIVE;
176 0126d91d 2022-09-07 op error = getaddrinfo(host, service, &hints, &res0);
177 0126d91d 2022-09-07 op if (error)
178 df5058c9 2023-06-05 op fatalx("%s", gai_strerror(error));
179 0126d91d 2022-09-07 op for (res = res0; res; res = res->ai_next) {
180 0126d91d 2022-09-07 op sock = socket(res->ai_family, res->ai_socktype,
181 0126d91d 2022-09-07 op res->ai_protocol);
182 0126d91d 2022-09-07 op if (sock == -1) {
183 0126d91d 2022-09-07 op cause = "socket";
184 0126d91d 2022-09-07 op continue;
185 0126d91d 2022-09-07 op }
186 0126d91d 2022-09-07 op
187 0126d91d 2022-09-07 op if (bind(sock, res->ai_addr, res->ai_addrlen) == -1) {
188 0126d91d 2022-09-07 op cause = "bind";
189 0126d91d 2022-09-07 op saved_errno = errno;
190 0126d91d 2022-09-07 op close(sock);
191 0126d91d 2022-09-07 op errno = saved_errno;
192 0126d91d 2022-09-07 op continue;
193 0126d91d 2022-09-07 op }
194 0126d91d 2022-09-07 op
195 0126d91d 2022-09-07 op if (listen(sock, 5) == -1)
196 0126d91d 2022-09-07 op fatal("listen");
197 0126d91d 2022-09-07 op
198 0126d91d 2022-09-07 op /*
199 0126d91d 2022-09-07 op * for the time being, we're happy as soon as
200 0126d91d 2022-09-07 op * something binds.
201 0126d91d 2022-09-07 op */
202 0126d91d 2022-09-07 op break;
203 0126d91d 2022-09-07 op }
204 0126d91d 2022-09-07 op
205 0126d91d 2022-09-07 op if (sock == -1)
206 0126d91d 2022-09-07 op fatal("%s", cause);
207 0126d91d 2022-09-07 op freeaddrinfo(res0);
208 0126d91d 2022-09-07 op
209 797c4609 2023-06-08 op event_init();
210 797c4609 2023-06-08 op
211 797c4609 2023-06-08 op /* cheating */
212 af1dab18 2023-06-09 op conf->sock4 = sock;
213 af1dab18 2023-06-09 op event_set(&conf->evsock4, conf->sock4, EV_READ|EV_PERSIST,
214 af1dab18 2023-06-09 op do_accept, conf);
215 797c4609 2023-06-08 op
216 797c4609 2023-06-08 op server_init(NULL, NULL, NULL);
217 af1dab18 2023-06-09 op if (server_configure_done(conf) == -1)
218 797c4609 2023-06-08 op fatalx("server configuration failed");
219 797c4609 2023-06-08 op
220 eae52ad4 2023-06-06 op log_info("serving %s on port %d", dir, port);
221 797c4609 2023-06-08 op event_dispatch();
222 797c4609 2023-06-08 op log_info("quitting");
223 797c4609 2023-06-08 op return 0;
224 0126d91d 2022-09-07 op }
225 0126d91d 2022-09-07 op
226 0126d91d 2022-09-07 op static __dead void
227 0126d91d 2022-09-07 op usage(void)
228 0126d91d 2022-09-07 op {
229 0126d91d 2022-09-07 op fprintf(stderr,
230 c5b4db93 2022-09-10 op "Version: " GE_STRING "\n"
231 0126d91d 2022-09-07 op "Usage: %s [-hVv] [-d certs-dir] [-H hostname] [-p port] [dir]\n",
232 0126d91d 2022-09-07 op getprogname());
233 0126d91d 2022-09-07 op exit(1);
234 0126d91d 2022-09-07 op }
235 0126d91d 2022-09-07 op
236 0126d91d 2022-09-07 op int
237 0126d91d 2022-09-07 op main(int argc, char **argv)
238 0126d91d 2022-09-07 op {
239 af1dab18 2023-06-09 op struct conf *conf;
240 0126d91d 2022-09-07 op struct vhost *host;
241 0126d91d 2022-09-07 op struct location *loc;
242 0126d91d 2022-09-07 op const char *errstr, *certs_dir = NULL, *hostname = "localhost";
243 0126d91d 2022-09-07 op char path[PATH_MAX];
244 0126d91d 2022-09-07 op int ch;
245 0126d91d 2022-09-07 op
246 0046c1fe 2023-06-06 op setlocale(LC_CTYPE, "");
247 0046c1fe 2023-06-06 op
248 eae52ad4 2023-06-06 op log_init(1, LOG_DAEMON);
249 eae52ad4 2023-06-06 op log_setverbose(0);
250 af1dab18 2023-06-09 op conf = config_new();
251 0126d91d 2022-09-07 op
252 0126d91d 2022-09-07 op while ((ch = getopt_long(argc, argv, "d:H:hp:Vv", opts, NULL)) != -1) {
253 0126d91d 2022-09-07 op switch (ch) {
254 0126d91d 2022-09-07 op case 'd':
255 0126d91d 2022-09-07 op certs_dir = optarg;
256 0126d91d 2022-09-07 op break;
257 0126d91d 2022-09-07 op case 'H':
258 0126d91d 2022-09-07 op hostname = optarg;
259 0126d91d 2022-09-07 op break;
260 0126d91d 2022-09-07 op case 'h':
261 0126d91d 2022-09-07 op usage();
262 0126d91d 2022-09-07 op break;
263 0126d91d 2022-09-07 op case 'p':
264 af1dab18 2023-06-09 op conf->port = strtonum(optarg, 0, UINT16_MAX, &errstr);
265 0126d91d 2022-09-07 op if (errstr)
266 df5058c9 2023-06-05 op fatalx("port number is %s: %s", errstr,
267 df5058c9 2023-06-05 op optarg);
268 0126d91d 2022-09-07 op break;
269 0126d91d 2022-09-07 op case 'V':
270 c5b4db93 2022-09-10 op puts("Version: " GE_STRING);
271 0126d91d 2022-09-07 op return 0;
272 0126d91d 2022-09-07 op default:
273 0126d91d 2022-09-07 op usage();
274 0126d91d 2022-09-07 op break;
275 0126d91d 2022-09-07 op }
276 0126d91d 2022-09-07 op }
277 0126d91d 2022-09-07 op argc -= optind;
278 0126d91d 2022-09-07 op argv += optind;
279 0126d91d 2022-09-07 op
280 0126d91d 2022-09-07 op if (argc > 1)
281 0126d91d 2022-09-07 op usage();
282 0126d91d 2022-09-07 op
283 0126d91d 2022-09-07 op /* prepare the configuration */
284 af1dab18 2023-06-09 op init_mime(&conf->mime);
285 0126d91d 2022-09-07 op
286 0126d91d 2022-09-07 op if (certs_dir == NULL)
287 0126d91d 2022-09-07 op certs_dir = data_dir();
288 0126d91d 2022-09-07 op
289 0126d91d 2022-09-07 op /* set up the implicit vhost and location */
290 0126d91d 2022-09-07 op host = xcalloc(1, sizeof(*host));
291 af1dab18 2023-06-09 op TAILQ_INSERT_HEAD(&conf->hosts, host, vhosts);
292 0126d91d 2022-09-07 op
293 0126d91d 2022-09-07 op loc = xcalloc(1, sizeof(*loc));
294 0126d91d 2022-09-07 op loc->fcgi = -1;
295 0126d91d 2022-09-07 op TAILQ_INSERT_HEAD(&host->locations, loc, locations);
296 0126d91d 2022-09-07 op
297 0126d91d 2022-09-07 op load_local_cert(host, hostname, certs_dir);
298 0126d91d 2022-09-07 op
299 534afd0d 2022-10-05 op strlcpy(host->domain, "*", sizeof(host->domain));
300 0126d91d 2022-09-07 op loc->auto_index = 1;
301 534afd0d 2022-10-05 op strlcpy(loc->match, "*", sizeof(loc->match));
302 0126d91d 2022-09-07 op
303 0126d91d 2022-09-07 op if (*argv == NULL) {
304 0126d91d 2022-09-07 op if (getcwd(path, sizeof(path)) == NULL)
305 0126d91d 2022-09-07 op fatal("getcwd");
306 534afd0d 2022-10-05 op strlcpy(loc->dir, path, sizeof(loc->dir));
307 534afd0d 2022-10-05 op } else {
308 534afd0d 2022-10-05 op char *tmp;
309 0126d91d 2022-09-07 op
310 534afd0d 2022-10-05 op tmp = absolutify_path(*argv);
311 534afd0d 2022-10-05 op strlcpy(loc->dir, tmp, sizeof(loc->dir));
312 534afd0d 2022-10-05 op free(tmp);
313 534afd0d 2022-10-05 op }
314 534afd0d 2022-10-05 op
315 0126d91d 2022-09-07 op /* start the server */
316 0126d91d 2022-09-07 op signal(SIGPIPE, SIG_IGN);
317 0126d91d 2022-09-07 op setproctitle("%s", loc->dir);
318 af1dab18 2023-06-09 op return serve(conf, hostname, conf->port, loc->dir);
319 0126d91d 2022-09-07 op }