94 lines
4.0 KiB
JavaScript
Raw Normal View History

/* eslint-env mocha */
import chai from 'chai'
import { decode, encode } from '../cborg.js'
import { fromHex, toHex } from '../lib/byte-utils.js'
const { assert } = chai
const fixtures = [
{ data: '80', expected: [], type: 'array empty' },
{ data: '8102', expected: [2], type: 'array 1 compact uint' },
{ data: '8118ff', expected: [255], type: 'array 1 uint8' },
{ data: '811901f4', expected: [500], type: 'array 1 uint16' },
{ data: '811a00010000', expected: [65536], type: 'array 1 uint32' },
{ data: '811b00000000000000ff', expected: [255], type: 'array 1 uint64', strict: false },
{ data: '811b0016db6db6db6db7', expected: [Number.MAX_SAFE_INTEGER / 1.4], type: 'array 1 uint64' },
{ data: '811b001fffffffffffff', expected: [Number.MAX_SAFE_INTEGER], type: 'array 1 uint64' },
{ data: '8403040506', expected: [3, 4, 5, 6], type: 'array 4 ints' },
{
data: '8c1b0016db6db6db6db71a000100001901f40200202238ff3aa5f702b33b0016db6db6db6db74261316fc48c6175657320c39f76c49b746521',
expected: [Number.MAX_SAFE_INTEGER / 1.4, 65536, 500, 2, 0, -1, -3, -256, -2784428724, Number.MIN_SAFE_INTEGER / 1.4 - 1, new TextEncoder().encode('a1'), 'Čaues ßvěte!'],
type: 'array mixed terminals',
label: '[]'
},
{
data: '8265617272617982626f66820582666e657374656482666172726179736121',
expected: ['array', ['of', [5, ['nested', ['arrays', '!']]]]],
type: 'array nested'
},
// testing lengths encoded as too-large ints
{ data: '980403040506', expected: [3, 4, 5, 6], type: 'array 4 ints, length8', strict: false },
{ data: '99000403040506', expected: [3, 4, 5, 6], type: 'array 4 ints, length16', strict: false },
{ data: '9a0000000403040506', expected: [3, 4, 5, 6], type: 'array 4 ints, length32', strict: false },
{ data: '9b000000000000000403040506', expected: [3, 4, 5, 6], type: 'array 4 ints, length64', strict: false }
]
describe('array', () => {
describe('decode', () => {
for (const fixture of fixtures) {
const data = fromHex(fixture.data)
it(`should decode ${fixture.type}=${fixture.label || fixture.expected}`, () => {
assert.deepStrictEqual(decode(data), fixture.expected, `decode ${fixture.type}`)
if (fixture.strict === false) {
assert.throws(() => decode(data, { strict: true }), Error, 'CBOR decode error: integer encoded in more bytes than necessary (strict decode)')
} else {
assert.deepStrictEqual(decode(data, { strict: true }), fixture.expected, `decode ${fixture.type}`)
}
})
it('should fail to decode very large length', () => {
assert.throws(
() => decode(fromHex('9ba5f702b3a5f7020403040506')),
/CBOR decode error: 64-bit integer array lengths not supported/)
})
}
})
describe('encode', () => {
for (const fixture of fixtures) {
it(`should encode ${fixture.type}=${fixture.label || fixture.expected}`, () => {
if (fixture.unsafe) {
assert.throws(encode.bind(null, fixture.expected), Error, /^CBOR encode error: number too large to encode \(\d+\)$/)
} else if (fixture.strict === false) {
assert.notDeepEqual(toHex(encode(fixture.expected)), fixture.data, `encode ${fixture.type} !strict`)
} else {
assert.strictEqual(toHex(encode(fixture.expected)), fixture.data, `encode ${fixture.type}`)
}
})
}
})
// mostly unnecessary, but feels good
describe('roundtrip', () => {
for (const fixture of fixtures) {
if (!fixture.unsafe && fixture.strict !== false) {
it(`should roundtrip ${fixture.type}=${fixture.label || fixture.expected}`, () => {
assert.deepStrictEqual(decode(encode(fixture.expected)), fixture.expected, `roundtrip ${fixture.type}`)
})
}
}
})
describe('specials', () => {
it('can decode indefinite length items', () => {
assert.deepStrictEqual(decode(fromHex('9f616f6174ff')), ['o', 't'])
})
it('can switch off indefinite length support', () => {
assert.throws(() => decode(fromHex('9f616f6174ff'), { allowIndefinite: false }), /indefinite/)
})
})
})