Skip to content

Commit 54fc9a1

Browse files
committed
http2: account for buffered data before creating HTTP2 socket wrap
If there is data stored in the socket, create a `JSStreamSocket` wrapper so that no data is lost when passing the socket to the C++ layer (which is unaware of data buffered in JS land). Fixes: nodejs#34532
1 parent a6ca822 commit 54fc9a1

File tree

2 files changed

+77
-1
lines changed

2 files changed

+77
-1
lines changed

lib/internal/http2/core.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -1122,7 +1122,11 @@ class Http2Session extends EventEmitter {
11221122
constructor(type, options, socket) {
11231123
super();
11241124

1125-
if (!socket._handle || !socket._handle.isStreamBase) {
1125+
// Ideally, the readableLength check would not be necessary, and we would
1126+
// have a way to handle the buffered data at the C++ StreamBase level.
1127+
if (!socket._handle ||
1128+
!socket._handle.isStreamBase ||
1129+
socket.readableLength > 0) {
11261130
socket = new JSStreamSocket(socket);
11271131
}
11281132

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
'use strict';
2+
const common = require('../common');
3+
if (!common.hasCrypto)
4+
common.skip('missing crypto');
5+
6+
const assert = require('assert');
7+
const net = require('net');
8+
const http = require('http');
9+
const http2 = require('http2');
10+
11+
// Example test for HTTP/1 vs HTTP/2 protocol autoselection.
12+
// Refs: https://github.com/nodejs/node/issues/34532
13+
14+
const h1Server = http.createServer(common.mustCall((req, res) => {
15+
res.end('HTTP/1 Response');
16+
}));
17+
18+
const h2Server = http2.createServer(common.mustCall((req, res) => {
19+
res.end('HTTP/2 Response');
20+
}));
21+
22+
const rawServer = net.createServer(common.mustCall(function listener(socket) {
23+
const data = socket.read(3);
24+
25+
if (!data) { // Repeat until data is available
26+
socket.once('readable', () => listener(socket));
27+
return;
28+
}
29+
30+
// Put the data back, so the real server can handle it:
31+
socket.unshift(data);
32+
33+
if (data.toString('ascii') === 'PRI') { // Very dumb preface check
34+
h2Server.emit('connection', socket);
35+
} else {
36+
h1Server.emit('connection', socket);
37+
}
38+
}, 2));
39+
40+
rawServer.listen(common.mustCall(() => {
41+
const { port } = rawServer.address();
42+
43+
let done = 0;
44+
{
45+
// HTTP/2 Request
46+
const client = http2.connect(`http://localhost:${port}`);
47+
const req = client.request({ ':path': '/' });
48+
req.end();
49+
50+
let content = '';
51+
req.setEncoding('utf8');
52+
req.on('data', (chunk) => content += chunk);
53+
req.on('end', common.mustCall(() => {
54+
assert.strictEqual(content, 'HTTP/2 Response');
55+
if (++done === 2) rawServer.close();
56+
client.close();
57+
}));
58+
}
59+
60+
{
61+
// HTTP/1 Request
62+
http.get(`http://localhost:${port}`, common.mustCall((res) => {
63+
let content = '';
64+
res.setEncoding('utf8');
65+
res.on('data', (chunk) => content += chunk);
66+
res.on('end', common.mustCall(() => {
67+
assert.strictEqual(content, 'HTTP/1 Response');
68+
if (++done === 2) rawServer.close();
69+
}));
70+
}));
71+
}
72+
}));

0 commit comments

Comments
 (0)