PeerTube/server/models/model-cache.ts

91 lines
2.1 KiB
TypeScript
Raw Normal View History

2020-02-04 04:26:51 -06:00
import { Model } from 'sequelize-typescript'
import { logger } from '@server/helpers/logger'
type ModelCacheType =
'local-account-name'
| 'local-actor-name'
| 'local-actor-url'
| 'load-video-immutable-id'
| 'load-video-immutable-url'
type DeleteKey =
'video'
2020-02-04 04:26:51 -06:00
class ModelCache {
private static instance: ModelCache
private readonly localCache: { [id in ModelCacheType]: Map<string, any> } = {
'local-account-name': new Map(),
'local-actor-name': new Map(),
'local-actor-url': new Map(),
'load-video-immutable-id': new Map(),
'load-video-immutable-url': new Map()
}
private readonly deleteIds: {
[deleteKey in DeleteKey]: Map<number, { cacheType: ModelCacheType, key: string }[]>
} = {
video: new Map()
2020-02-04 04:26:51 -06:00
}
private constructor () {
}
static get Instance () {
return this.instance || (this.instance = new this())
}
doCache<T extends Model> (options: {
cacheType: ModelCacheType
key: string
2020-12-08 07:30:29 -06:00
fun: () => Promise<T>
2020-02-04 04:26:51 -06:00
whitelist?: () => boolean
deleteKey?: DeleteKey
2020-02-04 04:26:51 -06:00
}) {
const { cacheType, key, fun, whitelist, deleteKey } = options
2020-02-04 04:26:51 -06:00
if (whitelist && whitelist() !== true) return fun()
const cache = this.localCache[cacheType]
if (cache.has(key)) {
logger.debug('Model cache hit for %s -> %s.', cacheType, key)
2021-08-27 07:32:44 -05:00
return Promise.resolve<T>(cache.get(key))
2020-02-04 04:26:51 -06:00
}
return fun().then(m => {
if (!m) return m
2020-02-04 04:26:51 -06:00
if (!whitelist || whitelist()) cache.set(key, m)
if (deleteKey) {
const map = this.deleteIds[deleteKey]
if (!map.has(m.id)) map.set(m.id, [])
const a = map.get(m.id)
a.push({ cacheType, key })
}
2020-02-04 04:26:51 -06:00
return m
})
}
invalidateCache (deleteKey: DeleteKey, modelId: number) {
const map = this.deleteIds[deleteKey]
if (!map.has(modelId)) return
for (const toDelete of map.get(modelId)) {
logger.debug('Removing %s -> %d of model cache %s -> %s.', deleteKey, modelId, toDelete.cacheType, toDelete.key)
this.localCache[toDelete.cacheType].delete(toDelete.key)
}
map.delete(modelId)
}
2020-02-04 04:26:51 -06:00
}
export {
ModelCache
}