gmid/ge.c

348 lines
7.9 KiB
C
Raw Permalink Normal View History

2022-09-07 22:47:33 +02:00
/*
2023-06-24 12:07:17 +02:00
* Copyright (c) 2022, 2023 Omar Polo <op@omarpolo.com>
2022-09-07 22:47:33 +02:00
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include "gmid.h"
#include <sys/stat.h>
#include <sys/wait.h>
#include <errno.h>
#include <fcntl.h>
#include <getopt.h>
2023-06-06 10:11:30 +02:00
#include <locale.h>
2022-09-07 22:47:33 +02:00
#include <libgen.h>
#include <signal.h>
#include <string.h>
2023-06-06 13:46:40 +02:00
#include <syslog.h>
2022-09-07 22:47:33 +02:00
#include <unistd.h>
#include <vis.h>
2022-09-07 22:47:33 +02:00
2023-06-06 13:46:40 +02:00
#include "log.h"
static int gen_eckey = 1;
2023-06-08 18:22:03 +02:00
int privsep_process;
2022-09-07 22:47:33 +02:00
static const struct option opts[] = {
{"help", no_argument, NULL, 'h'},
{"version", no_argument, NULL, 'V'},
{NULL, 0, NULL, 0},
};
2023-06-08 18:22:03 +02:00
void
log_request(struct client *c, int code, const char *meta)
2023-06-08 18:22:03 +02:00
{
char b[GEMINI_URL_LEN];
char cntmp[64], cn[64] = "-";
2023-06-08 18:22:03 +02:00
const char *t;
if (c->iri.schema != NULL) {
/* serialize the IRI */
strlcpy(b, c->iri.schema, sizeof(b));
strlcat(b, "://", sizeof(b));
/* log the decoded host name, but if it was invalid
* use the raw one. */
if (*c->domain != '\0')
strlcat(b, c->domain, sizeof(b));
else
strlcat(b, c->iri.host, sizeof(b));
if (*c->iri.path != '/')
strlcat(b, "/", sizeof(b));
strlcat(b, c->iri.path, sizeof(b)); /* TODO: sanitize UTF8 */
if (*c->iri.query != '\0') { /* TODO: sanitize UTF8 */
strlcat(b, "?", sizeof(b));
strlcat(b, c->iri.query, sizeof(b));
}
} else {
if ((t = c->req) == NULL)
t = "";
strlcpy(b, t, sizeof(b));
}
if (tls_peer_cert_provided(c->ctx)) {
const char *subj;
char *n;
subj = tls_peer_cert_subject(c->ctx);
if ((n = strstr(subj, "/CN=")) != NULL) {
strlcpy(cntmp, subj + 4, sizeof(cntmp));
if ((n = strchr(cntmp, '/')) != NULL)
*n = '\0';
strnvis(cn, cntmp, sizeof(cn), VIS_WHITE|VIS_DQ);
}
}
fprintf(stderr, "%s %s %s %s %d %s\n", c->rhost, cn,
*c->domain == '\0' ? c->iri.host : c->domain, b, code, meta);
2023-06-08 18:22:03 +02:00
}
2022-09-07 22:47:33 +02:00
void
load_local_cert(struct vhost *h, const char *hostname, const char *dir)
{
char *cert, *key;
if (asprintf(&cert, "%s/%s.pem", dir, hostname) == -1)
2023-06-06 13:52:43 +02:00
fatal("asprintf");
if (asprintf(&key, "%s/%s.key", dir, hostname) == -1)
2023-06-06 13:52:43 +02:00
fatal("asprintf");
2022-09-07 22:47:33 +02:00
if (access(cert, R_OK) == -1 || access(key, R_OK) == -1)
gencert(hostname, cert, key, gen_eckey);
2022-09-07 22:47:33 +02:00
2023-06-08 18:22:03 +02:00
h->cert = tls_load_file(cert, &h->certlen, NULL);
if (h->cert == NULL)
fatal("can't load %s", cert);
h->key = tls_load_file(key, &h->keylen, NULL);
if (h->key == NULL)
fatal("can't load %s", key);
strlcpy(h->domain, hostname, sizeof(h->domain));
2022-09-07 22:47:33 +02:00
}
/* wrapper around dirname(3). dn must be PATH_MAX+1 at least. */
static void
pdirname(const char *path, char *dn)
{
char p[PATH_MAX+1];
char *t;
strlcpy(p, path, sizeof(p));
t = dirname(p);
memmove(dn, t, strlen(t)+1);
}
static void
mkdirs(const char *path, mode_t mode)
{
char dname[PATH_MAX+1];
pdirname(path, dname);
if (!strcmp(dname, "/"))
return;
mkdirs(dname, mode);
if (mkdir(path, mode) != 0 && errno != EEXIST)
fatal("can't mkdir %s", path);
2022-09-07 22:47:33 +02:00
}
/* $XDG_DATA_HOME/gemexp */
2022-09-07 22:47:33 +02:00
char *
data_dir(void)
{
const char *home, *xdg;
char *t;
if ((xdg = getenv("XDG_DATA_HOME")) == NULL) {
if ((home = getenv("HOME")) == NULL)
2023-06-06 13:52:43 +02:00
fatalx("XDG_DATA_HOME and HOME both empty");
if (asprintf(&t, "%s/.local/share/gemexp", home) == -1)
2023-06-06 13:52:43 +02:00
fatalx("asprintf");
2022-09-07 22:47:33 +02:00
} else {
if (asprintf(&t, "%s/gemexp", xdg) == -1)
2023-06-06 13:52:43 +02:00
fatal("asprintf");
2022-09-07 22:47:33 +02:00
}
mkdirs(t, 0755);
return t;
}
static int
2023-06-09 19:18:04 +02:00
serve(struct conf *conf, const char *host, int port, const char *dir)
2022-09-07 22:47:33 +02:00
{
struct addrinfo hints, *res, *res0;
struct vhost *vh = TAILQ_FIRST(&conf->hosts);
struct address *addr, *acp;
2023-06-06 10:22:18 +02:00
int r, error, saved_errno, sock = -1;
2022-09-07 22:47:33 +02:00
const char *cause = NULL;
char service[32];
int any = 0;
event_init();
2022-09-07 22:47:33 +02:00
2023-06-06 10:22:18 +02:00
r = snprintf(service, sizeof(service), "%d", port);
if (r < 0 || (size_t)r >= sizeof(service))
2022-09-07 22:47:33 +02:00
fatal("snprintf");
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
error = getaddrinfo(host, service, &hints, &res0);
if (error)
fatalx("%s", gai_strerror(error));
2022-09-07 22:47:33 +02:00
for (res = res0; res; res = res->ai_next) {
sock = socket(res->ai_family, res->ai_socktype,
res->ai_protocol);
if (sock == -1) {
cause = "socket";
continue;
}
if (bind(sock, res->ai_addr, res->ai_addrlen) == -1) {
cause = "bind";
saved_errno = errno;
close(sock);
errno = saved_errno;
continue;
}
if (listen(sock, 5) == -1)
fatal("listen");
any = 1;
addr = xcalloc(1, sizeof(*addr));
addr->ai_flags = res->ai_flags;
addr->ai_family = res->ai_family;
addr->ai_socktype = res->ai_socktype;
addr->ai_protocol = res->ai_protocol;
addr->slen = res->ai_addrlen;
memcpy(&addr->ss, res->ai_addr, res->ai_addrlen);
addr->conf = conf;
addr->sock = sock;
event_set(&addr->evsock, addr->sock, EV_READ|EV_PERSIST,
2023-07-02 00:00:08 +02:00
server_accept, addr);
2023-06-24 11:50:30 +02:00
if ((addr->ctx = tls_server()) == NULL)
fatal("tls_server failure");
TAILQ_INSERT_HEAD(&conf->addrs, addr, addrs);
acp = xcalloc(1, sizeof(*acp));
memcpy(acp, addr, sizeof(*acp));
acp->sock = -1;
memset(&acp->evsock, 0, sizeof(acp->evsock));
TAILQ_INSERT_HEAD(&vh->addrs, addr, addrs);
2022-09-07 22:47:33 +02:00
}
if (!any)
2022-09-07 22:47:33 +02:00
fatal("%s", cause);
freeaddrinfo(res0);
2023-06-08 18:22:03 +02:00
server_init(NULL, NULL, NULL);
2023-06-09 19:18:04 +02:00
if (server_configure_done(conf) == -1)
2023-06-08 18:22:03 +02:00
fatalx("server configuration failed");
2023-06-06 13:46:40 +02:00
log_info("serving %s on port %d", dir, port);
2023-06-08 18:22:03 +02:00
event_dispatch();
log_info("quitting");
return 0;
2022-09-07 22:47:33 +02:00
}
static __dead void
usage(void)
{
fprintf(stderr,
2024-01-08 09:34:36 +01:00
"Version: " GEMEXP_STRING "\n"
"Usage: %s [-hRV] [-d certs-dir] [-H hostname] [-p port] [dir]\n",
2022-09-07 22:47:33 +02:00
getprogname());
exit(1);
}
int
main(int argc, char **argv)
{
2023-06-09 19:18:04 +02:00
struct conf *conf;
2022-09-07 22:47:33 +02:00
struct vhost *host;
struct location *loc;
const char *errstr, *certs_dir = NULL, *hostname = "localhost";
char path[PATH_MAX];
int ch, port = 1965;
2022-09-07 22:47:33 +02:00
2023-06-06 10:11:30 +02:00
setlocale(LC_CTYPE, "");
2023-06-06 13:46:40 +02:00
log_init(1, LOG_DAEMON);
log_setverbose(0);
2023-06-09 19:18:04 +02:00
conf = config_new();
2022-09-07 22:47:33 +02:00
/* ge doesn't do privsep so no privsep crypto engine. */
conf->use_privsep_crypto = 0;
while ((ch = getopt_long(argc, argv, "d:H:hp:RV", opts, NULL)) != -1) {
2022-09-07 22:47:33 +02:00
switch (ch) {
case 'd':
certs_dir = optarg;
break;
case 'H':
hostname = optarg;
break;
case 'h':
usage();
break;
case 'p':
port = strtonum(optarg, 0, UINT16_MAX, &errstr);
2022-09-07 22:47:33 +02:00
if (errstr)
fatalx("port number is %s: %s", errstr,
optarg);
2022-09-07 22:47:33 +02:00
break;
case 'R':
gen_eckey = 0;
break;
2022-09-07 22:47:33 +02:00
case 'V':
2024-01-08 09:34:36 +01:00
puts("Version: " GEMEXP_STRING);
2022-09-07 22:47:33 +02:00
return 0;
default:
usage();
break;
}
}
argc -= optind;
argv += optind;
if (argc > 1)
usage();
/* prepare the configuration */
2023-06-09 19:18:04 +02:00
init_mime(&conf->mime);
2022-09-07 22:47:33 +02:00
if (certs_dir == NULL)
certs_dir = data_dir();
/* set up the implicit vhost and location */
host = xcalloc(1, sizeof(*host));
2023-06-09 19:18:04 +02:00
TAILQ_INSERT_HEAD(&conf->hosts, host, vhosts);
2022-09-07 22:47:33 +02:00
loc = xcalloc(1, sizeof(*loc));
loc->fcgi = -1;
TAILQ_INSERT_HEAD(&host->locations, loc, locations);
load_local_cert(host, hostname, certs_dir);
strlcpy(host->domain, "*", sizeof(host->domain));
2022-09-07 22:47:33 +02:00
loc->auto_index = 1;
strlcpy(loc->match, "*", sizeof(loc->match));
2022-09-07 22:47:33 +02:00
if (*argv == NULL) {
if (getcwd(path, sizeof(path)) == NULL)
fatal("getcwd");
strlcpy(loc->dir, path, sizeof(loc->dir));
} else {
char *tmp;
tmp = absolutify_path(*argv);
strlcpy(loc->dir, tmp, sizeof(loc->dir));
free(tmp);
}
2022-09-07 22:47:33 +02:00
/* start the server */
signal(SIGPIPE, SIG_IGN);
setproctitle("%s", loc->dir);
return serve(conf, hostname, port, loc->dir);
2022-09-07 22:47:33 +02:00
}