Skip to content

Commit 9c7e664

Browse files
anonrigdanielleadams
authored andcommittedOct 11, 2022
http2: make early hints generic
PR-URL: #44820 Fixes: #44816 Reviewed-By: Matteo Collina <matteo.collina@gmail.com> Reviewed-By: Minwoo Jung <nodecorelab@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Rafael Gonzaga <rafael.nunu@hotmail.com>
1 parent 3627616 commit 9c7e664

11 files changed

+318
-163
lines changed
 

‎doc/api/http.md

+15-6
Original file line numberDiff line numberDiff line change
@@ -2124,32 +2124,41 @@ Sends an HTTP/1.1 100 Continue message to the client, indicating that
21242124
the request body should be sent. See the [`'checkContinue'`][] event on
21252125
`Server`.
21262126

2127-
### `response.writeEarlyHints(links[, callback])`
2127+
### `response.writeEarlyHints(hints[, callback])`
21282128

21292129
<!-- YAML
21302130
added: REPLACEME
2131+
changes:
2132+
- version: REPLACEME
2133+
pr-url: https://github.com/nodejs/node/pull/44820
2134+
description: Allow passing hints as an object.
21312135
-->
21322136

2133-
* `links` {string|Array}
2137+
* `hints` {Object}
21342138
* `callback` {Function}
21352139

21362140
Sends an HTTP/1.1 103 Early Hints message to the client with a Link header,
21372141
indicating that the user agent can preload/preconnect the linked resources.
2138-
The `links` can be a string or an array of strings containing the values
2139-
of the `Link` header. The optional `callback` argument will be called when
2142+
The `hints` is an object containing the values of headers to be sent with
2143+
early hints message. The optional `callback` argument will be called when
21402144
the response message has been written.
21412145

21422146
**Example**
21432147

21442148
```js
21452149
const earlyHintsLink = '</styles.css>; rel=preload; as=style';
2146-
response.writeEarlyHints(earlyHintsLink);
2150+
response.writeEarlyHints({
2151+
'link': earlyHintsLink,
2152+
});
21472153

21482154
const earlyHintsLinks = [
21492155
'</styles.css>; rel=preload; as=style',
21502156
'</scripts.js>; rel=preload; as=script',
21512157
];
2152-
response.writeEarlyHints(earlyHintsLinks);
2158+
response.writeEarlyHints({
2159+
'link': earlyHintsLinks,
2160+
'x-trace-id': 'id for diagnostics'
2161+
});
21532162

21542163
const earlyHintsCallback = () => console.log('early hints message sent');
21552164
response.writeEarlyHints(earlyHintsLinks, earlyHintsCallback);

‎lib/_http_server.js

+17-25
Original file line numberDiff line numberDiff line change
@@ -81,7 +81,8 @@ const {
8181
const {
8282
validateInteger,
8383
validateBoolean,
84-
validateLinkHeaderValue
84+
validateLinkHeaderValue,
85+
validateObject
8586
} = require('internal/validators');
8687
const Buffer = require('buffer').Buffer;
8788
const {
@@ -301,36 +302,27 @@ ServerResponse.prototype.writeProcessing = function writeProcessing(cb) {
301302
this._writeRaw('HTTP/1.1 102 Processing\r\n\r\n', 'ascii', cb);
302303
};
303304

304-
ServerResponse.prototype.writeEarlyHints = function writeEarlyHints(links, cb) {
305+
ServerResponse.prototype.writeEarlyHints = function writeEarlyHints(hints, cb) {
305306
let head = 'HTTP/1.1 103 Early Hints\r\n';
306307

307-
if (typeof links === 'string') {
308-
validateLinkHeaderValue(links, 'links');
309-
head += 'Link: ' + links + '\r\n';
310-
} else if (ArrayIsArray(links)) {
311-
if (!links.length) {
312-
return;
313-
}
308+
validateObject(hints, 'hints');
314309

315-
head += 'Link: ';
310+
if (hints.link === null || hints.link === undefined) {
311+
return;
312+
}
316313

317-
for (let i = 0; i < links.length; i++) {
318-
const link = links[i];
319-
validateLinkHeaderValue(link, 'links');
320-
head += link;
314+
const link = validateLinkHeaderValue(hints.link);
321315

322-
if (i !== links.length - 1) {
323-
head += ', ';
324-
}
325-
}
316+
if (link.length === 0) {
317+
return;
318+
}
326319

327-
head += '\r\n';
328-
} else {
329-
throw new ERR_INVALID_ARG_VALUE(
330-
'links',
331-
links,
332-
'must be an array or string of format "</styles.css>; rel=preload; as=style"'
333-
);
320+
head += 'Link: ' + link + '\r\n';
321+
322+
for (const key of ObjectKeys(hints)) {
323+
if (key !== 'link') {
324+
head += key + ': ' + hints[key] + '\r\n';
325+
}
334326
}
335327

336328
head += '\r\n';

‎lib/internal/http2/compat.js

+14-25
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,7 @@ const {
5757
validateFunction,
5858
validateString,
5959
validateLinkHeaderValue,
60+
validateObject,
6061
} = require('internal/validators');
6162
const {
6263
kSocket,
@@ -847,34 +848,21 @@ class Http2ServerResponse extends Stream {
847848
return true;
848849
}
849850

850-
writeEarlyHints(links) {
851-
let linkHeaderValue = '';
851+
writeEarlyHints(hints) {
852+
validateObject(hints, 'hints');
852853

853-
if (typeof links === 'string') {
854-
validateLinkHeaderValue(links, 'links');
855-
linkHeaderValue += links;
856-
} else if (ArrayIsArray(links)) {
857-
if (!links.length) {
858-
return;
859-
}
860-
861-
linkHeaderValue += '';
854+
const headers = ObjectCreate(null);
862855

863-
for (let i = 0; i < links.length; i++) {
864-
const link = links[i];
865-
validateLinkHeaderValue(link, 'links');
866-
linkHeaderValue += link;
856+
const linkHeaderValue = validateLinkHeaderValue(hints.link);
867857

868-
if (i !== links.length - 1) {
869-
linkHeaderValue += ', ';
870-
}
858+
for (const key of ObjectKeys(hints)) {
859+
if (key !== 'link') {
860+
headers[key] = hints[key];
871861
}
872-
} else {
873-
throw new ERR_INVALID_ARG_VALUE(
874-
'links',
875-
links,
876-
'must be an array or string of format "</styles.css>; rel=preload; as=style"'
877-
);
862+
}
863+
864+
if (linkHeaderValue.length === 0) {
865+
return false;
878866
}
879867

880868
const stream = this[kStream];
@@ -883,8 +871,9 @@ class Http2ServerResponse extends Stream {
883871
return false;
884872

885873
stream.additionalHeaders({
874+
...headers,
886875
[HTTP2_HEADER_STATUS]: HTTP_STATUS_EARLY_HINTS,
887-
'Link': linkHeaderValue
876+
'Link': linkHeaderValue,
888877
});
889878

890879
return true;

‎lib/internal/validators.js

+42-2
Original file line numberDiff line numberDiff line change
@@ -403,9 +403,13 @@ function validateUnion(value, name, union) {
403403
}
404404
}
405405

406-
function validateLinkHeaderValue(value, name) {
407-
const linkValueRegExp = /^(?:<[^>]*>;)\s*(?:rel=(")?[^;"]*\1;?)\s*(?:(?:as|anchor|title)=(")?[^;"]*\2)?$/;
406+
const linkValueRegExp = /^(?:<[^>]*>;)\s*(?:rel=(")?[^;"]*\1;?)\s*(?:(?:as|anchor|title)=(")?[^;"]*\2)?$/;
408407

408+
/**
409+
* @param {any} value
410+
* @param {string} name
411+
*/
412+
function validateLinkHeaderFormat(value, name) {
409413
if (
410414
typeof value === 'undefined' ||
411415
!RegExpPrototypeExec(linkValueRegExp, value)
@@ -418,6 +422,42 @@ function validateLinkHeaderValue(value, name) {
418422
}
419423
}
420424

425+
/**
426+
* @param {any} hints
427+
* @return {string}
428+
*/
429+
function validateLinkHeaderValue(hints) {
430+
if (typeof hints === 'string') {
431+
validateLinkHeaderFormat(hints, 'hints');
432+
return hints;
433+
} else if (ArrayIsArray(hints)) {
434+
const hintsLength = hints.length;
435+
let result = '';
436+
437+
if (hintsLength === 0) {
438+
return result;
439+
}
440+
441+
for (let i = 0; i < hintsLength; i++) {
442+
const link = hints[i];
443+
validateLinkHeaderFormat(link, 'hints');
444+
result += link;
445+
446+
if (i !== hintsLength - 1) {
447+
result += ', ';
448+
}
449+
}
450+
451+
return result;
452+
}
453+
454+
throw new ERR_INVALID_ARG_VALUE(
455+
'hints',
456+
hints,
457+
'must be an array or string of format "</styles.css>; rel=preload; as=style"'
458+
);
459+
}
460+
421461
module.exports = {
422462
isInt32,
423463
isUint32,

‎test/parallel/test-http-early-hints-invalid-argument-type.js

-33
This file was deleted.

‎test/parallel/test-http-early-hints-invalid-argument.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ const testResBody = 'response content\n';
88

99
const server = http.createServer(common.mustCall((req, res) => {
1010
debug('Server sending early hints...');
11-
res.writeEarlyHints('bad argument value');
11+
res.writeEarlyHints('bad argument type');
1212

1313
debug('Server sending full response...');
1414
res.end(testResBody);
@@ -27,7 +27,7 @@ server.listen(0, common.mustCall(() => {
2727
process.on('uncaughtException', (err) => {
2828
debug(`Caught an exception: ${JSON.stringify(err)}`);
2929
if (err.name === 'AssertionError') throw err;
30-
assert.strictEqual(err.code, 'ERR_INVALID_ARG_VALUE');
30+
assert.strictEqual(err.code, 'ERR_INVALID_ARG_TYPE');
3131
process.exit(0);
3232
});
3333
}));

‎test/parallel/test-http-early-hints.js

+150-6
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,9 @@ const testResBody = 'response content\n';
1111

1212
const server = http.createServer(common.mustCall((req, res) => {
1313
debug('Server sending early hints...');
14-
res.writeEarlyHints('</styles.css>; rel=preload; as=style');
14+
res.writeEarlyHints({
15+
link: '</styles.css>; rel=preload; as=style'
16+
});
1517

1618
debug('Server sending full response...');
1719
res.end(testResBody);
@@ -53,10 +55,12 @@ const testResBody = 'response content\n';
5355

5456
const server = http.createServer(common.mustCall((req, res) => {
5557
debug('Server sending early hints...');
56-
res.writeEarlyHints([
57-
'</styles.css>; rel=preload; as=style',
58-
'</scripts.js>; rel=preload; as=script',
59-
]);
58+
res.writeEarlyHints({
59+
link: [
60+
'</styles.css>; rel=preload; as=style',
61+
'</scripts.js>; rel=preload; as=script',
62+
]
63+
});
6064

6165
debug('Server sending full response...');
6266
res.end(testResBody);
@@ -100,7 +104,147 @@ const testResBody = 'response content\n';
100104

101105
const server = http.createServer(common.mustCall((req, res) => {
102106
debug('Server sending early hints...');
103-
res.writeEarlyHints([]);
107+
res.writeEarlyHints({
108+
link: []
109+
});
110+
111+
debug('Server sending full response...');
112+
res.end(testResBody);
113+
}));
114+
115+
server.listen(0, common.mustCall(() => {
116+
const req = http.request({
117+
port: server.address().port, path: '/'
118+
});
119+
debug('Client sending request...');
120+
121+
req.on('information', common.mustNotCall());
122+
123+
req.on('response', common.mustCall((res) => {
124+
let body = '';
125+
126+
assert.strictEqual(res.statusCode, 200, `Final status code was ${res.statusCode}, not 200.`);
127+
128+
res.on('data', (chunk) => {
129+
body += chunk;
130+
});
131+
132+
res.on('end', common.mustCall(() => {
133+
debug('Got full response.');
134+
assert.strictEqual(body, testResBody);
135+
server.close();
136+
}));
137+
}));
138+
139+
req.end();
140+
}));
141+
}
142+
143+
{
144+
// Happy flow - object argument with string
145+
146+
const server = http.createServer(common.mustCall((req, res) => {
147+
debug('Server sending early hints...');
148+
res.writeEarlyHints({
149+
'link': '</styles.css>; rel=preload; as=style',
150+
'x-trace-id': 'id for diagnostics'
151+
});
152+
153+
debug('Server sending full response...');
154+
res.end(testResBody);
155+
}));
156+
157+
server.listen(0, common.mustCall(() => {
158+
const req = http.request({
159+
port: server.address().port, path: '/'
160+
});
161+
debug('Client sending request...');
162+
163+
req.on('information', common.mustCall((res) => {
164+
assert.strictEqual(
165+
res.headers.link,
166+
'</styles.css>; rel=preload; as=style'
167+
);
168+
assert.strictEqual(res.headers['x-trace-id'], 'id for diagnostics');
169+
}));
170+
171+
req.on('response', common.mustCall((res) => {
172+
let body = '';
173+
174+
assert.strictEqual(res.statusCode, 200, `Final status code was ${res.statusCode}, not 200.`);
175+
176+
res.on('data', (chunk) => {
177+
body += chunk;
178+
});
179+
180+
res.on('end', common.mustCall(() => {
181+
debug('Got full response.');
182+
assert.strictEqual(body, testResBody);
183+
server.close();
184+
}));
185+
}));
186+
187+
req.end();
188+
}));
189+
}
190+
191+
{
192+
// Happy flow - object argument with array of strings
193+
194+
const server = http.createServer(common.mustCall((req, res) => {
195+
debug('Server sending early hints...');
196+
res.writeEarlyHints({
197+
'link': [
198+
'</styles.css>; rel=preload; as=style',
199+
'</scripts.js>; rel=preload; as=script',
200+
],
201+
'x-trace-id': 'id for diagnostics'
202+
});
203+
204+
debug('Server sending full response...');
205+
res.end(testResBody);
206+
}));
207+
208+
server.listen(0, common.mustCall(() => {
209+
const req = http.request({
210+
port: server.address().port, path: '/'
211+
});
212+
debug('Client sending request...');
213+
214+
req.on('information', common.mustCall((res) => {
215+
assert.strictEqual(
216+
res.headers.link,
217+
'</styles.css>; rel=preload; as=style, </scripts.js>; rel=preload; as=script'
218+
);
219+
assert.strictEqual(res.headers['x-trace-id'], 'id for diagnostics');
220+
}));
221+
222+
req.on('response', common.mustCall((res) => {
223+
let body = '';
224+
225+
assert.strictEqual(res.statusCode, 200, `Final status code was ${res.statusCode}, not 200.`);
226+
227+
res.on('data', (chunk) => {
228+
body += chunk;
229+
});
230+
231+
res.on('end', common.mustCall(() => {
232+
debug('Got full response.');
233+
assert.strictEqual(body, testResBody);
234+
server.close();
235+
}));
236+
}));
237+
238+
req.end();
239+
}));
240+
}
241+
242+
{
243+
// Happy flow - empty object
244+
245+
const server = http.createServer(common.mustCall((req, res) => {
246+
debug('Server sending early hints...');
247+
res.writeEarlyHints({});
104248

105249
debug('Server sending full response...');
106250
res.end(testResBody);

‎test/parallel/test-http2-compat-write-early-hints-invalid-argument-type.js

+24-20
Original file line numberDiff line numberDiff line change
@@ -9,30 +9,34 @@ const debug = require('node:util').debuglog('test');
99

1010
const testResBody = 'response content';
1111

12-
const server = http2.createServer();
12+
{
13+
// Invalid object value
1314

14-
server.on('request', common.mustCall((req, res) => {
15-
debug('Server sending early hints...');
16-
res.writeEarlyHints({ links: 'bad argument object' });
15+
const server = http2.createServer();
1716

18-
debug('Server sending full response...');
19-
res.end(testResBody);
20-
}));
17+
server.on('request', common.mustCall((req, res) => {
18+
debug('Server sending early hints...');
19+
res.writeEarlyHints('this should not be here');
2120

22-
server.listen(0);
21+
debug('Server sending full response...');
22+
res.end(testResBody);
23+
}));
2324

24-
server.on('listening', common.mustCall(() => {
25-
const client = http2.connect(`http://localhost:${server.address().port}`);
26-
const req = client.request();
25+
server.listen(0);
2726

28-
debug('Client sending request...');
27+
server.on('listening', common.mustCall(() => {
28+
const client = http2.connect(`http://localhost:${server.address().port}`);
29+
const req = client.request();
2930

30-
req.on('headers', common.mustNotCall());
31+
debug('Client sending request...');
3132

32-
process.on('uncaughtException', (err) => {
33-
debug(`Caught an exception: ${JSON.stringify(err)}`);
34-
if (err.name === 'AssertionError') throw err;
35-
assert.strictEqual(err.code, 'ERR_INVALID_ARG_VALUE');
36-
process.exit(0);
37-
});
38-
}));
33+
req.on('headers', common.mustNotCall());
34+
35+
process.on('uncaughtException', (err) => {
36+
debug(`Caught an exception: ${JSON.stringify(err)}`);
37+
if (err.name === 'AssertionError') throw err;
38+
assert.strictEqual(err.code, 'ERR_INVALID_ARG_TYPE');
39+
process.exit(0);
40+
});
41+
}));
42+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
if (!common.hasCrypto) common.skip('missing crypto');
5+
6+
const assert = require('node:assert');
7+
const http2 = require('node:http2');
8+
const debug = require('node:util').debuglog('test');
9+
10+
const testResBody = 'response content';
11+
12+
{
13+
// Invalid link header value
14+
15+
const server = http2.createServer();
16+
17+
server.on('request', common.mustCall((req, res) => {
18+
debug('Server sending early hints...');
19+
res.writeEarlyHints({ link: BigInt(100) });
20+
21+
debug('Server sending full response...');
22+
res.end(testResBody);
23+
}));
24+
25+
server.listen(0);
26+
27+
server.on('listening', common.mustCall(() => {
28+
const client = http2.connect(`http://localhost:${server.address().port}`);
29+
const req = client.request();
30+
31+
debug('Client sending request...');
32+
33+
req.on('headers', common.mustNotCall());
34+
35+
process.on('uncaughtException', (err) => {
36+
debug(`Caught an exception: ${JSON.stringify(err)}`);
37+
if (err.name === 'AssertionError') throw err;
38+
assert.strictEqual(err.code, 'ERR_INVALID_ARG_VALUE');
39+
process.exit(0);
40+
});
41+
}));
42+
}

‎test/parallel/test-http2-compat-write-early-hints-invalid-argument.js

-38
This file was deleted.

‎test/parallel/test-http2-compat-write-early-hints.js

+12-6
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,9 @@ const testResBody = 'response content';
1616

1717
server.on('request', common.mustCall((req, res) => {
1818
debug('Server sending early hints...');
19-
res.writeEarlyHints('</styles.css>; rel=preload; as=style');
19+
res.writeEarlyHints({
20+
link: '</styles.css>; rel=preload; as=style'
21+
});
2022

2123
debug('Server sending full response...');
2224
res.end(testResBody);
@@ -59,10 +61,12 @@ const testResBody = 'response content';
5961

6062
server.on('request', common.mustCall((req, res) => {
6163
debug('Server sending early hints...');
62-
res.writeEarlyHints([
63-
'</styles.css>; rel=preload; as=style',
64-
'</scripts.js>; rel=preload; as=script',
65-
]);
64+
res.writeEarlyHints({
65+
link: [
66+
'</styles.css>; rel=preload; as=style',
67+
'</scripts.js>; rel=preload; as=script',
68+
]
69+
});
6670

6771
debug('Server sending full response...');
6872
res.end(testResBody);
@@ -108,7 +112,9 @@ const testResBody = 'response content';
108112

109113
server.on('request', common.mustCall((req, res) => {
110114
debug('Server sending early hints...');
111-
res.writeEarlyHints([]);
115+
res.writeEarlyHints({
116+
link: []
117+
});
112118

113119
debug('Server sending full response...');
114120
res.end(testResBody);

0 commit comments

Comments
 (0)
Please sign in to comment.