1294 lines
38 KiB
C
1294 lines
38 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ This program is free software; you can redistribute it and/or modify │
|
|
│ it under the terms of the GNU General Public License as published by │
|
|
│ the Free Software Foundation; version 2 of the License. │
|
|
│ │
|
|
│ This program is distributed in the hope that it will be useful, but │
|
|
│ WITHOUT ANY WARRANTY; without even the implied warranty of │
|
|
│ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU │
|
|
│ General Public License for more details. │
|
|
│ │
|
|
│ You should have received a copy of the GNU General Public License │
|
|
│ along with this program; if not, write to the Free Software │
|
|
│ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA │
|
|
│ 02110-1301 USA │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/alg/arraylist2.internal.h"
|
|
#include "libc/bits/bits.h"
|
|
#include "libc/bits/bswap.h"
|
|
#include "libc/bits/safemacros.internal.h"
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/struct/iovec.h"
|
|
#include "libc/calls/struct/itimerval.h"
|
|
#include "libc/calls/struct/stat.h"
|
|
#include "libc/calls/weirdtypes.h"
|
|
#include "libc/conv/conv.h"
|
|
#include "libc/conv/itoa.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/fmt/fmt.h"
|
|
#include "libc/log/check.h"
|
|
#include "libc/log/log.h"
|
|
#include "libc/macros.h"
|
|
#include "libc/math.h"
|
|
#include "libc/mem/mem.h"
|
|
#include "libc/nexgen32e/crc32.h"
|
|
#include "libc/rand/rand.h"
|
|
#include "libc/runtime/gc.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/sock/epoll.h"
|
|
#include "libc/sock/sock.h"
|
|
#include "libc/stdio/stdio.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/str/undeflate.h"
|
|
#include "libc/sysv/consts/af.h"
|
|
#include "libc/sysv/consts/auxv.h"
|
|
#include "libc/sysv/consts/clock.h"
|
|
#include "libc/sysv/consts/epoll.h"
|
|
#include "libc/sysv/consts/ex.h"
|
|
#include "libc/sysv/consts/exit.h"
|
|
#include "libc/sysv/consts/f.h"
|
|
#include "libc/sysv/consts/fd.h"
|
|
#include "libc/sysv/consts/inaddr.h"
|
|
#include "libc/sysv/consts/ipproto.h"
|
|
#include "libc/sysv/consts/itimer.h"
|
|
#include "libc/sysv/consts/map.h"
|
|
#include "libc/sysv/consts/o.h"
|
|
#include "libc/sysv/consts/prot.h"
|
|
#include "libc/sysv/consts/sa.h"
|
|
#include "libc/sysv/consts/shut.h"
|
|
#include "libc/sysv/consts/sig.h"
|
|
#include "libc/sysv/consts/so.h"
|
|
#include "libc/sysv/consts/sock.h"
|
|
#include "libc/sysv/consts/sol.h"
|
|
#include "libc/sysv/consts/tcp.h"
|
|
#include "libc/sysv/errfuns.h"
|
|
#include "libc/time/struct/tm.h"
|
|
#include "libc/time/time.h"
|
|
#include "libc/x/x.h"
|
|
#include "libc/zip.h"
|
|
#include "libc/zipos/zipos.internal.h"
|
|
#include "net/http/http.h"
|
|
#include "third_party/getopt/getopt.h"
|
|
#include "third_party/zlib/zlib.h"
|
|
|
|
/**
|
|
* @fileoverview redbean.c modified for epoll tutorial
|
|
* This'll work on Linux and Windows but BSD is TODO
|
|
* TODO(jart): fix memory bug
|
|
*/
|
|
|
|
#define USAGE \
|
|
" [-hvdsm] [-p PORT]\n\
|
|
\n\
|
|
DESCRIPTION\n\
|
|
\n\
|
|
greenbean distributable static web server\n\
|
|
\n\
|
|
FLAGS\n\
|
|
\n\
|
|
-h help\n\
|
|
-v verbosity\n\
|
|
-d daemonize\n\
|
|
-s uniprocess\n\
|
|
-m log messages\n\
|
|
-c INT cache seconds\n\
|
|
-r /X=/Y redirect X to Y\n\
|
|
-l ADDR listen ip [default 0.0.0.0]\n\
|
|
-p PORT listen port [default 8080]\n\
|
|
-L PATH log file location\n\
|
|
-P PATH pid file location\n\
|
|
-U INT daemon set user id\n\
|
|
-G INT daemon set group id\n\
|
|
-B STR changes server header\n\
|
|
\n\
|
|
FEATURES\n\
|
|
\n\
|
|
- HTTP v1.1\n\
|
|
- Content-Encoding\n\
|
|
- Range / Content-Range\n\
|
|
- Last-Modified / If-Modified-Since\n\
|
|
\n\
|
|
USAGE\n\
|
|
\n\
|
|
This executable is also a ZIP file that contains static assets.\n\
|
|
\n\
|
|
unzip -vl greenbean.com # shows listing of zip contents\n\
|
|
\n\
|
|
Audio video content should not be compressed in your ZIP files.\n\
|
|
Uncompressed assets enable browsers to send Range HTTP request.\n\
|
|
On the other hand compressed assets are best for gzip encoding.\n\
|
|
\n\
|
|
zip greenbean.com index.html # adds file\n\
|
|
zip -0 greenbean.com video.mp4 # adds without compression\n\
|
|
\n\
|
|
Each connection uses a point in time snapshot of your ZIP file.\n\
|
|
If your ZIP is deleted then serving continues. If it's replaced\n\
|
|
then issuing SIGUSR1 (or SIGHUP if daemon) will reindex the zip\n\
|
|
for subsequent connections without interrupting active ones. If\n\
|
|
SIGINT or SIGTERM is issued then a graceful shutdown is started\n\
|
|
but if it's issued a second time, active connections are reset.\n\
|
|
\n"
|
|
|
|
#define HASH_LOAD_FACTOR /* 1. / */ 4
|
|
#define DEFAULT_PORT 8080
|
|
#define DEFAULT_SERVER "greenbean/0.1"
|
|
#define DEFAULT_CONTENT_TYPE "application/octet-stream"
|
|
#define DEFAULT_PATH "/tool/net/redbean.html"
|
|
#define FAVICON "tool/net/redbean.ico"
|
|
|
|
#define STPCPY(p, s) mempcpy(p, s, strlen(s))
|
|
#define AppendHeaderName(p, s) STPCPY(STPCPY(p, s), ": ")
|
|
|
|
struct Client {
|
|
uint32_t poison;
|
|
struct Client *prev;
|
|
struct Client *next;
|
|
int fd;
|
|
int iovlen;
|
|
bool close;
|
|
uint32_t insize;
|
|
uint32_t addrsize;
|
|
uint8_t gzip_footer[8];
|
|
struct sockaddr_in addr;
|
|
long double startrequest;
|
|
long double startconnection;
|
|
struct HttpRequest req;
|
|
struct iovec iov[8];
|
|
char addrstr[32];
|
|
char outbuf[1500];
|
|
char inbuf[2048];
|
|
};
|
|
|
|
static const struct itimerval kHeartbeat = {
|
|
{0, 500000},
|
|
{0, 500000},
|
|
};
|
|
|
|
static const uint8_t kGzipHeader[] = {
|
|
0x1F, // MAGNUM
|
|
0x8B, // MAGNUM
|
|
0x08, // CM: DEFLATE
|
|
0x00, // FLG: NONE
|
|
0x00, // MTIME: NONE
|
|
0x00, //
|
|
0x00, //
|
|
0x00, //
|
|
0x00, // XFL
|
|
kZipOsUnix, // OS
|
|
};
|
|
|
|
static const struct ContentTypeExtension {
|
|
unsigned char ext[8];
|
|
const char *mime;
|
|
} kContentTypeExtension[] = {
|
|
{"S", "text/plain"}, //
|
|
{"bmp", "image/x-ms-bmp"}, //
|
|
{"c", "text/plain"}, //
|
|
{"cc", "text/plain"}, //
|
|
{"css", "text/css"}, //
|
|
{"csv", "text/csv"}, //
|
|
{"gif", "image/gif"}, //
|
|
{"h", "text/plain"}, //
|
|
{"html", "text/html"}, //
|
|
{"i", "text/plain"}, //
|
|
{"ico", "image/vnd.microsoft.icon"}, //
|
|
{"jpeg", "image/jpeg"}, //
|
|
{"jpg", "image/jpeg"}, //
|
|
{"js", "application/javascript"}, //
|
|
{"json", "application/json"}, //
|
|
{"m4a", "audio/mpeg"}, //
|
|
{"mp2", "audio/mpeg"}, //
|
|
{"mp3", "audio/mpeg"}, //
|
|
{"mp4", "video/mp4"}, //
|
|
{"mpg", "video/mpeg"}, //
|
|
{"otf", "font/otf"}, //
|
|
{"pdf", "application/pdf"}, //
|
|
{"png", "image/png"}, //
|
|
{"png", "image/png"}, //
|
|
{"png", "image/png"}, //
|
|
{"png", "image/png"}, //
|
|
{"s", "text/plain"}, //
|
|
{"svg", "image/svg+xml"}, //
|
|
{"tiff", "image/tiff"}, //
|
|
{"ttf", "font/ttf"}, //
|
|
{"txt", "text/plain"}, //
|
|
{"wav", "audio/x-wav"}, //
|
|
{"woff", "font/woff"}, //
|
|
{"woff2", "font/woff2"}, //
|
|
{"xml", "application/xml"}, //
|
|
{"zip", "application/zip"}, //
|
|
};
|
|
|
|
static struct Redirects {
|
|
size_t i, n;
|
|
struct Redirect {
|
|
const char *path;
|
|
size_t pathlen;
|
|
const char *dest;
|
|
} * p;
|
|
} redirects;
|
|
|
|
static struct Assets {
|
|
uint32_t n;
|
|
struct Asset {
|
|
uint32_t hash;
|
|
uint32_t cf;
|
|
int64_t lastmodified;
|
|
char *lastmodifiedstr;
|
|
} * p;
|
|
} assets;
|
|
|
|
static bool killed;
|
|
static bool notimer;
|
|
static bool heartbeat;
|
|
static bool daemonize;
|
|
static bool terminated;
|
|
static bool uniprocess;
|
|
static bool legacyhttp;
|
|
static bool invalidated;
|
|
static bool logmessages;
|
|
|
|
static int epfd;
|
|
static int gmtoff;
|
|
static int server;
|
|
static int daemonuid;
|
|
static int daemongid;
|
|
static int cacheseconds;
|
|
|
|
static void *zmap;
|
|
static uint8_t *zbase;
|
|
static uint8_t *zcdir;
|
|
static size_t zmapsize;
|
|
static const char *pidpath;
|
|
static const char *logpath;
|
|
static int64_t programtime;
|
|
static struct Client *clients;
|
|
static const char *programfile;
|
|
static const char *serverheader;
|
|
|
|
static long double nowish;
|
|
static struct sockaddr_in serveraddr;
|
|
|
|
static char currentdate[32];
|
|
static char serveraddrstr[32];
|
|
|
|
static void OnHup(void) {
|
|
invalidated = true;
|
|
}
|
|
|
|
static void OnAlarm(void) {
|
|
heartbeat = true;
|
|
}
|
|
|
|
static void OnTerminate(void) {
|
|
if (terminated) {
|
|
killed = true;
|
|
} else {
|
|
terminated = true;
|
|
}
|
|
}
|
|
|
|
static void AddRedirect(const char *arg) {
|
|
const char *p;
|
|
struct Redirect r;
|
|
CHECK_NOTNULL((p = strchr(arg, '=')));
|
|
CHECK_GT(p - arg, 0);
|
|
r.path = arg;
|
|
r.pathlen = p - arg;
|
|
r.dest = strdup(p + 1);
|
|
APPEND(&redirects.p, &redirects.i, &redirects.n, &r);
|
|
}
|
|
|
|
static int CompareRedirects(const struct Redirect *a,
|
|
const struct Redirect *b) {
|
|
return strcmp(a->path, b->path);
|
|
}
|
|
|
|
static void SortRedirects(void) {
|
|
qsort(redirects.p, redirects.i, sizeof(struct Redirect),
|
|
(void *)CompareRedirects);
|
|
}
|
|
|
|
static const char *LookupRedirect(const char *path, size_t n) {
|
|
int c, m, l, r, z;
|
|
l = 0;
|
|
r = redirects.i - 1;
|
|
while (l <= r) {
|
|
m = (l + r) >> 1;
|
|
c = memcmp(redirects.p[m].path, path, MIN(redirects.p[m].pathlen, n));
|
|
if (c < 0) {
|
|
l = m + 1;
|
|
} else if (c > 0) {
|
|
r = m - 1;
|
|
} else if (redirects.p[m].pathlen < n) {
|
|
l = m + 1;
|
|
} else if (redirects.p[m].pathlen > n) {
|
|
r = m - 1;
|
|
} else {
|
|
return redirects.p[m].dest;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static int CompareInts(const uint64_t x, uint64_t y) {
|
|
return x > y ? 1 : x < y ? -1 : 0;
|
|
}
|
|
|
|
static const char *LookupContentType(uint64_t ext) {
|
|
int c, m, l, r;
|
|
l = 0;
|
|
r = ARRAYLEN(kContentTypeExtension) - 1;
|
|
while (l <= r) {
|
|
m = (l + r) >> 1;
|
|
c = CompareInts(READ64BE(kContentTypeExtension[m].ext), ext);
|
|
if (c < 0) {
|
|
l = m + 1;
|
|
} else if (c > 0) {
|
|
r = m - 1;
|
|
} else {
|
|
return kContentTypeExtension[m].mime;
|
|
}
|
|
}
|
|
return DEFAULT_CONTENT_TYPE;
|
|
}
|
|
|
|
static const char *GetContentType(const char *path, size_t n) {
|
|
size_t i;
|
|
uint64_t x;
|
|
const char *p;
|
|
if ((p = memrchr(path, '.', n))) {
|
|
for (x = 0, i = n; i-- > p + 1 - path;) {
|
|
x <<= 8;
|
|
x |= path[i] & 0xFF;
|
|
}
|
|
return LookupContentType(bswap_64(x));
|
|
} else {
|
|
return DEFAULT_CONTENT_TYPE;
|
|
}
|
|
}
|
|
|
|
static wontreturn void PrintUsage(FILE *f, int rc) {
|
|
fprintf(f, "SYNOPSIS\n\n %s%s", program_invocation_name, USAGE);
|
|
exit(rc);
|
|
}
|
|
|
|
static void DescribeAddress(char buf[32], const struct sockaddr_in *addr) {
|
|
char *p = buf;
|
|
const uint8_t *ip4 = (const uint8_t *)&addr->sin_addr.s_addr;
|
|
p += uint64toarray_radix10(ip4[0], p), *p++ = '.';
|
|
p += uint64toarray_radix10(ip4[1], p), *p++ = '.';
|
|
p += uint64toarray_radix10(ip4[2], p), *p++ = '.';
|
|
p += uint64toarray_radix10(ip4[3], p), *p++ = ':';
|
|
p += uint64toarray_radix10(ntohs(addr->sin_port), p);
|
|
*p = '\0';
|
|
}
|
|
|
|
void GetOpts(int argc, char *argv[]) {
|
|
int opt;
|
|
serveraddr.sin_family = AF_INET;
|
|
serveraddr.sin_port = htons(DEFAULT_PORT);
|
|
serveraddr.sin_addr.s_addr = INADDR_ANY;
|
|
while ((opt = getopt(argc, argv, "hduvml:p:w:r:c:L:P:U:G:B:")) != -1) {
|
|
switch (opt) {
|
|
case 'v':
|
|
g_loglevel++;
|
|
break;
|
|
case 'd':
|
|
daemonize = true;
|
|
break;
|
|
case 'u':
|
|
uniprocess = true;
|
|
break;
|
|
case 'm':
|
|
logmessages = true;
|
|
break;
|
|
case 'r':
|
|
AddRedirect(optarg);
|
|
break;
|
|
case 'c':
|
|
cacheseconds = atoi(optarg);
|
|
break;
|
|
case 'p':
|
|
CHECK_NE(0xFFFF, (serveraddr.sin_port = htons(parseport(optarg))));
|
|
break;
|
|
case 'l':
|
|
CHECK_EQ(1, inet_pton(AF_INET, optarg, &serveraddr.sin_addr));
|
|
break;
|
|
case 'B':
|
|
serverheader = optarg;
|
|
break;
|
|
case 'L':
|
|
logpath = optarg;
|
|
break;
|
|
case 'P':
|
|
pidpath = optarg;
|
|
break;
|
|
case 'U':
|
|
daemonuid = atoi(optarg);
|
|
break;
|
|
case 'G':
|
|
daemonuid = atoi(optarg);
|
|
break;
|
|
case 'h':
|
|
PrintUsage(stdout, EXIT_SUCCESS);
|
|
default:
|
|
PrintUsage(stderr, EX_USAGE);
|
|
}
|
|
}
|
|
SortRedirects();
|
|
if (logpath) {
|
|
CHECK_NOTNULL(freopen(logpath, "a", stderr));
|
|
}
|
|
}
|
|
|
|
static void Daemonize(void) {
|
|
char ibuf[21];
|
|
int i, fd, pid;
|
|
for (i = 0; i < 128; ++i) close(i);
|
|
xsigaction(SIGHUP, OnHup, 0, 0, 0);
|
|
CHECK_NE(-1, (pid = fork()));
|
|
if (pid > 0) exit(0);
|
|
if (pid == -1) return;
|
|
CHECK_NE(-1, setsid());
|
|
CHECK_NE(-1, (pid = fork()));
|
|
if (pid > 0) _exit(0);
|
|
LOGIFNEG1(umask(0));
|
|
if (pidpath) {
|
|
CHECK_NE(-1, (fd = open(pidpath, O_CREAT | O_EXCL | O_WRONLY, 0644)));
|
|
CHECK_NE(-1, write(fd, ibuf, uint64toarray_radix10(getpid(), ibuf)));
|
|
LOGIFNEG1(close(fd));
|
|
}
|
|
if (!logpath) logpath = "/dev/null";
|
|
CHECK_NOTNULL(freopen("/dev/null", "r", stdin));
|
|
CHECK_NOTNULL(freopen(logpath, "a", stdout));
|
|
CHECK_NOTNULL(freopen(logpath, "a", stderr));
|
|
}
|
|
|
|
static int CompareHeaderValue(struct Client *c, int h, const char *s) {
|
|
return strncmp(s, c->inbuf + c->req.headers[h].a,
|
|
c->req.headers[h].b - c->req.headers[h].a);
|
|
}
|
|
|
|
static size_t GetIovSize(struct iovec *iov, int iovlen) {
|
|
int i;
|
|
size_t size;
|
|
for (size = i = 0; i < iovlen; ++i) {
|
|
DCHECK_NOTNULL(iov[i].iov_base);
|
|
size += iov[i].iov_len;
|
|
}
|
|
return size;
|
|
}
|
|
|
|
static ssize_t WriteSome(int fd, struct iovec *iov, int iovlen) {
|
|
ssize_t rc;
|
|
size_t wrote, n;
|
|
if ((rc = writev(fd, iov, iovlen)) != -1) {
|
|
wrote = n = rc;
|
|
do {
|
|
if (n >= iov->iov_len) {
|
|
iov->iov_len = 0;
|
|
n -= iov->iov_len;
|
|
++iov;
|
|
--iovlen;
|
|
} else {
|
|
iov->iov_base = (char *)iov->iov_base + n;
|
|
iov->iov_len -= n;
|
|
n = 0;
|
|
}
|
|
} while (n);
|
|
return wrote;
|
|
} else if (errno == EINTR) {
|
|
return 0;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static uint32_t Hash(const void *data, size_t size) {
|
|
uint32_t h;
|
|
h = crc32c(0, data, size);
|
|
if (!h) h = 1;
|
|
return h;
|
|
}
|
|
|
|
static bool HasHeader(struct Client *c, int h) {
|
|
return c->req.headers[h].b > c->req.headers[h].a;
|
|
}
|
|
|
|
int64_t GetGmtOffset(void) {
|
|
int64_t t;
|
|
struct tm tm;
|
|
t = nowl();
|
|
localtime_r(&t, &tm);
|
|
return tm.tm_gmtoff;
|
|
}
|
|
|
|
static int64_t LocoTimeToZulu(int64_t x) {
|
|
return x - gmtoff;
|
|
}
|
|
|
|
static int64_t GetLastModifiedZip(uint8_t *cfile) {
|
|
uint8_t *p, *pe;
|
|
for (p = ZIP_CFILE_EXTRA(cfile), pe = p + ZIP_CFILE_EXTRASIZE(cfile); p < pe;
|
|
p += ZIP_EXTRA_SIZE(p)) {
|
|
if (ZIP_EXTRA_HEADERID(p) == kZipExtraNtfs) {
|
|
return LocoTimeToZulu(READ64LE(ZIP_EXTRA_CONTENT(p) + 8) /
|
|
HECTONANOSECONDS -
|
|
MODERNITYSECONDS);
|
|
} else if (ZIP_EXTRA_HEADERID(p) == kZipExtraExtendedTimestamp) {
|
|
return READ32LE(ZIP_EXTRA_CONTENT(p) + 1);
|
|
}
|
|
}
|
|
return LocoTimeToZulu(DosDateTimeToUnix(ZIP_CFILE_LASTMODIFIEDDATE(cfile),
|
|
ZIP_CFILE_LASTMODIFIEDTIME(cfile)));
|
|
}
|
|
|
|
static bool IsCompressed(struct Asset *a) {
|
|
return ZIP_CFILE_COMPRESSIONMETHOD(zbase + a->cf) == kZipCompressionDeflate;
|
|
}
|
|
|
|
static int GetHttpVersion(struct Client *c) {
|
|
return ParseHttpVersion(c->inbuf + c->req.version.a,
|
|
c->req.version.b - c->req.version.a);
|
|
}
|
|
|
|
static bool IsNotModified(struct Client *c, struct Asset *a) {
|
|
if (!HasHeader(c, kHttpIfModifiedSince)) return false;
|
|
return a->lastmodified >=
|
|
ParseHttpDateTime(c->inbuf + c->req.headers[kHttpIfModifiedSince].a,
|
|
c->req.headers[kHttpIfModifiedSince].b -
|
|
c->req.headers[kHttpIfModifiedSince].a);
|
|
}
|
|
|
|
static char *FormatUnixHttpDateTime(char *s, int64_t t) {
|
|
struct tm tm;
|
|
gmtime_r(&t, &tm);
|
|
FormatHttpDateTime(s, &tm);
|
|
return s;
|
|
}
|
|
|
|
static void FreeAssetsIndex(struct Asset *p, size_t n) {
|
|
int i;
|
|
if (p) {
|
|
for (i = 0; i < n; ++i) {
|
|
free(p[i].lastmodifiedstr);
|
|
}
|
|
free(p);
|
|
}
|
|
}
|
|
|
|
static bool IndexAssets(const uint8_t *base, const uint8_t *cdir) {
|
|
bool ok;
|
|
int64_t lm;
|
|
struct Asset *p;
|
|
uint32_t i, n, m, cf, step, hash;
|
|
DCHECK_GE(HASH_LOAD_FACTOR, 2);
|
|
n = ZIP_CDIR_RECORDS(cdir);
|
|
m = roundup2pow(MAX(1, n) * HASH_LOAD_FACTOR);
|
|
p = calloc(m, sizeof(struct Asset));
|
|
ok = ZIP_CDIR_MAGIC(cdir) == kZipCdirHdrMagic;
|
|
if (p && ok) {
|
|
for (cf = ZIP_CDIR_OFFSET(cdir); n--; cf += ZIP_CFILE_HDRSIZE(base + cf)) {
|
|
if (ZIP_CFILE_MAGIC(base + cf) == kZipCfileHdrMagic) {
|
|
hash = Hash(ZIP_CFILE_NAME(base + cf), ZIP_CFILE_NAMESIZE(base + cf));
|
|
step = 0;
|
|
do {
|
|
i = (hash + (step * (step + 1)) >> 1) & (m - 1);
|
|
++step;
|
|
} while (p[i].hash);
|
|
lm = GetLastModifiedZip(base + cf);
|
|
p[i].hash = hash;
|
|
p[i].cf = cf;
|
|
p[i].lastmodified = lm;
|
|
p[i].lastmodifiedstr = FormatUnixHttpDateTime(xmalloc(30), lm);
|
|
} else {
|
|
WARNF("corrupt zip central directory entry");
|
|
ok = false;
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
WARNF("corrupt zip central directory");
|
|
}
|
|
if (ok) {
|
|
FreeAssetsIndex(assets.p, assets.n);
|
|
assets.p = p;
|
|
assets.n = m;
|
|
} else {
|
|
FreeAssetsIndex(p, m);
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
static bool OpenZip(const char *path) {
|
|
int fd;
|
|
bool ok;
|
|
void *map;
|
|
struct stat st;
|
|
const uint8_t *cdir;
|
|
if (zmap) {
|
|
LOGIFNEG1(munmap(zmap, zmapsize));
|
|
}
|
|
if (!zmap && ZIP_CDIR_MAGIC(__zip_end) == kZipCdirHdrMagic) {
|
|
if (IndexAssets(_base, __zip_end)) {
|
|
ok = true;
|
|
zbase = _base;
|
|
zcdir = __zip_end;
|
|
} else {
|
|
ok = false;
|
|
}
|
|
} else {
|
|
fd = -1;
|
|
map = MAP_FAILED;
|
|
if ((fd = open(path, O_RDONLY)) != -1 && fstat(fd, &st) != -1 &&
|
|
st.st_size &&
|
|
(map = mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0)) &&
|
|
(cdir = zipfindcentraldir(zmap, zmapsize)) && IndexAssets(map, cdir)) {
|
|
ok = true;
|
|
zmap = map;
|
|
zbase = map;
|
|
zcdir = cdir;
|
|
map = MAP_FAILED;
|
|
zmapsize = st.st_size;
|
|
} else {
|
|
ok = false;
|
|
}
|
|
if (map != MAP_FAILED) LOGIFNEG1(munmap(map, st.st_size));
|
|
if (fd != -1) LOGIFNEG1(close(fd));
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
static struct Asset *FindAsset(const char *path, size_t pathlen) {
|
|
uint32_t i, step, hash;
|
|
if (pathlen && path[0] == '/') ++path, --pathlen;
|
|
hash = Hash(path, pathlen);
|
|
for (step = 0;; ++step) {
|
|
i = (hash + (step * (step + 1)) >> 1) & (assets.n - 1);
|
|
if (!assets.p[i].hash) return NULL;
|
|
if (hash == assets.p[i].hash &&
|
|
pathlen == ZIP_CFILE_NAMESIZE(zbase + assets.p[i].cf) &&
|
|
memcmp(path, ZIP_CFILE_NAME(zbase + assets.p[i].cf), pathlen) == 0) {
|
|
return &assets.p[i];
|
|
}
|
|
}
|
|
}
|
|
|
|
static struct Asset *FindFile(const char *path, size_t pathlen) {
|
|
char *p, *buf;
|
|
struct Asset *asset;
|
|
if ((asset = FindAsset(path, pathlen))) return asset;
|
|
if (pathlen == 12 && memcmp(path, "/favicon.ico", 12) == 0) {
|
|
return FindAsset(FAVICON, strlen(FAVICON));
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static void *AddRange(char *content, long start, long length) {
|
|
intptr_t mend, mstart;
|
|
if (!__builtin_add_overflow((intptr_t)content, start, &mstart) ||
|
|
!__builtin_add_overflow(mstart, length, &mend) ||
|
|
((intptr_t)zbase <= mstart && mstart <= (intptr_t)zbase + zmapsize) ||
|
|
((intptr_t)zbase <= mend && mend <= (intptr_t)zbase + zmapsize)) {
|
|
return (void *)mstart;
|
|
} else {
|
|
abort();
|
|
}
|
|
}
|
|
|
|
static bool IsConnectionClose(struct Client *c) {
|
|
int n;
|
|
char *p;
|
|
p = c->inbuf + c->req.headers[kHttpConnection].a;
|
|
n = c->req.headers[kHttpConnection].b - c->req.headers[kHttpConnection].a;
|
|
return n == 5 && memcmp(p, "close", 5) == 0;
|
|
}
|
|
|
|
static char *AppendCrlf(char *p) {
|
|
return STPCPY(p, "\r\n");
|
|
}
|
|
|
|
#define AppendStatus(p, c, s) AppendStatus(p, c, s, sizeof(s) - 1)
|
|
static char *(AppendStatus)(char *p, int c, const char *s, size_t n) {
|
|
if (legacyhttp) {
|
|
p = STPCPY(p, "HTTP/1.0 ");
|
|
} else {
|
|
p = STPCPY(p, "HTTP/1.1 ");
|
|
}
|
|
p += uint64toarray_radix10(c, p);
|
|
*p++ = ' ';
|
|
p = mempcpy(p, s, n);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static void UpdateCurrentDate(long double now) {
|
|
int64_t t;
|
|
struct tm tm;
|
|
t = nowish = now;
|
|
gmtime_r(&t, &tm);
|
|
FormatHttpDateTime(currentdate, &tm);
|
|
}
|
|
|
|
static char *AppendDate(char *p) {
|
|
p = AppendHeaderName(p, "Date");
|
|
p = mempcpy(p, currentdate, 29);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendLastModified(char *p, const char *s) {
|
|
p = AppendHeaderName(p, "Last-Modified");
|
|
p = mempcpy(p, s, 29);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendServer(char *p) {
|
|
const char *s;
|
|
if (*(s = firstnonnull(serverheader, DEFAULT_SERVER))) {
|
|
p = AppendHeaderName(p, "Server");
|
|
p = stpcpy(p, s);
|
|
p = AppendCrlf(p);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
static char *AppendConnectionClose(char *p) {
|
|
p = AppendHeaderName(p, "Connection");
|
|
p = STPCPY(p, "close");
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendAcceptRangesBytes(char *p) {
|
|
p = AppendHeaderName(p, "Accept-Ranges");
|
|
p = STPCPY(p, "bytes");
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendNosniff(char *p) {
|
|
p = AppendHeaderName(p, "X-Content-Type-Options");
|
|
p = STPCPY(p, "nosniff");
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendContentType(char *p, const char *ct) {
|
|
p = AppendHeaderName(p, "Content-Type");
|
|
p = stpcpy(p, ct);
|
|
if (startswith(ct, "text/")) {
|
|
p = STPCPY(p, "; charset=utf-8");
|
|
}
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendContentTypeTextPlain(char *p) {
|
|
return AppendContentType(p, "text/plain");
|
|
}
|
|
|
|
static char *AppendExpires(char *p, int64_t t) {
|
|
struct tm tm;
|
|
gmtime_r(&t, &tm);
|
|
p = AppendHeaderName(p, "Expires");
|
|
p = FormatHttpDateTime(p, &tm);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendVaryContentEncoding(char *p) {
|
|
p = AppendHeaderName(p, "Vary");
|
|
p = STPCPY(p, "Accept-Encoding");
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendCache(char *p) {
|
|
int x;
|
|
x = cacheseconds;
|
|
if (!x) return p;
|
|
x = MAX(0, x);
|
|
p = AppendHeaderName(p, "Cache-Control");
|
|
p = STPCPY(p, "max-age=");
|
|
p += uint64toarray_radix10(x, p);
|
|
if (x) p = STPCPY(p, ", public");
|
|
p = AppendCrlf(p);
|
|
return AppendExpires(p, nowish + cacheseconds);
|
|
}
|
|
|
|
static char *AppendContentLength(char *p, size_t n) {
|
|
p = AppendHeaderName(p, "Content-Length");
|
|
p += uint64toarray_radix10(n, p);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendContentRange(char *p, long rangestart, long rangelength,
|
|
long contentlength) {
|
|
long endrange;
|
|
CHECK_GE(rangestart + rangelength, rangestart);
|
|
CHECK_LE(rangestart + rangelength, contentlength);
|
|
if (__builtin_add_overflow(rangestart, rangelength, &endrange)) abort();
|
|
p = AppendHeaderName(p, "Content-Range");
|
|
p = STPCPY(p, "bytes ");
|
|
p += uint64toarray_radix10(rangestart, p);
|
|
*p++ = '-';
|
|
p += uint64toarray_radix10(endrange, p);
|
|
*p++ = '/';
|
|
p += uint64toarray_radix10(contentlength, p);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendContentEncodingGzip(char *p) {
|
|
p = AppendHeaderName(p, "Content-Encoding");
|
|
p = STPCPY(p, "gzip");
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static char *AppendRedirect(struct Client *c, char *p, const char *s) {
|
|
VERBOSEF("%s %s %.*s redirect %s", c->addrstr, kHttpMethod[c->req.method],
|
|
c->req.uri.b - c->req.uri.a, c->inbuf + c->req.uri.a, s);
|
|
p = AppendStatus(p, 302, "Temporary Redirect");
|
|
p = AppendHeaderName(p, "Location");
|
|
p = STPCPY(p, s);
|
|
return AppendCrlf(p);
|
|
}
|
|
|
|
static bool InflateTiny(uint8_t *outbuf, size_t outsize, const uint8_t *inbuf,
|
|
size_t insize) {
|
|
struct DeflateState ds;
|
|
return undeflate(outbuf, outsize, inbuf, insize, &ds) != -1;
|
|
}
|
|
|
|
static bool InflateZlib(uint8_t *outbuf, size_t outsize, const uint8_t *inbuf,
|
|
size_t insize) {
|
|
bool ok;
|
|
z_stream zs;
|
|
ok = false;
|
|
zs.next_in = inbuf;
|
|
zs.avail_in = insize;
|
|
zs.total_in = insize;
|
|
zs.next_out = outbuf;
|
|
zs.avail_out = outsize;
|
|
zs.total_out = outsize;
|
|
zs.zfree = Z_NULL;
|
|
zs.zalloc = Z_NULL;
|
|
if (inflateInit2(&zs, -MAX_WBITS) == Z_OK) {
|
|
switch (inflate(&zs, Z_NO_FLUSH)) {
|
|
case Z_STREAM_END:
|
|
ok = true;
|
|
break;
|
|
case Z_MEM_ERROR:
|
|
WARNF("Z_MEM_ERROR");
|
|
break;
|
|
case Z_DATA_ERROR:
|
|
WARNF("Z_DATA_ERROR");
|
|
break;
|
|
case Z_NEED_DICT:
|
|
WARNF("Z_NEED_DICT");
|
|
break;
|
|
default:
|
|
abort();
|
|
}
|
|
inflateEnd(&zs);
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
static bool Inflate(uint8_t *outbuf, size_t outsize, const uint8_t *inbuf,
|
|
size_t insize) {
|
|
if (IsTiny()) {
|
|
return InflateTiny(outbuf, outsize, inbuf, insize);
|
|
} else {
|
|
return InflateZlib(outbuf, outsize, inbuf, insize);
|
|
}
|
|
}
|
|
|
|
static void LogRequestLatency(struct Client *c) {
|
|
long double now = nowl();
|
|
LOGF("%s latency req %,16ldns conn %,16ldns", c->addrstr,
|
|
(long)((now - c->startrequest) * 1e9),
|
|
(long)((now - c->startconnection) * 1e9));
|
|
}
|
|
|
|
bool HandleRequest(struct Client *c) {
|
|
char *p;
|
|
int msgsize;
|
|
bool gzipped;
|
|
void *content;
|
|
size_t pathlen;
|
|
struct Asset *a;
|
|
unsigned version;
|
|
const char *path, *location;
|
|
long lf, contentlength, actualcontentlength, rangestart, rangelength;
|
|
p = c->outbuf;
|
|
content = "";
|
|
gzipped = false;
|
|
contentlength = -1;
|
|
c->close = false;
|
|
if ((msgsize = ParseHttpRequest(&c->req, c->inbuf, c->insize)) != -1) {
|
|
if (!msgsize) return false;
|
|
if (logmessages) {
|
|
LOGF("%s received %,d byte message\n%.*s", c->addrstr, c->req.length,
|
|
c->req.length - 4, c->inbuf);
|
|
}
|
|
version = GetHttpVersion(c);
|
|
if (version < 101) c->close = true, legacyhttp = true;
|
|
if (version <= 101) {
|
|
if (IsConnectionClose(c)) c->close = true;
|
|
path = c->inbuf + c->req.uri.a;
|
|
pathlen = c->req.uri.b - c->req.uri.a;
|
|
if (c->req.method == kHttpGet || c->req.method == kHttpHead) {
|
|
if (ParseContentLength(&c->req, c->inbuf)) c->close = true;
|
|
VERBOSEF(
|
|
"%s %s %.*s referer %.*s", c->addrstr, kHttpMethod[c->req.method],
|
|
pathlen, path,
|
|
c->req.headers[kHttpReferer].b - c->req.headers[kHttpReferer].a,
|
|
c->inbuf + c->req.headers[kHttpReferer].a);
|
|
if ((location = LookupRedirect(path, pathlen))) {
|
|
p = AppendRedirect(c, p, DEFAULT_PATH);
|
|
} else if ((a = FindFile(path, pathlen))) {
|
|
if (IsNotModified(c, a)) {
|
|
VERBOSEF("%s %s %.*s not modified", c->addrstr,
|
|
kHttpMethod[c->req.method], pathlen, path);
|
|
p = AppendStatus(p, 304, "Not Modified");
|
|
} else {
|
|
lf = ZIP_CFILE_OFFSET(zbase + a->cf);
|
|
content = ZIP_LFILE_CONTENT(zbase + lf);
|
|
contentlength = ZIP_CFILE_COMPRESSEDSIZE(zbase + a->cf);
|
|
if (IsCompressed(a)) {
|
|
if (memmem(c->inbuf + c->req.headers[kHttpAcceptEncoding].a,
|
|
c->req.headers[kHttpAcceptEncoding].b -
|
|
c->req.headers[kHttpAcceptEncoding].a,
|
|
"gzip", 4)) {
|
|
gzipped = true;
|
|
memcpy(c->gzip_footer + 0, zbase + a->cf + kZipCfileOffsetCrc32,
|
|
4);
|
|
memcpy(c->gzip_footer + 4,
|
|
zbase + a->cf + kZipCfileOffsetUncompressedsize, 4);
|
|
p = AppendStatus(p, 200, "OK");
|
|
p = AppendContentEncodingGzip(p);
|
|
} else if (Inflate(
|
|
(content = gc(xmalloc(
|
|
ZIP_CFILE_UNCOMPRESSEDSIZE(zbase + a->cf)))),
|
|
(contentlength =
|
|
ZIP_CFILE_UNCOMPRESSEDSIZE(zbase + a->cf)),
|
|
ZIP_LFILE_CONTENT(zbase + lf),
|
|
ZIP_CFILE_COMPRESSEDSIZE(zbase + a->cf))) {
|
|
p = AppendStatus(p, 200, "OK");
|
|
} else {
|
|
WARNF("%s %s %.*s internal server error", c->addrstr,
|
|
kHttpMethod[c->req.method], pathlen, path);
|
|
p = AppendStatus(p, 500, "Internal Server Error");
|
|
content = "Internal Server Error\r\n";
|
|
contentlength = -1;
|
|
}
|
|
} else if (HasHeader(c, kHttpRange)) {
|
|
if (ParseHttpRange(c->inbuf + c->req.headers[kHttpRange].a,
|
|
c->req.headers[kHttpRange].b -
|
|
c->req.headers[kHttpRange].a,
|
|
contentlength, &rangestart, &rangelength)) {
|
|
p = AppendStatus(p, 206, "Partial Content");
|
|
p = AppendContentRange(p, rangestart, rangelength,
|
|
contentlength);
|
|
content = AddRange(content, rangestart, rangelength);
|
|
contentlength = rangelength;
|
|
} else {
|
|
WARNF(
|
|
"%s %s %.*s bad range %`'.*s", c->addrstr,
|
|
kHttpMethod[c->req.method], pathlen, path,
|
|
c->req.headers[kHttpRange].b - c->req.headers[kHttpRange].a,
|
|
c->inbuf + c->req.headers[kHttpRange].a);
|
|
p = AppendStatus(p, 416, "Range Not Satisfiable");
|
|
p = AppendContentRange(p, rangestart, rangelength,
|
|
contentlength);
|
|
content = "";
|
|
contentlength = 0;
|
|
}
|
|
} else {
|
|
p = AppendStatus(p, 200, "OK");
|
|
}
|
|
}
|
|
p = AppendLastModified(p, a->lastmodifiedstr);
|
|
p = AppendContentType(p, GetContentType(path, pathlen));
|
|
p = AppendCache(p);
|
|
if (!IsCompressed(a)) {
|
|
p = AppendAcceptRangesBytes(p);
|
|
} else {
|
|
p = AppendVaryContentEncoding(p);
|
|
}
|
|
} else {
|
|
WARNF("%s %s %.*s not found", c->addrstr, kHttpMethod[c->req.method],
|
|
pathlen, path);
|
|
p = AppendStatus(p, 404, "Not Found");
|
|
p = AppendContentTypeTextPlain(p);
|
|
content = "Not Found\r\n";
|
|
}
|
|
} else {
|
|
WARNF("%s %s %.*s method not allowed", c->addrstr,
|
|
kHttpMethod[c->req.method], pathlen, path);
|
|
p = AppendStatus(p, 405, "method not allowed");
|
|
p = AppendContentTypeTextPlain(p);
|
|
content = "Method Not Allowed\r\n";
|
|
c->close = true;
|
|
}
|
|
} else {
|
|
WARNF("%s http version not supported %`'.*s", c->addrstr,
|
|
c->req.version.b - c->req.version.a, c->inbuf + c->req.version.a);
|
|
p = AppendStatus(p, 505, "HTTP Version Not Supported");
|
|
p = AppendContentTypeTextPlain(p);
|
|
content = "HTTP Version Not Supported\r\n";
|
|
c->close = true;
|
|
}
|
|
} else {
|
|
WARNF("%s parse error %s", c->addrstr, strerror(errno));
|
|
p = AppendStatus(p, 400, "Bad Request");
|
|
p = AppendContentTypeTextPlain(p);
|
|
content = "Bad Request\r\n";
|
|
c->close = true;
|
|
}
|
|
p = AppendDate(p);
|
|
p = AppendNosniff(p);
|
|
p = AppendServer(p);
|
|
if (c->close) p = AppendConnectionClose(p);
|
|
if (contentlength == -1) contentlength = strlen(content);
|
|
actualcontentlength = contentlength;
|
|
if (gzipped) {
|
|
actualcontentlength += sizeof(kGzipHeader) + sizeof(c->gzip_footer);
|
|
}
|
|
p = AppendContentLength(p, actualcontentlength);
|
|
p = AppendCrlf(p);
|
|
if (logmessages) {
|
|
LOGF("%s sending %,d byte message\n%.*s", c->addrstr, p - c->outbuf,
|
|
p - c->outbuf - 4, c->outbuf);
|
|
}
|
|
CHECK_LT(p, c->outbuf + sizeof(c->outbuf));
|
|
c->iovlen = 0;
|
|
c->iov[c->iovlen].iov_base = c->outbuf;
|
|
c->iov[c->iovlen].iov_len = p - c->outbuf;
|
|
++c->iovlen;
|
|
if (c->req.method != kHttpHead) {
|
|
if (gzipped) {
|
|
c->iov[c->iovlen].iov_base = kGzipHeader;
|
|
c->iov[c->iovlen].iov_len = sizeof(kGzipHeader);
|
|
++c->iovlen;
|
|
}
|
|
c->iov[c->iovlen].iov_base = content;
|
|
c->iov[c->iovlen].iov_len = contentlength;
|
|
++c->iovlen;
|
|
if (gzipped) {
|
|
c->iov[c->iovlen].iov_base = c->gzip_footer;
|
|
c->iov[c->iovlen].iov_len = sizeof(c->gzip_footer);
|
|
++c->iovlen;
|
|
}
|
|
}
|
|
/* LogRequestLatency(); */
|
|
if (WriteSome(c->fd, c->iov, c->iovlen) == -1) {
|
|
VERBOSEF("%s send error %s", c->addrstr, strerror(errno));
|
|
c->close = true;
|
|
c->insize = 0;
|
|
c->iovlen = 0;
|
|
}
|
|
if (!c->close) {
|
|
memcpy(c->inbuf, c->inbuf + msgsize, c->insize - msgsize);
|
|
c->insize -= msgsize;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void DeleteClient(struct Client *c) {
|
|
DEBUGF("%s close", c->addrstr);
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_DEL, c->fd, NULL));
|
|
LOGIFNEG1(close(c->fd));
|
|
if (c->next) {
|
|
c->next->prev = c->prev;
|
|
}
|
|
if (c->prev) {
|
|
c->prev->next = c->next;
|
|
} else {
|
|
clients = c->next;
|
|
}
|
|
memset(c, 0, sizeof(*c));
|
|
free(c);
|
|
}
|
|
|
|
static void ProcessInbuf(struct Client *c) {
|
|
long double now;
|
|
for (;;) {
|
|
c->startrequest = now = nowl();
|
|
if (now - nowish > 1) UpdateCurrentDate(now);
|
|
if (HandleRequest(c)) {
|
|
if (GetIovSize(c->iov, c->iovlen)) {
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_MOD, c->fd,
|
|
&(struct epoll_event){EPOLLOUT, {c}}));
|
|
} else {
|
|
if (c->close && !c->insize) {
|
|
DeleteClient(c);
|
|
} else if (c->insize) {
|
|
continue;
|
|
}
|
|
}
|
|
} else if (c->insize == sizeof(c->inbuf)) {
|
|
WARNF("%s http message too big", c->addrstr);
|
|
DeleteClient(c);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void ProcessRead(struct Client *c) {
|
|
char *p;
|
|
ssize_t rc;
|
|
rc = read(c->fd, c->inbuf + c->insize, sizeof(c->inbuf) - c->insize);
|
|
if (rc != -1) {
|
|
if (rc) {
|
|
c->insize += (size_t)rc;
|
|
ProcessInbuf(c);
|
|
} else {
|
|
DeleteClient(c);
|
|
}
|
|
} else if (errno != EINTR) {
|
|
if (errno == ECONNRESET) {
|
|
DEBUGF("%s reset", c->addrstr);
|
|
} else {
|
|
WARNF("%s recv error %s", c->addrstr, strerror(errno));
|
|
}
|
|
DeleteClient(c);
|
|
}
|
|
}
|
|
|
|
static void ProcessWrite(struct Client *c) {
|
|
if (WriteSome(c->fd, c->iov, c->iovlen) != -1) {
|
|
if (!GetIovSize(c->iov, c->iovlen)) {
|
|
if (!c->close) {
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_MOD, c->fd,
|
|
&(struct epoll_event){EPOLLIN, {c}}));
|
|
ProcessInbuf(c);
|
|
} else {
|
|
DeleteClient(c);
|
|
}
|
|
}
|
|
} else {
|
|
VERBOSEF("%s send error %s", c->addrstr, strerror(errno));
|
|
DeleteClient(c);
|
|
}
|
|
}
|
|
|
|
static void ProcessConnect(void) {
|
|
int fd;
|
|
struct Client *c;
|
|
c = xmalloc(sizeof(struct Client));
|
|
c->addrsize = sizeof(c->addr);
|
|
if ((fd = accept(server, &c->addr, &c->addrsize)) != -1) {
|
|
c->fd = fd;
|
|
c->prev = NULL;
|
|
c->next = clients;
|
|
c->insize = 0;
|
|
c->iovlen = 0;
|
|
c->close = false;
|
|
c->startconnection = nowl();
|
|
c->startrequest = c->startconnection;
|
|
DescribeAddress(c->addrstr, &c->addr);
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_ADD, c->fd,
|
|
&(struct epoll_event){EPOLLIN, {c}}));
|
|
if (clients) clients->prev = c;
|
|
clients = c;
|
|
DEBUGF("%s accept", c->addrstr);
|
|
} else if (errno != EINTR) {
|
|
FATALF("%s accept error %s", serveraddrstr, strerror(errno));
|
|
free(c);
|
|
}
|
|
}
|
|
|
|
static void HandleSignals(void) {
|
|
if (invalidated) {
|
|
if (OpenZip(programfile)) {
|
|
LOGF("%s reindexed zip", serveraddrstr);
|
|
} else {
|
|
WARNF("%s reindexing failed", serveraddrstr);
|
|
}
|
|
invalidated = false;
|
|
}
|
|
if (heartbeat | notimer) {
|
|
UpdateCurrentDate(nowl());
|
|
heartbeat = false;
|
|
}
|
|
}
|
|
|
|
static void ProcessEvents(void) {
|
|
int i, j, n;
|
|
struct epoll_event events[16];
|
|
for (;;) {
|
|
if (terminated) {
|
|
while (clients) {
|
|
if (!clients->insize) {
|
|
DeleteClient(clients);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (!clients) break;
|
|
}
|
|
if ((n = epoll_wait(epfd, events, ARRAYLEN(events), 200)) != -1) {
|
|
for (i = 0; i < n; ++i) {
|
|
if (!events[i].data.ptr) {
|
|
CHECK(events[i].events & EPOLLIN);
|
|
ProcessConnect();
|
|
} else if (events[i].events & (EPOLLIN | EPOLLERR | EPOLLHUP)) {
|
|
ProcessRead(events[i].data.ptr);
|
|
} else if (events[i].events & EPOLLOUT) {
|
|
ProcessWrite(events[i].data.ptr);
|
|
}
|
|
HandleSignals();
|
|
}
|
|
} else {
|
|
CHECK_EQ(EINTR, errno);
|
|
HandleSignals();
|
|
}
|
|
}
|
|
}
|
|
|
|
static void TuneServerSocket(void) {
|
|
int yes = 1;
|
|
LOGIFNEG1(setsockopt(server, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)));
|
|
LOGIFNEG1(setsockopt(server, SOL_SOCKET, SO_REUSEPORT, &yes, sizeof(yes)));
|
|
LOGIFNEG1(setsockopt(server, IPPROTO_TCP, TCP_NODELAY, &yes, sizeof(yes)));
|
|
LOGIFNEG1(setsockopt(server, IPPROTO_TCP, TCP_FASTOPEN, &yes, sizeof(yes)));
|
|
LOGIFNEG1(setsockopt(server, IPPROTO_TCP, TCP_QUICKACK, &yes, sizeof(yes)));
|
|
}
|
|
|
|
void GreenBean(void) {
|
|
gmtoff = GetGmtOffset();
|
|
programfile = (const char *)getauxval(AT_EXECFN);
|
|
CHECK(OpenZip(programfile));
|
|
xsigaction(SIGINT, OnTerminate, 0, 0, 0);
|
|
xsigaction(SIGHUP, OnTerminate, 0, 0, 0);
|
|
xsigaction(SIGTERM, OnTerminate, 0, 0, 0);
|
|
xsigaction(SIGCHLD, SIG_IGN, 0, 0, 0);
|
|
xsigaction(SIGPIPE, SIG_IGN, 0, 0, 0);
|
|
xsigaction(SIGUSR1, OnHup, 0, 0, 0);
|
|
xsigaction(SIGALRM, OnAlarm, 0, 0, 0);
|
|
if (setitimer(ITIMER_REAL, &kHeartbeat, NULL) == -1) notimer = true;
|
|
CHECK_NE(-1, (epfd = epoll_create1(0)));
|
|
CHECK_NE(-1, (server = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)));
|
|
TuneServerSocket();
|
|
CHECK_NE(-1, bind(server, &serveraddr, sizeof(serveraddr)));
|
|
CHECK_NE(-1, listen(server, 10));
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_ADD, server,
|
|
&(struct epoll_event){EPOLLIN}));
|
|
DescribeAddress(serveraddrstr, &serveraddr);
|
|
if (daemonize) Daemonize();
|
|
VERBOSEF("%s listen", serveraddrstr);
|
|
heartbeat = true;
|
|
ProcessEvents();
|
|
CHECK_NE(-1, epoll_ctl(epfd, EPOLL_CTL_DEL, server, NULL));
|
|
VERBOSEF("%s terminated", serveraddrstr);
|
|
LOGIFNEG1(close(server));
|
|
LOGIFNEG1(close(epfd));
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
showcrashreports();
|
|
GetOpts(argc, argv);
|
|
GreenBean();
|
|
return 0;
|
|
}
|