Skip to content

Commit b8366e7

Browse files
trevnorrisFishrock123
authored andcommitted
http_parser: use pushValueToArray for headers
For performance add headers to the headers Array by pushing them on from JS. Benchmark added to demonstrate this case. PR-URL: #3780 Reviewed-By: Fedor Indutny <[email protected]>
1 parent 8464667 commit b8366e7

File tree

2 files changed

+72
-6
lines changed

2 files changed

+72
-6
lines changed

benchmark/http/bench-parser.js

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const HTTPParser = process.binding('http_parser').HTTPParser;
5+
const REQUEST = HTTPParser.REQUEST;
6+
const kOnHeaders = HTTPParser.kOnHeaders | 0;
7+
const kOnHeadersComplete = HTTPParser.kOnHeadersComplete | 0;
8+
const kOnBody = HTTPParser.kOnBody | 0;
9+
const kOnMessageComplete = HTTPParser.kOnMessageComplete | 0;
10+
const CRLF = '\r\n';
11+
12+
const bench = common.createBenchmark(main, {
13+
fields: [4, 8, 16, 32],
14+
n: [1e5],
15+
});
16+
17+
18+
function main(conf) {
19+
const fields = conf.fields >>> 0;
20+
const n = conf.n >>> 0;
21+
var header = `GET /hello HTTP/1.1${CRLF}Content-Type: text/plain${CRLF}`;
22+
23+
for (var i = 0; i < fields; i++) {
24+
header += `X-Filler${i}: ${Math.random().toString(36).substr(2)}${CRLF}`;
25+
}
26+
header += CRLF;
27+
28+
processHeader(new Buffer(header), n);
29+
}
30+
31+
32+
function processHeader(header, n) {
33+
const parser = newParser(REQUEST);
34+
35+
bench.start();
36+
for (var i = 0; i < n; i++) {
37+
parser.execute(header, 0, header.length);
38+
parser.reinitialize(REQUEST);
39+
}
40+
bench.end(n);
41+
}
42+
43+
44+
function newParser(type) {
45+
const parser = new HTTPParser(type);
46+
47+
parser.headers = [];
48+
49+
parser[kOnHeaders] = function() { };
50+
parser[kOnHeadersComplete] = function() { };
51+
parser[kOnBody] = function() { };
52+
parser[kOnMessageComplete] = function() { };
53+
54+
return parser;
55+
}

src/node_http_parser.cc

+17-6
Original file line numberDiff line numberDiff line change
@@ -632,12 +632,23 @@ class Parser : public BaseObject {
632632
Local<Array> CreateHeaders() {
633633
// num_values_ is either -1 or the entry # of the last header
634634
// so num_values_ == 0 means there's a single header
635-
Local<Array> headers = Array::New(env()->isolate(), 2 * num_values_);
636-
637-
for (int i = 0; i < num_values_; ++i) {
638-
headers->Set(2 * i, fields_[i].ToString(env()));
639-
headers->Set(2 * i + 1, values_[i].ToString(env()));
640-
}
635+
Local<Array> headers = Array::New(env()->isolate());
636+
Local<Function> fn = env()->push_values_to_array_function();
637+
Local<Value> argv[NODE_PUSH_VAL_TO_ARRAY_MAX * 2];
638+
int i = 0;
639+
640+
do {
641+
size_t j = 0;
642+
while (i < num_values_ && j < ARRAY_SIZE(argv) / 2) {
643+
argv[j * 2] = fields_[i].ToString(env());
644+
argv[j * 2 + 1] = values_[i].ToString(env());
645+
i++;
646+
j++;
647+
}
648+
if (j > 0) {
649+
fn->Call(env()->context(), headers, j * 2, argv).ToLocalChecked();
650+
}
651+
} while (i < num_values_);
641652

642653
return headers;
643654
}

0 commit comments

Comments
 (0)