X-Git-Url: http://www.dolda2000.com/gitweb/?p=ashd.git;a=blobdiff_plain;f=src%2Fhtparser.c;h=de5bbc2a61f32f93fa1ebe1db51c197fcd857840;hp=a195fab3d81bd885771c06085ba0510b95a23a51;hb=ecd4208b888be7f60bc2ae013b799861959b7cb9;hpb=3ef788959b54c635638b84fd628fab7bfbbf1e46 diff --git a/src/htparser.c b/src/htparser.c index a195fab..de5bbc2 100644 --- a/src/htparser.c +++ b/src/htparser.c @@ -41,6 +41,7 @@ static int plex; static char *pidfile = NULL; static int daemonize, usesyslog; +struct mtbuf listeners; static void trimx(struct hthead *req) { @@ -148,6 +149,59 @@ static off_t passdata(FILE *in, FILE *out, off_t max) return(total); } +static int recvchunks(FILE *in, FILE *out) +{ + char buf[8192]; + size_t read, chlen; + int c, r; + + while(1) { + chlen = 0; + r = 0; + while(1) { + c = getc(in); + if(c == 10) { + if(!r) + return(-1); + break; + } else if(c == 13) { + } else if((c >= '0') && (c <= '9')) { + chlen = (chlen << 4) + (c - '0'); + r = 1; + } else if((c >= 'A') && (c <= 'F')) { + chlen = (chlen << 4) + (c + 10 - 'A'); + r = 1; + } else if((c >= 'a') && (c <= 'f')) { + chlen = (chlen << 4) + (c + 10 - 'a'); + r = 1; + } else { + /* XXX: Technically, there may be chunk extensions to + * be read, but since that will likely never actually + * happen in practice, I can just as well add support + * for that if it actually does become relevant. */ + return(-1); + } + } + if(chlen == 0) + break; + while(chlen > 0) { + read = fread(buf, 1, min(sizeof(buf), chlen), in); + if(feof(in) || ferror(in)) + return(-1); + if(fwrite(buf, 1, read, out) != read) + return(-1); + chlen -= read; + } + if((getc(in) != 13) || (getc(in) != 10)) + return(-1); + } + /* XXX: Technically, there may be trailers to be read, but that's + * just about as likely as chunk extensions. */ + if((getc(in) != 13) || (getc(in) != 10)) + return(-1); + return(0); +} + static int passchunks(FILE *in, FILE *out) { char buf[8192]; @@ -212,6 +266,20 @@ static int canonreq(struct hthead *req) return(0); } +static int http10keep(struct hthead *req, struct hthead *resp) +{ + int fc; + + fc = hasheader(resp, "connection", "close"); + headrmheader(resp, "connection"); + if(!fc && hasheader(req, "connection", "keep-alive")) { + headappheader(resp, "Connection", "Keep-Alive"); + return(1); + } else { + return(0); + } +} + void serve(FILE *in, struct conn *conn) { int pfds[2]; @@ -219,10 +287,11 @@ void serve(FILE *in, struct conn *conn) struct hthead *req, *resp; char *hd; off_t dlen; + int keep; out = NULL; req = resp = NULL; - while(1) { + while(plex >= 0) { if((req = parsereq(in)) == NULL) break; if(!canonreq(req)) @@ -231,7 +300,7 @@ void serve(FILE *in, struct conn *conn) if((conn->initreq != NULL) && conn->initreq(conn, req)) break; - if(block(plex, EV_WRITE, 60) <= 0) + if((plex < 0) || block(plex, EV_WRITE, 60) <= 0) break; if(socketpair(PF_UNIX, SOCK_STREAM, 0, pfds)) break; @@ -240,11 +309,19 @@ void serve(FILE *in, struct conn *conn) close(pfds[0]); out = mtstdopen(pfds[1], 1, 600, "r+"); - if((hd = getheader(req, "content-length")) != NULL) { - dlen = atoo(hd); - if(dlen > 0) { - if(passdata(in, out, dlen) != dlen) + if(getheader(req, "content-type") != NULL) { + if((hd = getheader(req, "content-length")) != NULL) { + dlen = atoo(hd); + if(dlen > 0) { + if(passdata(in, out, dlen) != dlen) + break; + } + } else if(((hd = getheader(req, "transfer-encoding")) != NULL) && !strcasecmp(hd, "chunked")) { + if(recvchunks(in, out)) break; + } else { + /* Ignore rather than abort, to be kinder to broken clients. */ + headrmheader(req, "content-type"); } } if(fflush(out)) @@ -259,25 +336,28 @@ void serve(FILE *in, struct conn *conn) if(!getheader(resp, "server")) headappheader(resp, "Server", sprintf3("ashd/%s", VERSION)); - if(!strcmp(req->ver, "HTTP/1.0")) { - writeresp(in, resp); - fprintf(in, "\r\n"); + if(!strcasecmp(req->ver, "HTTP/1.0")) { if(!strcasecmp(req->method, "head")) { - if(!hasheader(req, "connection", "keep-alive")) - break; + keep = http10keep(req, resp); + writeresp(in, resp); + fprintf(in, "\r\n"); } else if((hd = getheader(resp, "content-length")) != NULL) { + keep = http10keep(req, resp); dlen = atoo(hd); + writeresp(in, resp); + fprintf(in, "\r\n"); if(passdata(out, in, dlen) != dlen) break; - if(!hasheader(req, "connection", "keep-alive")) - break; } else { + headrmheader(resp, "connection"); + writeresp(in, resp); + fprintf(in, "\r\n"); passdata(out, in, -1); break; } - if(hasheader(req, "connection", "close") || hasheader(resp, "connection", "close")) + if(!keep) break; - } else if(!strcmp(req->ver, "HTTP/1.1")) { + } else if(!strcasecmp(req->ver, "HTTP/1.1")) { if(!strcasecmp(req->method, "head")) { writeresp(in, resp); fprintf(in, "\r\n"); @@ -325,30 +405,40 @@ static void plexwatch(struct muth *muth, va_list args) { vavar(int, fd); char *buf; - int ret; + int i, ret; while(1) { - block(fd, EV_READ, 0); + if(block(fd, EV_READ, 0) == 0) + break; buf = smalloc(65536); ret = recv(fd, buf, 65536, 0); if(ret < 0) { flog(LOG_WARNING, "received error on rootplex read channel: %s", strerror(errno)); exit(1); } else if(ret == 0) { - exit(0); + free(buf); + break; } /* Maybe I'd like to implement some protocol in this direction * some day... */ free(buf); } + shutdown(plex, SHUT_RDWR); + for(i = 0; i < listeners.d; i++) { + if(listeners.b[i] == muth) + bufdel(listeners, i); + } + flog(LOG_INFO, "root handler exited, so shutting down listening..."); + while(listeners.d > 0) + resume(listeners.b[0], 0); } static void initroot(void *uu) { int fd; + setsid(); if(daemonize) { - setsid(); chdir("/"); if((fd = open("/dev/null", O_RDWR)) >= 0) { dup2(fd, 0); @@ -415,9 +505,14 @@ static void addport(char *spec) buffree(vals); } +static void sighandler(int sig) +{ + exitioloop(1); +} + int main(int argc, char **argv) { - int c; + int c, d; int i, s1; char *root; FILE *pidout; @@ -469,7 +564,7 @@ int main(int argc, char **argv) flog(LOG_ERR, "could not spawn root multiplexer: %s", strerror(errno)); return(1); } - mustart(plexwatch, plex); + bufadd(listeners, mustart(plexwatch, plex)); pidout = NULL; if(pidfile != NULL) { if((pidout = fopen(pidfile, "w")) == NULL) { @@ -496,6 +591,9 @@ int main(int argc, char **argv) } } signal(SIGPIPE, SIG_IGN); + signal(SIGCHLD, SIG_IGN); + signal(SIGINT, sighandler); + signal(SIGTERM, sighandler); if(daemonize) { daemon(0, 0); } @@ -503,6 +601,22 @@ int main(int argc, char **argv) fprintf(pidout, "%i\n", getpid()); fclose(pidout); } - ioloop(); + d = 0; + while(!d) { + switch(ioloop()) { + case 0: + d = 1; + break; + case 1: + if(listeners.d > 0) { + while(listeners.d > 0) + resume(listeners.b[0], 0); + flog(LOG_INFO, "no longer listening"); + } else { + d = 1; + } + break; + } + } return(0); }