/* * Copyright (c) 2021 Omar Polo * * 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. */ /* * Test program for fastcgi. Can't handle more than one request at * the same time. */ #include "../config.h" #include #include #include #include #include #include #include #include #include #define FCGI_VERSION_1 1 /* subset of records that matters to us */ #define FCGI_BEGIN_REQUEST 1 #define FCGI_END_REQUEST 3 #define FCGI_PARAMS 4 #define FCGI_STDIN 5 #define FCGI_STDOUT 6 #define SUM(a, b) (((a) << 8) + (b)) #ifndef nitems #define nitems(_a) (sizeof((_a)) / sizeof((_a)[0])) #endif struct fcgi_header { uint8_t version; uint8_t type; uint8_t req_id1; uint8_t req_id0; uint8_t content_len1; uint8_t content_len0; uint8_t padding; uint8_t reserved; }; struct fcgi_end_req_body { unsigned char app_status3; unsigned char app_status2; unsigned char app_status1; unsigned char app_status0; unsigned char proto_status; unsigned char reserved[3]; }; struct param { char name[32]; char value[128]; }; static int prepare_header(struct fcgi_header *h, int type, int id, size_t size, size_t padding) { memset(h, 0, sizeof(*h)); h->version = FCGI_VERSION_1; h->type = type; h->req_id1 = (id >> 8); h->req_id0 = (id & 0xFF); h->content_len1 = (size >> 8); h->content_len0 = (size & 0xFF); h->padding = padding; return 0; } static void must_read(int sock, void *d, size_t len) { uint8_t *data = d; ssize_t r; while (len > 0) { switch (r = read(sock, data, len)) { case -1: err(1, "read"); case 0: errx(1, "EOF"); default: len -= r; data += r; } } } static void must_write(int sock, const void *d, size_t len) { const uint8_t *data = d; ssize_t w; while (len > 0) { switch (w = write(sock, data, len)) { case -1: err(1, "write"); case 0: errx(1, "EOF"); default: len -= w; data += w; } } } static int consume(int fd, size_t len) { size_t l; char buf[64]; while (len != 0) { if ((l = len) > sizeof(buf)) l = sizeof(buf); must_read(fd, buf, l); len -= l; } return 1; } static void read_header(int fd, struct fcgi_header *hdr) { must_read(fd, hdr, sizeof(*hdr)); } /* read and consume a record of the given type */ static void assert_record(int fd, int type) { struct fcgi_header hdr; read_header(fd, &hdr); if (hdr.type != type) errx(1, "expected record type %d; got %d", type, hdr.type); consume(fd, SUM(hdr.content_len1, hdr.content_len0)); consume(fd, hdr.padding); } static int parse_len(int sock) { unsigned char c, x[3]; must_read(sock, &c, 1); if (c >> 7 == 0) return (c); must_read(sock, &x, sizeof(x)); return (((c & 0x7F) << 24) | (x[0] << 16) | (x[1] << 8) | x[2]); } static void parse_params(int sock, struct param *param, struct fcgi_header *hdr) { int tot; int nlen, vlen; tot = SUM(hdr->content_len1, hdr->content_len0); tot -= sizeof(*hdr); nlen = parse_len(sock); vlen = parse_len(sock); if (nlen + vlen != tot) errx(1, "unexpected length, name=%d val=%d exp=%d", nlen, vlen, tot); if ((size_t)nlen > sizeof(param->name) - 1 || (size_t)vlen > sizeof(param->value) - 1) errx(1, "parameter name or value too long"); memset(param, 0, sizeof(*param)); must_read(sock, param->name, nlen); must_read(sock, param->value, vlen); if (!strcmp(param->name, "SERVER_NAME")) strlcpy(param->value, "", sizeof(param->value)); consume(sock, hdr->padding); } static int param_cmp(const void *a, const void *b) { const struct param *x = a, *y = b; return (strcmp(x->name, y->name)); } int main(int argc, char **argv) { struct fcgi_header hdr; struct fcgi_end_req_body end; struct sockaddr_un sun; struct param params[64]; size_t i, nparam; const char *path; const char *msg; size_t len; int ch, sock, s; while ((ch = getopt(argc, argv, "")) != -1) errx(1, "wrong usage"); argc -= optind; argv += optind; if (argc != 1) errx(1, "wrong usage"); path = argv[0]; if ((sock = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) err(1, "socket"); memset(&sun, 0, sizeof(sun)); sun.sun_family = AF_UNIX; (void)strlcpy(sun.sun_path, path, sizeof(sun.sun_path)); if (unlink(path) == -1 && errno != ENOENT) err(1, "unlink %s", path); if (bind(sock, (struct sockaddr *)&sun, sizeof(sun)) == -1) err(1, "bind"); if (listen(sock, 5) == -1) err(1, "listen"); for (;;) { if ((s = accept(sock, NULL, NULL)) == -1) { warn("retrying; accept failed"); continue; } assert_record(s, FCGI_BEGIN_REQUEST); /* read params */ nparam = 0; for (;;) { read_header(s, &hdr); if (hdr.type != FCGI_PARAMS) errx(1, "got %d; expecting PARAMS", hdr.type); if (hdr.content_len0 == 0 && hdr.content_len1 == 0 && hdr.padding == 0) { consume(s, SUM(hdr.content_len1, hdr.content_len0)); consume(s, hdr.padding); break; } if (nparam == nitems(params)) errx(1, "too many parameters"); parse_params(s, ¶ms[nparam++], &hdr); } assert_record(s, FCGI_STDIN); msg = "20 text/gemini\r\n"; len = strlen(msg); prepare_header(&hdr, FCGI_STDOUT, 1, len, 0); must_write(s, &hdr, sizeof(hdr)); must_write(s, msg, len); msg = "Here's the parameters I've got:\n"; len = strlen(msg); prepare_header(&hdr, FCGI_STDOUT, 1, len, 0); must_write(s, &hdr, sizeof(hdr)); must_write(s, msg, len); qsort(params, nparam, sizeof(params[0]), param_cmp); for (i = 0; i < nparam; ++i) { char line[256]; int r; r = snprintf(line, sizeof(line), "* %s=%s\n", params[i].name, params[i].value); if (r < 0 || (size_t)r >= sizeof(line)) errx(1, "line too short"); prepare_header(&hdr, FCGI_STDOUT, 1, r, 0); must_write(s, &hdr, sizeof(hdr)); must_write(s, line, r); } prepare_header(&hdr, FCGI_END_REQUEST, 1, sizeof(end), 0); write(s, &hdr, sizeof(hdr)); memset(&end, 0, sizeof(end)); write(s, &end, sizeof(end)); } }