Skip to content

Commit 323a6f1

Browse files
committedDec 23, 2020
deps: update http-parser to http-parser@ec8b5ee63f
PR-URL: nodejs-private/node-private#235 Reviewed-By: Fedor Indutny <fedor.indutny@gmail.com>
1 parent 357e285 commit 323a6f1

File tree

7 files changed

+220
-30
lines changed

7 files changed

+220
-30
lines changed
 

‎deps/http_parser/Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ BINEXT ?=
2424
SOLIBNAME = libhttp_parser
2525
SOMAJOR = 2
2626
SOMINOR = 9
27-
SOREV = 3
27+
SOREV = 4
2828
ifeq (darwin,$(PLATFORM))
2929
SOEXT ?= dylib
3030
SONAME ?= $(SOLIBNAME).$(SOMAJOR).$(SOMINOR).$(SOEXT)

‎deps/http_parser/README.md

+3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
HTTP Parser
22
===========
33

4+
http-parser is [**not** actively maintained](https://github.com/nodejs/http-parser/issues/522).
5+
New projects and projects looking to migrate should consider [llhttp](https://github.com/nodejs/llhttp).
6+
47
[![Build Status](https://api.travis-ci.org/nodejs/http-parser.svg?branch=master)](https://travis-ci.org/nodejs/http-parser)
58

69
This is a parser for HTTP messages written in C. It parses both requests and
+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
#include <stdint.h>
2+
#include <stdlib.h>
3+
#include <string.h>
4+
#include "http_parser.h"
5+
6+
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size)
7+
{
8+
static const http_parser_settings settings_null = {
9+
.on_message_begin = 0
10+
, .on_header_field = 0
11+
,.on_header_value = 0
12+
,.on_url = 0
13+
,.on_status = 0
14+
,.on_body = 0
15+
,.on_headers_complete = 0
16+
,.on_message_complete = 0
17+
,.on_chunk_header = 0
18+
,.on_chunk_complete = 0
19+
};
20+
21+
http_parser parser;
22+
http_parser_init(&parser, HTTP_BOTH);
23+
http_parser_execute(&parser, &settings_null, (char*)data, size);
24+
25+
return 0;
26+
}

‎deps/http_parser/fuzzers/fuzz_url.c

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#include <stdint.h>
2+
#include <stdlib.h>
3+
#include <string.h>
4+
#include "http_parser.h"
5+
6+
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size)
7+
{
8+
struct http_parser_url u;
9+
http_parser_url_init(&u);
10+
http_parser_parse_url((char*)data, size, 0, &u);
11+
http_parser_parse_url((char*)data, size, 1, &u);
12+
13+
return 0;
14+
}

‎deps/http_parser/http_parser.c

+18-8
Original file line numberDiff line numberDiff line change
@@ -653,6 +653,8 @@ size_t http_parser_execute (http_parser *parser,
653653
const char *status_mark = 0;
654654
enum state p_state = (enum state) parser->state;
655655
const unsigned int lenient = parser->lenient_http_headers;
656+
const unsigned int allow_chunked_length = parser->allow_chunked_length;
657+
656658
uint32_t nread = parser->nread;
657659

658660
/* We're in an error state. Don't bother doing anything. */
@@ -731,6 +733,7 @@ size_t http_parser_execute (http_parser *parser,
731733
if (ch == CR || ch == LF)
732734
break;
733735
parser->flags = 0;
736+
parser->uses_transfer_encoding = 0;
734737
parser->content_length = ULLONG_MAX;
735738

736739
if (ch == 'H') {
@@ -768,6 +771,7 @@ size_t http_parser_execute (http_parser *parser,
768771
if (ch == CR || ch == LF)
769772
break;
770773
parser->flags = 0;
774+
parser->uses_transfer_encoding = 0;
771775
parser->content_length = ULLONG_MAX;
772776

773777
if (ch == 'H') {
@@ -925,6 +929,7 @@ size_t http_parser_execute (http_parser *parser,
925929
if (ch == CR || ch == LF)
926930
break;
927931
parser->flags = 0;
932+
parser->uses_transfer_encoding = 0;
928933
parser->content_length = ULLONG_MAX;
929934

930935
if (UNLIKELY(!IS_ALPHA(ch))) {
@@ -1338,7 +1343,7 @@ size_t http_parser_execute (http_parser *parser,
13381343
parser->header_state = h_general;
13391344
} else if (parser->index == sizeof(TRANSFER_ENCODING)-2) {
13401345
parser->header_state = h_transfer_encoding;
1341-
parser->flags |= F_TRANSFER_ENCODING;
1346+
parser->uses_transfer_encoding = 1;
13421347
}
13431348
break;
13441349

@@ -1798,14 +1803,19 @@ size_t http_parser_execute (http_parser *parser,
17981803
REEXECUTE();
17991804
}
18001805

1801-
/* Cannot us transfer-encoding and a content-length header together
1806+
/* Cannot use transfer-encoding and a content-length header together
18021807
per the HTTP specification. (RFC 7230 Section 3.3.3) */
1803-
if ((parser->flags & F_TRANSFER_ENCODING) &&
1808+
if ((parser->uses_transfer_encoding == 1) &&
18041809
(parser->flags & F_CONTENTLENGTH)) {
18051810
/* Allow it for lenient parsing as long as `Transfer-Encoding` is
1806-
* not `chunked`
1811+
* not `chunked` or allow_length_with_encoding is set
18071812
*/
1808-
if (!lenient || (parser->flags & F_CHUNKED)) {
1813+
if (parser->flags & F_CHUNKED) {
1814+
if (!allow_chunked_length) {
1815+
SET_ERRNO(HPE_UNEXPECTED_CONTENT_LENGTH);
1816+
goto error;
1817+
}
1818+
} else if (!lenient) {
18091819
SET_ERRNO(HPE_UNEXPECTED_CONTENT_LENGTH);
18101820
goto error;
18111821
}
@@ -1886,7 +1896,7 @@ size_t http_parser_execute (http_parser *parser,
18861896
/* chunked encoding - ignore Content-Length header,
18871897
* prepare for a chunk */
18881898
UPDATE_STATE(s_chunk_size_start);
1889-
} else if (parser->flags & F_TRANSFER_ENCODING) {
1899+
} else if (parser->uses_transfer_encoding == 1) {
18901900
if (parser->type == HTTP_REQUEST && !lenient) {
18911901
/* RFC 7230 3.3.3 */
18921902

@@ -2162,7 +2172,7 @@ http_message_needs_eof (const http_parser *parser)
21622172
}
21632173

21642174
/* RFC 7230 3.3.3, see `s_headers_almost_done` */
2165-
if ((parser->flags & F_TRANSFER_ENCODING) &&
2175+
if ((parser->uses_transfer_encoding == 1) &&
21662176
(parser->flags & F_CHUNKED) == 0) {
21672177
return 1;
21682178
}
@@ -2514,7 +2524,7 @@ http_parser_parse_url(const char *buf, size_t buflen, int is_connect,
25142524
end = buf + off + len;
25152525

25162526
/* NOTE: The characters are already validated and are in the [0-9] range */
2517-
assert(off + len <= buflen && "Port number overflow");
2527+
assert((size_t) (off + len) <= buflen && "Port number overflow");
25182528
v = 0;
25192529
for (p = buf + off; p < end; p++) {
25202530
v *= 10;

‎deps/http_parser/http_parser.h

+15-8
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ extern "C" {
2727
/* Also update SONAME in the Makefile whenever you change these. */
2828
#define HTTP_PARSER_VERSION_MAJOR 2
2929
#define HTTP_PARSER_VERSION_MINOR 9
30-
#define HTTP_PARSER_VERSION_PATCH 3
30+
#define HTTP_PARSER_VERSION_PATCH 4
3131

3232
#include <stddef.h>
3333
#if defined(_WIN32) && !defined(__MINGW32__) && \
@@ -41,6 +41,8 @@ typedef __int32 int32_t;
4141
typedef unsigned __int32 uint32_t;
4242
typedef __int64 int64_t;
4343
typedef unsigned __int64 uint64_t;
44+
#elif (defined(__sun) || defined(__sun__)) && defined(__SunOS_5_9)
45+
#include <sys/inttypes.h>
4446
#else
4547
#include <stdint.h>
4648
#endif
@@ -225,7 +227,6 @@ enum flags
225227
, F_UPGRADE = 1 << 5
226228
, F_SKIPBODY = 1 << 6
227229
, F_CONTENTLENGTH = 1 << 7
228-
, F_TRANSFER_ENCODING = 1 << 8
229230
};
230231

231232

@@ -272,13 +273,13 @@ enum flags
272273
"unexpected content-length header") \
273274
XX(INVALID_CHUNK_SIZE, \
274275
"invalid character in chunk size header") \
275-
XX(INVALID_TRANSFER_ENCODING, \
276-
"request has invalid transfer-encoding") \
277276
XX(INVALID_CONSTANT, "invalid constant string") \
278277
XX(INVALID_INTERNAL_STATE, "encountered unexpected internal state")\
279278
XX(STRICT, "strict mode assertion failed") \
280279
XX(PAUSED, "parser is paused") \
281-
XX(UNKNOWN, "an unknown error occurred")
280+
XX(UNKNOWN, "an unknown error occurred") \
281+
XX(INVALID_TRANSFER_ENCODING, \
282+
"request has invalid transfer-encoding") \
282283

283284

284285
/* Define HPE_* values for each errno value above */
@@ -296,14 +297,20 @@ enum http_errno {
296297
struct http_parser {
297298
/** PRIVATE **/
298299
unsigned int type : 2; /* enum http_parser_type */
300+
unsigned int flags : 8; /* F_* values from 'flags' enum; semi-public */
299301
unsigned int state : 7; /* enum state from http_parser.c */
300302
unsigned int header_state : 7; /* enum header_state from http_parser.c */
301-
unsigned int index : 7; /* index into current matcher */
303+
unsigned int index : 5; /* index into current matcher */
304+
unsigned int uses_transfer_encoding : 1; /* Transfer-Encoding header is present */
305+
unsigned int allow_chunked_length : 1; /* Allow headers with both
306+
* `Content-Length` and
307+
* `Transfer-Encoding: chunked` set */
302308
unsigned int lenient_http_headers : 1;
303-
unsigned int flags : 16; /* F_* values from 'flags' enum; semi-public */
304309

305310
uint32_t nread; /* # bytes read in various scenarios */
306-
uint64_t content_length; /* # bytes in body (0 if no Content-Length header) */
311+
uint64_t content_length; /* # bytes in body. `(uint64_t) -1` (all bits one)
312+
* if no Content-Length header.
313+
*/
307314

308315
/** READ-ONLY **/
309316
unsigned short http_major;

‎deps/http_parser/test.c

+143-13
Large diffs are not rendered by default.

0 commit comments

Comments
 (0)
Please sign in to comment.