2020-06-15 14:18:57 +00:00
|
|
|
/*-*- 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 │
|
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
2020-06-15 14:18:57 +00:00
|
|
|
│ │
|
2020-12-28 01:18:44 +00:00
|
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
2020-06-15 14:18:57 +00:00
|
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
|
|
#include "libc/bits/bits.h"
|
2021-03-25 09:21:13 +00:00
|
|
|
#include "libc/errno.h"
|
2021-03-29 01:00:29 +00:00
|
|
|
#include "libc/log/check.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
#include "libc/mem/mem.h"
|
2021-03-08 04:23:29 +00:00
|
|
|
#include "libc/runtime/gc.internal.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
#include "libc/stdio/stdio.h"
|
|
|
|
#include "libc/str/str.h"
|
2021-03-29 01:00:29 +00:00
|
|
|
#include "libc/testlib/ezbench.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
#include "libc/testlib/testlib.h"
|
2020-09-07 04:39:00 +00:00
|
|
|
#include "libc/x/x.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
#include "net/http/http.h"
|
2021-03-25 09:21:13 +00:00
|
|
|
#include "net/http/uri.h"
|
2020-06-15 14:18:57 +00:00
|
|
|
|
2020-09-07 04:39:00 +00:00
|
|
|
struct HttpRequest req[1];
|
|
|
|
|
|
|
|
static char *slice(const char *m, struct HttpRequestSlice s) {
|
2020-11-13 09:27:49 +00:00
|
|
|
char *p;
|
|
|
|
p = xmalloc(s.b - s.a + 1);
|
|
|
|
memcpy(p, m + s.a, s.b - s.a);
|
|
|
|
p[s.b - s.a] = 0;
|
|
|
|
return p;
|
2020-09-07 04:39:00 +00:00
|
|
|
}
|
|
|
|
|
2021-03-25 09:21:13 +00:00
|
|
|
static unsigned version(const char *m) {
|
|
|
|
return ParseHttpVersion(m + req->version.a, req->version.b - req->version.a);
|
|
|
|
}
|
|
|
|
|
2021-03-27 14:29:55 +00:00
|
|
|
void SetUp(void) {
|
2021-03-25 09:21:13 +00:00
|
|
|
InitHttpRequest(req);
|
2021-03-27 14:29:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TearDown(void) {
|
|
|
|
DestroyHttpRequest(req);
|
|
|
|
}
|
|
|
|
|
2021-03-28 07:10:17 +00:00
|
|
|
/* TEST(ParseHttpRequest, soLittleState) { */
|
|
|
|
/* ASSERT_EQ(280, sizeof(struct HttpRequest)); */
|
|
|
|
/* } */
|
|
|
|
|
2021-03-27 14:29:55 +00:00
|
|
|
TEST(ParseHttpRequest, testEmpty_tooShort) {
|
2020-11-28 20:01:51 +00:00
|
|
|
EXPECT_EQ(0, ParseHttpRequest(req, "", 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testTooShort) {
|
|
|
|
EXPECT_EQ(0, ParseHttpRequest(req, "\r\n", 2));
|
2020-06-15 14:18:57 +00:00
|
|
|
}
|
|
|
|
|
2020-09-07 04:39:00 +00:00
|
|
|
TEST(ParseHttpRequest, testNoHeaders) {
|
|
|
|
static const char m[] = "GET /foo HTTP/1.0\r\n\r\n";
|
2020-11-28 20:01:51 +00:00
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
2020-09-07 04:39:00 +00:00
|
|
|
EXPECT_EQ(kHttpGet, req->method);
|
|
|
|
EXPECT_STREQ("/foo", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.0", gc(slice(m, req->version)));
|
2020-06-15 14:18:57 +00:00
|
|
|
}
|
|
|
|
|
2020-09-07 04:39:00 +00:00
|
|
|
TEST(ParseHttpRequest, testSomeHeaders) {
|
|
|
|
static const char m[] = "\
|
|
|
|
POST /foo?bar%20hi HTTP/1.0\r\n\
|
|
|
|
Host: foo.example\r\n\
|
|
|
|
Content-Length: 0\r\n\
|
|
|
|
\r\n";
|
2021-03-25 09:21:13 +00:00
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(kHttpPost, req->method);
|
|
|
|
EXPECT_STREQ("/foo?bar%20hi", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.0", gc(slice(m, req->version)));
|
|
|
|
EXPECT_STREQ("foo.example", gc(slice(m, req->headers[kHttpHost])));
|
|
|
|
EXPECT_STREQ("0", gc(slice(m, req->headers[kHttpContentLength])));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->headers[kHttpEtag])));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testHttp101) {
|
|
|
|
static const char m[] = "GET / HTTP/1.1\r\n\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(kHttpGet, req->method);
|
|
|
|
EXPECT_STREQ("/", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.1", gc(slice(m, req->version)));
|
|
|
|
EXPECT_EQ(101, version(m));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testHttp100) {
|
|
|
|
static const char m[] = "GET / HTTP/1.0\r\n\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(kHttpGet, req->method);
|
|
|
|
EXPECT_STREQ("/", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.0", gc(slice(m, req->version)));
|
|
|
|
EXPECT_EQ(100, version(m));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testHttp009) {
|
|
|
|
static const char m[] = "GET /\r\n\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(kHttpGet, req->method);
|
|
|
|
EXPECT_STREQ("/", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->version)));
|
|
|
|
EXPECT_EQ(9, version(m));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testLeadingLineFeeds_areIgnored) {
|
|
|
|
static const char m[] = "\
|
|
|
|
\r\n\
|
|
|
|
GET /foo?bar%20hi HTTP/1.0\r\n\
|
|
|
|
User-Agent: hi\r\n\
|
|
|
|
\r\n";
|
2020-11-28 20:01:51 +00:00
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
2021-03-25 09:21:13 +00:00
|
|
|
EXPECT_STREQ("/foo?bar%20hi", gc(slice(m, req->uri)));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testLineFolding_isRejected) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /foo?bar%20hi HTTP/1.0\r\n\
|
|
|
|
User-Agent: hi\r\n\
|
|
|
|
there\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(-1, ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(EBADMSG, errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testEmptyHeaderName_isRejected) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /foo?bar%20hi HTTP/1.0\r\n\
|
|
|
|
User-Agent: hi\r\n\
|
|
|
|
: hi\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(-1, ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(EBADMSG, errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testUnixNewlines) {
|
|
|
|
static const char m[] = "\
|
|
|
|
POST /foo?bar%20hi HTTP/1.0\n\
|
|
|
|
Host: foo.example\n\
|
|
|
|
Content-Length: 0\n\
|
|
|
|
\n\
|
|
|
|
\n";
|
|
|
|
EXPECT_EQ(strlen(m) - 1, ParseHttpRequest(req, m, strlen(m)));
|
2020-09-07 04:39:00 +00:00
|
|
|
EXPECT_EQ(kHttpPost, req->method);
|
|
|
|
EXPECT_STREQ("/foo?bar%20hi", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.0", gc(slice(m, req->version)));
|
|
|
|
EXPECT_STREQ("foo.example", gc(slice(m, req->headers[kHttpHost])));
|
|
|
|
EXPECT_STREQ("0", gc(slice(m, req->headers[kHttpContentLength])));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->headers[kHttpEtag])));
|
2020-06-15 14:18:57 +00:00
|
|
|
}
|
2021-03-27 14:29:55 +00:00
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testChromeMessage) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /tool/net/redbean.png HTTP/1.1\r\n\
|
|
|
|
Host: 10.10.10.124:8080\r\n\
|
|
|
|
Connection: keep-alive\r\n\
|
|
|
|
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\r\n\
|
|
|
|
DNT: \t1\r\n\
|
|
|
|
Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8\r\n\
|
|
|
|
Referer: http://10.10.10.124:8080/\r\n\
|
|
|
|
Accept-Encoding: gzip, deflate\r\n\
|
|
|
|
Accept-Language: en-US,en;q=0.9\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(kHttpGet, req->method);
|
|
|
|
EXPECT_STREQ("/tool/net/redbean.png", gc(slice(m, req->uri)));
|
|
|
|
EXPECT_STREQ("HTTP/1.1", gc(slice(m, req->version)));
|
|
|
|
EXPECT_STREQ("10.10.10.124:8080", gc(slice(m, req->headers[kHttpHost])));
|
|
|
|
EXPECT_STREQ("1", gc(slice(m, req->headers[kHttpDnt])));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->headers[kHttpExpect])));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->headers[kHttpContentLength])));
|
|
|
|
EXPECT_STREQ("", gc(slice(m, req->headers[kHttpExpect])));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testExtendedHeaders) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /foo?bar%20hi HTTP/1.0\r\n\
|
|
|
|
X-User-Agent: hi\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
ASSERT_EQ(1, req->xheaders.n);
|
|
|
|
EXPECT_STREQ("X-User-Agent", gc(slice(m, req->xheaders.p[0].k)));
|
|
|
|
EXPECT_STREQ("hi", gc(slice(m, req->xheaders.p[0].v)));
|
|
|
|
}
|
2021-03-28 07:10:17 +00:00
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testHeaderValuesWithWhitespace_getsTrimmed) {
|
|
|
|
static const char m[] = "\
|
|
|
|
OPTIONS * HTTP/1.0\r\n\
|
|
|
|
User-Agent: \t hi there \t \r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_STREQ("hi there", gc(slice(m, req->headers[kHttpUserAgent])));
|
|
|
|
}
|
2021-03-29 01:00:29 +00:00
|
|
|
|
2021-04-18 18:34:59 +00:00
|
|
|
TEST(ParseHttpRequest, testAbsentHost_setsSliceToZero) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET / HTTP/1.1\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_EQ(0, req->headers[kHttpHost].a);
|
|
|
|
EXPECT_EQ(0, req->headers[kHttpHost].b);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testEmptyHost_setsSliceToNonzeroValue) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET / HTTP/1.1\r\n\
|
|
|
|
Host:\r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_NE(0, req->headers[kHttpHost].a);
|
|
|
|
EXPECT_EQ(req->headers[kHttpHost].a, req->headers[kHttpHost].b);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ParseHttpRequest, testEmptyHost2_setsSliceToNonzeroValue) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET / HTTP/1.1\r\n\
|
|
|
|
Host: \r\n\
|
|
|
|
\r\n";
|
|
|
|
EXPECT_EQ(strlen(m), ParseHttpRequest(req, m, strlen(m)));
|
|
|
|
EXPECT_NE(0, req->headers[kHttpHost].a);
|
|
|
|
EXPECT_EQ(req->headers[kHttpHost].a, req->headers[kHttpHost].b);
|
|
|
|
}
|
|
|
|
|
2021-03-29 01:00:29 +00:00
|
|
|
void DoTiniestHttpRequest(void) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /\r\n\
|
|
|
|
\r\n";
|
|
|
|
InitHttpRequest(req);
|
|
|
|
ParseHttpRequest(req, m, sizeof(m));
|
|
|
|
DestroyHttpRequest(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoTinyHttpRequest(void) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /\r\n\
|
|
|
|
Accept-Encoding: gzip\r\n\
|
|
|
|
\r\n";
|
|
|
|
InitHttpRequest(req);
|
|
|
|
ParseHttpRequest(req, m, sizeof(m));
|
|
|
|
DestroyHttpRequest(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoStandardChromeRequest(void) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /tool/net/redbean.png HTTP/1.1\r\n\
|
|
|
|
Host: 10.10.10.124:8080\r\n\
|
|
|
|
Connection: keep-alive\r\n\
|
|
|
|
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\r\n\
|
|
|
|
DNT: \t1 \r\n\
|
|
|
|
Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8\r\n\
|
|
|
|
Referer: http://10.10.10.124:8080/\r\n\
|
|
|
|
Accept-Encoding: gzip, deflate\r\n\
|
|
|
|
Accept-Language: en-US,en;q=0.9\r\n\
|
|
|
|
\r\n";
|
|
|
|
InitHttpRequest(req);
|
|
|
|
CHECK_EQ(sizeof(m) - 1, ParseHttpRequest(req, m, sizeof(m)));
|
|
|
|
DestroyHttpRequest(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoUnstandardChromeRequest(void) {
|
|
|
|
static const char m[] = "\
|
|
|
|
GET /tool/net/redbean.png HTTP/1.1\r\n\
|
|
|
|
X-Host: 10.10.10.124:8080\r\n\
|
|
|
|
X-Connection: keep-alive\r\n\
|
|
|
|
X-User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\r\n\
|
|
|
|
X-DNT: \t1 \r\n\
|
|
|
|
X-Accept: image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8\r\n\
|
|
|
|
X-Referer: http://10.10.10.124:8080/\r\n\
|
|
|
|
X-Accept-Encoding: gzip, deflate\r\n\
|
|
|
|
X-Accept-Language: en-US,en;q=0.9\r\n\
|
|
|
|
\r\n";
|
|
|
|
InitHttpRequest(req);
|
|
|
|
CHECK_EQ(sizeof(m) - 1, ParseHttpRequest(req, m, sizeof(m)));
|
|
|
|
DestroyHttpRequest(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
BENCH(ParseHttpRequest, bench) {
|
|
|
|
EZBENCH2("DoTiniestHttpRequest", donothing, DoTiniestHttpRequest());
|
|
|
|
EZBENCH2("DoTinyHttpRequest", donothing, DoTinyHttpRequest());
|
|
|
|
EZBENCH2("DoStandardChromeRequest", donothing, DoStandardChromeRequest());
|
|
|
|
EZBENCH2("DoUnstandardChromeRequest", donothing, DoUnstandardChromeRequest());
|
|
|
|
}
|