budibase/packages/backend-core/src/cache/appMetadata.ts

89 lines
2.5 KiB
TypeScript
Raw Normal View History

import { getAppClient } from "../redis/init"
import { doWithDB, DocumentType } from "../db"
import { Database, App } from "@budibase/types"
export enum AppState {
2023-07-20 17:56:31 +02:00
INVALID = "invalid",
}
2023-07-21 10:57:37 +02:00
export interface DeletedApp {
state: AppState
}
const EXPIRY_SECONDS = 3600
const INVALID_EXPIRY_SECONDS = 60
/**
* The default populate app metadata function
*/
async function populateFromDB(appId: string) {
return doWithDB(
appId,
(db: Database) => {
2023-11-08 17:17:24 +01:00
return db.get<App>(DocumentType.APP_METADATA)
},
{ skip_setup: true }
)
}
function isInvalid(metadata?: { state: string }) {
return !metadata || metadata.state === AppState.INVALID
}
/**
* Get the requested app metadata by id.
* Use redis cache to first read the app metadata.
* If not present fallback to loading the app metadata directly and re-caching.
* @param appId the id of the app to get metadata from.
* @returns the app metadata.
*/
2023-07-21 10:57:37 +02:00
export async function getAppMetadata(appId: string): Promise<App | DeletedApp> {
const client = await getAppClient()
// try cache
let metadata = await client.get(appId)
if (!metadata) {
let expiry: number | undefined = EXPIRY_SECONDS
try {
metadata = await populateFromDB(appId)
} catch (err: any) {
// app DB left around, but no metadata, it is invalid
if (err && err.status === 404) {
metadata = { state: AppState.INVALID }
// expire invalid apps regularly, in-case it was only briefly invalid
expiry = INVALID_EXPIRY_SECONDS
} else {
throw err
}
}
2023-09-07 15:37:22 +02:00
// needed for some scenarios where the caching happens
// so quickly the requests can get slightly out of sync
// might store its invalid just before it stores its valid
if (isInvalid(metadata)) {
const temp = await client.get(appId)
if (temp) {
metadata = temp
}
}
await client.store(appId, metadata, expiry)
}
2023-07-21 10:43:25 +02:00
return metadata
}
/**
* Invalidate/reset the cached metadata when a change occurs in the db.
* @param appId the cache key to bust/update.
* @param newMetadata optional - can simply provide the new metadata to update with.
* @return will respond with success when cache is updated.
*/
export async function invalidateAppMetadata(appId: string, newMetadata?: any) {
if (!appId) {
throw "Cannot invalidate if no app ID provided."
}
const client = await getAppClient()
await client.delete(appId)
if (newMetadata) {
await client.store(appId, newMetadata, EXPIRY_SECONDS)
}
}