Skip to content

Commit 255650f

Browse files
committed
http: Handle hex/base64 encodings properly
This is a backport of 6d3d60aced39d59eaa5e705b7d822c227d0d3dae for v0.10.
1 parent b1acb2e commit 255650f

2 files changed

Lines changed: 67 additions & 23 deletions

File tree

lib/http.js

Lines changed: 14 additions & 23 deletions
Original file line numberDiff line numberDiff line change
@@ -495,7 +495,9 @@ OutgoingMessage.prototype._send = function(data, encoding) {
495495
// the same packet. Future versions of Node are going to take care of
496496
// this at a lower level and in a more general way.
497497
if (!this._headerSent) {
498-
if (typeof data === 'string') {
498+
if (typeof data === 'string' &&
499+
encoding !== 'hex' &&
500+
encoding !== 'base64') {
499501
data = this._header + data;
500502
} else {
501503
this.output.unshift(this._header);
@@ -542,25 +544,6 @@ OutgoingMessage.prototype._writeRaw = function(data, encoding) {
542544

543545

544546
OutgoingMessage.prototype._buffer = function(data, encoding) {
545-
if (data.length === 0) return;
546-
547-
var length = this.output.length;
548-
549-
if (length === 0 || typeof data != 'string') {
550-
this.output.push(data);
551-
this.outputEncodings.push(encoding);
552-
return false;
553-
}
554-
555-
var lastEncoding = this.outputEncodings[length - 1];
556-
var lastData = this.output[length - 1];
557-
558-
if ((encoding && lastEncoding === encoding) ||
559-
(!encoding && data.constructor === lastData.constructor)) {
560-
this.output[length - 1] = lastData + data;
561-
return false;
562-
}
563-
564547
this.output.push(data);
565548
this.outputEncodings.push(encoding);
566549

@@ -888,8 +871,12 @@ OutgoingMessage.prototype.write = function(chunk, encoding) {
888871
ret = this._send(buf, encoding);
889872
} else {
890873
// Non-toString-friendly encoding.
891-
len = chunk.length;
892-
this._send(len.toString(16) + CRLF);
874+
if (typeof chunk === 'string')
875+
len = Buffer.byteLength(chunk, encoding);
876+
else
877+
len = chunk.length;
878+
879+
this._send(len.toString(16) + CRLF, 'ascii');
893880
this._send(chunk, encoding);
894881
ret = this._send(CRLF);
895882
}
@@ -957,6 +944,10 @@ OutgoingMessage.prototype.end = function(data, encoding) {
957944
if (hot && Buffer.isBuffer(data) && data.length > 120 * 1024)
958945
hot = false;
959946

947+
// Can't concatenate safely with hex or base64 encodings.
948+
if (encoding === 'hex' || encoding === 'base64')
949+
hot = false;
950+
960951
if (hot) {
961952
// Hot path. They're doing
962953
// res.writeHead();
@@ -995,7 +986,7 @@ OutgoingMessage.prototype.end = function(data, encoding) {
995986

996987
if (!hot) {
997988
if (this.chunkedEncoding) {
998-
ret = this._send('0\r\n' + this._trailer + '\r\n'); // Last chunk.
989+
ret = this._send('0\r\n' + this._trailer + '\r\n', 'ascii');
999990
} else {
1000991
// Force a flush, HACK.
1001992
ret = this._send('');

test/simple/test-http-hex-write.js

Lines changed: 53 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
// Copyright Joyent, Inc. and other Node contributors.
2+
//
3+
// Permission is hereby granted, free of charge, to any person obtaining a
4+
// copy of this software and associated documentation files (the
5+
// "Software"), to deal in the Software without restriction, including
6+
// without limitation the rights to use, copy, modify, merge, publish,
7+
// distribute, sublicense, and/or sell copies of the Software, and to permit
8+
// persons to whom the Software is furnished to do so, subject to the
9+
// following conditions:
10+
//
11+
// The above copyright notice and this permission notice shall be included
12+
// in all copies or substantial portions of the Software.
13+
//
14+
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15+
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16+
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17+
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18+
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19+
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20+
// USE OR OTHER DEALINGS IN THE SOFTWARE.
21+
22+
var common = require('../common');
23+
var assert = require('assert');
24+
25+
var http = require('http');
26+
27+
var expect = 'hex\nutf8\n';
28+
var data = '';
29+
var ended = false;
30+
31+
process.on('exit', function() {
32+
assert(ended);
33+
assert.equal(data, expect);
34+
console.log('ok');
35+
});
36+
37+
http.createServer(function(q, s) {
38+
s.setHeader('content-length', expect.length);
39+
s.write('6865780a', 'hex');
40+
s.write('utf8\n');
41+
s.end();
42+
this.close();
43+
}).listen(common.PORT, function() {
44+
http.request({ port: common.PORT }).on('response', function(res) {
45+
res.setEncoding('ascii');
46+
res.on('data', function(c) {
47+
data += c;
48+
});
49+
res.on('end', function() {
50+
ended = true;
51+
});
52+
}).end();
53+
});

0 commit comments

Comments
 (0)