// Copyright (c) 2018, Compiler Explorer Authors // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // * Redistributions of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in the // documentation and/or other materials provided with the distribution. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE // POSSIBILITY OF SUCH DAMAGE. import type {GetResult} from '../../types/cache.interfaces.js'; import {unwrap} from '../assert.js'; import {BaseCache} from './base.js'; // A write-through multiple cache. // Writes get pushed to all caches, but reads are serviced from the first cache that returns // a hit. export class MultiCache extends BaseCache { private readonly upstream: BaseCache[]; constructor(cacheName, ...upstream) { super(cacheName, 'Multi', 'multi'); this.countersEnabled = false; this.upstream = upstream; } override statString(): string { return `${super.statString()}. ${this.upstream.map(c => `${c.details}: ${c.statString()}`).join('. ')}`; } private async propagateHitToMissedCaches(caches: BaseCache[], key: string, data: Uint8Array): Promise { await Promise.all(caches.map(c => c.put(key, data))); } override async getInternal(key: string): Promise { const misses: BaseCache[] = []; for (const cache of this.upstream) { const result = await cache.get(key); if (result.hit) { await this.propagateHitToMissedCaches(misses, key, unwrap(result.data)); return result; } // keep track of caches that missed so we can write to them if we do hit misses.push(cache); } // all caches missed return {hit: false}; } async putInternal(object: string, value: Buffer, creator?: string): Promise { await Promise.all(this.upstream.map(cache => cache.put(object, value, creator))); } }