/* 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: '8601f5f4f6f720', expected: [1, true, false, null, undefined, -1], type: 'array of float specials' }, { data: 'f93800', expected: 0.5, type: 'float16' }, { data: 'f9b800', expected: -0.5, type: 'float16' }, { data: 'fa33c00000', expected: 8.940696716308594e-08, type: 'float32' }, { data: 'fab3c00000', expected: -8.940696716308594e-08, type: 'float32' }, { data: 'fb3ff199999999999a', expected: 1.1, type: 'float64' }, { data: 'fbbff199999999999a', expected: -1.1, type: 'float64' }, { data: 'fb3ff1c71c71c71c72', expected: 1.11111111111111111111111111111, type: 'float64' }, // eslint-disable-line { data: 'fb0000000000000002', expected: 1e-323, type: 'float64' }, { data: 'fb8000000000000002', expected: -1e-323, type: 'float64' }, { data: 'fb3fefffffffffffff', expected: 0.9999999999999999, type: 'float64' }, { data: 'fbbfefffffffffffff', expected: -0.9999999999999999, type: 'float64' }, { data: 'f97c00', expected: Infinity, type: 'Infinity' }, // special CBOR token for -Infinity { data: 'fb7ff0000000000000', expected: Infinity, type: 'Infinity', strict: false }, // an IEEE 754 representation of Infinity { data: 'f9fc00', expected: -Infinity, type: '-Infinity' }, // special CBOR token for -Infinity { data: 'fbfff0000000000000', expected: -Infinity, type: '-Infinity', strict: false }, // an IEEE 754 representation of Infinity { data: 'f97e00', expected: NaN, type: 'NaN' }, // special CBOR token for NaN { data: 'f97ff8', expected: NaN, type: 'NaN', strict: false }, // one of the many IEEE 754 representations of NaN { data: 'fa7ff80000', expected: NaN, type: 'NaN', strict: false }, { data: 'fb7ff8000000000000', expected: NaN, type: 'NaN', strict: false }, { data: 'fb7ff8cafedeadbeef', expected: NaN, type: 'NaN', strict: false }, // yep, that's NaN too { data: 'fb40f4241a31a5a515', expected: 82497.63712086187, type: 'float64' } ] describe('float', () => { describe('decode', () => { for (const fixture of fixtures) { const data = fromHex(fixture.data) it(`should decode ${fixture.type}=${fixture.expected}`, () => { assert.deepStrictEqual(decode(data), fixture.expected, `decode ${fixture.type}`) assert.deepStrictEqual(decode(data, { strict: true }), fixture.expected, `decode ${fixture.type}`) }) } }) it('error', () => { // minor number 28, too high for uint assert.throws(() => decode(fromHex('f80000')), Error, 'simple values are not supported') assert.throws(() => decode(fromHex('f900')), Error, 'not enough data for float16') assert.throws(() => decode(fromHex('fa0000')), Error, 'not enough data for float32') assert.throws(() => decode(fromHex('fb00000000')), Error, 'not enough data for float64') }) describe('encode', () => { for (const fixture of fixtures) { if (fixture.strict !== false) { it(`should encode ${fixture.type}=${fixture.expected}`, () => { assert.strictEqual(toHex(encode(fixture.expected)), fixture.data, `encode ${fixture.type}`) }) } } }) describe('encode float64', () => { for (const fixture of fixtures) { if (fixture.type.startsWith('float')) { it(`should encode ${fixture.type}=${fixture.expected}`, () => { const encoded = encode(fixture.expected, { float64: true }) assert.strictEqual(encoded.length, 9) // always encode as 9 bytes, regardless of size assert.strictEqual(encoded[0], 0xfb) assert.strictEqual(decode(encoded), fixture.expected, `encode float64 ${fixture.type}`) }) } } }) describe('roundtrip', () => { for (const fixture of fixtures) { if (!fixture.unsafe && fixture.strict !== false) { it(`should roundtrip ${fixture.type}=${fixture.expected}`, () => { assert.deepStrictEqual(decode(encode(fixture.expected)), fixture.expected, `roundtrip ${fixture.type}`) }) } } }) describe('specials', () => { // This is a bit of a hack, the CBOR is invalid because it's a standard fixed-length array // followed by a BREAK, which should normally error ("too many terminals"), but we want to // exercise the allowIndefinite switch in the major-7 decode and it should error before it // even gets to looking at terminals and whether the tokens make sense. it('indefinite length switch fails on BREAK', () => { // sanity check, BREAK doesn't belong there assert.throws(() => decode(Uint8Array.from([131, 1, 2, 0xff])), /unexpected break to lengthed array/) // throw earlier because we're disallowing BREAK entirely assert.throws(() => decode(Uint8Array.from([131, 1, 2, 0xff]), { allowIndefinite: false }), /indefinite/) }) it('can switch off undefined support', () => { assert.deepStrictEqual(decode(fromHex('f7')), undefined) assert.throws(() => decode(fromHex('f7'), { allowUndefined: false }), /undefined/) assert.deepStrictEqual(decode(fromHex('830102f7')), [1, 2, undefined]) assert.throws(() => decode(fromHex('830102f7'), { allowUndefined: false }), /undefined/) }) it('can coerce undefined to null', () => { assert.deepStrictEqual(decode(fromHex('f7'), { coerceUndefinedToNull: false }), undefined) assert.deepStrictEqual(decode(fromHex('f7'), { coerceUndefinedToNull: true }), null) assert.deepStrictEqual(decode(fromHex('830102f7'), { coerceUndefinedToNull: false }), [1, 2, undefined]) assert.deepStrictEqual(decode(fromHex('830102f7'), { coerceUndefinedToNull: true }), [1, 2, null]) }) it('can switch off Infinity support', () => { assert.deepStrictEqual(decode(fromHex('830102f97c00')), [1, 2, Infinity]) assert.deepStrictEqual(decode(fromHex('830102f9fc00')), [1, 2, -Infinity]) assert.throws(() => decode(fromHex('830102f97c00'), { allowInfinity: false }), /Infinity/) assert.throws(() => decode(fromHex('830102f9fc00'), { allowInfinity: false }), /Infinity/) for (const fixture of fixtures.filter((f) => f.type.endsWith('Infinity'))) { assert.throws(() => decode(fromHex(fixture.data), { allowInfinity: false }), /Infinity/) } }) it('can switch off NaN support', () => { assert.deepStrictEqual(decode(fromHex('830102f97e00')), [1, 2, NaN]) assert.throws(() => decode(fromHex('830102f97e00'), { allowNaN: false }), /NaN/) for (const fixture of fixtures.filter((f) => f.type === 'NaN')) { assert.throws(() => decode(fromHex(fixture.data), { allowNaN: false }), /NaN/) } }) }) })