test('TernarySearchTree - set', function () { let trie = TernarySearchTree.forStrings<number>(); trie.set('foobar', 1); trie.set('foobaz', 2); assertTernarySearchTree(trie, ['foobar', 1], ['foobaz', 2]); // longer trie = TernarySearchTree.forStrings<number>(); trie.set('foobar', 1); trie.set('fooba', 2); assertTernarySearchTree(trie, ['foobar', 1], ['fooba', 2]); // shorter trie = TernarySearchTree.forStrings<number>(); trie.set('foo', 1); trie.set('foo', 2); assertTernarySearchTree(trie, ['foo', 2]); trie = TernarySearchTree.forStrings<number>(); trie.set('foo', 1); trie.set('foobar', 2); trie.set('bar', 3); trie.set('foob', 4); trie.set('bazz', 5); assertTernarySearchTree(trie, ['foo', 1], ['foobar', 2], ['bar', 3], ['foob', 4], ['bazz', 5] ); });
test('TernarySearchTree (PathSegments) - lookup', function () { const map = new TernarySearchTree<number>(new PathIterator()); map.set('/user/foo/bar', 1); map.set('/user/foo', 2); map.set('/user/foo/flip/flop', 3); assert.equal(map.get('/foo'), undefined); assert.equal(map.get('/user'), undefined); assert.equal(map.get('/user/foo'), 2); assert.equal(map.get('/user/foo/bar'), 1); assert.equal(map.get('/user/foo/bar/boo'), undefined); });
function assertTernarySearchTree<E>(trie: TernarySearchTree<E>, ...elements: [string, E][]) { const map = new Map<string, E>(); for (const [key, value] of elements) { map.set(key, value); } map.forEach((value, key) => { assert.equal(trie.get(key), value); }); trie.forEach((element, key) => { assert.equal(element, map.get(key)); map.delete(key); }); assert.equal(map.size, 0); }
test('TernarySearchTree - delete & cleanup', function () { let trie = new TernarySearchTree<number>(new StringIterator()); trie.set('foo', 1); trie.set('foobar', 2); trie.set('bar', 3); trie.delete('foo'); trie.delete('foobar'); });
test('TernarySearchTree - findLongestMatch', function () { let trie = TernarySearchTree.forStrings<number>(); trie.set('foo', 1); trie.set('foobar', 2); trie.set('foobaz', 3); assert.equal(trie.findSubstr('f'), undefined); assert.equal(trie.findSubstr('z'), undefined); assert.equal(trie.findSubstr('foo'), 1); assert.equal(trie.findSubstr('fooö'), 1); assert.equal(trie.findSubstr('fooba'), 1); assert.equal(trie.findSubstr('foobarr'), 2); assert.equal(trie.findSubstr('foobazrr'), 3); });
test('TernarySearchTree (PathSegments) - superstr', function () { const map = new TernarySearchTree<number>(new PathIterator()); map.set('/user/foo/bar', 1); map.set('/user/foo', 2); map.set('/user/foo/flip/flop', 3); map.set('/usr/foo', 4); let item: IteratorResult<number>; let iter = map.findSuperstr('/user'); item = iter.next(); assert.equal(item.value, 2); assert.equal(item.done, false); item = iter.next(); assert.equal(item.value, 1); assert.equal(item.done, false); item = iter.next(); assert.equal(item.value, 3); assert.equal(item.done, false); item = iter.next(); assert.equal(item.value, undefined); assert.equal(item.done, true); iter = map.findSuperstr('/usr'); item = iter.next(); assert.equal(item.value, 4); assert.equal(item.done, false); item = iter.next(); assert.equal(item.value, undefined); assert.equal(item.done, true); assert.equal(map.findSuperstr('/not'), undefined); assert.equal(map.findSuperstr('/us'), undefined); assert.equal(map.findSuperstr('/usrr'), undefined); assert.equal(map.findSuperstr('/userr'), undefined); });
test('TernarySearchTree (PathSegments) - basics', function () { let trie = new TernarySearchTree<number>(new PathIterator()); trie.set('/user/foo/bar', 1); trie.set('/user/foo', 2); trie.set('/user/foo/flip/flop', 3); assert.equal(trie.get('/user/foo/bar'), 1); assert.equal(trie.get('/user/foo'), 2); assert.equal(trie.get('/user//foo'), 2); assert.equal(trie.get('/user\\foo'), 2); assert.equal(trie.get('/user/foo/flip/flop'), 3); assert.equal(trie.findSubstr('/user/bar'), undefined); assert.equal(trie.findSubstr('/user/foo'), 2); assert.equal(trie.findSubstr('\\user\\foo'), 2); assert.equal(trie.findSubstr('/user//foo'), 2); assert.equal(trie.findSubstr('/user/foo/ba'), 2); assert.equal(trie.findSubstr('/user/foo/far/boo'), 2); assert.equal(trie.findSubstr('/user/foo/bar'), 1); assert.equal(trie.findSubstr('/user/foo/bar/far/boo'), 1); });
test('TernarySearchTree - basics', function () { let trie = new TernarySearchTree<number>(new StringIterator()); trie.set('foo', 1); trie.set('bar', 2); trie.set('foobar', 3); assert.equal(trie.get('foo'), 1); assert.equal(trie.get('bar'), 2); assert.equal(trie.get('foobar'), 3); assert.equal(trie.get('foobaz'), undefined); assert.equal(trie.get('foobarr'), undefined); assert.equal(trie.findSubstr('fo'), undefined); assert.equal(trie.findSubstr('foo'), 1); assert.equal(trie.findSubstr('foooo'), 1); trie.delete('foobar'); trie.delete('bar'); assert.equal(trie.get('foobar'), undefined); assert.equal(trie.get('bar'), undefined); trie.set('foobar', 17); trie.set('barr', 18); assert.equal(trie.get('foobar'), 17); assert.equal(trie.get('barr'), 18); assert.equal(trie.get('bar'), undefined); });
map.forEach((value, key) => { assert.equal(trie.get(key), value); });
test('TernarySearchTree (PathSegments) - superstr', function () { const map = new TernarySearchTree<number>(new PathIterator()); map.set('/user/foo/bar', 1); map.set('/user/foo', 2); map.set('/user/foo/flip/flop', 3); map.set('/usr/foo', 4); const elements = map.findSuperstr('/user'); assertTernarySearchTree(elements, ['foo/bar', 1], ['foo', 2], ['foo/flip/flop', 3]); // assert.equal(elements.length, 3); assert.equal(elements.get('foo/bar'), 1); assert.equal(elements.get('foo'), 2); assert.equal(elements.get('foo/flip/flop'), 3); assertTernarySearchTree(map.findSuperstr('/usr'), ['foo', 4]); assert.equal(map.findSuperstr('/usr/foo'), undefined); assert.equal(map.get('/usr/foo'), 4); assert.equal(map.findSuperstr('/not'), undefined); assert.equal(map.findSuperstr('/us'), undefined); assert.equal(map.findSuperstr('/usrr'), undefined); assert.equal(map.findSuperstr('/userr'), undefined); });