202 lines
5.7 KiB
JavaScript
202 lines
5.7 KiB
JavaScript
import { TimeoutError, timeoutSignal } from '@/lib/time'
|
|
|
|
export class FetchTimeoutError extends TimeoutError {
|
|
constructor (method, url, timeout) {
|
|
super(timeout)
|
|
this.name = 'FetchTimeoutError'
|
|
this.message = timeout
|
|
? `${method} ${url}: timeout after ${timeout / 1000}s`
|
|
: `${method} ${url}: timeout`
|
|
}
|
|
}
|
|
|
|
export async function fetchWithTimeout (resource, { signal, timeout = 1000, ...options } = {}) {
|
|
try {
|
|
return await fetch(resource, {
|
|
...options,
|
|
signal: signal ?? timeoutSignal(timeout)
|
|
})
|
|
} catch (err) {
|
|
if (err.name === 'TimeoutError') {
|
|
// use custom error message
|
|
throw new FetchTimeoutError(options.method ?? 'GET', resource, err.timeout)
|
|
}
|
|
throw err
|
|
}
|
|
}
|
|
|
|
class LRUCache {
|
|
constructor (maxSize = 100) {
|
|
this.maxSize = maxSize
|
|
this.cache = new Map()
|
|
}
|
|
|
|
get (key) {
|
|
if (!this.cache.has(key)) return undefined
|
|
const value = this.cache.get(key)
|
|
// refresh the entry
|
|
this.cache.delete(key)
|
|
this.cache.set(key, value)
|
|
return value
|
|
}
|
|
|
|
delete (key) {
|
|
this.cache.delete(key)
|
|
}
|
|
|
|
set (key, value) {
|
|
if (this.cache.has(key)) this.cache.delete(key)
|
|
else if (this.cache.size >= this.maxSize) {
|
|
// Remove the least recently used item
|
|
this.cache.delete(this.cache.keys().next().value)
|
|
}
|
|
this.cache.set(key, value)
|
|
}
|
|
}
|
|
|
|
function createDebugLogger (name, cache, debug) {
|
|
const noop = () => {}
|
|
|
|
if (!debug) {
|
|
return {
|
|
log: noop,
|
|
errorLog: noop,
|
|
startPeriodicLogging: noop,
|
|
stopPeriodicLogging: noop,
|
|
incrementTotalFetches: noop,
|
|
incrementCacheHits: noop,
|
|
incrementCacheMisses: noop,
|
|
incrementBackgroundRefreshes: noop
|
|
}
|
|
}
|
|
|
|
let totalFetches = 0
|
|
let cacheMisses = 0
|
|
let cacheHits = 0
|
|
let backgroundRefreshes = 0
|
|
let intervalId = null
|
|
|
|
const log = (message) => console.log(`[CACHE:${name}] ${message}`)
|
|
const errorLog = (message, error) => console.error(`[CACHE:${name}] ${message}`, error)
|
|
|
|
function estimateCacheSize () {
|
|
let size = 0
|
|
for (const [key, value] of cache.cache) {
|
|
size += key.length * 2
|
|
size += JSON.stringify(value).length * 2
|
|
}
|
|
return size
|
|
}
|
|
|
|
function startPeriodicLogging () {
|
|
if (intervalId) return // Prevent multiple intervals
|
|
intervalId = setInterval(() => {
|
|
const cacheSize = cache.cache.size
|
|
const memorySizeBytes = estimateCacheSize()
|
|
log(`Stats: total=${totalFetches}, hits=${cacheHits}, misses=${cacheMisses}, backgroundRefreshes=${backgroundRefreshes}, cacheSize=${cacheSize}, memoryFootprint=${memorySizeBytes} bytes`)
|
|
}, 60000)
|
|
}
|
|
|
|
function stopPeriodicLogging () {
|
|
if (intervalId) {
|
|
clearInterval(intervalId)
|
|
intervalId = null
|
|
}
|
|
}
|
|
|
|
return {
|
|
log,
|
|
errorLog,
|
|
startPeriodicLogging,
|
|
stopPeriodicLogging,
|
|
incrementTotalFetches: () => totalFetches++,
|
|
incrementCacheHits: () => cacheHits++,
|
|
incrementCacheMisses: () => cacheMisses++,
|
|
incrementBackgroundRefreshes: () => backgroundRefreshes++
|
|
}
|
|
}
|
|
|
|
export function cachedFetcher (fetcher, {
|
|
maxSize = 100, cacheExpiry, forceRefreshThreshold,
|
|
keyGenerator, debug = process.env.DEBUG_CACHED_FETCHER
|
|
}) {
|
|
const cache = new LRUCache(maxSize)
|
|
const name = fetcher.name || fetcher.toString().slice(0, 20).replace(/\s+/g, '_')
|
|
const logger = createDebugLogger(name, cache, debug)
|
|
|
|
logger.log(`initializing with maxSize=${maxSize}, cacheExpiry=${cacheExpiry}, forceRefreshThreshold=${forceRefreshThreshold}`)
|
|
logger.startPeriodicLogging()
|
|
|
|
if (!keyGenerator) {
|
|
throw new Error('keyGenerator is required')
|
|
}
|
|
|
|
const cachedFetch = async function (...args) {
|
|
const key = keyGenerator(...args)
|
|
const now = Date.now()
|
|
logger.incrementTotalFetches()
|
|
|
|
async function fetchAndCache () {
|
|
logger.log(`Fetching data for key: ${key}`)
|
|
const result = await fetcher(...args)
|
|
cache.set(key, { data: result, createdAt: now })
|
|
logger.log(`Data fetched and cached for key: ${key}`)
|
|
return result
|
|
}
|
|
|
|
const cached = cache.get(key)
|
|
|
|
if (cached) {
|
|
const age = now - cached.createdAt
|
|
|
|
if (cacheExpiry === 0 || age < cacheExpiry) {
|
|
logger.incrementCacheHits()
|
|
logger.log(`Cache hit for key: ${key}, age: ${age}ms`)
|
|
return cached.data
|
|
} else if (forceRefreshThreshold === 0 || age < forceRefreshThreshold) {
|
|
if (cached.pendingPromise) {
|
|
logger.log(`Already background refreshing key: ${key}`)
|
|
return cached.data
|
|
}
|
|
|
|
logger.incrementBackgroundRefreshes()
|
|
logger.log(`Background refresh for key: ${key}, age: ${age}ms`)
|
|
cached.pendingPromise = fetchAndCache().catch(error => {
|
|
logger.errorLog(`Background refresh failed for key: ${key}`, error)
|
|
return cached.data
|
|
}).finally(() => {
|
|
logger.log(`Background refresh completed for key: ${key}`)
|
|
delete cached.pendingPromise
|
|
})
|
|
return cached.data
|
|
}
|
|
|
|
if (cached.pendingPromise) {
|
|
logger.log(`Waiting for pending force refresh for key: ${key}`)
|
|
return await cached.pendingPromise
|
|
}
|
|
}
|
|
|
|
logger.incrementCacheMisses()
|
|
logger.log(`Cache miss for key: ${key}`)
|
|
const entry = { createdAt: now, pendingPromise: fetchAndCache() }
|
|
try {
|
|
entry.data = await entry.pendingPromise
|
|
cache.set(key, entry)
|
|
return entry.data
|
|
} catch (error) {
|
|
logger.errorLog(`Error fetching data for key: ${key}`, error)
|
|
cache.delete(key)
|
|
throw error
|
|
} finally {
|
|
logger.log(`Fetch completed for key: ${key}`)
|
|
delete entry.pendingPromise
|
|
}
|
|
}
|
|
|
|
// Attach the stopPeriodicLogging method to the returned function
|
|
cachedFetch.stopPeriodicLogging = logger.stopPeriodicLogging
|
|
|
|
return cachedFetch
|
|
}
|