2018-02-04 20:27:33 +00:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
2018-02-12 19:29:51 +00:00
|
|
|
#include <netinet/in.h>
|
2018-02-04 20:27:33 +00:00
|
|
|
#include <regex.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2018-03-01 14:18:20 +00:00
|
|
|
#include <strings.h>
|
2018-02-12 19:29:51 +00:00
|
|
|
#include <sys/socket.h>
|
2018-02-04 20:27:33 +00:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
2018-03-01 14:18:20 +00:00
|
|
|
#include <time.h>
|
2018-02-04 20:27:33 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include "http.h"
|
|
|
|
#include "resp.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
const char *req_field_str[] = {
|
2020-08-05 13:46:03 +00:00
|
|
|
[REQ_HOST] = "Host",
|
|
|
|
[REQ_RANGE] = "Range",
|
|
|
|
[REQ_IF_MODIFIED_SINCE] = "If-Modified-Since",
|
2018-02-04 20:27:33 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const char *req_method_str[] = {
|
|
|
|
[M_GET] = "GET",
|
|
|
|
[M_HEAD] = "HEAD",
|
|
|
|
};
|
|
|
|
|
|
|
|
const char *status_str[] = {
|
|
|
|
[S_OK] = "OK",
|
|
|
|
[S_PARTIAL_CONTENT] = "Partial Content",
|
|
|
|
[S_MOVED_PERMANENTLY] = "Moved Permanently",
|
|
|
|
[S_NOT_MODIFIED] = "Not Modified",
|
|
|
|
[S_BAD_REQUEST] = "Bad Request",
|
|
|
|
[S_FORBIDDEN] = "Forbidden",
|
|
|
|
[S_NOT_FOUND] = "Not Found",
|
|
|
|
[S_METHOD_NOT_ALLOWED] = "Method Not Allowed",
|
|
|
|
[S_REQUEST_TIMEOUT] = "Request Time-out",
|
|
|
|
[S_RANGE_NOT_SATISFIABLE] = "Range Not Satisfiable",
|
|
|
|
[S_REQUEST_TOO_LARGE] = "Request Header Fields Too Large",
|
|
|
|
[S_INTERNAL_SERVER_ERROR] = "Internal Server Error",
|
|
|
|
[S_VERSION_NOT_SUPPORTED] = "HTTP Version not supported",
|
|
|
|
};
|
|
|
|
|
2020-08-05 11:41:44 +00:00
|
|
|
const char *res_field_str[] = {
|
|
|
|
[RES_ACCEPT_RANGES] = "Accept-Ranges",
|
|
|
|
[RES_ALLOW] = "Allow",
|
|
|
|
[RES_LOCATION] = "Location",
|
|
|
|
[RES_LAST_MODIFIED] = "Last-Modified",
|
|
|
|
[RES_CONTENT_LENGTH] = "Content-Length",
|
|
|
|
[RES_CONTENT_RANGE] = "Content-Range",
|
|
|
|
[RES_CONTENT_TYPE] = "Content-Type",
|
|
|
|
};
|
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
enum status
|
2020-08-05 11:41:44 +00:00
|
|
|
http_send_header(int fd, const struct response *res)
|
2018-02-04 20:27:33 +00:00
|
|
|
{
|
2020-08-05 11:41:44 +00:00
|
|
|
char t[FIELD_MAX];
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (timestamp(t, sizeof(t), time(NULL))) {
|
|
|
|
return S_INTERNAL_SERVER_ERROR;
|
|
|
|
}
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
if (dprintf(fd,
|
|
|
|
"HTTP/1.1 %d %s\r\n"
|
|
|
|
"Date: %s\r\n"
|
2020-08-05 11:41:44 +00:00
|
|
|
"Connection: close\r\n",
|
|
|
|
res->status, status_str[res->status], t) < 0) {
|
|
|
|
return S_REQUEST_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < NUM_RES_FIELDS; i++) {
|
|
|
|
if (res->field[i][0] != '\0') {
|
|
|
|
if (dprintf(fd, "%s: %s\r\n", res_field_str[i],
|
|
|
|
res->field[i]) < 0) {
|
|
|
|
return S_REQUEST_TIMEOUT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dprintf(fd, "\r\n") < 0) {
|
|
|
|
return S_REQUEST_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res->status;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum status
|
|
|
|
http_send_status(int fd, enum status s)
|
|
|
|
{
|
|
|
|
enum status sendstatus;
|
|
|
|
|
|
|
|
struct response res = {
|
|
|
|
.status = s,
|
|
|
|
.field[RES_CONTENT_TYPE] = "text/html; charset=utf-8",
|
|
|
|
};
|
|
|
|
|
|
|
|
if (s == S_METHOD_NOT_ALLOWED) {
|
|
|
|
if (esnprintf(res.field[RES_ALLOW],
|
|
|
|
sizeof(res.field[RES_ALLOW]), "%s",
|
|
|
|
"Allow: GET, HEAD")) {
|
|
|
|
return S_INTERNAL_SERVER_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sendstatus = http_send_header(fd, &res)) != s) {
|
|
|
|
return sendstatus;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dprintf(fd,
|
2018-02-04 20:27:33 +00:00
|
|
|
"<!DOCTYPE html>\n<html>\n\t<head>\n"
|
|
|
|
"\t\t<title>%d %s</title>\n\t</head>\n\t<body>\n"
|
|
|
|
"\t\t<h1>%d %s</h1>\n\t</body>\n</html>\n",
|
|
|
|
s, status_str[s], s, status_str[s]) < 0) {
|
|
|
|
return S_REQUEST_TIMEOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-08-05 16:28:21 +00:00
|
|
|
decode(const char src[PATH_MAX], char dest[PATH_MAX])
|
2018-02-04 20:27:33 +00:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
uint8_t n;
|
2020-08-05 16:28:21 +00:00
|
|
|
const char *s;
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
for (s = src, i = 0; *s; s++, i++) {
|
2019-01-10 21:02:23 +00:00
|
|
|
if (*s == '%' && (sscanf(s + 1, "%2hhx", &n) == 1)) {
|
2018-02-04 20:27:33 +00:00
|
|
|
dest[i] = n;
|
|
|
|
s += 2;
|
|
|
|
} else {
|
|
|
|
dest[i] = *s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dest[i] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2020-08-05 13:43:29 +00:00
|
|
|
http_get_request(int fd, struct request *req)
|
2018-02-04 20:27:33 +00:00
|
|
|
{
|
2020-08-05 11:41:44 +00:00
|
|
|
struct in6_addr addr;
|
2018-02-04 20:27:33 +00:00
|
|
|
size_t hlen, i, mlen;
|
|
|
|
ssize_t off;
|
|
|
|
char h[HEADER_MAX], *p, *q;
|
|
|
|
|
|
|
|
/* empty all fields */
|
2020-08-05 13:43:29 +00:00
|
|
|
memset(req, 0, sizeof(*req));
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* receive header
|
|
|
|
*/
|
|
|
|
for (hlen = 0; ;) {
|
|
|
|
if ((off = read(fd, h + hlen, sizeof(h) - hlen)) < 0) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TIMEOUT);
|
|
|
|
} else if (off == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hlen += off;
|
|
|
|
if (hlen >= 4 && !memcmp(h + hlen - 4, "\r\n\r\n", 4)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (hlen == sizeof(h)) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove terminating empty line */
|
|
|
|
if (hlen < 2) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
hlen -= 2;
|
|
|
|
|
|
|
|
/* null-terminate the header */
|
|
|
|
h[hlen] = '\0';
|
|
|
|
|
|
|
|
/*
|
|
|
|
* parse request line
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* METHOD */
|
|
|
|
for (i = 0; i < NUM_REQ_METHODS; i++) {
|
|
|
|
mlen = strlen(req_method_str[i]);
|
|
|
|
if (!strncmp(req_method_str[i], h, mlen)) {
|
2020-08-05 13:43:29 +00:00
|
|
|
req->method = i;
|
2018-02-04 20:27:33 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == NUM_REQ_METHODS) {
|
|
|
|
return http_send_status(fd, S_METHOD_NOT_ALLOWED);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* a single space must follow the method */
|
|
|
|
if (h[mlen] != ' ') {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* basis for next step */
|
|
|
|
p = h + mlen + 1;
|
|
|
|
|
|
|
|
/* TARGET */
|
|
|
|
if (!(q = strchr(p, ' '))) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
*q = '\0';
|
|
|
|
if (q - p + 1 > PATH_MAX) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
2020-08-05 13:43:29 +00:00
|
|
|
memcpy(req->target, p, q - p + 1);
|
|
|
|
decode(req->target, req->target);
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
/* basis for next step */
|
|
|
|
p = q + 1;
|
|
|
|
|
|
|
|
/* HTTP-VERSION */
|
|
|
|
if (strncmp(p, "HTTP/", sizeof("HTTP/") - 1)) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
p += sizeof("HTTP/") - 1;
|
|
|
|
if (strncmp(p, "1.0", sizeof("1.0") - 1) &&
|
|
|
|
strncmp(p, "1.1", sizeof("1.1") - 1)) {
|
|
|
|
return http_send_status(fd, S_VERSION_NOT_SUPPORTED);
|
|
|
|
}
|
|
|
|
p += sizeof("1.*") - 1;
|
|
|
|
|
|
|
|
/* check terminator */
|
|
|
|
if (strncmp(p, "\r\n", sizeof("\r\n") - 1)) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* basis for next step */
|
|
|
|
p += sizeof("\r\n") - 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* parse request-fields
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* match field type */
|
|
|
|
for (; *p != '\0';) {
|
|
|
|
for (i = 0; i < NUM_REQ_FIELDS; i++) {
|
|
|
|
if (!strncasecmp(p, req_field_str[i],
|
|
|
|
strlen(req_field_str[i]))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i == NUM_REQ_FIELDS) {
|
|
|
|
/* unmatched field, skip this line */
|
|
|
|
if (!(q = strstr(p, "\r\n"))) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
p = q + (sizeof("\r\n") - 1);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
p += strlen(req_field_str[i]);
|
|
|
|
|
|
|
|
/* a single colon must follow the field name */
|
|
|
|
if (*p != ':') {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* skip whitespace */
|
|
|
|
for (++p; *p == ' ' || *p == '\t'; p++)
|
|
|
|
;
|
|
|
|
|
|
|
|
/* extract field content */
|
|
|
|
if (!(q = strstr(p, "\r\n"))) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
*q = '\0';
|
|
|
|
if (q - p + 1 > FIELD_MAX) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
2020-08-05 13:43:29 +00:00
|
|
|
memcpy(req->field[i], p, q - p + 1);
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
/* go to next line */
|
|
|
|
p = q + (sizeof("\r\n") - 1);
|
|
|
|
}
|
|
|
|
|
2018-04-02 22:55:52 +00:00
|
|
|
/*
|
|
|
|
* clean up host
|
|
|
|
*/
|
|
|
|
|
2020-08-05 13:43:29 +00:00
|
|
|
p = strrchr(req->field[REQ_HOST], ':');
|
|
|
|
q = strrchr(req->field[REQ_HOST], ']');
|
2018-04-02 22:55:52 +00:00
|
|
|
|
|
|
|
/* strip port suffix but don't interfere with IPv6 bracket notation
|
|
|
|
* as per RFC 2732 */
|
|
|
|
if (p && (!q || p > q)) {
|
|
|
|
/* port suffix must not be empty */
|
|
|
|
if (*(p + 1) == '\0') {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* strip the brackets from the IPv6 notation and validate the address */
|
|
|
|
if (q) {
|
|
|
|
/* brackets must be on the outside */
|
2020-08-05 13:43:29 +00:00
|
|
|
if (req->field[REQ_HOST][0] != '[' || *(q + 1) != '\0') {
|
2018-04-02 22:55:52 +00:00
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove the right bracket */
|
|
|
|
*q = '\0';
|
2020-08-05 13:43:29 +00:00
|
|
|
p = req->field[REQ_HOST] + 1;
|
2018-04-02 22:55:52 +00:00
|
|
|
|
|
|
|
/* validate the contained IPv6 address */
|
2020-08-05 11:41:44 +00:00
|
|
|
if (inet_pton(AF_INET6, p, &addr) != 1) {
|
2018-04-02 22:55:52 +00:00
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy it into the host field */
|
2020-08-05 13:43:29 +00:00
|
|
|
memmove(req->field[REQ_HOST], p, q - p + 1);
|
2018-04-02 22:55:52 +00:00
|
|
|
}
|
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-08-05 16:28:21 +00:00
|
|
|
encode(const char src[PATH_MAX], char dest[PATH_MAX])
|
2018-02-04 20:27:33 +00:00
|
|
|
{
|
|
|
|
size_t i;
|
2020-08-05 16:28:21 +00:00
|
|
|
const char *s;
|
2018-02-04 20:27:33 +00:00
|
|
|
|
|
|
|
for (s = src, i = 0; *s && i < (PATH_MAX - 4); s++) {
|
|
|
|
if (iscntrl(*s) || (unsigned char)*s > 127) {
|
|
|
|
i += snprintf(dest + i, PATH_MAX - i, "%%%02X",
|
|
|
|
(unsigned char)*s);
|
|
|
|
} else {
|
|
|
|
dest[i] = *s;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dest[i] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
normabspath(char *path)
|
|
|
|
{
|
|
|
|
size_t len;
|
|
|
|
int last = 0;
|
|
|
|
char *p, *q;
|
|
|
|
|
|
|
|
/* require and skip first slash */
|
|
|
|
if (path[0] != '/') {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
p = path + 1;
|
|
|
|
|
|
|
|
/* get length of path */
|
|
|
|
len = strlen(p);
|
|
|
|
|
|
|
|
for (; !last; ) {
|
|
|
|
/* bound path component within (p,q) */
|
|
|
|
if (!(q = strchr(p, '/'))) {
|
|
|
|
q = strchr(p, '\0');
|
|
|
|
last = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (p == q || (q - p == 1 && p[0] == '.')) {
|
|
|
|
/* "/" or "./" */
|
|
|
|
goto squash;
|
|
|
|
} else if (q - p == 2 && p[0] == '.' && p[1] == '.') {
|
|
|
|
/* "../" */
|
|
|
|
if (p != path + 1) {
|
|
|
|
/* place p right after the previous / */
|
|
|
|
for (p -= 2; p > path && *p != '/'; p--);
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
goto squash;
|
|
|
|
} else {
|
|
|
|
/* move on */
|
|
|
|
p = q + 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
squash:
|
|
|
|
/* squash (p,q) into void */
|
|
|
|
if (last) {
|
|
|
|
*p = '\0';
|
|
|
|
len = p - path;
|
|
|
|
} else {
|
|
|
|
memmove(p, q + 1, len - ((q + 1) - path) + 2);
|
|
|
|
len -= (q + 1) - p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-08-04 14:31:08 +00:00
|
|
|
static enum status
|
2020-08-05 16:59:55 +00:00
|
|
|
parse_range(const char *str, size_t size, size_t *lower, size_t *upper)
|
2020-08-04 14:31:08 +00:00
|
|
|
{
|
2020-08-05 16:28:21 +00:00
|
|
|
char first[FIELD_MAX], last[FIELD_MAX];
|
|
|
|
const char *p, *q, *r, *err;
|
2020-08-04 14:31:08 +00:00
|
|
|
|
|
|
|
/* default to the complete range */
|
|
|
|
*lower = 0;
|
|
|
|
*upper = size - 1;
|
|
|
|
|
|
|
|
/* done if no range-string is given */
|
2020-08-05 16:28:21 +00:00
|
|
|
if (str == NULL || *str == '\0') {
|
2020-08-04 14:31:08 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* skip opening statement */
|
2020-08-05 16:28:21 +00:00
|
|
|
if (strncmp(str, "bytes=", sizeof("bytes=") - 1)) {
|
2020-08-04 14:31:08 +00:00
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
2020-08-05 16:28:21 +00:00
|
|
|
p = str + (sizeof("bytes=") - 1);
|
2020-08-04 14:31:08 +00:00
|
|
|
|
2020-08-05 16:28:21 +00:00
|
|
|
/* check string (should only contain numbers and a hyphen) */
|
|
|
|
for (r = p, q = NULL; *r != '\0'; r++) {
|
|
|
|
if (*r < '0' || *r > '9') {
|
|
|
|
if (*r == '-') {
|
|
|
|
if (q != NULL) {
|
|
|
|
/* we have already seen a hyphen */
|
|
|
|
return S_BAD_REQUEST;
|
|
|
|
} else {
|
|
|
|
/* place q after the hyphen */
|
|
|
|
q = r + 1;
|
|
|
|
}
|
|
|
|
} else if (*r == ',' && r > p) {
|
|
|
|
/*
|
|
|
|
* we refuse to accept range-lists out
|
|
|
|
* of spite towards this horrible part
|
|
|
|
* of the spec
|
|
|
|
*/
|
|
|
|
return S_RANGE_NOT_SATISFIABLE;
|
|
|
|
} else {
|
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (q == NULL) {
|
|
|
|
/* the input string must contain a hyphen */
|
2020-08-04 14:31:08 +00:00
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
2020-08-05 16:28:21 +00:00
|
|
|
r = q + strlen(q);
|
2020-08-04 14:31:08 +00:00
|
|
|
|
|
|
|
/*
|
2020-08-05 16:28:21 +00:00
|
|
|
* byte-range=first-last\0
|
|
|
|
* ^ ^ ^
|
|
|
|
* | | |
|
|
|
|
* p q r
|
2020-08-04 14:31:08 +00:00
|
|
|
*/
|
|
|
|
|
2020-08-05 16:28:21 +00:00
|
|
|
/* copy 'first' and 'last' to their respective arrays */
|
|
|
|
if ((size_t)((q - 1) - p + 1) > sizeof(first) ||
|
|
|
|
(size_t)(r - q + 1) > sizeof(last)) {
|
|
|
|
return S_REQUEST_TOO_LARGE;
|
2020-08-04 14:31:08 +00:00
|
|
|
}
|
2020-08-05 16:28:21 +00:00
|
|
|
memcpy(first, p, (q - 1) - p);
|
|
|
|
first[(q - 1) - p] = '\0';
|
|
|
|
memcpy(last, q, r - q);
|
|
|
|
last[r - q] = '\0';
|
2020-08-04 14:31:08 +00:00
|
|
|
|
2020-08-05 16:28:21 +00:00
|
|
|
if (first[0] != '\0') {
|
2020-08-04 14:31:08 +00:00
|
|
|
/*
|
2020-08-05 16:28:21 +00:00
|
|
|
* range has format "first-last" or "first-",
|
2020-08-04 14:31:08 +00:00
|
|
|
* i.e. return bytes 'first' to 'last' (or the
|
|
|
|
* last byte if 'last' is not given),
|
|
|
|
* inclusively, and byte-numbering beginning at 0
|
|
|
|
*/
|
2020-08-05 16:59:55 +00:00
|
|
|
*lower = strtonum(first, 0, SIZE_MAX, &err);
|
2020-08-04 14:31:08 +00:00
|
|
|
if (!err) {
|
2020-08-05 16:28:21 +00:00
|
|
|
if (last[0] != '\0') {
|
2020-08-05 16:59:55 +00:00
|
|
|
*upper = strtonum(last, 0, SIZE_MAX, &err);
|
2020-08-04 14:31:08 +00:00
|
|
|
} else {
|
|
|
|
*upper = size - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (err) {
|
|
|
|
/* one of the strtonum()'s failed */
|
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check ranges */
|
|
|
|
if (*lower > *upper || *lower >= size) {
|
|
|
|
return S_RANGE_NOT_SATISFIABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* adjust upper limit to be at most the last byte */
|
|
|
|
*upper = MIN(*upper, size - 1);
|
|
|
|
} else {
|
2020-08-05 16:28:21 +00:00
|
|
|
/* last must not also be empty */
|
|
|
|
if (last[0] == '\0') {
|
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
|
|
|
|
2020-08-04 14:31:08 +00:00
|
|
|
/*
|
|
|
|
* Range has format "-num", i.e. return the 'num'
|
|
|
|
* last bytes
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* use upper as a temporary storage for 'num',
|
|
|
|
* as we know 'upper' is size - 1
|
|
|
|
*/
|
2020-08-05 16:59:55 +00:00
|
|
|
*upper = strtonum(last, 0, SIZE_MAX, &err);
|
2020-08-04 14:31:08 +00:00
|
|
|
if (err) {
|
|
|
|
return S_BAD_REQUEST;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* determine lower */
|
|
|
|
if (*upper > size) {
|
|
|
|
/* more bytes requested than we have */
|
|
|
|
*lower = 0;
|
|
|
|
} else {
|
|
|
|
*lower = size - *upper;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set upper to the correct value */
|
|
|
|
*upper = size - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
#undef RELPATH
|
|
|
|
#define RELPATH(x) ((!*(x) || !strcmp(x, "/")) ? "." : ((x) + 1))
|
|
|
|
|
|
|
|
enum status
|
2020-08-05 16:28:21 +00:00
|
|
|
http_send_response(int fd, const struct request *req)
|
2018-02-04 20:27:33 +00:00
|
|
|
{
|
2020-08-05 16:28:21 +00:00
|
|
|
enum status returnstatus;
|
2020-08-05 11:41:44 +00:00
|
|
|
struct in6_addr addr;
|
|
|
|
struct response res = { 0 };
|
2018-02-04 20:27:33 +00:00
|
|
|
struct stat st;
|
2020-07-23 14:54:21 +00:00
|
|
|
struct tm tm = { 0 };
|
2018-02-04 20:27:33 +00:00
|
|
|
size_t len, i;
|
2020-08-05 16:59:55 +00:00
|
|
|
size_t lower, upper;
|
2018-02-04 20:27:33 +00:00
|
|
|
int hasport, ipv6host;
|
2020-08-05 11:41:44 +00:00
|
|
|
static char realtarget[PATH_MAX], tmptarget[PATH_MAX];
|
2020-08-04 14:31:08 +00:00
|
|
|
char *p, *mime;
|
|
|
|
const char *vhostmatch, *targethost;
|
2018-02-04 20:27:33 +00:00
|
|
|
|
2018-02-27 10:36:24 +00:00
|
|
|
/* make a working copy of the target */
|
2020-08-05 13:43:29 +00:00
|
|
|
memcpy(realtarget, req->target, sizeof(realtarget));
|
2018-02-27 10:36:24 +00:00
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
/* match vhost */
|
|
|
|
vhostmatch = NULL;
|
2018-03-04 23:14:25 +00:00
|
|
|
if (s.vhost) {
|
|
|
|
for (i = 0; i < s.vhost_len; i++) {
|
2018-02-04 20:27:33 +00:00
|
|
|
/* switch to vhost directory if there is a match */
|
2020-08-05 13:43:29 +00:00
|
|
|
if (!regexec(&s.vhost[i].re, req->field[REQ_HOST], 0,
|
2018-02-04 20:27:33 +00:00
|
|
|
NULL, 0)) {
|
2018-03-04 23:14:25 +00:00
|
|
|
if (chdir(s.vhost[i].dir) < 0) {
|
2018-02-04 20:27:33 +00:00
|
|
|
return http_send_status(fd, (errno == EACCES) ?
|
|
|
|
S_FORBIDDEN : S_NOT_FOUND);
|
|
|
|
}
|
2018-03-05 08:51:29 +00:00
|
|
|
vhostmatch = s.vhost[i].chost;
|
2018-02-04 20:27:33 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-03-04 23:14:25 +00:00
|
|
|
if (i == s.vhost_len) {
|
2018-02-04 20:27:33 +00:00
|
|
|
return http_send_status(fd, S_NOT_FOUND);
|
|
|
|
}
|
2018-02-27 10:36:24 +00:00
|
|
|
|
|
|
|
/* if we have a vhost prefix, prepend it to the target */
|
2018-03-04 23:14:25 +00:00
|
|
|
if (s.vhost[i].prefix) {
|
2018-03-05 00:12:09 +00:00
|
|
|
if (esnprintf(tmptarget, sizeof(tmptarget), "%s%s",
|
2018-03-04 23:59:37 +00:00
|
|
|
s.vhost[i].prefix, realtarget)) {
|
2018-02-27 10:36:24 +00:00
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
2018-03-05 00:12:09 +00:00
|
|
|
memcpy(realtarget, tmptarget, sizeof(realtarget));
|
2018-02-27 10:36:24 +00:00
|
|
|
}
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|
|
|
|
|
2018-02-27 11:43:05 +00:00
|
|
|
/* apply target prefix mapping */
|
2018-03-04 23:14:25 +00:00
|
|
|
for (i = 0; i < s.map_len; i++) {
|
|
|
|
len = strlen(s.map[i].from);
|
|
|
|
if (!strncmp(realtarget, s.map[i].from, len)) {
|
2019-02-23 23:40:46 +00:00
|
|
|
/* match canonical host if vhosts are enabled and
|
|
|
|
* the mapping specifies a canonical host */
|
|
|
|
if (s.vhost && s.map[i].chost &&
|
|
|
|
strcmp(s.map[i].chost, vhostmatch)) {
|
2018-02-27 11:43:05 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* swap out target prefix */
|
2018-03-04 23:59:37 +00:00
|
|
|
if (esnprintf(tmptarget, sizeof(tmptarget), "%s%s",
|
|
|
|
s.map[i].to, realtarget + len)) {
|
2018-02-27 11:43:05 +00:00
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
2018-03-04 23:14:25 +00:00
|
|
|
memcpy(realtarget, tmptarget, sizeof(realtarget));
|
2018-02-27 11:43:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
/* normalize target */
|
|
|
|
if (normabspath(realtarget)) {
|
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* reject hidden target */
|
|
|
|
if (realtarget[0] == '.' || strstr(realtarget, "/.")) {
|
|
|
|
return http_send_status(fd, S_FORBIDDEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stat the target */
|
|
|
|
if (stat(RELPATH(realtarget), &st) < 0) {
|
|
|
|
return http_send_status(fd, (errno == EACCES) ?
|
|
|
|
S_FORBIDDEN : S_NOT_FOUND);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
/* add / to target if not present */
|
|
|
|
len = strlen(realtarget);
|
2018-07-16 20:46:09 +00:00
|
|
|
if (len >= PATH_MAX - 2) {
|
2018-02-04 20:27:33 +00:00
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
|
|
|
if (len && realtarget[len - 1] != '/') {
|
|
|
|
realtarget[len] = '/';
|
|
|
|
realtarget[len + 1] = '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-27 10:36:24 +00:00
|
|
|
/* redirect if targets differ, host is non-canonical or we prefixed */
|
2020-08-05 13:43:29 +00:00
|
|
|
if (strcmp(req->target, realtarget) || (s.vhost && vhostmatch &&
|
|
|
|
strcmp(req->field[REQ_HOST], vhostmatch))) {
|
2020-08-05 11:41:44 +00:00
|
|
|
res.status = S_MOVED_PERMANENTLY;
|
|
|
|
|
2018-02-04 20:27:33 +00:00
|
|
|
/* encode realtarget */
|
|
|
|
encode(realtarget, tmptarget);
|
|
|
|
|
2020-08-05 11:41:44 +00:00
|
|
|
/* determine target location */
|
2018-07-02 16:43:06 +00:00
|
|
|
if (s.vhost) {
|
|
|
|
/* absolute redirection URL */
|
2020-08-05 13:43:29 +00:00
|
|
|
targethost = req->field[REQ_HOST][0] ? vhostmatch ?
|
|
|
|
vhostmatch : req->field[REQ_HOST] : s.host ?
|
2018-07-02 16:43:06 +00:00
|
|
|
s.host : "localhost";
|
|
|
|
|
|
|
|
/* do we need to add a port to the Location? */
|
|
|
|
hasport = s.port && strcmp(s.port, "80");
|
|
|
|
|
|
|
|
/* RFC 2732 specifies to use brackets for IPv6-addresses
|
|
|
|
* in URLs, so we need to check if our host is one and
|
|
|
|
* honor that later when we fill the "Location"-field */
|
|
|
|
if ((ipv6host = inet_pton(AF_INET6, targethost,
|
2020-08-05 11:41:44 +00:00
|
|
|
&addr)) < 0) {
|
2018-07-02 16:43:06 +00:00
|
|
|
return http_send_status(fd,
|
|
|
|
S_INTERNAL_SERVER_ERROR);
|
|
|
|
}
|
|
|
|
|
2020-08-05 11:41:44 +00:00
|
|
|
/* write location to response struct */
|
|
|
|
if (esnprintf(res.field[RES_LOCATION],
|
|
|
|
sizeof(res.field[RES_LOCATION]),
|
|
|
|
"//%s%s%s%s%s%s",
|
|
|
|
ipv6host ? "[" : "",
|
|
|
|
targethost,
|
|
|
|
ipv6host ? "]" : "", hasport ? ":" : "",
|
|
|
|
hasport ? s.port : "", tmptarget)) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
2018-07-02 16:43:06 +00:00
|
|
|
}
|
|
|
|
} else {
|
2020-08-05 11:41:44 +00:00
|
|
|
/* write relative redirection URL to response struct */
|
|
|
|
if (esnprintf(res.field[RES_LOCATION],
|
|
|
|
sizeof(res.field[RES_LOCATION]),
|
|
|
|
tmptarget)) {
|
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
2018-07-02 16:43:06 +00:00
|
|
|
}
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|
|
|
|
|
2020-08-05 11:41:44 +00:00
|
|
|
return http_send_header(fd, &res);
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
/* append docindex to target */
|
2018-03-04 23:59:37 +00:00
|
|
|
if (esnprintf(realtarget, sizeof(realtarget), "%s%s",
|
2020-08-05 13:43:29 +00:00
|
|
|
req->target, s.docindex)) {
|
2018-02-04 20:27:33 +00:00
|
|
|
return http_send_status(fd, S_REQUEST_TOO_LARGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stat the docindex, which must be a regular file */
|
|
|
|
if (stat(RELPATH(realtarget), &st) < 0 || !S_ISREG(st.st_mode)) {
|
2018-03-04 23:14:25 +00:00
|
|
|
if (s.listdirs) {
|
2018-02-04 20:27:33 +00:00
|
|
|
/* remove index suffix and serve dir */
|
|
|
|
realtarget[strlen(realtarget) -
|
2018-03-04 23:14:25 +00:00
|
|
|
strlen(s.docindex)] = '\0';
|
2020-08-05 13:43:29 +00:00
|
|
|
return resp_dir(fd, RELPATH(realtarget), req);
|
2018-02-04 20:27:33 +00:00
|
|
|
} else {
|
|
|
|
/* reject */
|
|
|
|
if (!S_ISREG(st.st_mode) || errno == EACCES) {
|
|
|
|
return http_send_status(fd, S_FORBIDDEN);
|
|
|
|
} else {
|
|
|
|
return http_send_status(fd, S_NOT_FOUND);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* modified since */
|
2020-08-05 13:46:03 +00:00
|
|
|
if (req->field[REQ_IF_MODIFIED_SINCE][0]) {
|
2018-02-04 20:27:33 +00:00
|
|
|
/* parse field */
|
2020-08-05 13:46:03 +00:00
|
|
|
if (!strptime(req->field[REQ_IF_MODIFIED_SINCE],
|
|
|
|
"%a, %d %b %Y %T GMT", &tm)) {
|
2018-02-04 20:27:33 +00:00
|
|
|
return http_send_status(fd, S_BAD_REQUEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* compare with last modification date of the file */
|
2020-07-23 14:48:34 +00:00
|
|
|
if (difftime(st.st_mtim.tv_sec, timegm(&tm)) <= 0) {
|
2020-08-05 11:41:44 +00:00
|
|
|
res.status = S_NOT_MODIFIED;
|
|
|
|
return http_send_header(fd, &res);
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* range */
|
2020-08-05 16:28:21 +00:00
|
|
|
if ((returnstatus = parse_range(req->field[REQ_RANGE],
|
|
|
|
st.st_size, &lower, &upper))) {
|
|
|
|
if (returnstatus == S_RANGE_NOT_SATISFIABLE) {
|
|
|
|
res.status = S_RANGE_NOT_SATISFIABLE;
|
|
|
|
|
|
|
|
if (esnprintf(res.field[RES_CONTENT_RANGE],
|
|
|
|
sizeof(res.field[RES_CONTENT_RANGE]),
|
|
|
|
"bytes */%zu", st.st_size)) {
|
|
|
|
return http_send_status(fd,
|
|
|
|
S_INTERNAL_SERVER_ERROR);
|
|
|
|
}
|
2020-08-05 11:41:44 +00:00
|
|
|
|
2020-08-05 16:28:21 +00:00
|
|
|
return http_send_header(fd, &res);
|
|
|
|
} else {
|
|
|
|
return http_send_status(fd, returnstatus);
|
2020-07-23 16:16:08 +00:00
|
|
|
}
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* mime */
|
|
|
|
mime = "application/octet-stream";
|
|
|
|
if ((p = strrchr(realtarget, '.'))) {
|
|
|
|
for (i = 0; i < sizeof(mimes) / sizeof(*mimes); i++) {
|
|
|
|
if (!strcmp(mimes[i].ext, p + 1)) {
|
|
|
|
mime = mimes[i].type;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-05 13:43:29 +00:00
|
|
|
return resp_file(fd, RELPATH(realtarget), req, &st, mime, lower, upper);
|
2018-02-04 20:27:33 +00:00
|
|
|
}
|