From 161904f65af2da19d601019563ca2cea580f31ef Mon Sep 17 00:00:00 2001 From: HumanoidSandvichDispenser Date: Sat, 21 Dec 2024 17:23:10 -0800 Subject: [PATCH] Add caching mechanism to API calls --- availabili.tf/src/stores/client.ts | 42 +++++++++++++++++++++++++++++- availabili.tf/src/stores/teams.ts | 4 +-- 2 files changed, 43 insertions(+), 3 deletions(-) diff --git a/availabili.tf/src/stores/client.ts b/availabili.tf/src/stores/client.ts index dbbcd00..1fc40a7 100644 --- a/availabili.tf/src/stores/client.ts +++ b/availabili.tf/src/stores/client.ts @@ -8,6 +8,39 @@ export const useClientStore = defineStore("client", () => { const calls = new Map>(); + type CachedValue = { + value: T, + timestamp: number, + }; + + const cached = new Map(); + + function callCached( + key: string, + apiCall: () => CancelablePromise, + maxLifetime: number = 2000, + thenOnce?: (result: T) => T, + catchOnce?: (error: any) => any, + finallyOnce?: () => void, + ): Promise { + if (cached.has(key)) { + const cachedValue = cached.get(key) as CachedValue; + if (Date.now() - cachedValue.timestamp < maxLifetime) { + return Promise.resolve(cachedValue.value); + } + } + + // cache miss + let promise = call(key, apiCall, thenOnce, catchOnce, finallyOnce); + promise.then((result) => { + cached.set(key, { + value: result, + timestamp: Date.now(), + }); + }); + return promise; + } + function call( key: string, apiCall: () => CancelablePromise, @@ -17,11 +50,15 @@ export const useClientStore = defineStore("client", () => { ): Promise { console.log("Fetching call " + key); if (!calls.has(key)) { + console.log("Making new call " + key); const promise = apiCall(); calls.set(key, promise); // remove from calls once completed - promise.finally(() => calls.delete(key)); + promise.finally(() => { + console.log("Call " + key + " completed"); + calls.delete(key); + }); // only execute this "then" once if the call was just freshly made if (thenOnce) { @@ -37,6 +74,8 @@ export const useClientStore = defineStore("client", () => { } return promise; + } else { + console.log("Returning concurrent call " + key); } return calls.get(key) as Promise; } @@ -44,6 +83,7 @@ export const useClientStore = defineStore("client", () => { return { client, call, + callCached, calls, } }); diff --git a/availabili.tf/src/stores/teams.ts b/availabili.tf/src/stores/teams.ts index fe2cea2..ad1cb70 100644 --- a/availabili.tf/src/stores/teams.ts +++ b/availabili.tf/src/stores/teams.ts @@ -33,7 +33,7 @@ export const useTeamsStore = defineStore("teams", () => { } async function fetchTeam(id: number) { - const response = await clientStore.call( + const response = await clientStore.callCached( fetchTeam.name, () => client.default.getTeam(id) ); @@ -42,7 +42,7 @@ export const useTeamsStore = defineStore("teams", () => { } async function fetchTeamMembers(id: number) { - const response = await clientStore.call( + const response = await clientStore.callCached( fetchTeamMembers.name, () => client.default.getTeamMembers(id) );