+ Caching expiry strategies
Some checks failed
Build / Build NPM Project (push) Failing after 34s
Build / Publish Documentation (push) Has been skipped
Build / Tag Version (push) Has been skipped

+ Prefix PathEvents
This commit is contained in:
Zakary Timson 2025-05-12 19:46:23 -04:00
parent cdcaeda67c
commit d938996a66
4 changed files with 92 additions and 25 deletions

View File

@ -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",

View File

@ -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;
} }

View File

@ -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
View 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()
});
});
});