var BufferBuilder = require('./bufferbuilder').BufferBuilder; var binaryFeatures = require('./bufferbuilder').binaryFeatures; var BinaryPack = { unpack: function (data) { var unpacker = new Unpacker(data); return unpacker.unpack(); }, pack: function (data) { var packer = new Packer(); packer.pack(data); var buffer = packer.getBuffer(); return buffer; } }; module.exports = BinaryPack; function Unpacker (data) { // Data is ArrayBuffer this.index = 0; this.dataBuffer = data; this.dataView = new Uint8Array(this.dataBuffer); this.length = this.dataBuffer.byteLength; } Unpacker.prototype.unpack = function () { var type = this.unpack_uint8(); if (type < 0x80) { return type; } else if ((type ^ 0xe0) < 0x20) { return (type ^ 0xe0) - 0x20; } var size; if ((size = type ^ 0xa0) <= 0x0f) { return this.unpack_raw(size); } else if ((size = type ^ 0xb0) <= 0x0f) { return this.unpack_string(size); } else if ((size = type ^ 0x90) <= 0x0f) { return this.unpack_array(size); } else if ((size = type ^ 0x80) <= 0x0f) { return this.unpack_map(size); } switch (type) { case 0xc0: return null; case 0xc1: return undefined; case 0xc2: return false; case 0xc3: return true; case 0xca: return this.unpack_float(); case 0xcb: return this.unpack_double(); case 0xcc: return this.unpack_uint8(); case 0xcd: return this.unpack_uint16(); case 0xce: return this.unpack_uint32(); case 0xcf: return this.unpack_uint64(); case 0xd0: return this.unpack_int8(); case 0xd1: return this.unpack_int16(); case 0xd2: return this.unpack_int32(); case 0xd3: return this.unpack_int64(); case 0xd4: return undefined; case 0xd5: return undefined; case 0xd6: return undefined; case 0xd7: return undefined; case 0xd8: size = this.unpack_uint16(); return this.unpack_string(size); case 0xd9: size = this.unpack_uint32(); return this.unpack_string(size); case 0xda: size = this.unpack_uint16(); return this.unpack_raw(size); case 0xdb: size = this.unpack_uint32(); return this.unpack_raw(size); case 0xdc: size = this.unpack_uint16(); return this.unpack_array(size); case 0xdd: size = this.unpack_uint32(); return this.unpack_array(size); case 0xde: size = this.unpack_uint16(); return this.unpack_map(size); case 0xdf: size = this.unpack_uint32(); return this.unpack_map(size); } }; Unpacker.prototype.unpack_uint8 = function () { var byte = this.dataView[this.index] & 0xff; this.index++; return byte; }; Unpacker.prototype.unpack_uint16 = function () { var bytes = this.read(2); var uint16 = ((bytes[0] & 0xff) * 256) + (bytes[1] & 0xff); this.index += 2; return uint16; }; Unpacker.prototype.unpack_uint32 = function () { var bytes = this.read(4); var uint32 = ((bytes[0] * 256 + bytes[1]) * 256 + bytes[2]) * 256 + bytes[3]; this.index += 4; return uint32; }; Unpacker.prototype.unpack_uint64 = function () { var bytes = this.read(8); var uint64 = ((((((bytes[0] * 256 + bytes[1]) * 256 + bytes[2]) * 256 + bytes[3]) * 256 + bytes[4]) * 256 + bytes[5]) * 256 + bytes[6]) * 256 + bytes[7]; this.index += 8; return uint64; }; Unpacker.prototype.unpack_int8 = function () { var uint8 = this.unpack_uint8(); return (uint8 < 0x80) ? uint8 : uint8 - (1 << 8); }; Unpacker.prototype.unpack_int16 = function () { var uint16 = this.unpack_uint16(); return (uint16 < 0x8000) ? uint16 : uint16 - (1 << 16); }; Unpacker.prototype.unpack_int32 = function () { var uint32 = this.unpack_uint32(); return (uint32 < Math.pow(2, 31)) ? uint32 : uint32 - Math.pow(2, 32); }; Unpacker.prototype.unpack_int64 = function () { var uint64 = this.unpack_uint64(); return (uint64 < Math.pow(2, 63)) ? uint64 : uint64 - Math.pow(2, 64); }; Unpacker.prototype.unpack_raw = function (size) { if (this.length < this.index + size) { throw new Error('BinaryPackFailure: index is out of range' + ' ' + this.index + ' ' + size + ' ' + this.length); } var buf = this.dataBuffer.slice(this.index, this.index + size); this.index += size; // buf = util.bufferToString(buf); return buf; }; Unpacker.prototype.unpack_string = function (size) { var bytes = this.read(size); var i = 0; var str = ''; var c; var code; while (i < size) { c = bytes[i]; if (c < 128) { str += String.fromCharCode(c); i++; } else if ((c ^ 0xc0) < 32) { code = ((c ^ 0xc0) << 6) | (bytes[i + 1] & 63); str += String.fromCharCode(code); i += 2; } else { code = ((c & 15) << 12) | ((bytes[i + 1] & 63) << 6) | (bytes[i + 2] & 63); str += String.fromCharCode(code); i += 3; } } this.index += size; return str; }; Unpacker.prototype.unpack_array = function (size) { var objects = new Array(size); for (var i = 0; i < size; i++) { objects[i] = this.unpack(); } return objects; }; Unpacker.prototype.unpack_map = function (size) { var map = {}; for (var i = 0; i < size; i++) { var key = this.unpack(); var value = this.unpack(); map[key] = value; } return map; }; Unpacker.prototype.unpack_float = function () { var uint32 = this.unpack_uint32(); var sign = uint32 >> 31; var exp = ((uint32 >> 23) & 0xff) - 127; var fraction = (uint32 & 0x7fffff) | 0x800000; return (sign === 0 ? 1 : -1) * fraction * Math.pow(2, exp - 23); }; Unpacker.prototype.unpack_double = function () { var h32 = this.unpack_uint32(); var l32 = this.unpack_uint32(); var sign = h32 >> 31; var exp = ((h32 >> 20) & 0x7ff) - 1023; var hfrac = (h32 & 0xfffff) | 0x100000; var frac = hfrac * Math.pow(2, exp - 20) + l32 * Math.pow(2, exp - 52); return (sign === 0 ? 1 : -1) * frac; }; Unpacker.prototype.read = function (length) { var j = this.index; if (j + length <= this.length) { return this.dataView.subarray(j, j + length); } else { throw new Error('BinaryPackFailure: read index out of range'); } }; function Packer () { this.bufferBuilder = new BufferBuilder(); } Packer.prototype.getBuffer = function () { return this.bufferBuilder.getBuffer(); }; Packer.prototype.pack = function (value) { var type = typeof (value); if (type === 'string') { this.pack_string(value); } else if (type === 'number') { if (Math.floor(value) === value) { this.pack_integer(value); } else { this.pack_double(value); } } else if (type === 'boolean') { if (value === true) { this.bufferBuilder.append(0xc3); } else if (value === false) { this.bufferBuilder.append(0xc2); } } else if (type === 'undefined') { this.bufferBuilder.append(0xc0); } else if (type === 'object') { if (value === null) { this.bufferBuilder.append(0xc0); } else { var constructor = value.constructor; if (constructor == Array) { this.pack_array(value); } else if (constructor == Blob || constructor == File || value instanceof Blob || value instanceof File) { this.pack_bin(value); } else if (constructor == ArrayBuffer) { if (binaryFeatures.useArrayBufferView) { this.pack_bin(new Uint8Array(value)); } else { this.pack_bin(value); } } else if ('BYTES_PER_ELEMENT' in value) { if (binaryFeatures.useArrayBufferView) { this.pack_bin(new Uint8Array(value.buffer)); } else { this.pack_bin(value.buffer); } } else if ((constructor == Object) || (constructor.toString().startsWith('class'))) { this.pack_object(value); } else if (constructor == Date) { this.pack_string(value.toString()); } else if (typeof value.toBinaryPack === 'function') { this.bufferBuilder.append(value.toBinaryPack()); } else { throw new Error('Type "' + constructor.toString() + '" not yet supported'); } } } else { throw new Error('Type "' + type + '" not yet supported'); } this.bufferBuilder.flush(); }; Packer.prototype.pack_bin = function (blob) { var length = blob.length || blob.byteLength || blob.size; if (length <= 0x0f) { this.pack_uint8(0xa0 + length); } else if (length <= 0xffff) { this.bufferBuilder.append(0xda); this.pack_uint16(length); } else if (length <= 0xffffffff) { this.bufferBuilder.append(0xdb); this.pack_uint32(length); } else { throw new Error('Invalid length'); } this.bufferBuilder.append(blob); }; Packer.prototype.pack_string = function (str) { var length = utf8Length(str); if (length <= 0x0f) { this.pack_uint8(0xb0 + length); } else if (length <= 0xffff) { this.bufferBuilder.append(0xd8); this.pack_uint16(length); } else if (length <= 0xffffffff) { this.bufferBuilder.append(0xd9); this.pack_uint32(length); } else { throw new Error('Invalid length'); } this.bufferBuilder.append(str); }; Packer.prototype.pack_array = function (ary) { var length = ary.length; if (length <= 0x0f) { this.pack_uint8(0x90 + length); } else if (length <= 0xffff) { this.bufferBuilder.append(0xdc); this.pack_uint16(length); } else if (length <= 0xffffffff) { this.bufferBuilder.append(0xdd); this.pack_uint32(length); } else { throw new Error('Invalid length'); } for (var i = 0; i < length; i++) { this.pack(ary[i]); } }; Packer.prototype.pack_integer = function (num) { if (num >= -0x20 && num <= 0x7f) { this.bufferBuilder.append(num & 0xff); } else if (num >= 0x00 && num <= 0xff) { this.bufferBuilder.append(0xcc); this.pack_uint8(num); } else if (num >= -0x80 && num <= 0x7f) { this.bufferBuilder.append(0xd0); this.pack_int8(num); } else if (num >= 0x0000 && num <= 0xffff) { this.bufferBuilder.append(0xcd); this.pack_uint16(num); } else if (num >= -0x8000 && num <= 0x7fff) { this.bufferBuilder.append(0xd1); this.pack_int16(num); } else if (num >= 0x00000000 && num <= 0xffffffff) { this.bufferBuilder.append(0xce); this.pack_uint32(num); } else if (num >= -0x80000000 && num <= 0x7fffffff) { this.bufferBuilder.append(0xd2); this.pack_int32(num); } else if (num >= -0x8000000000000000 && num <= 0x7FFFFFFFFFFFFFFF) { this.bufferBuilder.append(0xd3); this.pack_int64(num); } else if (num >= 0x0000000000000000 && num <= 0xFFFFFFFFFFFFFFFF) { this.bufferBuilder.append(0xcf); this.pack_uint64(num); } else { throw new Error('Invalid integer'); } }; Packer.prototype.pack_double = function (num) { var sign = 0; if (num < 0) { sign = 1; num = -num; } var exp = Math.floor(Math.log(num) / Math.LN2); var frac0 = num / Math.pow(2, exp) - 1; var frac1 = Math.floor(frac0 * Math.pow(2, 52)); var b32 = Math.pow(2, 32); var h32 = (sign << 31) | ((exp + 1023) << 20) | (frac1 / b32) & 0x0fffff; var l32 = frac1 % b32; this.bufferBuilder.append(0xcb); this.pack_int32(h32); this.pack_int32(l32); }; Packer.prototype.pack_object = function (obj) { var keys = Object.keys(obj); var length = keys.length; if (length <= 0x0f) { this.pack_uint8(0x80 + length); } else if (length <= 0xffff) { this.bufferBuilder.append(0xde); this.pack_uint16(length); } else if (length <= 0xffffffff) { this.bufferBuilder.append(0xdf); this.pack_uint32(length); } else { throw new Error('Invalid length'); } for (var prop in obj) { if (obj.hasOwnProperty(prop)) { this.pack(prop); this.pack(obj[prop]); } } }; Packer.prototype.pack_uint8 = function (num) { this.bufferBuilder.append(num); }; Packer.prototype.pack_uint16 = function (num) { this.bufferBuilder.append(num >> 8); this.bufferBuilder.append(num & 0xff); }; Packer.prototype.pack_uint32 = function (num) { var n = num & 0xffffffff; this.bufferBuilder.append((n & 0xff000000) >>> 24); this.bufferBuilder.append((n & 0x00ff0000) >>> 16); this.bufferBuilder.append((n & 0x0000ff00) >>> 8); this.bufferBuilder.append((n & 0x000000ff)); }; Packer.prototype.pack_uint64 = function (num) { var high = num / Math.pow(2, 32); var low = num % Math.pow(2, 32); this.bufferBuilder.append((high & 0xff000000) >>> 24); this.bufferBuilder.append((high & 0x00ff0000) >>> 16); this.bufferBuilder.append((high & 0x0000ff00) >>> 8); this.bufferBuilder.append((high & 0x000000ff)); this.bufferBuilder.append((low & 0xff000000) >>> 24); this.bufferBuilder.append((low & 0x00ff0000) >>> 16); this.bufferBuilder.append((low & 0x0000ff00) >>> 8); this.bufferBuilder.append((low & 0x000000ff)); }; Packer.prototype.pack_int8 = function (num) { this.bufferBuilder.append(num & 0xff); }; Packer.prototype.pack_int16 = function (num) { this.bufferBuilder.append((num & 0xff00) >> 8); this.bufferBuilder.append(num & 0xff); }; Packer.prototype.pack_int32 = function (num) { this.bufferBuilder.append((num >>> 24) & 0xff); this.bufferBuilder.append((num & 0x00ff0000) >>> 16); this.bufferBuilder.append((num & 0x0000ff00) >>> 8); this.bufferBuilder.append((num & 0x000000ff)); }; Packer.prototype.pack_int64 = function (num) { var high = Math.floor(num / Math.pow(2, 32)); var low = num % Math.pow(2, 32); this.bufferBuilder.append((high & 0xff000000) >>> 24); this.bufferBuilder.append((high & 0x00ff0000) >>> 16); this.bufferBuilder.append((high & 0x0000ff00) >>> 8); this.bufferBuilder.append((high & 0x000000ff)); this.bufferBuilder.append((low & 0xff000000) >>> 24); this.bufferBuilder.append((low & 0x00ff0000) >>> 16); this.bufferBuilder.append((low & 0x0000ff00) >>> 8); this.bufferBuilder.append((low & 0x000000ff)); }; function _utf8Replace (m) { var code = m.charCodeAt(0); if (code <= 0x7ff) return '00'; if (code <= 0xffff) return '000'; if (code <= 0x1fffff) return '0000'; if (code <= 0x3ffffff) return '00000'; return '000000'; } function utf8Length (str) { if (str.length > 600) { // Blob method faster for large strings return (new Blob([str])).size; } else { return str.replace(/[^\u0000-\u007F]/g, _utf8Replace).length; } }