From 809bb0a373e82f728f03fa792122128741b821ea Mon Sep 17 00:00:00 2001 From: Fedor Indutny Date: Tue, 9 Feb 2016 16:00:24 -0500 Subject: [PATCH] tls: nullify `.ssl` on handle close This is an intermediate fix for an issue of accessing `TLSWrap` fields after the parent handle was destroyed. While `close` listener cleans up this field automatically, it can be done even earlier at the `TLSWrap.close` call. Proper fix is going to be submitted and landed after this one. Fix: #5108 --- lib/_tls_wrap.js | 3 ++ test/parallel/test-tls-regr-gh-5108.js | 45 ++++++++++++++++++++++++++ 2 files changed, 48 insertions(+) create mode 100644 test/parallel/test-tls-regr-gh-5108.js diff --git a/lib/_tls_wrap.js b/lib/_tls_wrap.js index a888dc1554a495..69e197ca46c771 100644 --- a/lib/_tls_wrap.js +++ b/lib/_tls_wrap.js @@ -299,6 +299,9 @@ proxiedMethods.forEach(function(name) { }); tls_wrap.TLSWrap.prototype.close = function closeProxy(cb) { + if (this.owner) + this.owner.ssl = null; + if (this._parentWrap && this._parentWrap._handle === this._parent) { this._parentWrap.once('close', cb); return this._parentWrap.destroy(); diff --git a/test/parallel/test-tls-regr-gh-5108.js b/test/parallel/test-tls-regr-gh-5108.js new file mode 100644 index 00000000000000..5bc71b46fef631 --- /dev/null +++ b/test/parallel/test-tls-regr-gh-5108.js @@ -0,0 +1,45 @@ +'use strict'; + +const common = require('../common'); + +if (!common.hasCrypto) { + console.log('1..0 # Skipped: node compiled without crypto.'); + return; +} + +const assert = require('assert'); +const tls = require('tls'); +const fs = require('fs'); +const path = require('path'); + +const keyDir = path.join(common.fixturesDir, 'keys'); +const key = fs.readFileSync(path.join(keyDir, 'agent1-key.pem')); +const cert = fs.readFileSync(path.join(keyDir, 'agent1-cert.pem')); + +const server = tls.createServer({ + key: key, + cert: cert +}, function(c) { + c.end(); +}).listen(common.PORT, function() { + var client = tls.connect({ + port: common.PORT, + rejectUnauthorized: false + }, common.mustCall(function() { + client.destroy(); + if (!client._events.close) + client._events.close = []; + else if (!Array.isArray(client._events.close)) + client._events.close = [ client._events.close ]; + + client._events.close.unshift(common.mustCall(function() { + setImmediate(function() { + // Make it crash on unpatched node.js + var fd = client.ssl && client.ssl.fd; + assert(client.ssl === null); + assert(!fd); + }); + })); + server.close(); + })); +});