Skip to content

Commit 061f207

Browse files
committed
string_decoder: Add 'end' method, do base64 properly
1 parent d7c45ea commit 061f207

File tree

7 files changed

+132
-2
lines changed

7 files changed

+132
-2
lines changed

doc/api/string_decoder.markdown

+6-2
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,10 @@ additional support for utf8.
1919

2020
Accepts a single argument, `encoding` which defaults to `utf8`.
2121

22-
### StringDecoder.write(buffer)
22+
### decoder.write(buffer)
2323

24-
Returns a decoded string.
24+
Returns a decoded string.
25+
26+
### decoder.end()
27+
28+
Returns any trailing bytes that were left in the buffer.

lib/fs.js

+5
Original file line numberDiff line numberDiff line change
@@ -1402,6 +1402,11 @@ ReadStream.prototype._read = function() {
14021402
}
14031403

14041404
if (bytesRead === 0) {
1405+
if (this._decoder) {
1406+
var ret = this._decoder.end();
1407+
if (ret)
1408+
this.emit('data', ret);
1409+
}
14051410
self.emit('end');
14061411
self.destroy();
14071412
return;

lib/http.js

+10
Original file line numberDiff line numberDiff line change
@@ -363,6 +363,11 @@ IncomingMessage.prototype._emitData = function(d) {
363363

364364
IncomingMessage.prototype._emitEnd = function() {
365365
if (!this._endEmitted) {
366+
if (this._decoder) {
367+
var ret = this._decoder.end();
368+
if (ret)
369+
this.emit('data', ret);
370+
}
366371
this.emit('end');
367372
}
368373

@@ -1859,6 +1864,11 @@ Client.prototype.request = function(method, path, headers) {
18591864
// but it will get removed when we remove this legacy interface.
18601865
c.on('socket', function(s) {
18611866
s.on('end', function() {
1867+
if (self._decoder) {
1868+
var ret = self._decoder.end();
1869+
if (ret)
1870+
self.emit('data', ret);
1871+
}
18621872
self.emit('end');
18631873
});
18641874
});

lib/net.js

+5
Original file line numberDiff line numberDiff line change
@@ -414,6 +414,11 @@ function onread(buffer, offset, length) {
414414
if (!self.writable) self._destroy();
415415

416416
if (!self.allowHalfOpen) self.end();
417+
if (self._decoder) {
418+
var ret = self._decoder.end();
419+
if (ret)
420+
self.emit('data', ret);
421+
}
417422
if (self._events && self._events['end']) self.emit('end');
418423
if (self.onend) self.onend();
419424
} else {

lib/string_decoder.js

+26
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,11 @@ var StringDecoder = exports.StringDecoder = function(encoding) {
3232
this.surrogateSize = 2;
3333
this.detectIncompleteChar = utf16DetectIncompleteChar;
3434
break;
35+
case 'base64':
36+
// Base-64 stores 3 bytes in 4 chars, and pads the remainder.
37+
this.surrogateSize = 3;
38+
this.detectIncompleteChar = base64DetectIncompleteChar;
39+
break;
3540
default:
3641
this.write = passThroughWrite;
3742
return;
@@ -145,6 +150,21 @@ StringDecoder.prototype.detectIncompleteChar = function(buffer) {
145150
return i;
146151
};
147152

153+
StringDecoder.prototype.end = function(buffer) {
154+
var res = '';
155+
if (buffer && buffer.length)
156+
res = this.write(buffer);
157+
158+
if (this.charReceived) {
159+
var cr = this.charReceived;
160+
var buf = this.charBuffer;
161+
var enc = this.encoding;
162+
res += buf.slice(0, cr).toString(enc);
163+
}
164+
165+
return res;
166+
};
167+
148168
function passThroughWrite(buffer) {
149169
return buffer.toString(this.encoding);
150170
}
@@ -154,3 +174,9 @@ function utf16DetectIncompleteChar(buffer) {
154174
this.charLength = incomplete ? 2 : 0;
155175
return incomplete;
156176
}
177+
178+
function base64DetectIncompleteChar(buffer) {
179+
var incomplete = this.charReceived = buffer.length % 3;
180+
this.charLength = incomplete ? 3 : 0;
181+
return incomplete;
182+
}

lib/tls.js

+5
Original file line numberDiff line numberDiff line change
@@ -967,6 +967,11 @@ SecurePair.prototype.destroy = function() {
967967
self.cleartext.writable = self.cleartext.readable = false;
968968

969969
process.nextTick(function() {
970+
if (self.cleartext._decoder) {
971+
var ret = self.cleartext._decoder.end();
972+
if (ret)
973+
self.cleartext.emit('data', ret);
974+
}
970975
self.cleartext.emit('end');
971976
self.encrypted.emit('close');
972977
self.cleartext.emit('close');
+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
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+
// verify that the string decoder works getting 1 byte at a time,
23+
// the whole buffer at once, and that both match the .toString(enc)
24+
// result of the entire buffer.
25+
26+
var assert = require('assert');
27+
var SD = require('string_decoder').StringDecoder;
28+
var encodings = ['base64', 'hex', 'utf8', 'utf16le', 'ucs2'];
29+
30+
var bufs = [ '☃💩', 'asdf' ].map(function(b) {
31+
return new Buffer(b);
32+
});
33+
34+
// also test just arbitrary bytes from 0-15.
35+
for (var i = 1; i <= 16; i++) {
36+
var bytes = new Array(i).join('.').split('.').map(function(_, j) {
37+
return j + 0x78;
38+
});
39+
bufs.push(new Buffer(bytes));
40+
}
41+
42+
encodings.forEach(testEncoding);
43+
44+
console.log('ok');
45+
46+
function testEncoding(encoding) {
47+
bufs.forEach(function(buf) {
48+
testBuf(encoding, buf);
49+
});
50+
}
51+
52+
function testBuf(encoding, buf) {
53+
console.error('# %s', encoding, buf);
54+
55+
// write one byte at a time.
56+
var s = new SD(encoding);
57+
var res1 = '';
58+
for (var i = 0; i < buf.length; i++) {
59+
res1 += s.write(buf.slice(i, i + 1));
60+
}
61+
res1 += s.end();
62+
63+
// write the whole buffer at once.
64+
var res2 = '';
65+
var s = new SD(encoding);
66+
res2 += s.write(buf);
67+
res2 += s.end();
68+
69+
// .toString() on the buffer
70+
var res3 = buf.toString(encoding);
71+
72+
console.log('expect=%j', res3);
73+
assert.equal(res1, res3, 'one byte at a time should match toString');
74+
assert.equal(res2, res3, 'all bytes at once should match toString');
75+
}

0 commit comments

Comments
 (0)