123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 |
- /**
- * @fileoverview Tests for reader.js.
- */
- goog.module('protobuf.binary.ReaderTest');
- goog.setTestOnly();
- // Note to the reader:
- // Since the reader behavior changes with the checking level some of the
- // tests in this file have to know which checking level is enable to make
- // correct assertions.
- const BufferDecoder = goog.require('protobuf.binary.BufferDecoder');
- const ByteString = goog.require('protobuf.ByteString');
- const reader = goog.require('protobuf.binary.reader');
- const {CHECK_CRITICAL_STATE} = goog.require('protobuf.internal.checks');
- const {createBufferDecoder} = goog.require('protobuf.binary.bufferDecoderHelper');
- const {encode} = goog.require('protobuf.binary.textencoding');
- const {getBoolPairs} = goog.require('protobuf.binary.boolTestPairs');
- const {getDoublePairs} = goog.require('protobuf.binary.doubleTestPairs');
- const {getFixed32Pairs} = goog.require('protobuf.binary.fixed32TestPairs');
- const {getFloatPairs} = goog.require('protobuf.binary.floatTestPairs');
- const {getInt32Pairs} = goog.require('protobuf.binary.int32TestPairs');
- const {getInt64Pairs} = goog.require('protobuf.binary.int64TestPairs');
- const {getPackedBoolPairs} = goog.require('protobuf.binary.packedBoolTestPairs');
- const {getPackedDoublePairs} = goog.require('protobuf.binary.packedDoubleTestPairs');
- const {getPackedFixed32Pairs} = goog.require('protobuf.binary.packedFixed32TestPairs');
- const {getPackedFloatPairs} = goog.require('protobuf.binary.packedFloatTestPairs');
- const {getPackedInt32Pairs} = goog.require('protobuf.binary.packedInt32TestPairs');
- const {getPackedInt64Pairs} = goog.require('protobuf.binary.packedInt64TestPairs');
- const {getPackedSfixed32Pairs} = goog.require('protobuf.binary.packedSfixed32TestPairs');
- const {getPackedSfixed64Pairs} = goog.require('protobuf.binary.packedSfixed64TestPairs');
- const {getPackedSint32Pairs} = goog.require('protobuf.binary.packedSint32TestPairs');
- const {getPackedSint64Pairs} = goog.require('protobuf.binary.packedSint64TestPairs');
- const {getPackedUint32Pairs} = goog.require('protobuf.binary.packedUint32TestPairs');
- const {getSfixed32Pairs} = goog.require('protobuf.binary.sfixed32TestPairs');
- const {getSfixed64Pairs} = goog.require('protobuf.binary.sfixed64TestPairs');
- const {getSint32Pairs} = goog.require('protobuf.binary.sint32TestPairs');
- const {getSint64Pairs} = goog.require('protobuf.binary.sint64TestPairs');
- const {getUint32Pairs} = goog.require('protobuf.binary.uint32TestPairs');
- /******************************************************************************
- * Optional FUNCTIONS
- ******************************************************************************/
- describe('Read bool does', () => {
- for (const pair of getBoolPairs()) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readBool(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readBool(
- pair.bufferDecoder, pair.bufferDecoder.startIndex());
- expect(d).toEqual(pair.boolValue);
- }
- });
- }
- });
- describe('readBytes does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder();
- expect(() => reader.readBytes(bufferDecoder, 0)).toThrow();
- });
- it('read bytes by index', () => {
- const bufferDecoder = createBufferDecoder(3, 1, 2, 3);
- const byteString = reader.readBytes(bufferDecoder, 0);
- expect(ByteString.fromArrayBuffer(new Uint8Array([1, 2, 3]).buffer))
- .toEqual(byteString);
- });
- });
- describe('readDouble does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder();
- expect(() => reader.readDouble(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getDoublePairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readDouble(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.doubleValue);
- });
- }
- });
- describe('readFixed32 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder();
- expect(() => reader.readFixed32(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getFixed32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readFixed32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.intValue);
- });
- }
- });
- describe('readFloat does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder();
- expect(() => reader.readFloat(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getFloatPairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readFloat(pair.bufferDecoder, 0);
- expect(d).toEqual(Math.fround(pair.floatValue));
- });
- }
- });
- describe('readInt32 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readInt32(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getInt32Pairs()) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readInt32(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readInt32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.intValue);
- }
- });
- }
- });
- describe('readSfixed32 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readSfixed32(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getSfixed32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readSfixed32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.intValue);
- });
- }
- });
- describe('readSfixed64 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readSfixed64(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getSfixed64Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readSfixed64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.longValue);
- });
- }
- });
- describe('readSint32 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readSint32(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getSint32Pairs()) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readSint32(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readSint32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.intValue);
- }
- });
- }
- });
- describe('readInt64 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readInt64(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getInt64Pairs()) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readInt64(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readInt64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.longValue);
- }
- });
- }
- });
- describe('readSint64 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readSint64(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getSint64Pairs()) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readSint64(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readSint64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.longValue);
- }
- });
- }
- });
- describe('readUint32 does', () => {
- it('throw exception if data is too short', () => {
- const bufferDecoder = createBufferDecoder(0x80);
- expect(() => reader.readUint32(bufferDecoder, 0)).toThrow();
- });
- for (const pair of getUint32Pairs()) {
- if (!pair.skip_reader) {
- it(`decode ${pair.name}`, () => {
- if (pair.error && CHECK_CRITICAL_STATE) {
- expect(() => reader.readUint32(pair.bufferDecoder, 0)).toThrow();
- } else {
- const d = reader.readUint32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.intValue);
- }
- });
- }
- }
- });
- /**
- *
- * @param {string} s
- * @return {!Uint8Array}
- */
- function encodeString(s) {
- if (typeof TextEncoder !== 'undefined') {
- const textEncoder = new TextEncoder('utf-8');
- return textEncoder.encode(s);
- } else {
- return encode(s);
- }
- }
- /** @param {string} s */
- function expectEncodedStringToMatch(s) {
- const array = encodeString(s);
- const length = array.length;
- if (length > 127) {
- throw new Error('Test only works for strings shorter than 128');
- }
- const encodedArray = new Uint8Array(length + 1);
- encodedArray[0] = length;
- encodedArray.set(array, 1);
- const bufferDecoder = BufferDecoder.fromArrayBuffer(encodedArray.buffer);
- expect(reader.readString(bufferDecoder, 0)).toEqual(s);
- }
- describe('readString does', () => {
- it('return empty string for zero length string', () => {
- const s = reader.readString(createBufferDecoder(0x00), 0);
- expect(s).toEqual('');
- });
- it('decode random strings', () => {
- // 1 byte strings
- expectEncodedStringToMatch('hello');
- expectEncodedStringToMatch('HELLO1!');
- // 2 byte String
- expectEncodedStringToMatch('©');
- // 3 byte string
- expectEncodedStringToMatch('❄');
- // 4 byte string
- expectEncodedStringToMatch('😁');
- });
- it('decode 1 byte strings', () => {
- for (let i = 0; i < 0x80; i++) {
- const s = String.fromCharCode(i);
- expectEncodedStringToMatch(s);
- }
- });
- it('decode 2 byte strings', () => {
- for (let i = 0xC0; i < 0x7FF; i++) {
- const s = String.fromCharCode(i);
- expectEncodedStringToMatch(s);
- }
- });
- it('decode 3 byte strings', () => {
- for (let i = 0x7FF; i < 0x8FFF; i++) {
- const s = String.fromCharCode(i);
- expectEncodedStringToMatch(s);
- }
- });
- it('throw exception on invalid bytes', () => {
- // This test will only succeed with the native TextDecoder since
- // our polyfill does not do any validation. IE10 and IE11 don't support
- // TextDecoder.
- // TODO: Remove this check once we no longer need to support IE
- if (typeof TextDecoder !== 'undefined') {
- expect(
- () => reader.readString(
- createBufferDecoder(0x01, /* invalid utf data point*/ 0xFF), 0))
- .toThrow();
- }
- });
- it('throw exception if data is too short', () => {
- const array = createBufferDecoder(0x02, '?'.charCodeAt(0));
- expect(() => reader.readString(array, 0)).toThrow();
- });
- });
- /******************************************************************************
- * REPEATED FUNCTIONS
- ******************************************************************************/
- describe('readPackedBool does', () => {
- for (const pair of getPackedBoolPairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedBool(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.boolValues);
- });
- }
- });
- describe('readPackedDouble does', () => {
- for (const pair of getPackedDoublePairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedDouble(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.doubleValues);
- });
- }
- });
- describe('readPackedFixed32 does', () => {
- for (const pair of getPackedFixed32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedFixed32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.fixed32Values);
- });
- }
- });
- describe('readPackedFloat does', () => {
- for (const pair of getPackedFloatPairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedFloat(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.floatValues);
- });
- }
- });
- describe('readPackedInt32 does', () => {
- for (const pair of getPackedInt32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedInt32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.int32Values);
- });
- }
- });
- describe('readPackedInt64 does', () => {
- for (const pair of getPackedInt64Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedInt64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.int64Values);
- });
- }
- });
- describe('readPackedSfixed32 does', () => {
- for (const pair of getPackedSfixed32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedSfixed32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.sfixed32Values);
- });
- }
- });
- describe('readPackedSfixed64 does', () => {
- for (const pair of getPackedSfixed64Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedSfixed64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.sfixed64Values);
- });
- }
- });
- describe('readPackedSint32 does', () => {
- for (const pair of getPackedSint32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedSint32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.sint32Values);
- });
- }
- });
- describe('readPackedSint64 does', () => {
- for (const pair of getPackedSint64Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedSint64(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.sint64Values);
- });
- }
- });
- describe('readPackedUint32 does', () => {
- for (const pair of getPackedUint32Pairs()) {
- it(`decode ${pair.name}`, () => {
- const d = reader.readPackedUint32(pair.bufferDecoder, 0);
- expect(d).toEqual(pair.uint32Values);
- });
- }
- });
|