Skip to content

Commit

Permalink
Replace createSecurePair with TLSSocket + TCP server
Browse files Browse the repository at this point in the history
This is stupid. Not how to properly replace createSecurePair. SecurePair
fires ‘secure’ event synchronously. Using a couple of sockets instead of
it means that the flow becomes more asynchronous. Therefore socket ‘end’
can not be used directly to control the Connection state, the data first
needs to pass through the TCP server and TLSSocket.
  • Loading branch information
Antti Risteli committed Dec 14, 2017
1 parent e2152d4 commit 182e4e0
Show file tree
Hide file tree
Showing 3 changed files with 86 additions and 37 deletions.
9 changes: 1 addition & 8 deletions src/connection.js
Original file line number Diff line number Diff line change
Expand Up @@ -357,7 +357,6 @@ class Connection extends EventEmitter {

this.reset = this.reset.bind(this);
this.socketClose = this.socketClose.bind(this);
this.socketEnd = this.socketEnd.bind(this);
this.socketConnect = this.socketConnect.bind(this);
this.socketError = this.socketError.bind(this);
this.requestTimeout = this.requestTimeout.bind(this);
Expand Down Expand Up @@ -691,12 +690,11 @@ class Connection extends EventEmitter {

this.socket = socket;
this.socket.on('error', this.socketError);
this.socket.on('close', this.socketClose);
this.socket.on('end', this.socketEnd);
this.messageIo = new MessageIO(this.socket, this.config.options.packetSize, this.debug);
this.messageIo.on('data', (data) => { this.dispatchEvent('data', data); });
this.messageIo.on('message', () => { this.dispatchEvent('message'); });
this.messageIo.on('secure', this.emit.bind(this, 'secure'));
this.messageIo.on('close', this.socketClose);

this.socketConnect();
});
Expand Down Expand Up @@ -815,11 +813,6 @@ class Connection extends EventEmitter {
return this.dispatchEvent('socketConnect');
}

socketEnd() {
this.debug.log('socket ended');
return this.transitionTo(this.STATE.FINAL);
}

socketClose() {
this.debug.log('connection to ' + this.config.server + ':' + this.config.options.port + ' closed');
if (this.state === this.STATE.REROUTING) {
Expand Down
111 changes: 83 additions & 28 deletions src/message-io.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
const tls = require('tls');
const crypto = require('crypto');
const net = require('net');
const EventEmitter = require('events').EventEmitter;
const Transform = require('readable-stream').Transform;

Expand Down Expand Up @@ -50,6 +51,51 @@ class ReadablePacketStream extends Transform {
}
}

class TLSHandler extends EventEmitter {
constructor(secureContext) {
super();
const self = this;
self.server = net.createServer();

self.server.listen(0, '127.0.0.1', () => {
self.cleartext = tls.connect({
host: '127.0.0.1',
port: self.server.address().port,
secureContext: secureContext,
rejectUnauthorized: false
});
self.cleartext.on('secureConnect', () => {
self.emit('secure');
self.cleartext.write('');
});
});

const encryptedOnQueue = [];
self.encrypted = {
on: (event, cb) => {
encryptedOnQueue.push([event, cb]);
}
};

self.server.on('connection', (socket) => {
self.encrypted = socket;
encryptedOnQueue.forEach(([event, cb]) => {
self.encrypted.on(event, cb);
});
self.server.close();
});
}

destroy() {
if (this.encrypted && this.encrypted.destroy) {
this.encrypted.destroy();
}
if (this.cleartext && this.cleartext.destroy) {
this.cleartext.destroy();
}
}
}

module.exports = class MessageIO extends EventEmitter {
constructor(socket, _packetSize, debug) {
super();
Expand All @@ -70,6 +116,10 @@ module.exports = class MessageIO extends EventEmitter {

this.socket.pipe(this.packetStream);
this.packetDataSize = this._packetSize - packetHeaderLength;

this.socket.on('close', () => {
this.emit('close');
});
}

packetSize(packetSize) {
Expand All @@ -84,54 +134,59 @@ module.exports = class MessageIO extends EventEmitter {
startTls(credentialsDetails, hostname, trustServerCertificate) {
const credentials = tls.createSecureContext ? tls.createSecureContext(credentialsDetails) : crypto.createCredentials(credentialsDetails);

this.securePair = tls.createSecurePair(credentials);
this.tlsHandler = new TLSHandler(credentials);

this.tlsNegotiationComplete = false;

this.securePair.on('secure', () => {
const cipher = this.securePair.cleartext.getCipher();
this.tlsHandler.on('secure', () => {
const cipher = this.tlsHandler.cleartext.getCipher();

if (!trustServerCertificate) {
let verifyError = this.securePair.ssl.verifyError();

// Verify that server's identity matches it's certificate's names
if (!verifyError) {
verifyError = tls.checkServerIdentity(hostname, this.securePair.cleartext.getPeerCertificate());
}

if (verifyError) {
this.securePair.destroy();
this.socket.destroy(verifyError);
if (!this.tlsHandler.cleartext.authorized) {
this.tlsHandler.destroy();
this.socket.destroy(this.tlsHandler.cleartext.authorizationError);
return;
}
}

this.debug.log('TLS negotiated (' + cipher.name + ', ' + cipher.version + ')');
this.emit('secure', this.securePair.cleartext);
this.emit('secure', this.tlsHandler.cleartext);
this.encryptAllFutureTraffic();
});

this.securePair.encrypted.on('data', (data) => {
this.tlsHandler.encrypted.on('data', (data) => {
this.sendMessage(TYPE.PRELOGIN, data);
});

// On Node >= 0.12, the encrypted stream automatically starts spewing out
// data once we attach a `data` listener. But on Node <= 0.10.x, this is not
// the case. We need to kick the cleartext stream once to get the
// encrypted end of the secure pair to emit the TLS handshake data.
this.securePair.cleartext.write('');
}

encryptAllFutureTraffic() {
this.socket.unpipe(this.packetStream);
this.securePair.encrypted.removeAllListeners('data');
this.socket.pipe(this.securePair.encrypted);
this.securePair.encrypted.pipe(this.socket);
this.securePair.cleartext.pipe(this.packetStream);
this.tlsHandler.encrypted.removeAllListeners('data');
this.socket.pipe(this.tlsHandler.encrypted);
this.socket.removeAllListeners('close');
this.socket.on('close', () => {
this.tlsHandler.encrypted.end();
});
this.socket.on('error', () => {
this.tlsHandler.encrypted.end();
});
this.tlsHandler.cleartext.on('close', () => {
this.emit('close');
});
this.tlsHandler.encrypted.pipe(this.socket);
this.tlsHandler.cleartext.pipe(this.packetStream);
this.tlsNegotiationComplete = true;
if (!this.socket.destroyed) {
// the old SecurePair worked synchronously and fired the
// 'secure' event before the packet was handled by
// RedablePacketStream. this is not the case anymore so
// emit 'message' manually to fire SENT_TLSSSLNEGOTIATION.message again
this.emit('message');
}
}

tlsHandshakeData(data) {
this.securePair.encrypted.write(data);
this.tlsHandler.encrypted.write(data);
}

// TODO listen for 'drain' event when socket.write returns false.
Expand Down Expand Up @@ -168,8 +223,8 @@ module.exports = class MessageIO extends EventEmitter {

sendPacket(packet) {
this.logPacket('Sent', packet);
if (this.securePair && this.tlsNegotiationComplete) {
this.securePair.cleartext.write(packet.buffer);
if (this.tlsHandler && this.tlsNegotiationComplete) {
this.tlsHandler.cleartext.write(packet.buffer);
} else {
this.socket.write(packet.buffer);
}
Expand Down
3 changes: 2 additions & 1 deletion test/integration/connection-retry-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,8 @@ exports['connection retry tests'] = {
connection.on('end', (info) => {
test.done();
});
},
}
,

'no retries on non-transient errors': function(test) {
const config = getConfig();
Expand Down

0 comments on commit 182e4e0

Please sign in to comment.