From 280092ba08c9325e1f5f749c438b786c996dc5ff Mon Sep 17 00:00:00 2001 From: Bowen Tan Date: Fri, 26 Jan 2024 13:58:29 +0800 Subject: [PATCH] cache global store get requests --- src/global-store.ts | 103 +++++++++++++++++++++++++++++--------------- 1 file changed, 69 insertions(+), 34 deletions(-) diff --git a/src/global-store.ts b/src/global-store.ts index 49cca05..be84ba4 100644 --- a/src/global-store.ts +++ b/src/global-store.ts @@ -1,4 +1,5 @@ import { MetaQuery } from '@refinedev/core'; +import { isEqual } from 'lodash'; import { KubeApi, UnstructuredList, @@ -39,6 +40,11 @@ export class GlobalStore { fieldManager?: string; private store = new Map(); + private requestsCache: Array<{ + resource: string; + meta?: MetaQuery; + promise: Promise; + }> = []; private subscribers = new Map void)[]>(); private stopWatchHandlers = new Map(); private cancelControllers = new Map(); @@ -60,42 +66,71 @@ export class GlobalStore { } get(resource: string, meta?: MetaQuery): Promise { - return new Promise((resolve, reject) => { - if (!this.store.has(resource)) { - const kubeApi = new KubeApi({ - basePath: this._apiUrl, - watchWsBasePath: this.watchWsApiUrl, - objectConstructor: getObjectConstructor(resource, meta), - kubeApiTimeout: this.kubeApiTimeout, - }); - const controller = new AbortController(); - const { signal } = controller; - this.cancelControllers.set(resource, controller); - let resolved = false; - kubeApi - .listWatch({ - onResponse: async res => { - const processedRes = await this.processList(res); - if (!resolved) { - resolve(processedRes as unknown as T); - resolved = true; - } - this.store.set(resource, processedRes); - }, - onEvent: async event => { - await this.processItem(event.object); - this.notify(resource, event); - }, - signal, - }) - .then(stop => { - this.stopWatchHandlers.set(resource, stop); - }) - .catch(e => reject(e)); - } else { + if (this.store.has(resource)) { + // return cached data + return new Promise(resolve => { resolve(this.store.get(resource)! as unknown as T); - } + }); + } + + const cacheRequest = this.requestsCache.find(f => { + return f.resource === resource && isEqual(f.meta, meta); + }); + + // return cached fetching request + if (cacheRequest) { + return cacheRequest.promise as Promise; + } + + const promise = new Promise((resolve, reject) => { + const kubeApi = new KubeApi({ + basePath: this._apiUrl, + watchWsBasePath: this.watchWsApiUrl, + objectConstructor: getObjectConstructor(resource, meta), + kubeApiTimeout: this.kubeApiTimeout, + }); + const controller = new AbortController(); + const { signal } = controller; + this.cancelControllers.set(resource, controller); + let resolved = false; + kubeApi + .listWatch({ + onResponse: async res => { + const processedRes = await this.processList(res); + if (!resolved) { + resolve(processedRes as unknown as T); + resolved = true; + } + this.store.set(resource, processedRes); + }, + onEvent: async event => { + await this.processItem(event.object); + this.notify(resource, event); + }, + signal, + }) + .then(stop => { + this.stopWatchHandlers.set(resource, stop); + }) + .catch(e => reject(e)) + .finally(() => { + // clear cache request + const cacheRequestIndex = this.requestsCache.findIndex(f => { + return f.promise === promise; + }); + if (cacheRequestIndex > -1) { + this.requestsCache.splice(cacheRequestIndex, 1); + } + }); + }); + + // cache the promise + this.requestsCache.push({ + resource, + meta, + promise, }); + return promise; } subscribe(resource: string, onEvent: (data: WatchEvent) => void) {