Skip to content

Commit

Permalink
http2: wait for secureConnect before initializing
Browse files Browse the repository at this point in the history
PR-URL: #32958
Fixes: #32922
Reviewed-By: James M Snell <jasnell@gmail.com>
Reviewed-By: Anna Henningsen <anna@addaleax.net>
  • Loading branch information
bcoe authored and targos committed May 13, 2020
1 parent 1811a10 commit d883024
Show file tree
Hide file tree
Showing 3 changed files with 83 additions and 1 deletion.
2 changes: 2 additions & 0 deletions lib/_tls_wrap.js
Expand Up @@ -466,6 +466,7 @@ function TLSSocket(socket, opts) {
this._securePending = false;
this._newSessionPending = false;
this._controlReleased = false;
this.secureConnecting = true;
this._SNICallback = null;
this.servername = null;
this.alpnProtocol = null;
Expand Down Expand Up @@ -1035,6 +1036,7 @@ function onServerSocketSecure() {

if (!this.destroyed && this._releaseControl()) {
debug('server emit secureConnection');
this.secureConnecting = false;
this._tlsOptions.server.emit('secureConnection', this);
}
}
Expand Down
2 changes: 1 addition & 1 deletion lib/internal/http2/core.js
Expand Up @@ -1115,7 +1115,7 @@ class Http2Session extends EventEmitter {
socket.disableRenegotiation();

const setupFn = setupHandle.bind(this, socket, type, options);
if (socket.connecting) {
if (socket.connecting || socket.secureConnecting) {
const connectEvent =
socket instanceof tls.TLSSocket ? 'secureConnect' : 'connect';
socket.once(connectEvent, () => {
Expand Down
80 changes: 80 additions & 0 deletions test/internet/test-http2-issue-32922.js
@@ -0,0 +1,80 @@
'use strict';
const common = require('../common');
const assert = require('assert');

if (!common.hasCrypto)
common.skip('missing crypto');

const http2 = require('http2');
const net = require('net');

const {
HTTP2_HEADER_PATH,
} = http2.constants;

// Create a normal session, as a control case
function normalSession(cb) {
http2.connect('https://google.com', (clientSession) => {
let error = null;
const req = clientSession.request({ [HTTP2_HEADER_PATH]: '/' });
req.on('error', (err) => {
error = err;
});
req.on('response', (_headers) => {
req.on('data', (_chunk) => { });
req.on('end', () => {
clientSession.close();
return cb(error);
});
});
});
}
normalSession(common.mustCall(function(err) {
assert.ifError(err);
}));

// Create a session using a socket that has not yet finished connecting
function socketNotFinished(done) {
const socket2 = net.connect(443, 'google.com');
http2.connect('https://google.com', { socket2 }, (clientSession) => {
let error = null;
const req = clientSession.request({ [HTTP2_HEADER_PATH]: '/' });
req.on('error', (err) => {
error = err;
});
req.on('response', (_headers) => {
req.on('data', (_chunk) => { });
req.on('end', () => {
clientSession.close();
socket2.destroy();
return done(error);
});
});
});
}
socketNotFinished(common.mustCall(function(err) {
assert.ifError(err);
}));

// Create a session using a socket that has finished connecting
function socketFinished(done) {
const socket = net.connect(443, 'google.com', () => {
http2.connect('https://google.com', { socket }, (clientSession) => {
let error = null;
const req = clientSession.request({ [HTTP2_HEADER_PATH]: '/' });
req.on('error', (err) => {
error = err;
});
req.on('response', (_headers) => {
req.on('data', (_chunk) => { });
req.on('end', () => {
clientSession.close();
return done(error);
});
});
});
});
}
socketFinished(common.mustCall(function(err) {
assert.ifError(err);
}));

0 comments on commit d883024

Please sign in to comment.