1121 lines
28 KiB
C
1121 lines
28 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
* information regarding copyright ownership.
|
|
*/
|
|
|
|
/*! \file */
|
|
|
|
#include <ctype.h>
|
|
#include <inttypes.h>
|
|
#include <stdbool.h>
|
|
#include <string.h>
|
|
|
|
#include <isc/buffer.h>
|
|
#include <isc/httpd.h>
|
|
#include <isc/list.h>
|
|
#include <isc/mem.h>
|
|
#include <isc/netmgr.h>
|
|
#include <isc/refcount.h>
|
|
#include <isc/sockaddr.h>
|
|
#include <isc/string.h>
|
|
#include <isc/time.h>
|
|
#include <isc/url.h>
|
|
#include <isc/util.h>
|
|
|
|
#include "netmgr/netmgr-int.h"
|
|
#include "picohttpparser.h"
|
|
|
|
#ifdef HAVE_ZLIB
|
|
#include <zlib.h>
|
|
#endif /* ifdef HAVE_ZLIB */
|
|
|
|
#define CHECK(m) \
|
|
do { \
|
|
result = (m); \
|
|
if (result != ISC_R_SUCCESS) { \
|
|
goto cleanup; \
|
|
} \
|
|
} while (0)
|
|
|
|
/*
|
|
* Size the recv buffer to hold at maximum two full buffers from isc_nm_read(),
|
|
* so we don't have to handle the truncation.
|
|
*/
|
|
#define HTTP_RECVLEN ISC_NETMGR_TCP_RECVBUF_SIZE * 2
|
|
#define HTTP_SENDLEN ISC_NETMGR_TCP_RECVBUF_SIZE
|
|
#define HTTP_HEADERS_NUM 100
|
|
#define HTTP_MAX_REQUEST_LEN 4096
|
|
|
|
typedef enum httpd_flags {
|
|
CONNECTION_CLOSE = 1 << 0, /* connection must close */
|
|
CONNECTION_KEEP_ALIVE = 1 << 1, /* response needs a keep-alive header */
|
|
ACCEPT_DEFLATE = 1 << 2, /* response can be compressed */
|
|
} httpd_flags_t;
|
|
|
|
#define HTTPD_MAGIC ISC_MAGIC('H', 't', 'p', 'd')
|
|
#define VALID_HTTPD(m) ISC_MAGIC_VALID(m, HTTPD_MAGIC)
|
|
|
|
#define HTTPDMGR_MAGIC ISC_MAGIC('H', 'p', 'd', 'm')
|
|
#define VALID_HTTPDMGR(m) ISC_MAGIC_VALID(m, HTTPDMGR_MAGIC)
|
|
|
|
/*%
|
|
* HTTP methods.
|
|
*/
|
|
typedef enum { METHOD_UNKNOWN = 0, METHOD_GET = 1, METHOD_POST = 2 } method_t;
|
|
|
|
/*%
|
|
* HTTP urls. These are the URLs we manage, and the function to call to
|
|
* provide the data for it. We pass in the base url (so the same function
|
|
* can handle multiple requests), and a structure to fill in to return a
|
|
* result to the client. We also pass in a pointer to be filled in for
|
|
* the data cleanup function.
|
|
*/
|
|
struct isc_httpdurl {
|
|
char *url;
|
|
isc_httpdaction_t *action;
|
|
void *action_arg;
|
|
bool isstatic;
|
|
isc_time_t loadtime;
|
|
ISC_LINK(isc_httpdurl_t) link;
|
|
};
|
|
|
|
/*% http client */
|
|
struct isc_httpd {
|
|
unsigned int magic; /* HTTPD_MAGIC */
|
|
isc_refcount_t references;
|
|
|
|
isc_httpdmgr_t *mgr; /*%< our parent */
|
|
ISC_LINK(isc_httpd_t) link;
|
|
|
|
isc_nmhandle_t *handle; /* Permanent pointer to handle */
|
|
|
|
/*%
|
|
* Received data state.
|
|
*/
|
|
char recvbuf[HTTP_RECVLEN]; /*%< receive buffer */
|
|
size_t recvlen; /*%< length recv'd */
|
|
size_t consume; /*%< length of last command */
|
|
|
|
method_t method;
|
|
int minor_version;
|
|
httpd_flags_t flags;
|
|
const char *path;
|
|
isc_url_parser_t up;
|
|
isc_time_t if_modified_since;
|
|
};
|
|
|
|
#if ISC_HTTPD_TRACE
|
|
#define isc_httpd_ref(ptr) isc_httpd__ref(ptr, __func__, __FILE__, __LINE__)
|
|
#define isc_httpd_unref(ptr) isc_httpd__unref(ptr, __func__, __FILE__, __LINE__)
|
|
#define isc_httpd_attach(ptr, ptrp) \
|
|
isc_httpd__attach(ptr, ptrp, __func__, __FILE__, __LINE__)
|
|
#define isc_httpd_detach(ptrp) \
|
|
isc_httpd__detach(ptrp, __func__, __FILE__, __LINE__)
|
|
ISC_REFCOUNT_TRACE_DECL(isc_httpd);
|
|
#else
|
|
ISC_REFCOUNT_DECL(isc_httpd);
|
|
#endif
|
|
|
|
struct isc_httpdmgr {
|
|
unsigned int magic; /* HTTPDMGR_MAGIC */
|
|
isc_refcount_t references;
|
|
isc_mem_t *mctx;
|
|
isc_nmsocket_t *sock;
|
|
|
|
isc_httpdclientok_t *client_ok; /*%< client validator */
|
|
isc_httpdondestroy_t *ondestroy; /*%< cleanup callback */
|
|
void *cb_arg; /*%< argument for the above */
|
|
|
|
unsigned int flags;
|
|
ISC_LIST(isc_httpd_t) running; /*%< running clients */
|
|
|
|
isc_mutex_t lock;
|
|
|
|
ISC_LIST(isc_httpdurl_t) urls; /*%< urls we manage */
|
|
isc_httpdaction_t *render_404;
|
|
isc_httpdaction_t *render_500;
|
|
};
|
|
|
|
#if ISC_HTTPD_TRACE
|
|
#define isc_httpdmgr_ref(ptr) \
|
|
isc_httpdmgr__ref(ptr, __func__, __FILE__, __LINE__)
|
|
#define isc_httpdmgr_unref(ptr) \
|
|
isc_httpdmgr__unref(ptr, __func__, __FILE__, __LINE__)
|
|
#define isc_httpdmgr_attach(ptr, ptrp) \
|
|
isc_httpdmgr__attach(ptr, ptrp, __func__, __FILE__, __LINE__)
|
|
#define isc_httpdmgr_detach(ptrp) \
|
|
isc_httpdmgr__detach(ptrp, __func__, __FILE__, __LINE__)
|
|
ISC_REFCOUNT_TRACE_DECL(isc_httpdmgr);
|
|
#else
|
|
ISC_REFCOUNT_DECL(isc_httpdmgr);
|
|
#endif
|
|
|
|
typedef struct isc_httpd_sendreq {
|
|
isc_mem_t *mctx;
|
|
isc_httpd_t *httpd;
|
|
|
|
/*%
|
|
* Transmit data state.
|
|
*
|
|
* This is the data buffer we will transmit.
|
|
*
|
|
* This free function pointer is filled in by the rendering function
|
|
* we call. The free function is called after the data is transmitted
|
|
* to the client.
|
|
*
|
|
* We currently use three buffers total:
|
|
*
|
|
* sendbuffer - gets filled as we gather the data
|
|
*
|
|
* bodybuffer - for the client to fill in (which it manages, it provides
|
|
* the space for it, etc) -- we will pass that buffer structure back to
|
|
* the caller, who is responsible for managing the space it may have
|
|
* allocated as backing store for it. we only allocate the buffer
|
|
* itself, not the backing store.
|
|
*
|
|
* compbuffer - managed by us, that contains the compressed HTTP data,
|
|
* if compression is used.
|
|
*/
|
|
isc_buffer_t *sendbuffer;
|
|
isc_buffer_t *compbuffer;
|
|
|
|
isc_buffer_t bodybuffer;
|
|
|
|
const char *mimetype;
|
|
unsigned int retcode;
|
|
const char *retmsg;
|
|
isc_httpdfree_t *freecb;
|
|
void *freecb_arg;
|
|
|
|
} isc_httpd_sendreq_t;
|
|
|
|
static isc_result_t
|
|
httpd_newconn(isc_nmhandle_t *, isc_result_t, void *);
|
|
static void
|
|
httpd_request(isc_nmhandle_t *, isc_result_t, isc_region_t *, void *);
|
|
static void
|
|
httpd_senddone(isc_nmhandle_t *, isc_result_t, void *);
|
|
static void
|
|
httpd_free(isc_httpd_t *httpd);
|
|
|
|
static void
|
|
httpd_addheader(isc_httpd_sendreq_t *, const char *, const char *);
|
|
static void
|
|
httpd_addheaderuint(isc_httpd_sendreq_t *, const char *, int);
|
|
static void
|
|
httpd_endheaders(isc_httpd_sendreq_t *);
|
|
static void
|
|
httpd_response(isc_httpd_t *, isc_httpd_sendreq_t *);
|
|
|
|
static isc_result_t
|
|
process_request(isc_httpd_t *, size_t);
|
|
|
|
static isc_httpdaction_t render_404;
|
|
static isc_httpdaction_t render_500;
|
|
|
|
#if ENABLE_AFL
|
|
static void (*finishhook)(void) = NULL;
|
|
#endif /* ENABLE_AFL */
|
|
|
|
isc_result_t
|
|
isc_httpdmgr_create(isc_nm_t *nm, isc_mem_t *mctx, isc_sockaddr_t *addr,
|
|
isc_httpdclientok_t *client_ok,
|
|
isc_httpdondestroy_t *ondestroy, void *cb_arg,
|
|
isc_httpdmgr_t **httpdmgrp) {
|
|
isc_result_t result;
|
|
isc_httpdmgr_t *httpdmgr = NULL;
|
|
|
|
REQUIRE(nm != NULL);
|
|
REQUIRE(mctx != NULL);
|
|
REQUIRE(httpdmgrp != NULL && *httpdmgrp == NULL);
|
|
|
|
httpdmgr = isc_mem_get(mctx, sizeof(isc_httpdmgr_t));
|
|
*httpdmgr = (isc_httpdmgr_t){ .client_ok = client_ok,
|
|
.ondestroy = ondestroy,
|
|
.cb_arg = cb_arg,
|
|
.render_404 = render_404,
|
|
.render_500 = render_500 };
|
|
|
|
isc_mutex_init(&httpdmgr->lock);
|
|
isc_mem_attach(mctx, &httpdmgr->mctx);
|
|
|
|
ISC_LIST_INIT(httpdmgr->running);
|
|
ISC_LIST_INIT(httpdmgr->urls);
|
|
|
|
isc_refcount_init(&httpdmgr->references, 1);
|
|
|
|
CHECK(isc_nm_listentcp(nm, ISC_NM_LISTEN_ONE, addr, httpd_newconn,
|
|
httpdmgr, 5, NULL, &httpdmgr->sock));
|
|
|
|
httpdmgr->magic = HTTPDMGR_MAGIC;
|
|
*httpdmgrp = httpdmgr;
|
|
|
|
return ISC_R_SUCCESS;
|
|
|
|
cleanup:
|
|
httpdmgr->magic = 0;
|
|
isc_refcount_decrementz(&httpdmgr->references);
|
|
isc_refcount_destroy(&httpdmgr->references);
|
|
isc_mem_detach(&httpdmgr->mctx);
|
|
isc_mutex_destroy(&httpdmgr->lock);
|
|
isc_mem_put(mctx, httpdmgr, sizeof(isc_httpdmgr_t));
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
destroy_httpdmgr(isc_httpdmgr_t *httpdmgr) {
|
|
isc_refcount_destroy(&httpdmgr->references);
|
|
|
|
LOCK(&httpdmgr->lock);
|
|
|
|
REQUIRE((httpdmgr->flags & ISC_HTTPDMGR_SHUTTINGDOWN) != 0);
|
|
REQUIRE(ISC_LIST_EMPTY(httpdmgr->running));
|
|
|
|
httpdmgr->magic = 0;
|
|
|
|
if (httpdmgr->sock != NULL) {
|
|
isc_nmsocket_close(&httpdmgr->sock);
|
|
}
|
|
|
|
/*
|
|
* Clear out the list of all actions we know about. Just free the
|
|
* memory.
|
|
*/
|
|
isc_httpdurl_t *url, *next;
|
|
ISC_LIST_FOREACH_SAFE (httpdmgr->urls, url, link, next) {
|
|
isc_mem_free(httpdmgr->mctx, url->url);
|
|
ISC_LIST_UNLINK(httpdmgr->urls, url, link);
|
|
isc_mem_put(httpdmgr->mctx, url, sizeof(isc_httpdurl_t));
|
|
}
|
|
|
|
UNLOCK(&httpdmgr->lock);
|
|
isc_mutex_destroy(&httpdmgr->lock);
|
|
|
|
if (httpdmgr->ondestroy != NULL) {
|
|
(httpdmgr->ondestroy)(httpdmgr->cb_arg);
|
|
}
|
|
isc_mem_putanddetach(&httpdmgr->mctx, httpdmgr, sizeof(isc_httpdmgr_t));
|
|
}
|
|
|
|
#if ISC_HTTPD_TRACE
|
|
ISC_REFCOUNT_TRACE_IMPL(isc_httpdmgr, destroy_httpdmgr)
|
|
#else
|
|
ISC_REFCOUNT_IMPL(isc_httpdmgr, destroy_httpdmgr);
|
|
#endif
|
|
|
|
static bool
|
|
name_match(const struct phr_header *header, const char *match) {
|
|
size_t match_len = strlen(match);
|
|
if (match_len != header->name_len) {
|
|
return false;
|
|
}
|
|
return strncasecmp(header->name, match, match_len) == 0;
|
|
}
|
|
|
|
static bool
|
|
value_match(const struct phr_header *header, const char *match) {
|
|
size_t match_len = strlen(match);
|
|
size_t limit;
|
|
|
|
if (match_len > header->value_len) {
|
|
return false;
|
|
}
|
|
|
|
limit = header->value_len - match_len + 1;
|
|
|
|
for (size_t i = 0; i < limit; i++) {
|
|
if (isspace((unsigned char)header->value[i])) {
|
|
while (i < limit &&
|
|
isspace((unsigned char)header->value[i]))
|
|
{
|
|
i++;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (strncasecmp(&header->value[i], match, match_len) == 0) {
|
|
i += match_len;
|
|
/*
|
|
* Sanity check; f.e. for 'deflate' match only
|
|
* 'deflate[,;]', but not 'deflateyou'
|
|
*/
|
|
if (i == header->value_len || header->value[i] == ',' ||
|
|
header->value[i] == ';')
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
while (i < limit && header->value[i] != ',') {
|
|
i++;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static isc_result_t
|
|
process_request(isc_httpd_t *httpd, size_t last_len) {
|
|
int pret;
|
|
const char *method = NULL;
|
|
size_t method_len = 0;
|
|
const char *path;
|
|
size_t path_len = 0;
|
|
struct phr_header headers[HTTP_HEADERS_NUM];
|
|
size_t num_headers;
|
|
isc_result_t result;
|
|
|
|
num_headers = ARRAY_SIZE(headers);
|
|
|
|
pret = phr_parse_request(httpd->recvbuf, httpd->recvlen, &method,
|
|
&method_len, &path, &path_len,
|
|
&httpd->minor_version, headers, &num_headers,
|
|
last_len);
|
|
|
|
if (pret == -1) {
|
|
/* Parse Error */
|
|
return ISC_R_UNEXPECTED;
|
|
} else if (pret == -2) {
|
|
/* Need more data */
|
|
return ISC_R_NOMORE;
|
|
}
|
|
|
|
INSIST(pret > 0);
|
|
|
|
if (pret > HTTP_MAX_REQUEST_LEN) {
|
|
return ISC_R_RANGE;
|
|
}
|
|
|
|
httpd->consume = pret;
|
|
|
|
/*
|
|
* Determine if this is a POST or GET method. Any other values will
|
|
* cause an error to be returned.
|
|
*/
|
|
if (strncmp(method, "GET ", method_len) == 0) {
|
|
httpd->method = METHOD_GET;
|
|
} else if (strncmp(method, "POST ", method_len) == 0) {
|
|
httpd->method = METHOD_POST;
|
|
} else {
|
|
return ISC_R_RANGE;
|
|
}
|
|
|
|
/*
|
|
* Parse the URL
|
|
*/
|
|
result = isc_url_parse(path, path_len, 0, &httpd->up);
|
|
if (result != ISC_R_SUCCESS) {
|
|
return result;
|
|
}
|
|
httpd->path = path;
|
|
|
|
/*
|
|
* Examine headers that can affect this request's response
|
|
*/
|
|
httpd->flags = 0;
|
|
|
|
size_t content_len = 0;
|
|
bool keep_alive = false;
|
|
bool host_header = false;
|
|
|
|
isc_time_set(&httpd->if_modified_since, 0, 0);
|
|
|
|
for (size_t i = 0; i < num_headers; i++) {
|
|
struct phr_header *header = &headers[i];
|
|
|
|
if (name_match(header, "Content-Length")) {
|
|
char *endptr;
|
|
long val = strtol(header->value, &endptr, 10);
|
|
|
|
errno = 0;
|
|
|
|
/* ensure we consumed all digits */
|
|
if ((header->value + header->value_len) != endptr) {
|
|
return ISC_R_BADNUMBER;
|
|
}
|
|
/* ensure there was no minus sign */
|
|
if (val < 0) {
|
|
return ISC_R_BADNUMBER;
|
|
}
|
|
/* ensure it did not overflow */
|
|
if (errno != 0) {
|
|
return ISC_R_RANGE;
|
|
}
|
|
content_len = val;
|
|
} else if (name_match(header, "Connection")) {
|
|
/* multiple fields in a connection header are allowed */
|
|
if (value_match(header, "close")) {
|
|
httpd->flags |= CONNECTION_CLOSE;
|
|
}
|
|
if (value_match(header, "keep-alive")) {
|
|
keep_alive = true;
|
|
}
|
|
} else if (name_match(header, "Host")) {
|
|
host_header = true;
|
|
} else if (name_match(header, "Accept-Encoding")) {
|
|
if (value_match(header, "deflate")) {
|
|
httpd->flags |= ACCEPT_DEFLATE;
|
|
}
|
|
} else if (name_match(header, "If-Modified-Since") &&
|
|
header->value_len < ISC_FORMATHTTPTIMESTAMP_SIZE)
|
|
{
|
|
char timestamp[ISC_FORMATHTTPTIMESTAMP_SIZE + 1];
|
|
memmove(timestamp, header->value, header->value_len);
|
|
timestamp[header->value_len] = 0;
|
|
|
|
/* Ignore the value if it can't be parsed */
|
|
(void)isc_time_parsehttptimestamp(
|
|
timestamp, &httpd->if_modified_since);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* The Content-Length is optional in an HTTP request.
|
|
* For a GET the length must be zero.
|
|
*/
|
|
if (httpd->method == METHOD_GET && content_len != 0) {
|
|
return ISC_R_BADNUMBER;
|
|
}
|
|
|
|
if (content_len >= HTTP_MAX_REQUEST_LEN) {
|
|
return ISC_R_RANGE;
|
|
}
|
|
|
|
size_t consume = httpd->consume + content_len;
|
|
if (consume > httpd->recvlen) {
|
|
/* The request data isn't complete yet. */
|
|
return ISC_R_NOMORE;
|
|
}
|
|
|
|
/* Consume the request's data, which we do not use. */
|
|
httpd->consume = consume;
|
|
|
|
switch (httpd->minor_version) {
|
|
case 0:
|
|
/*
|
|
* RFC 9112 section 9.3 says close takes priority if
|
|
* keep-alive is also present
|
|
*/
|
|
if ((httpd->flags & CONNECTION_CLOSE) == 0 && keep_alive) {
|
|
httpd->flags |= CONNECTION_KEEP_ALIVE;
|
|
} else {
|
|
httpd->flags |= CONNECTION_CLOSE;
|
|
}
|
|
break;
|
|
case 1:
|
|
if (!host_header) {
|
|
return ISC_R_RANGE;
|
|
}
|
|
break;
|
|
default:
|
|
return ISC_R_UNEXPECTED;
|
|
}
|
|
|
|
/*
|
|
* Looks like a a valid request, so now we know we won't have
|
|
* to process this buffer again. We can NULL-terminate the
|
|
* URL for the caller's benefit, and set recvlen to 0 so
|
|
* the next read will overwrite this one instead of appending
|
|
* to the buffer.
|
|
*/
|
|
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
static void
|
|
httpd_free(isc_httpd_t *httpd) {
|
|
isc_httpdmgr_t *httpdmgr = NULL;
|
|
|
|
REQUIRE(VALID_HTTPD(httpd));
|
|
|
|
httpdmgr = httpd->mgr;
|
|
|
|
REQUIRE(VALID_HTTPDMGR(httpdmgr));
|
|
|
|
LOCK(&httpdmgr->lock);
|
|
ISC_LIST_UNLINK(httpdmgr->running, httpd, link);
|
|
UNLOCK(&httpdmgr->lock);
|
|
|
|
httpd->recvbuf[0] = 0;
|
|
httpd->recvlen = 0;
|
|
httpd->consume = 0;
|
|
httpd->method = METHOD_UNKNOWN;
|
|
httpd->flags = 0;
|
|
|
|
httpd->minor_version = -1;
|
|
httpd->path = NULL;
|
|
httpd->up = (isc_url_parser_t){ 0 };
|
|
isc_time_set(&httpd->if_modified_since, 0, 0);
|
|
|
|
httpd->magic = 0;
|
|
httpd->mgr = NULL;
|
|
|
|
isc_mem_put(httpdmgr->mctx, httpd, sizeof(*httpd));
|
|
|
|
isc_httpdmgr_detach(&httpdmgr);
|
|
|
|
#if ENABLE_AFL
|
|
if (finishhook != NULL) {
|
|
finishhook();
|
|
}
|
|
#endif /* ENABLE_AFL */
|
|
}
|
|
|
|
#if ISC_HTTPD_TRACE
|
|
ISC_REFCOUNT_TRACE_IMPL(isc_httpd, httpd_free)
|
|
#else
|
|
ISC_REFCOUNT_IMPL(isc_httpd, httpd_free);
|
|
#endif
|
|
|
|
static void
|
|
isc__httpd_sendreq_free(isc_httpd_sendreq_t *req) {
|
|
/* Clean up buffers */
|
|
|
|
isc_buffer_free(&req->sendbuffer);
|
|
|
|
isc_mem_putanddetach(&req->mctx, req, sizeof(*req));
|
|
}
|
|
|
|
static isc_httpd_sendreq_t *
|
|
isc__httpd_sendreq_new(isc_httpd_t *httpd) {
|
|
isc_httpdmgr_t *httpdmgr = httpd->mgr;
|
|
isc_httpd_sendreq_t *req;
|
|
|
|
REQUIRE(VALID_HTTPDMGR(httpdmgr));
|
|
|
|
req = isc_mem_get(httpdmgr->mctx, sizeof(*req));
|
|
*req = (isc_httpd_sendreq_t){ 0 };
|
|
|
|
isc_mem_attach(httpdmgr->mctx, &req->mctx);
|
|
|
|
/*
|
|
* Initialize the buffer for our headers.
|
|
*/
|
|
isc_buffer_allocate(req->mctx, &req->sendbuffer, HTTP_SENDLEN);
|
|
isc_buffer_clear(req->sendbuffer);
|
|
|
|
isc_buffer_initnull(&req->bodybuffer);
|
|
|
|
isc_httpd_attach(httpd, &req->httpd);
|
|
|
|
return req;
|
|
}
|
|
|
|
static void
|
|
new_httpd(isc_httpdmgr_t *httpdmgr, isc_nmhandle_t *handle) {
|
|
isc_httpd_t *httpd = NULL;
|
|
|
|
REQUIRE(VALID_HTTPDMGR(httpdmgr));
|
|
|
|
httpd = isc_mem_get(httpdmgr->mctx, sizeof(*httpd));
|
|
*httpd = (isc_httpd_t){
|
|
.magic = HTTPD_MAGIC,
|
|
.link = ISC_LINK_INITIALIZER,
|
|
.references = ISC_REFCOUNT_INITIALIZER(1),
|
|
};
|
|
|
|
isc_nmhandle_attach(handle, &httpd->handle);
|
|
|
|
isc_httpdmgr_attach(httpdmgr, &httpd->mgr);
|
|
|
|
LOCK(&httpdmgr->lock);
|
|
ISC_LIST_APPEND(httpdmgr->running, httpd, link);
|
|
UNLOCK(&httpdmgr->lock);
|
|
|
|
isc_nm_read(handle, httpd_request, httpd);
|
|
}
|
|
|
|
static isc_result_t
|
|
httpd_newconn(isc_nmhandle_t *handle, isc_result_t result, void *arg) {
|
|
isc_httpdmgr_t *httpdmgr = (isc_httpdmgr_t *)arg;
|
|
isc_sockaddr_t peeraddr;
|
|
|
|
REQUIRE(VALID_HTTPDMGR(httpdmgr));
|
|
|
|
if ((httpdmgr->flags & ISC_HTTPDMGR_SHUTTINGDOWN) != 0) {
|
|
return ISC_R_CANCELED;
|
|
} else if (result == ISC_R_CANCELED) {
|
|
isc_httpdmgr_shutdown(&httpdmgr);
|
|
return result;
|
|
} else if (result != ISC_R_SUCCESS) {
|
|
return result;
|
|
}
|
|
|
|
peeraddr = isc_nmhandle_peeraddr(handle);
|
|
if (httpdmgr->client_ok != NULL &&
|
|
!(httpdmgr->client_ok)(&peeraddr, httpdmgr->cb_arg))
|
|
{
|
|
return ISC_R_FAILURE;
|
|
}
|
|
|
|
new_httpd(httpdmgr, handle);
|
|
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
static isc_result_t
|
|
render_404(const isc_httpd_t *httpd, const isc_httpdurl_t *urlinfo, void *arg,
|
|
unsigned int *retcode, const char **retmsg, const char **mimetype,
|
|
isc_buffer_t *b, isc_httpdfree_t **freecb, void **freecb_args) {
|
|
static char msg[] = "No such URL.\r\n";
|
|
|
|
UNUSED(httpd);
|
|
UNUSED(urlinfo);
|
|
UNUSED(arg);
|
|
|
|
*retcode = 404;
|
|
*retmsg = "No such URL";
|
|
*mimetype = "text/plain";
|
|
isc_buffer_reinit(b, msg, strlen(msg));
|
|
isc_buffer_add(b, strlen(msg));
|
|
*freecb = NULL;
|
|
*freecb_args = NULL;
|
|
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
static isc_result_t
|
|
render_500(const isc_httpd_t *httpd, const isc_httpdurl_t *urlinfo, void *arg,
|
|
unsigned int *retcode, const char **retmsg, const char **mimetype,
|
|
isc_buffer_t *b, isc_httpdfree_t **freecb, void **freecb_args) {
|
|
static char msg[] = "Internal server failure.\r\n";
|
|
|
|
UNUSED(httpd);
|
|
UNUSED(urlinfo);
|
|
UNUSED(arg);
|
|
|
|
*retcode = 500;
|
|
*retmsg = "Internal server failure";
|
|
*mimetype = "text/plain";
|
|
isc_buffer_reinit(b, msg, strlen(msg));
|
|
isc_buffer_add(b, strlen(msg));
|
|
*freecb = NULL;
|
|
*freecb_args = NULL;
|
|
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
#ifdef HAVE_ZLIB
|
|
/*%<
|
|
* Tries to compress httpd->bodybuffer to httpd->compbuffer, extending it
|
|
* if necessary.
|
|
*
|
|
* Requires:
|
|
*\li httpd a valid isc_httpd_t object
|
|
*
|
|
* Returns:
|
|
*\li #ISC_R_SUCCESS -- all is well.
|
|
*\li #ISC_R_NOMEMORY -- not enough memory to compress data
|
|
*\li #ISC_R_FAILURE -- error during compression or compressed
|
|
* data would be larger than input data
|
|
*/
|
|
static isc_result_t
|
|
httpd_compress(isc_httpd_sendreq_t *req) {
|
|
z_stream zstr;
|
|
int ret, inputlen;
|
|
|
|
/*
|
|
* We're setting output buffer size to input size so it fails if the
|
|
* compressed data size would be bigger than the input size.
|
|
*/
|
|
inputlen = isc_buffer_usedlength(&req->bodybuffer);
|
|
if (inputlen == 0) {
|
|
return ISC_R_FAILURE;
|
|
}
|
|
|
|
isc_buffer_allocate(req->mctx, &req->compbuffer, inputlen);
|
|
isc_buffer_clear(req->compbuffer);
|
|
|
|
zstr = (z_stream){
|
|
.total_in = inputlen,
|
|
.avail_out = inputlen,
|
|
.avail_in = inputlen,
|
|
.next_in = isc_buffer_base(&req->bodybuffer),
|
|
.next_out = isc_buffer_base(req->compbuffer),
|
|
};
|
|
|
|
ret = deflateInit(&zstr, Z_DEFAULT_COMPRESSION);
|
|
if (ret == Z_OK) {
|
|
ret = deflate(&zstr, Z_FINISH);
|
|
}
|
|
deflateEnd(&zstr);
|
|
if (ret == Z_STREAM_END) {
|
|
isc_buffer_add(req->compbuffer, zstr.total_out);
|
|
return ISC_R_SUCCESS;
|
|
} else {
|
|
isc_buffer_free(&req->compbuffer);
|
|
return ISC_R_FAILURE;
|
|
}
|
|
}
|
|
#endif /* ifdef HAVE_ZLIB */
|
|
|
|
static void
|
|
prepare_response(void *arg) {
|
|
isc_httpd_sendreq_t *req = arg;
|
|
isc_httpd_t *httpd = req->httpd;
|
|
isc_httpdmgr_t *mgr = httpd->mgr;
|
|
isc_time_t now = isc_time_now();
|
|
char datebuf[ISC_FORMATHTTPTIMESTAMP_SIZE];
|
|
const char *path = "/";
|
|
size_t path_len = 1;
|
|
bool is_compressed = false;
|
|
isc_httpdurl_t *url = NULL;
|
|
isc_result_t result;
|
|
|
|
REQUIRE(VALID_HTTPD(httpd));
|
|
REQUIRE(req != NULL);
|
|
|
|
isc_time_formathttptimestamp(&now, datebuf, sizeof(datebuf));
|
|
|
|
if (httpd->up.field_set & (1 << ISC_UF_PATH)) {
|
|
path = &httpd->path[httpd->up.field_data[ISC_UF_PATH].off];
|
|
path_len = httpd->up.field_data[ISC_UF_PATH].len;
|
|
}
|
|
|
|
LOCK(&mgr->lock);
|
|
ISC_LIST_FOREACH (mgr->urls, url, link) {
|
|
if (strncmp(path, url->url, path_len) == 0) {
|
|
break;
|
|
}
|
|
}
|
|
UNLOCK(&mgr->lock);
|
|
|
|
if (url == NULL) {
|
|
result = mgr->render_404(httpd, NULL, NULL, &req->retcode,
|
|
&req->retmsg, &req->mimetype,
|
|
&req->bodybuffer, &req->freecb,
|
|
&req->freecb_arg);
|
|
} else {
|
|
result = url->action(httpd, url, url->action_arg, &req->retcode,
|
|
&req->retmsg, &req->mimetype,
|
|
&req->bodybuffer, &req->freecb,
|
|
&req->freecb_arg);
|
|
}
|
|
if (result != ISC_R_SUCCESS) {
|
|
result = mgr->render_500(httpd, url, NULL, &req->retcode,
|
|
&req->retmsg, &req->mimetype,
|
|
&req->bodybuffer, &req->freecb,
|
|
&req->freecb_arg);
|
|
RUNTIME_CHECK(result == ISC_R_SUCCESS);
|
|
}
|
|
|
|
#ifdef HAVE_ZLIB
|
|
if ((httpd->flags & ACCEPT_DEFLATE) != 0) {
|
|
result = httpd_compress(req);
|
|
if (result == ISC_R_SUCCESS) {
|
|
is_compressed = true;
|
|
}
|
|
}
|
|
#endif /* ifdef HAVE_ZLIB */
|
|
|
|
httpd_response(httpd, req);
|
|
/* RFC 9112 § 9.6: SHOULD send Connection: close in last response */
|
|
if ((httpd->flags & CONNECTION_CLOSE) != 0) {
|
|
httpd_addheader(req, "Connection", "close");
|
|
} else if ((httpd->flags & CONNECTION_KEEP_ALIVE) != 0) {
|
|
httpd_addheader(req, "Connection", "Keep-Alive");
|
|
}
|
|
httpd_addheader(req, "Content-Type", req->mimetype);
|
|
httpd_addheader(req, "Date", datebuf);
|
|
httpd_addheader(req, "Expires", datebuf);
|
|
|
|
if (url != NULL && url->isstatic) {
|
|
char loadbuf[ISC_FORMATHTTPTIMESTAMP_SIZE];
|
|
isc_time_formathttptimestamp(&url->loadtime, loadbuf,
|
|
sizeof(loadbuf));
|
|
httpd_addheader(req, "Last-Modified", loadbuf);
|
|
httpd_addheader(req, "Cache-Control: public", NULL);
|
|
} else {
|
|
httpd_addheader(req, "Last-Modified", datebuf);
|
|
httpd_addheader(req, "Pragma: no-cache", NULL);
|
|
httpd_addheader(req, "Cache-Control: no-cache", NULL);
|
|
}
|
|
|
|
httpd_addheader(req, "Server: libisc", NULL);
|
|
|
|
if (is_compressed) {
|
|
httpd_addheader(req, "Content-Encoding", "deflate");
|
|
httpd_addheaderuint(req, "Content-Length",
|
|
isc_buffer_usedlength(req->compbuffer));
|
|
} else {
|
|
httpd_addheaderuint(req, "Content-Length",
|
|
isc_buffer_usedlength(&req->bodybuffer));
|
|
}
|
|
|
|
httpd_endheaders(req); /* done */
|
|
|
|
/*
|
|
* Append either the compressed or the non-compressed response body to
|
|
* the response headers and store the result in httpd->sendbuffer.
|
|
*/
|
|
if (is_compressed) {
|
|
isc_buffer_putmem(req->sendbuffer,
|
|
isc_buffer_base(req->compbuffer),
|
|
isc_buffer_usedlength(req->compbuffer));
|
|
isc_buffer_free(&req->compbuffer);
|
|
} else {
|
|
isc_buffer_putmem(req->sendbuffer,
|
|
isc_buffer_base(&req->bodybuffer),
|
|
isc_buffer_usedlength(&req->bodybuffer));
|
|
}
|
|
|
|
/* Free the bodybuffer */
|
|
if (req->freecb != NULL && isc_buffer_length(&req->bodybuffer) > 0) {
|
|
req->freecb(&req->bodybuffer, req->freecb_arg);
|
|
}
|
|
|
|
/* Consume the request from the recv buffer. */
|
|
INSIST(httpd->consume != 0);
|
|
INSIST(httpd->consume <= httpd->recvlen);
|
|
if (httpd->consume < httpd->recvlen) {
|
|
memmove(httpd->recvbuf, httpd->recvbuf + httpd->consume,
|
|
httpd->recvlen - httpd->consume);
|
|
}
|
|
httpd->recvlen -= httpd->consume;
|
|
httpd->consume = 0;
|
|
}
|
|
|
|
static void
|
|
prepare_response_done(void *arg) {
|
|
isc_region_t r;
|
|
isc_httpd_sendreq_t *req = arg;
|
|
isc_httpd_t *httpd = req->httpd;
|
|
|
|
/*
|
|
* Determine total response size.
|
|
*/
|
|
isc_buffer_usedregion(req->sendbuffer, &r);
|
|
|
|
isc_nm_send(httpd->handle, &r, httpd_senddone, req);
|
|
}
|
|
|
|
static void
|
|
httpd_request(isc_nmhandle_t *handle, isc_result_t eresult,
|
|
isc_region_t *region, void *arg) {
|
|
isc_httpd_t *httpd = arg;
|
|
isc_httpdmgr_t *mgr = httpd->mgr;
|
|
size_t last_len = 0;
|
|
isc_result_t result;
|
|
|
|
REQUIRE(VALID_HTTPD(httpd));
|
|
|
|
REQUIRE(httpd->handle == handle);
|
|
|
|
if (eresult != ISC_R_SUCCESS) {
|
|
goto close_readhandle;
|
|
}
|
|
|
|
REQUIRE((mgr->flags & ISC_HTTPDMGR_SHUTTINGDOWN) == 0);
|
|
|
|
isc_nm_read_stop(handle);
|
|
|
|
/*
|
|
* If we are being called from httpd_senddone(), the last HTTP request
|
|
* was processed successfully, reset the last_len to 0, even if there's
|
|
* data in the httpd->recvbuf.
|
|
*/
|
|
last_len = (region == NULL) ? 0 : httpd->recvlen;
|
|
|
|
/* Store the received data into the recvbuf */
|
|
if (region != NULL) {
|
|
if (httpd->recvlen + region->length > sizeof(httpd->recvbuf)) {
|
|
goto close_readhandle;
|
|
}
|
|
|
|
memmove(httpd->recvbuf + httpd->recvlen, region->base,
|
|
region->length);
|
|
httpd->recvlen += region->length;
|
|
}
|
|
|
|
result = process_request(httpd, last_len);
|
|
|
|
if (result == ISC_R_NOMORE) {
|
|
if (httpd->recvlen > HTTP_MAX_REQUEST_LEN) {
|
|
goto close_readhandle;
|
|
}
|
|
|
|
/* Wait for more data, the handle is still attached */
|
|
isc_nm_read(handle, httpd_request, arg);
|
|
return;
|
|
}
|
|
|
|
/* XXXFANF it would be more polite to reply 400 bad request */
|
|
if (result != ISC_R_SUCCESS) {
|
|
goto close_readhandle;
|
|
}
|
|
|
|
isc_httpd_sendreq_t *req = isc__httpd_sendreq_new(httpd);
|
|
isc_nmhandle_ref(handle);
|
|
isc_work_enqueue(isc_loop(), prepare_response, prepare_response_done,
|
|
req);
|
|
return;
|
|
|
|
close_readhandle:
|
|
isc_nmhandle_close(httpd->handle);
|
|
isc_nmhandle_detach(&httpd->handle);
|
|
|
|
isc_httpd_detach(&httpd);
|
|
}
|
|
|
|
void
|
|
isc_httpdmgr_shutdown(isc_httpdmgr_t **httpdmgrp) {
|
|
isc_httpdmgr_t *httpdmgr = NULL;
|
|
|
|
REQUIRE(httpdmgrp != NULL);
|
|
REQUIRE(VALID_HTTPDMGR(*httpdmgrp));
|
|
|
|
httpdmgr = *httpdmgrp;
|
|
*httpdmgrp = NULL;
|
|
|
|
isc_nm_stoplistening(httpdmgr->sock);
|
|
|
|
LOCK(&httpdmgr->lock);
|
|
|
|
isc_httpd_t *httpd = NULL, *next = NULL;
|
|
ISC_LIST_FOREACH_SAFE (httpdmgr->running, httpd, link, next) {
|
|
if (httpd->handle != NULL) {
|
|
httpd_request(httpd->handle, ISC_R_SUCCESS, NULL,
|
|
httpd);
|
|
}
|
|
}
|
|
|
|
httpdmgr->flags |= ISC_HTTPDMGR_SHUTTINGDOWN;
|
|
|
|
UNLOCK(&httpdmgr->lock);
|
|
|
|
isc_nmsocket_close(&httpdmgr->sock);
|
|
|
|
isc_httpdmgr_detach(&httpdmgr);
|
|
}
|
|
|
|
static void
|
|
httpd_response(isc_httpd_t *httpd, isc_httpd_sendreq_t *req) {
|
|
isc_result_t result;
|
|
|
|
result = isc_buffer_printf(req->sendbuffer, "HTTP/1.%u %03u %s\r\n",
|
|
httpd->minor_version, req->retcode,
|
|
req->retmsg);
|
|
|
|
RUNTIME_CHECK(result == ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
httpd_addheader(isc_httpd_sendreq_t *req, const char *name, const char *val) {
|
|
isc_result_t result;
|
|
|
|
if (val != NULL) {
|
|
result = isc_buffer_printf(req->sendbuffer, "%s: %s\r\n", name,
|
|
val);
|
|
} else {
|
|
result = isc_buffer_printf(req->sendbuffer, "%s\r\n", name);
|
|
}
|
|
|
|
RUNTIME_CHECK(result == ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
httpd_endheaders(isc_httpd_sendreq_t *req) {
|
|
isc_result_t result;
|
|
|
|
result = isc_buffer_printf(req->sendbuffer, "\r\n");
|
|
|
|
RUNTIME_CHECK(result == ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
httpd_addheaderuint(isc_httpd_sendreq_t *req, const char *name, int val) {
|
|
isc_result_t result;
|
|
|
|
result = isc_buffer_printf(req->sendbuffer, "%s: %d\r\n", name, val);
|
|
|
|
RUNTIME_CHECK(result == ISC_R_SUCCESS);
|
|
}
|
|
|
|
static void
|
|
httpd_senddone(isc_nmhandle_t *handle, isc_result_t eresult, void *arg) {
|
|
isc_httpd_sendreq_t *req = (isc_httpd_sendreq_t *)arg;
|
|
isc_httpd_t *httpd = req->httpd;
|
|
|
|
REQUIRE(VALID_HTTPD(httpd));
|
|
|
|
if ((httpd->mgr->flags & ISC_HTTPDMGR_SHUTTINGDOWN) != 0) {
|
|
goto detach;
|
|
}
|
|
|
|
if (eresult == ISC_R_SUCCESS && (httpd->flags & CONNECTION_CLOSE) != 0)
|
|
{
|
|
eresult = ISC_R_EOF;
|
|
}
|
|
|
|
/*
|
|
* Calling httpd_request() with region NULL restarts reading.
|
|
*/
|
|
httpd_request(handle, eresult, NULL, httpd);
|
|
|
|
detach:
|
|
isc_nmhandle_detach(&handle);
|
|
isc__httpd_sendreq_free(req);
|
|
isc_httpd_detach(&httpd);
|
|
}
|
|
|
|
isc_result_t
|
|
isc_httpdmgr_addurl(isc_httpdmgr_t *httpdmgr, const char *url, bool isstatic,
|
|
isc_httpdaction_t *func, void *arg) {
|
|
isc_httpdurl_t *item;
|
|
|
|
REQUIRE(VALID_HTTPDMGR(httpdmgr));
|
|
|
|
if (url == NULL) {
|
|
httpdmgr->render_404 = func;
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
item = isc_mem_get(httpdmgr->mctx, sizeof(isc_httpdurl_t));
|
|
|
|
item->url = isc_mem_strdup(httpdmgr->mctx, url);
|
|
|
|
item->action = func;
|
|
item->action_arg = arg;
|
|
item->isstatic = isstatic;
|
|
item->loadtime = isc_time_now();
|
|
|
|
ISC_LINK_INIT(item, link);
|
|
|
|
LOCK(&httpdmgr->lock);
|
|
ISC_LIST_APPEND(httpdmgr->urls, item, link);
|
|
UNLOCK(&httpdmgr->lock);
|
|
|
|
return ISC_R_SUCCESS;
|
|
}
|
|
|
|
void
|
|
isc_httpd_setfinishhook(void (*fn)(void)) {
|
|
#if ENABLE_AFL
|
|
finishhook = fn;
|
|
#else /* ENABLE_AFL */
|
|
UNUSED(fn);
|
|
#endif /* ENABLE_AFL */
|
|
}
|
|
|
|
bool
|
|
isc_httpdurl_isstatic(const isc_httpdurl_t *url) {
|
|
return url->isstatic;
|
|
}
|
|
|
|
const isc_time_t *
|
|
isc_httpdurl_loadtime(const isc_httpdurl_t *url) {
|
|
return &url->loadtime;
|
|
}
|
|
|
|
const isc_time_t *
|
|
isc_httpd_if_modified_since(const isc_httpd_t *httpd) {
|
|
return (const isc_time_t *)&httpd->if_modified_since;
|
|
}
|