+ Caching expiry strategies
+ Prefix PathEvents
This commit is contained in:
parent
cdcaeda67c
commit
d938996a66
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@ztimson/utils",
|
"name": "@ztimson/utils",
|
||||||
"version": "0.24.7",
|
"version": "0.24.8",
|
||||||
"description": "Utility library",
|
"description": "Utility library",
|
||||||
"author": "Zak Timson",
|
"author": "Zak Timson",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
48
src/cache.ts
48
src/cache.ts
@ -7,16 +7,20 @@ export type CacheOptions = {
|
|||||||
storage?: Storage;
|
storage?: Storage;
|
||||||
/** Key cache will be stored under */
|
/** Key cache will be stored under */
|
||||||
storageKey?: string;
|
storageKey?: string;
|
||||||
|
/** Keep or delete cached items once expired, defaults to delete */
|
||||||
|
expiryPolicy?: 'delete' | 'keep';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type CachedValue<T> = T | {_expired?: boolean};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Map of data which tracks whether it is a complete collection & offers optional expiry of cached values
|
* Map of data which tracks whether it is a complete collection & offers optional expiry of cached values
|
||||||
*/
|
*/
|
||||||
export class Cache<K extends string | number | symbol, T> {
|
export class Cache<K extends string | number | symbol, T> {
|
||||||
private store = <Record<K, T>>{};
|
private store: Record<K, T> = <any>{};
|
||||||
|
|
||||||
/** Support index lookups */
|
/** Support index lookups */
|
||||||
[key: string | number | symbol]: T | any;
|
[key: string | number | symbol]: CachedValue<T> | any;
|
||||||
/** Whether cache is complete */
|
/** Whether cache is complete */
|
||||||
complete = false;
|
complete = false;
|
||||||
|
|
||||||
@ -38,7 +42,7 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
return new Proxy(this, {
|
return new Proxy(this, {
|
||||||
get: (target: this, prop: string | symbol) => {
|
get: (target: this, prop: string | symbol) => {
|
||||||
if(prop in target) return (target as any)[prop];
|
if(prop in target) return (target as any)[prop];
|
||||||
return deepCopy(target.store[prop as K]);
|
return this.get(prop as K, true);
|
||||||
},
|
},
|
||||||
set: (target: this, prop: string | symbol, value: any) => {
|
set: (target: this, prop: string | symbol, value: any) => {
|
||||||
if(prop in target) (target as any)[prop] = value;
|
if(prop in target) (target as any)[prop] = value;
|
||||||
@ -57,8 +61,9 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
* Get all cached items
|
* Get all cached items
|
||||||
* @return {T[]} Array of items
|
* @return {T[]} Array of items
|
||||||
*/
|
*/
|
||||||
all(): T[] {
|
all(expired?: boolean): CachedValue<T>[] {
|
||||||
return deepCopy(Object.values(this.store));
|
return deepCopy<any>(Object.values(this.store)
|
||||||
|
.filter((v: any) => expired || !v._expired));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -90,7 +95,8 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
* Remove all keys from cache
|
* Remove all keys from cache
|
||||||
*/
|
*/
|
||||||
clear() {
|
clear() {
|
||||||
this.store = <Record<K, T>>{};
|
this.complete = false;
|
||||||
|
this.store = <any>{};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -107,8 +113,9 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
* Return cache as an array of key-value pairs
|
* Return cache as an array of key-value pairs
|
||||||
* @return {[K, T][]} Key-value pairs array
|
* @return {[K, T][]} Key-value pairs array
|
||||||
*/
|
*/
|
||||||
entries(): [K, T][] {
|
entries(expired?: boolean): [K, CachedValue<T>][] {
|
||||||
return <[K, T][]>Object.entries(this.store);
|
return deepCopy<any>(Object.entries(this.store)
|
||||||
|
.filter((v: any) => expired || !v._expired));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -116,24 +123,31 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
* @param {K} key Key to lookup
|
* @param {K} key Key to lookup
|
||||||
* @return {T} Cached item
|
* @return {T} Cached item
|
||||||
*/
|
*/
|
||||||
get(key: K): T {
|
get(key: K, expired?: boolean): T | null {
|
||||||
return deepCopy(this.store[key]);
|
const cached = deepCopy<any>(this.store[key] ?? null);
|
||||||
|
if(expired || !cached._expired) return cached;
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a list of cached keys
|
* Get a list of cached keys
|
||||||
* @return {K[]} Array of keys
|
* @return {K[]} Array of keys
|
||||||
*/
|
*/
|
||||||
keys(): K[] {
|
keys(expired?: boolean): K[] {
|
||||||
return <K[]>Object.keys(this.store);
|
return <K[]>Object.keys(this.store)
|
||||||
|
.filter(k => expired || !(<any>this.store)[k]._expired);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get map of cached items
|
* Get map of cached items
|
||||||
* @return {Record<K, T>}
|
* @return {Record<K, T>}
|
||||||
*/
|
*/
|
||||||
map(): Record<K, T> {
|
map(expired?: boolean): Record<K, CachedValue<T>> {
|
||||||
return deepCopy(this.store);
|
const copy: any = deepCopy(this.store);
|
||||||
|
if(!expired) Object.keys(copy).forEach(k => {
|
||||||
|
if(copy[k]._expired) delete copy[k]
|
||||||
|
});
|
||||||
|
return copy;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -144,12 +158,16 @@ export class Cache<K extends string | number | symbol, T> {
|
|||||||
* @return {this}
|
* @return {this}
|
||||||
*/
|
*/
|
||||||
set(key: K, value: T, ttl = this.options.ttl): this {
|
set(key: K, value: T, ttl = this.options.ttl): this {
|
||||||
|
if(this.options.expiryPolicy == 'keep') delete (<any>this.store[key])._expired;
|
||||||
this.store[key] = value;
|
this.store[key] = value;
|
||||||
if(this.options.storageKey && this.options.storage)
|
if(this.options.storageKey && this.options.storage)
|
||||||
this.options.storage.setItem(this.options.storageKey, JSON.stringify(this.store));
|
this.options.storage.setItem(this.options.storageKey, JSON.stringify(this.store));
|
||||||
if(ttl) setTimeout(() => {
|
if(ttl) setTimeout(() => {
|
||||||
this.complete = false;
|
this.complete = false;
|
||||||
this.delete(key);
|
if(this.options.expiryPolicy == 'keep') (<any>this.store[key])._expired = true;
|
||||||
|
else this.delete(key);
|
||||||
|
if(this.options.storageKey && this.options.storage)
|
||||||
|
this.options.storage.setItem(this.options.storageKey, JSON.stringify(this.store));
|
||||||
}, ttl * 1000);
|
}, ttl * 1000);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
@ -223,7 +223,7 @@ export class PathEvent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create event string from its components
|
* Create event string from its components
|
||||||
*
|
*
|
||||||
* @return {string} String representation of Event
|
* @return {string} String representation of Event
|
||||||
*/
|
*/
|
||||||
@ -235,11 +235,12 @@ export class PathEvent {
|
|||||||
export type PathListener = (event: PathEvent, ...args: any[]) => any;
|
export type PathListener = (event: PathEvent, ...args: any[]) => any;
|
||||||
export type PathUnsubscribe = () => void;
|
export type PathUnsubscribe = () => void;
|
||||||
|
|
||||||
|
export type Event = string | PathEvent;
|
||||||
export interface IPathEventEmitter {
|
export interface IPathEventEmitter {
|
||||||
emit(event: string, ...args: any[]): void;
|
emit(event: Event, ...args: any[]): void;
|
||||||
off(listener: PathListener): void;
|
off(listener: PathListener): void;
|
||||||
on(event: string, listener: PathListener): PathUnsubscribe;
|
on(event: Event | Event[], listener: PathListener): PathUnsubscribe;
|
||||||
once(event: string, listener?: PathListener): Promise<any>;
|
once(event: Event | Event[], listener?: PathListener): Promise<any>;
|
||||||
relayEvents(emitter: PathEventEmitter): void;
|
relayEvents(emitter: PathEventEmitter): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -249,8 +250,10 @@ export interface IPathEventEmitter {
|
|||||||
export class PathEventEmitter implements IPathEventEmitter{
|
export class PathEventEmitter implements IPathEventEmitter{
|
||||||
private listeners: [PathEvent, PathListener][] = [];
|
private listeners: [PathEvent, PathListener][] = [];
|
||||||
|
|
||||||
emit(event: string | PathEvent, ...args: any[]) {
|
constructor(public readonly prefix: string = '') { }
|
||||||
const parsed = new PathEvent(event);
|
|
||||||
|
emit(event: Event, ...args: any[]) {
|
||||||
|
const parsed = new PathEvent(`${this.prefix}/${typeof event == 'string' ? event : event.toString()}`);
|
||||||
this.listeners.filter(l => PathEvent.has(l[0], event))
|
this.listeners.filter(l => PathEvent.has(l[0], event))
|
||||||
.forEach(async l => l[1](parsed, ...args));
|
.forEach(async l => l[1](parsed, ...args));
|
||||||
};
|
};
|
||||||
@ -259,12 +262,15 @@ export class PathEventEmitter implements IPathEventEmitter{
|
|||||||
this.listeners = this.listeners.filter(l => l[1] != listener);
|
this.listeners = this.listeners.filter(l => l[1] != listener);
|
||||||
}
|
}
|
||||||
|
|
||||||
on(event: string | string[], listener: PathListener): PathUnsubscribe {
|
on(event: Event | Event[], listener: PathListener): PathUnsubscribe {
|
||||||
makeArray(event).forEach(e => this.listeners.push([new PathEvent(e), listener]));
|
makeArray(event).forEach(e => this.listeners.push([
|
||||||
|
new PathEvent(`${this.prefix}/${typeof e == 'string' ? event : event.toString()}`),
|
||||||
|
listener
|
||||||
|
]));
|
||||||
return () => this.off(listener);
|
return () => this.off(listener);
|
||||||
}
|
}
|
||||||
|
|
||||||
once(event: string | string[], listener?: PathListener): Promise<any> {
|
once(event: Event | Event[], listener?: PathListener): Promise<any> {
|
||||||
return new Promise(res => {
|
return new Promise(res => {
|
||||||
const unsubscribe = this.on(event, (event: PathEvent, ...args: any[]) => {
|
const unsubscribe = this.on(event, (event: PathEvent, ...args: any[]) => {
|
||||||
res(args.length < 2 ? args[0] : args);
|
res(args.length < 2 ? args[0] : args);
|
||||||
|
43
tests/path-events.spec.ts
Normal file
43
tests/path-events.spec.ts
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
import {PathEvent} from '../src';
|
||||||
|
|
||||||
|
describe('Path Events', () => {
|
||||||
|
describe('malformed', () => {
|
||||||
|
test('starting slash', async () =>
|
||||||
|
expect(new PathEvent('/module').toString()).toEqual('module:*'));
|
||||||
|
test('trailing slash', async () =>
|
||||||
|
expect(new PathEvent('module/').toString()).toEqual('module:*'));
|
||||||
|
test('double slash', async () =>
|
||||||
|
expect(new PathEvent('module////path').toString()).toEqual('module/path:*'));
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('methods', () => {
|
||||||
|
test('custom', async () => {
|
||||||
|
expect(new PathEvent('module:t').methods.includes('t')).toBeTruthy();
|
||||||
|
expect(new PathEvent('module:t').methods.includes('z')).toBeFalsy();
|
||||||
|
});
|
||||||
|
test('create', async () =>
|
||||||
|
expect(new PathEvent('module:crud').create).toBeTruthy());
|
||||||
|
test('read', async () =>
|
||||||
|
expect(new PathEvent('module:crud').read).toBeTruthy());
|
||||||
|
test('update', async () =>
|
||||||
|
expect(new PathEvent('module:crud').update).toBeTruthy());
|
||||||
|
test('delete', async () =>
|
||||||
|
expect(new PathEvent('module:crud').delete).toBeTruthy());
|
||||||
|
test('none', async () => {
|
||||||
|
const event = new PathEvent('module:n');
|
||||||
|
expect(event.none).toBeTruthy();
|
||||||
|
expect(event.create).toBeFalsy();
|
||||||
|
expect(event.read).toBeFalsy();
|
||||||
|
expect(event.update).toBeFalsy();
|
||||||
|
expect(event.delete).toBeFalsy()
|
||||||
|
});
|
||||||
|
test('wildcard', async () => {
|
||||||
|
const event = new PathEvent('module:*');
|
||||||
|
expect(event.none).toBeFalsy();
|
||||||
|
expect(event.create).toBeTruthy();
|
||||||
|
expect(event.read).toBeTruthy();
|
||||||
|
expect(event.update).toBeTruthy();
|
||||||
|
expect(event.delete).toBeTruthy()
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
Loading…
x
Reference in New Issue
Block a user