http: fix out-of-order 'finish' bug in pipelining
authorFedor Indutny <fedor@indutny.com>
Sat, 26 Sep 2015 03:36:06 +0000 (23:36 -0400)
committerRod Vagg <rod@vagg.org>
Mon, 5 Oct 2015 11:33:08 +0000 (22:33 +1100)
Changes to `stream_base.cc` are required to support empty writes.
Fixes CVE-2015-7384, https://github.com/nodejs/node/issues/3138

Fix: https://github.com/nodejs/node/issues/2639
PR-URL: https://github.com/nodejs/node/pull/3128

lib/_http_outgoing.js
src/stream_base.cc
test/parallel/test-http-pipeline-regr-2639.js [new file with mode: 0644]

index 0522ecb..0b725d5 100644 (file)
@@ -133,12 +133,6 @@ OutgoingMessage.prototype._writeRaw = function(data, encoding, callback) {
     encoding = null;
   }
 
-  if (data.length === 0) {
-    if (typeof callback === 'function')
-      process.nextTick(callback);
-    return true;
-  }
-
   var connection = this.connection;
   if (connection &&
       connection._httpMessage === this &&
@@ -158,6 +152,10 @@ OutgoingMessage.prototype._writeRaw = function(data, encoding, callback) {
       this.output = [];
       this.outputEncodings = [];
       this.outputCallbacks = [];
+    } else if (data.length === 0) {
+      if (typeof callback === 'function')
+        process.nextTick(callback);
+      return true;
     }
 
     // Directly write to socket.
index 3139274..27ae0fe 100644 (file)
@@ -154,7 +154,7 @@ int StreamBase::Writev(const FunctionCallbackInfo<Value>& args) {
 
     // Write string
     offset = ROUND_UP(offset, WriteWrap::kAlignSize);
-    CHECK_LT(offset, storage_size);
+    CHECK_LE(offset, storage_size);
     char* str_storage = req_wrap->Extra(offset);
     size_t str_size = storage_size - offset;
 
diff --git a/test/parallel/test-http-pipeline-regr-2639.js b/test/parallel/test-http-pipeline-regr-2639.js
new file mode 100644 (file)
index 0000000..f1f75bf
--- /dev/null
@@ -0,0 +1,34 @@
+'use strict';
+const common = require('../common');
+const assert = require('assert');
+const http = require('http');
+const net = require('net');
+
+const COUNT = 10;
+
+var received = 0;
+
+var server = http.createServer(function(req, res) {
+  // Close the server, we have only one TCP connection anyway
+  if (received++ === 0)
+    server.close();
+
+  res.writeHead(200);
+  res.write('data');
+
+  setTimeout(function() {
+    res.end();
+  }, (Math.random() * 100) | 0);
+}).listen(common.PORT, function() {
+  const s = net.connect(common.PORT);
+
+  var big = '';
+  for (var i = 0; i < COUNT; i++)
+    big += 'GET / HTTP/1.0\r\n\r\n';
+  s.write(big);
+  s.resume();
+});
+
+process.on('exit', function() {
+  assert.equal(received, COUNT);
+});