Skip to content
This repository was archived by the owner on Apr 22, 2023. It is now read-only.

Commit 0aa1a8a

Browse files
isaacsry
authored andcommitted
Closes GH-695 Add 'hex' encoding to Buffer
1 parent 9851574 commit 0aa1a8a

File tree

6 files changed

+84
-1
lines changed

6 files changed

+84
-1
lines changed

doc/api/buffers.markdown

+2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,8 @@ the first 8 bits of each character. This encoding method is depreciated and
2626
should be avoided in favor of `Buffer` objects where possible. This encoding
2727
will be removed in future versions of Node.
2828

29+
* `'hex'` - Encode each byte as two hexidecimal characters.
30+
2931

3032
### new Buffer(size)
3133

lib/buffer.js

+45
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,21 @@ SlowBuffer.prototype.inspect = function() {
1717
};
1818

1919

20+
SlowBuffer.prototype.hexSlice = function(start, end) {
21+
var len = this.length;
22+
23+
if (!start || start < 0) start = 0;
24+
if (end < 0 || start + end > len) end = len - start;
25+
26+
var out = '';
27+
for (var i = start; i < end; i ++) {
28+
out += toHex(this[i]);
29+
}
30+
return out;
31+
};
32+
33+
34+
2035
SlowBuffer.prototype.toString = function(encoding, start, end) {
2136
encoding = String(encoding || 'utf8').toLowerCase();
2237
start = +start || 0;
@@ -28,6 +43,9 @@ SlowBuffer.prototype.toString = function(encoding, start, end) {
2843
}
2944

3045
switch (encoding) {
46+
case 'hex':
47+
return this.hexSlice(start, end);
48+
3149
case 'utf8':
3250
case 'utf-8':
3351
return this.utf8Slice(start, end);
@@ -51,6 +69,23 @@ SlowBuffer.prototype.toString = function(encoding, start, end) {
5169
};
5270

5371

72+
SlowBuffer.prototype.hexWrite = function(string, offset) {
73+
var len = string.length;
74+
offset = +offset || 0;
75+
76+
// must be an even number of digits
77+
if (len % 2) {
78+
throw new Error('Invalid hex string');
79+
}
80+
for (var i = 0; i < len / 2; i ++) {
81+
var byte = parseInt(string.substr(i * 2, 2), 16);
82+
if (isNaN(byte)) throw new Error('Invalid hex string');
83+
this[offset + i] = byte;
84+
}
85+
return i;
86+
}
87+
88+
5489
SlowBuffer.prototype.write = function(string, offset, encoding) {
5590
// Support both (string, offset, encoding)
5691
// and the legacy (string, encoding, offset)
@@ -64,6 +99,9 @@ SlowBuffer.prototype.write = function(string, offset, encoding) {
6499
encoding = String(encoding || 'utf8').toLowerCase();
65100

66101
switch (encoding) {
102+
case 'hex':
103+
return this.hexWrite(string, offset);
104+
67105
case 'utf8':
68106
case 'utf-8':
69107
return this.utf8Write(string, offset);
@@ -224,6 +262,10 @@ Buffer.prototype.write = function(string, offset, encoding) {
224262

225263
var ret;
226264
switch (encoding) {
265+
case 'hex':
266+
ret = this.parent.hexWrite(string, this.offset + offset, maxLength);
267+
break;
268+
227269
case 'utf8':
228270
case 'utf-8':
229271
ret = this.parent.utf8Write(string, this.offset + offset, maxLength);
@@ -277,6 +319,9 @@ Buffer.prototype.toString = function(encoding, start, end) {
277319
end = end + this.offset;
278320

279321
switch (encoding) {
322+
case 'hex':
323+
return this.parent.hexSlice(start, end);
324+
280325
case 'utf8':
281326
case 'utf-8':
282327
return this.parent.utf8Slice(start, end);

src/node.cc

+3
Original file line numberDiff line numberDiff line change
@@ -1084,6 +1084,8 @@ enum encoding ParseEncoding(Handle<Value> encoding_v, enum encoding _default) {
10841084
return UCS2;
10851085
} else if (strcasecmp(*encoding, "binary") == 0) {
10861086
return BINARY;
1087+
} else if (strcasecmp(*encoding, "hex") == 0) {
1088+
return HEX;
10871089
} else if (strcasecmp(*encoding, "raw") == 0) {
10881090
fprintf(stderr, "'raw' (array of integers) has been removed. "
10891091
"Use 'binary'.\n");
@@ -1134,6 +1136,7 @@ ssize_t DecodeBytes(v8::Handle<v8::Value> val, enum encoding encoding) {
11341136

11351137
if (encoding == UTF8) return str->Utf8Length();
11361138
else if (encoding == UCS2) return str->Length() * 2;
1139+
else if (encoding == HEX) return str->Length() / 2;
11371140

11381141
return str->Length();
11391142
}

src/node.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ do { \
4444
__callback##_TEM); \
4545
} while (0)
4646

47-
enum encoding {ASCII, UTF8, BASE64, UCS2, BINARY};
47+
enum encoding {ASCII, UTF8, BASE64, UCS2, BINARY, HEX};
4848
enum encoding ParseEncoding(v8::Handle<v8::Value> encoding_v,
4949
enum encoding _default = BINARY);
5050
void FatalException(v8::TryCatch &try_catch);

src/node_buffer.cc

+2
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,8 @@ static size_t ByteLength (Handle<String> string, enum encoding enc) {
8686
return base64_decoded_size(*v, v.length());
8787
} else if (enc == UCS2) {
8888
return string->Length() * 2;
89+
} else if (enc == HEX) {
90+
return string->Length() / 2;
8991
} else {
9092
return string->Length();
9193
}

test/simple/test-buffer.js

+31
Original file line numberDiff line numberDiff line change
@@ -410,3 +410,34 @@ assert.equal(12, Buffer.byteLength('Il était tué', 'binary'));
410410

411411
// slice(0,0).length === 0
412412
assert.equal(0, Buffer('hello').slice(0, 0).length);
413+
414+
// test hex toString
415+
console.log('Create hex string from buffer');
416+
var hexb = new Buffer(256);
417+
for (var i = 0; i < 256; i ++) {
418+
hexb[i] = i;
419+
}
420+
var hexStr = hexb.toString('hex');
421+
assert.equal(hexStr,
422+
'000102030405060708090a0b0c0d0e0f'+
423+
'101112131415161718191a1b1c1d1e1f'+
424+
'202122232425262728292a2b2c2d2e2f'+
425+
'303132333435363738393a3b3c3d3e3f'+
426+
'404142434445464748494a4b4c4d4e4f'+
427+
'505152535455565758595a5b5c5d5e5f'+
428+
'606162636465666768696a6b6c6d6e6f'+
429+
'707172737475767778797a7b7c7d7e7f'+
430+
'808182838485868788898a8b8c8d8e8f'+
431+
'909192939495969798999a9b9c9d9e9f'+
432+
'a0a1a2a3a4a5a6a7a8a9aaabacadaeaf'+
433+
'b0b1b2b3b4b5b6b7b8b9babbbcbdbebf'+
434+
'c0c1c2c3c4c5c6c7c8c9cacbcccdcecf'+
435+
'd0d1d2d3d4d5d6d7d8d9dadbdcdddedf'+
436+
'e0e1e2e3e4e5e6e7e8e9eaebecedeeef'+
437+
'f0f1f2f3f4f5f6f7f8f9fafbfcfdfeff');
438+
439+
console.log('Create buffer from hex string');
440+
var hexb2 = new Buffer(hexStr, 'hex');
441+
for (var i = 0; i < 256; i ++) {
442+
assert.equal(hexb2[i], hexb[i]);
443+
}

0 commit comments

Comments
 (0)