implement kvp-array.ts utility
This commit is contained in:
parent
457fc38e6a
commit
5a28c4e110
2 changed files with 126 additions and 0 deletions
48
packages/backend/src/misc/kvp-array.ts
Normal file
48
packages/backend/src/misc/kvp-array.ts
Normal file
|
|
@ -0,0 +1,48 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: hazelnoot and other Sharkey contributors
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Key-Value Pair Array - stores a collection of Key/Value pairs with helper methods to access ordered keys/values.
|
||||||
|
* Keys and Values can be of any type, and Keys default to type "string" if unspecified.
|
||||||
|
*/
|
||||||
|
export type KVPArray<T, K = string> = KVPs<T, K> & {
|
||||||
|
/**
|
||||||
|
* Lazy-loaded array of all keys in the array, matching the order of the pairs.
|
||||||
|
*/
|
||||||
|
readonly keys: readonly K[],
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lazy-loaded array of all values in the array, matching the order of the pairs.
|
||||||
|
*/
|
||||||
|
readonly values: readonly T[],
|
||||||
|
};
|
||||||
|
|
||||||
|
type KVPs<V, K = string> = Omit<readonly KVP<V, K>[], 'keys' | 'values' | 'entries'>;
|
||||||
|
type KVP<T, K = string> = readonly [key: K, value: T];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wraps an array of Key/Value pairs into a KVPArray.
|
||||||
|
*/
|
||||||
|
export function makeKVPArray<T, K = string>(pairs: KVPs<T, K>): KVPArray<T, K> {
|
||||||
|
let keys: K[] | null = null;
|
||||||
|
let values: T[] | null = null;
|
||||||
|
|
||||||
|
Object.defineProperties(pairs, {
|
||||||
|
keys: {
|
||||||
|
get() {
|
||||||
|
return keys ??= pairs.map(pair => pair[0]);
|
||||||
|
},
|
||||||
|
enumerable: false,
|
||||||
|
},
|
||||||
|
values: {
|
||||||
|
get() {
|
||||||
|
return values ??= pairs.map(pair => pair[1]);
|
||||||
|
},
|
||||||
|
enumerable: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return pairs as KVPArray<T, K>;
|
||||||
|
}
|
||||||
78
packages/backend/test/unit/misc/kvp-array.ts
Normal file
78
packages/backend/test/unit/misc/kvp-array.ts
Normal file
|
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* SPDX-FileCopyrightText: hazelnoot and other Sharkey contributors
|
||||||
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { makeKVPArray } from '@/misc/kvp-array.js';
|
||||||
|
|
||||||
|
describe(makeKVPArray, () => {
|
||||||
|
it('should add keys property', () => {
|
||||||
|
const array = [['1', 1], ['2', 2], ['3', 3]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result).toHaveProperty('keys');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should add values property', () => {
|
||||||
|
const array = [['1', 1], ['2', 2], ['3', 3]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result).toHaveProperty('values');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should preserve values', () => {
|
||||||
|
const array: [string, number][] = [['1', 1], ['2', 2], ['3', 3]];
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result).toEqual(array);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should accept empty array', () => {
|
||||||
|
const array = [] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result).toHaveProperty('keys');
|
||||||
|
expect(result).toHaveProperty('values');
|
||||||
|
expect(result).toHaveLength(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('keys', () => {
|
||||||
|
it('should return all keys', () => {
|
||||||
|
const array = [['1', 1], ['2', 2], ['3', 3]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result.keys).toEqual(['1', '2', '3']);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should preserve duplicates', () => {
|
||||||
|
const array = [['1', 1], ['1', 1], ['1', 1]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result.keys).toEqual(['1', '1', '1']);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('values', () => {
|
||||||
|
it('should return all values', () => {
|
||||||
|
const array = [['1', 1], ['2', 2], ['3', 3]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result.values).toEqual([1, 2, 3]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should preserve duplicates', () => {
|
||||||
|
const array = [['1', 1], ['1', 1], ['1', 1]] as const;
|
||||||
|
|
||||||
|
const result = makeKVPArray(array);
|
||||||
|
|
||||||
|
expect(result.values).toEqual([1, 1, 1]);
|
||||||
|
});
|
||||||
|
});
|
||||||
Loading…
Add table
Add a link
Reference in a new issue