|  | // Protocol Buffers - Google's data interchange format | 
|  | // Copyright 2008 Google Inc.  All rights reserved. | 
|  | // https://developers.google.com/protocol-buffers/ | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions are | 
|  | // met: | 
|  | // | 
|  | //     * Redistributions of source code must retain the above copyright | 
|  | // notice, this list of conditions and the following disclaimer. | 
|  | //     * Redistributions in binary form must reproduce the above | 
|  | // copyright notice, this list of conditions and the following disclaimer | 
|  | // in the documentation and/or other materials provided with the | 
|  | // distribution. | 
|  | //     * Neither the name of Google Inc. nor the names of its | 
|  | // contributors may be used to endorse or promote products derived from | 
|  | // this software without specific prior written permission. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
|  | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
|  | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
|  | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
|  | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|  | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|  | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|  | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | /** | 
|  | * @fileoverview Test cases for jspb's binary protocol buffer decoder. | 
|  | * | 
|  | * There are two particular magic numbers that need to be pointed out - | 
|  | * 2^64-1025 is the largest number representable as both a double and an | 
|  | * unsigned 64-bit integer, and 2^63-513 is the largest number representable as | 
|  | * both a double and a signed 64-bit integer. | 
|  | * | 
|  | * Test suite is written using Jasmine -- see http://jasmine.github.io/ | 
|  | * | 
|  | * @author aappleby@google.com (Austin Appleby) | 
|  | */ | 
|  |  | 
|  | goog.require('goog.testing.asserts'); | 
|  | goog.require('jspb.BinaryConstants'); | 
|  | goog.require('jspb.BinaryDecoder'); | 
|  | goog.require('jspb.BinaryEncoder'); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests encoding and decoding of unsigned types. | 
|  | * @param {Function} readValue | 
|  | * @param {Function} writeValue | 
|  | * @param {number} epsilon | 
|  | * @param {number} upperLimit | 
|  | * @param {Function} filter | 
|  | * @suppress {missingProperties|visibility} | 
|  | */ | 
|  | function doTestUnsignedValue(readValue, | 
|  | writeValue, epsilon, upperLimit, filter) { | 
|  | var encoder = new jspb.BinaryEncoder(); | 
|  |  | 
|  | // Encode zero and limits. | 
|  | writeValue.call(encoder, filter(0)); | 
|  | writeValue.call(encoder, filter(epsilon)); | 
|  | writeValue.call(encoder, filter(upperLimit)); | 
|  |  | 
|  | // Encode positive values. | 
|  | for (var cursor = epsilon; cursor < upperLimit; cursor *= 1.1) { | 
|  | writeValue.call(encoder, filter(cursor)); | 
|  | } | 
|  |  | 
|  | var decoder = jspb.BinaryDecoder.alloc(encoder.end()); | 
|  |  | 
|  | // Check zero and limits. | 
|  | assertEquals(filter(0), readValue.call(decoder)); | 
|  | assertEquals(filter(epsilon), readValue.call(decoder)); | 
|  | assertEquals(filter(upperLimit), readValue.call(decoder)); | 
|  |  | 
|  | // Check positive values. | 
|  | for (var cursor = epsilon; cursor < upperLimit; cursor *= 1.1) { | 
|  | if (filter(cursor) != readValue.call(decoder)) throw 'fail!'; | 
|  | } | 
|  |  | 
|  | // Encoding values outside the valid range should assert. | 
|  | assertThrows(function() {writeValue.call(encoder, -1);}); | 
|  | assertThrows(function() {writeValue.call(encoder, upperLimit * 1.1);}); | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests encoding and decoding of signed types. | 
|  | * @param {Function} readValue | 
|  | * @param {Function} writeValue | 
|  | * @param {number} epsilon | 
|  | * @param {number} lowerLimit | 
|  | * @param {number} upperLimit | 
|  | * @param {Function} filter | 
|  | * @suppress {missingProperties} | 
|  | */ | 
|  | function doTestSignedValue(readValue, | 
|  | writeValue, epsilon, lowerLimit, upperLimit, filter) { | 
|  | var encoder = new jspb.BinaryEncoder(); | 
|  |  | 
|  | // Encode zero and limits. | 
|  | writeValue.call(encoder, filter(lowerLimit)); | 
|  | writeValue.call(encoder, filter(-epsilon)); | 
|  | writeValue.call(encoder, filter(0)); | 
|  | writeValue.call(encoder, filter(epsilon)); | 
|  | writeValue.call(encoder, filter(upperLimit)); | 
|  |  | 
|  | var inputValues = []; | 
|  |  | 
|  | // Encode negative values. | 
|  | for (var cursor = lowerLimit; cursor < -epsilon; cursor /= 1.1) { | 
|  | var val = filter(cursor); | 
|  | writeValue.call(encoder, val); | 
|  | inputValues.push(val); | 
|  | } | 
|  |  | 
|  | // Encode positive values. | 
|  | for (var cursor = epsilon; cursor < upperLimit; cursor *= 1.1) { | 
|  | var val = filter(cursor); | 
|  | writeValue.call(encoder, val); | 
|  | inputValues.push(val); | 
|  | } | 
|  |  | 
|  | var decoder = jspb.BinaryDecoder.alloc(encoder.end()); | 
|  |  | 
|  | // Check zero and limits. | 
|  | assertEquals(filter(lowerLimit), readValue.call(decoder)); | 
|  | assertEquals(filter(-epsilon), readValue.call(decoder)); | 
|  | assertEquals(filter(0), readValue.call(decoder)); | 
|  | assertEquals(filter(epsilon), readValue.call(decoder)); | 
|  | assertEquals(filter(upperLimit), readValue.call(decoder)); | 
|  |  | 
|  | // Verify decoded values. | 
|  | for (var i = 0; i < inputValues.length; i++) { | 
|  | assertEquals(inputValues[i], readValue.call(decoder)); | 
|  | } | 
|  |  | 
|  | // Encoding values outside the valid range should assert. | 
|  | assertThrows(function() {writeValue.call(encoder, lowerLimit * 1.1);}); | 
|  | assertThrows(function() {writeValue.call(encoder, upperLimit * 1.1);}); | 
|  | } | 
|  |  | 
|  | describe('binaryDecoderTest', function() { | 
|  | /** | 
|  | * Tests the decoder instance cache. | 
|  | */ | 
|  | it('testInstanceCache', /** @suppress {visibility} */ function() { | 
|  | // Empty the instance caches. | 
|  | jspb.BinaryDecoder.instanceCache_ = []; | 
|  |  | 
|  | // Allocating and then freeing a decoder should put it in the instance | 
|  | // cache. | 
|  | jspb.BinaryDecoder.alloc().free(); | 
|  |  | 
|  | assertEquals(1, jspb.BinaryDecoder.instanceCache_.length); | 
|  |  | 
|  | // Allocating and then freeing three decoders should leave us with three in | 
|  | // the cache. | 
|  |  | 
|  | var decoder1 = jspb.BinaryDecoder.alloc(); | 
|  | var decoder2 = jspb.BinaryDecoder.alloc(); | 
|  | var decoder3 = jspb.BinaryDecoder.alloc(); | 
|  | decoder1.free(); | 
|  | decoder2.free(); | 
|  | decoder3.free(); | 
|  |  | 
|  | assertEquals(3, jspb.BinaryDecoder.instanceCache_.length); | 
|  | }); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests reading 64-bit integers as hash strings. | 
|  | */ | 
|  | it('testHashStrings', function() { | 
|  | var encoder = new jspb.BinaryEncoder(); | 
|  |  | 
|  | var hashA = String.fromCharCode(0x00, 0x00, 0x00, 0x00, | 
|  | 0x00, 0x00, 0x00, 0x00); | 
|  | var hashB = String.fromCharCode(0x12, 0x34, 0x00, 0x00, | 
|  | 0x00, 0x00, 0x00, 0x00); | 
|  | var hashC = String.fromCharCode(0x12, 0x34, 0x56, 0x78, | 
|  | 0x87, 0x65, 0x43, 0x21); | 
|  | var hashD = String.fromCharCode(0xFF, 0xFF, 0xFF, 0xFF, | 
|  | 0xFF, 0xFF, 0xFF, 0xFF); | 
|  |  | 
|  | encoder.writeVarintHash64(hashA); | 
|  | encoder.writeVarintHash64(hashB); | 
|  | encoder.writeVarintHash64(hashC); | 
|  | encoder.writeVarintHash64(hashD); | 
|  |  | 
|  | encoder.writeFixedHash64(hashA); | 
|  | encoder.writeFixedHash64(hashB); | 
|  | encoder.writeFixedHash64(hashC); | 
|  | encoder.writeFixedHash64(hashD); | 
|  |  | 
|  | var decoder = jspb.BinaryDecoder.alloc(encoder.end()); | 
|  |  | 
|  | assertEquals(hashA, decoder.readVarintHash64()); | 
|  | assertEquals(hashB, decoder.readVarintHash64()); | 
|  | assertEquals(hashC, decoder.readVarintHash64()); | 
|  | assertEquals(hashD, decoder.readVarintHash64()); | 
|  |  | 
|  | assertEquals(hashA, decoder.readFixedHash64()); | 
|  | assertEquals(hashB, decoder.readFixedHash64()); | 
|  | assertEquals(hashC, decoder.readFixedHash64()); | 
|  | assertEquals(hashD, decoder.readFixedHash64()); | 
|  | }); | 
|  |  | 
|  | /** | 
|  | * Tests reading and writing large strings | 
|  | */ | 
|  | it('testLargeStrings', function() { | 
|  | var encoder = new jspb.BinaryEncoder(); | 
|  |  | 
|  | var len = 150000; | 
|  | var long_string = ''; | 
|  | for (var i = 0; i < len; i++) { | 
|  | long_string += 'a'; | 
|  | } | 
|  |  | 
|  | encoder.writeString(long_string); | 
|  |  | 
|  | var decoder = jspb.BinaryDecoder.alloc(encoder.end()); | 
|  |  | 
|  | assertEquals(long_string, decoder.readString(len)); | 
|  | }); | 
|  |  | 
|  | /** | 
|  | * Test encoding and decoding utf-8. | 
|  | */ | 
|  | it('testUtf8', function() { | 
|  | var encoder = new jspb.BinaryEncoder(); | 
|  |  | 
|  | var ascii = "ASCII should work in 3, 2, 1..."; | 
|  | var utf8_two_bytes = "©"; | 
|  | var utf8_three_bytes = "❄"; | 
|  | var utf8_four_bytes = "😁"; | 
|  |  | 
|  | encoder.writeString(ascii); | 
|  | encoder.writeString(utf8_two_bytes); | 
|  | encoder.writeString(utf8_three_bytes); | 
|  | encoder.writeString(utf8_four_bytes); | 
|  |  | 
|  | var decoder = jspb.BinaryDecoder.alloc(encoder.end()); | 
|  |  | 
|  | assertEquals(ascii, decoder.readString(ascii.length)); | 
|  | assertEquals(utf8_two_bytes, decoder.readString(utf8_two_bytes.length)); | 
|  | assertEquals(utf8_three_bytes, decoder.readString(utf8_three_bytes.length)); | 
|  | assertEquals(utf8_four_bytes, decoder.readString(utf8_four_bytes.length)); | 
|  | }); | 
|  |  | 
|  | /** | 
|  | * Verifies that misuse of the decoder class triggers assertions. | 
|  | */ | 
|  | it('testDecodeErrors', function() { | 
|  | // Reading a value past the end of the stream should trigger an assertion. | 
|  | var decoder = jspb.BinaryDecoder.alloc([0, 1, 2]); | 
|  | assertThrows(function() {decoder.readUint64()}); | 
|  |  | 
|  | // Overlong varints should trigger assertions. | 
|  | decoder.setBlock([255, 255, 255, 255, 255, 255, | 
|  | 255, 255, 255, 255, 255, 0]); | 
|  | assertThrows(function() {decoder.readUnsignedVarint64()}); | 
|  | decoder.reset(); | 
|  | assertThrows(function() {decoder.readSignedVarint64()}); | 
|  | decoder.reset(); | 
|  | assertThrows(function() {decoder.readZigzagVarint64()}); | 
|  | decoder.reset(); | 
|  | assertThrows(function() {decoder.readUnsignedVarint32()}); | 
|  | }); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests encoding and decoding of unsigned integers. | 
|  | */ | 
|  | it('testUnsignedIntegers', function() { | 
|  | doTestUnsignedValue( | 
|  | jspb.BinaryDecoder.prototype.readUint8, | 
|  | jspb.BinaryEncoder.prototype.writeUint8, | 
|  | 1, 0xFF, Math.round); | 
|  |  | 
|  | doTestUnsignedValue( | 
|  | jspb.BinaryDecoder.prototype.readUint16, | 
|  | jspb.BinaryEncoder.prototype.writeUint16, | 
|  | 1, 0xFFFF, Math.round); | 
|  |  | 
|  | doTestUnsignedValue( | 
|  | jspb.BinaryDecoder.prototype.readUint32, | 
|  | jspb.BinaryEncoder.prototype.writeUint32, | 
|  | 1, 0xFFFFFFFF, Math.round); | 
|  |  | 
|  | doTestUnsignedValue( | 
|  | jspb.BinaryDecoder.prototype.readUint64, | 
|  | jspb.BinaryEncoder.prototype.writeUint64, | 
|  | 1, Math.pow(2, 64) - 1025, Math.round); | 
|  | }); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests encoding and decoding of signed integers. | 
|  | */ | 
|  | it('testSignedIntegers', function() { | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readInt8, | 
|  | jspb.BinaryEncoder.prototype.writeInt8, | 
|  | 1, -0x80, 0x7F, Math.round); | 
|  |  | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readInt16, | 
|  | jspb.BinaryEncoder.prototype.writeInt16, | 
|  | 1, -0x8000, 0x7FFF, Math.round); | 
|  |  | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readInt32, | 
|  | jspb.BinaryEncoder.prototype.writeInt32, | 
|  | 1, -0x80000000, 0x7FFFFFFF, Math.round); | 
|  |  | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readInt64, | 
|  | jspb.BinaryEncoder.prototype.writeInt64, | 
|  | 1, -Math.pow(2, 63), Math.pow(2, 63) - 513, Math.round); | 
|  | }); | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Tests encoding and decoding of floats. | 
|  | */ | 
|  | it('testFloats', function() { | 
|  | /** | 
|  | * @param {number} x | 
|  | * @return {number} | 
|  | */ | 
|  | function truncate(x) { | 
|  | var temp = new Float32Array(1); | 
|  | temp[0] = x; | 
|  | return temp[0]; | 
|  | } | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readFloat, | 
|  | jspb.BinaryEncoder.prototype.writeFloat, | 
|  | jspb.BinaryConstants.FLOAT32_EPS, | 
|  | -jspb.BinaryConstants.FLOAT32_MAX, | 
|  | jspb.BinaryConstants.FLOAT32_MAX, | 
|  | truncate); | 
|  |  | 
|  | doTestSignedValue( | 
|  | jspb.BinaryDecoder.prototype.readDouble, | 
|  | jspb.BinaryEncoder.prototype.writeDouble, | 
|  | jspb.BinaryConstants.FLOAT64_EPS * 10, | 
|  | -jspb.BinaryConstants.FLOAT64_MAX, | 
|  | jspb.BinaryConstants.FLOAT64_MAX, | 
|  | function(x) { return x; }); | 
|  | }); | 
|  | }); |