Dorian 0d073fa89e Add comprehensive installation and setup documentation
- Add GETTING_STARTED.md with quick start guide and development modes
- Add INSTALL.sh automated installation script
- Add INSTALLATION_CHECKLIST.md, INSTALLATION_SUCCESS.md, and INSTALLATION_SUMMARY.md
- Add QUICK_REFERENCE.md for common commands
- Add SETUP_GUIDE.md with detailed setup instructions
- Update README.md with improved project overview
- Add did-wallet app dependencies and node_modules
2026-01-27 17:18:21 +00:00

285 lines
11 KiB
JavaScript

/* eslint-env mocha */
import chai from 'chai'
import { decode, encode } from '../cborg.js'
import { fromHex, toHex } from '../lib/byte-utils.js'
const { assert } = chai
// TODO: reject duplicate keys from encoded form
const fixtures = [
{ data: 'a0', expected: {}, type: 'map empty' },
{ data: 'a0', expected: new Map(), type: 'map empty (useMaps)', useMaps: true },
{ data: 'a1616101', expected: { a: 1 }, type: 'map 1 pair' },
{ data: 'a161316161', expected: { 1: 'a' }, type: 'map 1 pair (rev)' },
{
data: 'a1016161',
expected: toMap([[1, 'a']]),
type: 'map 1 pair (int key as Map w/ useMaps)',
useMaps: true
},
{
data: 'a243010203633132334302030463323334',
expected: toMap([[Uint8Array.from([1, 2, 3]), '123'], [Uint8Array.from([2, 3, 4]), '234']]),
type: 'map 2 pair (bytes keys Map w/ useMaps)',
useMaps: true
},
{
data: 'a1666f626a656374a16477697468a26134666e6573746564676f626a65637473a161216121',
expected: { object: { with: { 4: 'nested', objects: { '!': '!' } } } },
type: 'map nested'
},
{ // same as previous but as a Map and an int key
data: 'a1666f626a656374a16477697468a204666e6573746564676f626a65637473a161216121',
expected: toMap([['object', toMap([['with', toMap([[4, 'nested'], ['objects', toMap([['!', '!']])]])]])]]),
type: 'map nested w/ useMaps',
useMaps: true
},
{
data: 'ae636f6e651b0016db6db6db6db763736978206374656e3b0016db6db6db6db76374776f1a0001000064666976650064666f757202646e696e653aa5f702b365656967687438ff65736576656e226574687265651901f466656c6576656e426131667477656c76656fc48c6175657320c39f76c49b74652168666f75727465656ea4616664666f7572616f016174026274680368746869727465656e840203046466697665',
encode: {
one: Number.MAX_SAFE_INTEGER / 1.4,
two: 65536,
three: 500,
four: 2,
five: 0,
six: -1,
seven: -3,
eight: -256,
nine: -2784428724,
ten: Number.MIN_SAFE_INTEGER / 1.4 - 1,
eleven: new TextEncoder().encode('a1'),
twelve: 'Čaues ßvěte!',
thirteen: [2, 3, 4, 'five'],
fourteen: { o: 1, t: 2, th: 3, f: 'four' }
},
expected: {
one: Number.MAX_SAFE_INTEGER / 1.4,
six: -1,
ten: Number.MIN_SAFE_INTEGER / 1.4 - 1,
two: 65536,
five: 0,
four: 2,
nine: -2784428724,
eight: -256,
seven: -3,
three: 500,
eleven: new TextEncoder().encode('a1'),
twelve: 'Čaues ßvěte!',
fourteen: { f: 'four', o: 1, t: 2, th: 3 },
thirteen: [2, 3, 4, 'five']
},
type: 'map with complex entries',
label: '{}'
},
{
data: 'ad01636f6e65026374776f1901f46c666976652068756e647265641902586b7369782068756e647265641a00010000636269671b0016db6db6db6db76662696767657220696d696e7573206f6e6521696d696e75732074776f38ff781f6d696e75732074776f2068756e6472656420616e64206669667479207369783901f4781a6d696e757820666976652068756e6472656420616e64206f6e653901f5781a6d696e757820666976652068756e6472656420616e642074776f3aa5f702b367626967206e65673b0016db6db6db6db76a626967676572206e6567',
encode: toMap([
[2, 'two'],
[1, 'one'],
[-2, 'minus two'],
[-1, 'minus one'],
[600, 'six hundred'],
[500, 'five hundred'],
[-256, 'minus two hundred and fifty six'],
[-502, 'minux five hundred and two'],
[-501, 'minux five hundred and one'],
[65536, 'big'],
[-2784428724, 'big neg'],
[6433713753386423, 'bigger'],
[-6433713753386424, 'bigger neg']
]),
expected: toMap([
[1, 'one'],
[2, 'two'],
[500, 'five hundred'],
[600, 'six hundred'],
[65536, 'big'],
[6433713753386423, 'bigger'],
[-1, 'minus one'],
[-2, 'minus two'],
[-256, 'minus two hundred and fifty six'],
[-501, 'minux five hundred and one'],
[-502, 'minux five hundred and two'],
[-2784428724, 'big neg'],
[-6433713753386424, 'bigger neg']
]),
type: 'map with ints and negints',
useMaps: true
},
{
data: 'a44104636f6e65430102026374776f430102036574687265654301020464666f7572',
encode: toMap([
[Uint8Array.from([1, 2, 3]), 'three'],
[Uint8Array.from([4]), 'one'],
[Uint8Array.from([1, 2, 4]), 'four'],
[Uint8Array.from([1, 2, 2]), 'two']
]),
expected: toMap([
[Uint8Array.from([4]), 'one'],
[Uint8Array.from([1, 2, 2]), 'two'],
[Uint8Array.from([1, 2, 3]), 'three'],
[Uint8Array.from([1, 2, 4]), 'four']
]),
type: 'map with bytes keys',
useMaps: true
},
// testing lengths encoded as too-large ints
{ data: 'b801616101', expected: { a: 1 }, type: 'map 1 pair, length8', strict: false },
{ data: 'b90001616101', expected: { a: 1 }, type: 'map 1 pair, length16', strict: false },
{ data: 'ba00000001616101', expected: { a: 1 }, type: 'map 1 pair, length32', strict: false },
{ data: 'bb0000000000000001616101', expected: { a: 1 }, type: 'map 1 pair, length64', strict: false }
]
function toMap (arr) {
const m = new Map()
for (const [key, value] of arr) {
m.set(key, value)
}
return m
}
function entries (map) {
function nest (a) {
for (const e of a) {
e[0] = entries(e[0])
e[1] = entries(e[1])
}
return a
}
if (Object.getPrototypeOf(map) === Map.prototype) {
return nest([...map.entries()])
}
if (typeof map === 'object') {
return nest([...Object.entries(map)])
}
return map
}
describe('map', () => {
describe('decode', () => {
for (const fixture of fixtures) {
const data = fromHex(fixture.data)
it(`should decode ${fixture.type}=${fixture.label || JSON.stringify(fixture.expected)}`, () => {
let options = fixture.useMaps ? { useMaps: true } : undefined
const decoded = decode(data, options)
if (fixture.useMaps) {
assert.strictEqual(Object.getPrototypeOf(decoded), Map.prototype, 'is Map')
} else {
assert.isObject(decoded, 'is object')
}
assert.deepStrictEqual(entries(decoded), entries(fixture.expected), `decode ${fixture.type}`)
options = Object.assign({ strict: true }, options)
if (fixture.strict === false) {
assert.throws(() => decode(data, options), Error, 'CBOR decode error: integer encoded in more bytes than necessary (strict decode)')
} else {
assert.deepStrictEqual(
entries(decode(data, options)),
entries(fixture.expected),
`decode ${fixture.type}`)
}
})
it('should fail to decode very large length', () => {
assert.throws(
() => decode(fromHex('bba5f702b3a5f70201616101')),
/CBOR decode error: 64-bit integer map lengths not supported/)
})
}
it('errors', () => {
assert.throws(() => decode(fromHex('a1016161')), /non-string keys not supported \(got number\)/)
})
})
describe('encode', () => {
for (const fixture of fixtures) {
it(`should encode ${fixture.type}=${fixture.label || JSON.stringify(fixture.expected)}`, () => {
const toEncode = fixture.encode || fixture.expected
if (fixture.unsafe) {
assert.throws(encode.bind(null, toEncode), Error, /^CBOR encode error: number too large to encode \(\d+\)$/)
} else if (fixture.strict === false || fixture.roundtrip === false) {
assert.notDeepEqual(toHex(encode(toEncode)), fixture.data, `encode ${fixture.type} !strict`)
} else {
assert.strictEqual(toHex(encode(toEncode)), fixture.data, `encode ${fixture.type}`)
}
})
}
})
// mostly unnecessary, but feels good
describe('roundtrip', () => {
for (const fixture of fixtures) {
if (!fixture.unsafe && fixture.strict !== false && fixture.roundtrip !== false) {
it(`should roundtrip ${fixture.type}=${fixture.label || JSON.stringify(fixture.expected)}`, () => {
const toEncode = fixture.encode || fixture.expected
const options = fixture.useMaps ? { useMaps: true } : undefined
const rt = decode(encode(toEncode), options)
if (fixture.useMaps) {
assert.strictEqual(Object.getPrototypeOf(rt), Map.prototype, 'is Map')
} else {
assert.isObject(rt, 'is object')
}
assert.deepStrictEqual(entries(rt), entries(fixture.expected), `roundtrip ${fixture.type}`)
})
}
}
})
describe('specials', () => {
it('can decode indefinite length items', () => {
assert.deepStrictEqual(decode(fromHex('bf616f01617402ff')), { o: 1, t: 2 })
})
it('can switch off indefinite length support', () => {
assert.throws(() => decode(fromHex('bf616f01617402ff'), { allowIndefinite: false }), /indefinite/)
})
})
describe('sorting', () => {
it('sorts int map keys', () => {
assert.strictEqual(toHex(encode(new Map([[1, 1], [2, 2]]))), 'a201010202')
assert.strictEqual(toHex(encode(new Map([[2, 1], [1, 2]]))), 'a201020201')
})
it('sorts negint map keys', () => {
assert.strictEqual(toHex(encode(new Map([[-1, 1], [-2, 2]]))), 'a220012102')
assert.strictEqual(toHex(encode(new Map([[-2, 1], [-1, 2]]))), 'a220022101')
})
it('sorts bytes map keys', () => {
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([1, 2]), 1], [Uint8Array.from([2, 1]), 2]]))), 'a24201020142020102')
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([2, 1]), 1], [Uint8Array.from([1, 2]), 2]]))), 'a24201020242020101')
// shortest first
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([1, 2]), 1], [Uint8Array.from([2, 1]), 2], [Uint8Array.from([200]), 3]]))), 'a341c8034201020142020102')
})
it('sorts bytes map keys', () => {
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([1, 2]), 1], [Uint8Array.from([2, 1]), 2]]))), 'a24201020142020102')
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([2, 1]), 1], [Uint8Array.from([1, 2]), 2]]))), 'a24201020242020101')
// shortest first
assert.strictEqual(toHex(encode(new Map([[Uint8Array.from([1, 2]), 1], [Uint8Array.from([2, 1]), 2], [Uint8Array.from([200]), 3]]))), 'a341c8034201020142020102')
})
it('sorts array map keys (length only)', () => {
assert.strictEqual(toHex(encode(new Map([[[1], 1], [[1, 1], 2]]))), 'a281010182010102')
assert.strictEqual(toHex(encode(new Map([[[1, 1], 1], [[1], 2]]))), 'a281010282010101')
})
it('sorts map map keys (length only)', () => {
assert.strictEqual(toHex(encode(new Map([[{ a: 1 }, 1], [{ a: 1, b: 1 }, 2]]))), 'a2a161610101a261610161620102')
assert.strictEqual(toHex(encode(new Map([[{ a: 1, b: 1 }, 1], [{ a: 1 }, 2]]))), 'a2a161610102a261610161620101')
})
// TODO: tag keys .. but why would you do this!?
})
})