diff --git a/lib/components/media-cache-provider.react.js b/lib/components/media-cache-provider.react.js new file mode 100644 --- /dev/null +++ b/lib/components/media-cache-provider.react.js @@ -0,0 +1,122 @@ +// @flow + +import * as React from 'react'; + +/** + * This represents a persistent cache layer (e.g filesystem) + * underneath the memory Holder->URI map. + */ +export type MediaCachePersistence = { + // returns true if the URI is a cached media URI. This check should be fast + +hasURI: (uri: string) => Promise, + // returns URI if holder cached or null if not + +getCachedFile: (holder: string) => Promise, + // returns URI of saved file + +saveFile: (holder: string, uri: string) => Promise, + // clears cache (deletes all files) + +clearCache: () => Promise, + // returns size of cache in bytes + +getCacheSize: () => Promise, + // cleans up old files until cache size is less than cacheSizeLimit (bytes) + // returns true if some files were deleted and memory cache should be + // invalidated + +cleanupOldFiles: (cacheSizeLimit: number) => Promise, +}; + +const DEFAULT_CACHE_SIZE_LIMIT = 100 * 1024 * 1024; // 100 MiB in bytes + +type MediaCacheContextType = { + /** + * Gets the URI for a given holder, or `null` if it's not cached. + */ + +get: (holder: string) => Promise, + /** + * Saves the URI for a given holder. Accepts both file and data URIs. + */ + +set: (holder: string, uri: string) => Promise, +}; + +function createMediaCacheContext( + persistence: MediaCachePersistence, + options: { +cacheSizeLimit?: number }, +): MediaCacheContextType { + // holder -> URI + let uriCache = new Map(); + + async function get(holder: string): Promise { + const cachedURI = uriCache.get(holder); + if (cachedURI) { + // even though we have the URI in memory, we still need to check if it's + // still valid (e.g. file was deleted from the platform cache) + const uriExists = await persistence.hasURI(cachedURI); + if (uriExists) { + return cachedURI; + } else { + uriCache.delete(holder); + } + } + // if the in-memory cache doesn't have it, check the platform cache + const cachedFile = await persistence.getCachedFile(holder); + if (cachedFile) { + uriCache.set(holder, cachedFile); + } + return cachedFile; + } + + async function set(holder: string, uri: string): Promise { + const cachedURI = await persistence.saveFile(holder, uri); + uriCache.set(holder, cachedURI); + + // do the cleanup, but don't wait for it + cleanupCacheIfNecessary(); + } + + async function cleanupCacheIfNecessary() { + // store original memory cache before invalidating it. + // The cache should be invalidated before the cleanup, so that + // memory cache doesn't hit files that are being deleted + const backupMemoryCache = new Map(uriCache); + uriCache.clear(); + let filesWereDeleted = false; + try { + filesWereDeleted = await persistence.cleanupOldFiles( + options.cacheSizeLimit ?? DEFAULT_CACHE_SIZE_LIMIT, + ); + } finally { + if (!filesWereDeleted) { + // if no files were deleted, restore the original memory cache. + // we could possibly filter out files that were deleted, but it's + // probably not worth the effort - despite the memory cache miss, + // file cache should still be hit and thus rebuild the in-memory cache. + uriCache = backupMemoryCache; + } + } + } + + return { get, set }; +} + +const MediaCacheContext: React.Context = + React.createContext(null); + +type Props = { + +children: React.Node, + +persistence: MediaCachePersistence, + +cacheSizeLimit?: number, +}; +function MediaCacheProvider(props: Props): React.Node { + const { children, persistence, cacheSizeLimit } = props; + + const cacheContext = React.useMemo( + () => createMediaCacheContext(persistence, { cacheSizeLimit }), + [persistence, cacheSizeLimit], + ); + + return ( + + {children} + + ); +} + +export { MediaCacheContext, MediaCacheProvider };