2 * Copyright (c) 2022, 2023 Omar Polo <op@omarpolo.com>
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
37 static const struct option opts[] = {
38 {"help", no_argument, NULL, 'h'},
39 {"version", no_argument, NULL, 'V'},
44 log_request(struct client *c, int code, const char *meta)
46 char b[GEMINI_URL_LEN];
47 char cntmp[64], cn[64] = "-";
53 if ((now = time(NULL)) == -1)
55 if ((tm = localtime(&now)) == NULL)
57 if (strftime(rfc3339, sizeof(rfc3339), "%FT%T%z", tm) == 0)
60 if (c->iri.schema != NULL) {
61 /* serialize the IRI */
62 strlcpy(b, c->iri.schema, sizeof(b));
63 strlcat(b, "://", sizeof(b));
65 /* log the decoded host name, but if it was invalid
67 if (*c->domain != '\0')
68 strlcat(b, c->domain, sizeof(b));
70 strlcat(b, c->iri.host, sizeof(b));
72 if (*c->iri.path != '/')
73 strlcat(b, "/", sizeof(b));
74 strlcat(b, c->iri.path, sizeof(b)); /* TODO: sanitize UTF8 */
75 if (*c->iri.query != '\0') { /* TODO: sanitize UTF8 */
76 strlcat(b, "?", sizeof(b));
77 strlcat(b, c->iri.query, sizeof(b));
80 if ((t = c->req) == NULL)
82 strlcpy(b, t, sizeof(b));
85 if (tls_peer_cert_provided(c->ctx)) {
89 subj = tls_peer_cert_subject(c->ctx);
90 if ((n = strstr(subj, "/CN=")) != NULL) {
91 strlcpy(cntmp, subj + 4, sizeof(cntmp));
92 if ((n = strchr(cntmp, '/')) != NULL)
94 strnvis(cn, cntmp, sizeof(cn), VIS_WHITE|VIS_DQ);
98 fprintf(stderr, "%s %s %s %s %s 0 %d %s\n", rfc3339, c->rhost, cn,
99 *c->domain == '\0' ? c->iri.host : c->domain, b, code, meta);
103 load_local_cert(struct vhost *h, const char *hostname, const char *dir)
107 if (asprintf(&cert, "%s/%s.pem", dir, hostname) == -1)
109 if (asprintf(&key, "%s/%s.key", dir, hostname) == -1)
112 if (access(cert, R_OK) == -1 || access(key, R_OK) == -1)
113 gen_certificate(hostname, cert, key);
115 h->cert = tls_load_file(cert, &h->certlen, NULL);
117 fatal("can't load %s", cert);
119 h->key = tls_load_file(key, &h->keylen, NULL);
121 fatal("can't load %s", key);
123 strlcpy(h->domain, hostname, sizeof(h->domain));
126 /* wrapper around dirname(3). dn must be PATH_MAX+1 at least. */
128 pdirname(const char *path, char *dn)
133 strlcpy(p, path, sizeof(p));
135 memmove(dn, t, strlen(t)+1);
139 mkdirs(const char *path, mode_t mode)
141 char dname[PATH_MAX+1];
143 pdirname(path, dname);
144 if (!strcmp(dname, "/"))
147 if (mkdir(path, mode) != 0 && errno != EEXIST)
148 fatal("can't mkdir %s", path);
151 /* $XDG_DATA_HOME/gemexp */
155 const char *home, *xdg;
158 if ((xdg = getenv("XDG_DATA_HOME")) == NULL) {
159 if ((home = getenv("HOME")) == NULL)
160 fatalx("XDG_DATA_HOME and HOME both empty");
161 if (asprintf(&t, "%s/.local/share/gemexp", home) == -1)
164 if (asprintf(&t, "%s/gemexp", xdg) == -1)
173 serve(struct conf *conf, const char *host, int port, const char *dir)
175 struct addrinfo hints, *res, *res0;
176 struct vhost *vh = TAILQ_FIRST(&conf->hosts);
177 struct address *addr, *acp;
178 int r, error, saved_errno, sock = -1;
179 const char *cause = NULL;
185 r = snprintf(service, sizeof(service), "%d", port);
186 if (r < 0 || (size_t)r >= sizeof(service))
189 memset(&hints, 0, sizeof(hints));
190 hints.ai_family = AF_UNSPEC;
191 hints.ai_socktype = SOCK_STREAM;
192 hints.ai_flags = AI_PASSIVE;
193 error = getaddrinfo(host, service, &hints, &res0);
195 fatalx("%s", gai_strerror(error));
196 for (res = res0; res; res = res->ai_next) {
197 sock = socket(res->ai_family, res->ai_socktype,
204 if (bind(sock, res->ai_addr, res->ai_addrlen) == -1) {
212 if (listen(sock, 5) == -1)
217 addr = xcalloc(1, sizeof(*addr));
218 addr->ai_flags = res->ai_flags;
219 addr->ai_family = res->ai_family;
220 addr->ai_socktype = res->ai_socktype;
221 addr->ai_protocol = res->ai_protocol;
222 addr->slen = res->ai_addrlen;
223 memcpy(&addr->ss, res->ai_addr, res->ai_addrlen);
227 event_set(&addr->evsock, addr->sock, EV_READ|EV_PERSIST,
228 server_accept, addr);
230 if ((addr->ctx = tls_server()) == NULL)
231 fatal("tls_server failure");
233 TAILQ_INSERT_HEAD(&conf->addrs, addr, addrs);
235 acp = xcalloc(1, sizeof(*acp));
236 memcpy(acp, addr, sizeof(*acp));
238 memset(&acp->evsock, 0, sizeof(acp->evsock));
239 TAILQ_INSERT_HEAD(&vh->addrs, addr, addrs);
246 server_init(NULL, NULL, NULL);
247 if (server_configure_done(conf) == -1)
248 fatalx("server configuration failed");
250 log_info("serving %s on port %d", dir, port);
252 log_info("quitting");
260 "Version: " GE_STRING "\n"
261 "Usage: %s [-hVv] [-d certs-dir] [-H hostname] [-p port] [dir]\n",
267 main(int argc, char **argv)
271 struct location *loc;
272 const char *errstr, *certs_dir = NULL, *hostname = "localhost";
276 setlocale(LC_CTYPE, "");
278 log_init(1, LOG_DAEMON);
282 /* ge doesn't do privsep so no privsep crypto engine. */
283 conf->use_privsep_crypto = 0;
285 while ((ch = getopt_long(argc, argv, "d:H:hp:Vv", opts, NULL)) != -1) {
297 port = strtonum(optarg, 0, UINT16_MAX, &errstr);
299 fatalx("port number is %s: %s", errstr,
303 puts("Version: " GE_STRING);
316 /* prepare the configuration */
317 init_mime(&conf->mime);
319 if (certs_dir == NULL)
320 certs_dir = data_dir();
322 /* set up the implicit vhost and location */
323 host = xcalloc(1, sizeof(*host));
324 TAILQ_INSERT_HEAD(&conf->hosts, host, vhosts);
326 loc = xcalloc(1, sizeof(*loc));
328 TAILQ_INSERT_HEAD(&host->locations, loc, locations);
330 load_local_cert(host, hostname, certs_dir);
332 strlcpy(host->domain, "*", sizeof(host->domain));
334 strlcpy(loc->match, "*", sizeof(loc->match));
337 if (getcwd(path, sizeof(path)) == NULL)
339 strlcpy(loc->dir, path, sizeof(loc->dir));
343 tmp = absolutify_path(*argv);
344 strlcpy(loc->dir, tmp, sizeof(loc->dir));
348 /* start the server */
349 signal(SIGPIPE, SIG_IGN);
350 setproctitle("%s", loc->dir);
351 return serve(conf, hostname, port, loc->dir);