summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFedor Indutny <fedor.indutny@gmail.com>2013-11-08 01:22:02 +0400
committerFedor Indutny <fedor.indutny@gmail.com>2013-11-09 02:07:36 +0400
commitac2263b77f3f346458d06fc019de27e24c63cee0 (patch)
tree6843d17d227b69016cb7ea44c746de713943d752
parent8f221bc43d02d86da94186d5290c29a426039d15 (diff)
downloadnode-new-ac2263b77f3f346458d06fc019de27e24c63cee0.tar.gz
tls: prevent stalls by using read(0)
Do not `.push()` the same data as just passed to `.ondata()`, it may be read by 'data' event listeners. fix #6277
-rw-r--r--lib/tls.js14
-rw-r--r--test/simple/test-https-req-split.js75
2 files changed, 85 insertions, 4 deletions
diff --git a/lib/tls.js b/lib/tls.js
index dcdd99a193..f575bd69df 100644
--- a/lib/tls.js
+++ b/lib/tls.js
@@ -506,10 +506,16 @@ CryptoStream.prototype._read = function read(size) {
if (this.ondata) {
this.ondata(pool, start, start + bytesRead);
- // Consume data automatically
- // simple/test-https-drain fails without it
- this.push(pool.slice(start, start + bytesRead));
- this.read(bytesRead);
+ // Deceive streams2
+ var self = this;
+
+ setImmediate(function() {
+ // Force state.reading to set to false
+ self.push('');
+
+ // Try reading more, we most likely have some data
+ self.read(0);
+ });
} else {
this.push(pool.slice(start, start + bytesRead));
}
diff --git a/test/simple/test-https-req-split.js b/test/simple/test-https-req-split.js
new file mode 100644
index 0000000000..db54d72377
--- /dev/null
+++ b/test/simple/test-https-req-split.js
@@ -0,0 +1,75 @@
+// Copyright Joyent, Inc. and other Node contributors.
+//
+// Permission is hereby granted, free of charge, to any person obtaining a
+// copy of this software and associated documentation files (the
+// "Software"), to deal in the Software without restriction, including
+// without limitation the rights to use, copy, modify, merge, publish,
+// distribute, sublicense, and/or sell copies of the Software, and to permit
+// persons to whom the Software is furnished to do so, subject to the
+// following conditions:
+//
+// The above copyright notice and this permission notice shall be included
+// in all copies or substantial portions of the Software.
+//
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
+// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
+// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
+// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
+// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
+// USE OR OTHER DEALINGS IN THE SOFTWARE.
+
+if (!process.versions.openssl) {
+ console.error('Skipping because node compiled without OpenSSL.');
+ process.exit(0);
+}
+
+// disable strict server certificate validation by the client
+process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
+
+var common = require('../common');
+var assert = require('assert');
+var https = require('https');
+var tls = require('tls');
+var fs = require('fs');
+
+var seen_req = false;
+
+var options = {
+ key: fs.readFileSync(common.fixturesDir + '/keys/agent1-key.pem'),
+ cert: fs.readFileSync(common.fixturesDir + '/keys/agent1-cert.pem')
+};
+
+// Force splitting incoming data
+tls.SLAB_BUFFER_SIZE = 1;
+
+var server = https.createServer(options);
+server.on('upgrade', function(req, socket, upgrade) {
+ socket.on('data', function(data) {
+ throw new Error('Unexpected data: ' + data);
+ });
+ socket.end('HTTP/1.1 200 Ok\r\n\r\n');
+ seen_req = true;
+});
+
+server.listen(common.PORT, function() {
+ var req = https.request({
+ host: '127.0.0.1',
+ port: common.PORT,
+ agent: false,
+ headers: {
+ Connection: 'Upgrade',
+ Upgrade: 'Websocket'
+ }
+ }, function() {
+ req.socket.destroy();
+ server.close();
+ });
+
+ req.end();
+});
+
+process.on('exit', function() {
+ assert(seen_req);
+ console.log('ok');
+});