|  | // 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 path from 'path'; | 
|  |  | 
|  | import fs from 'fs-extra'; | 
|  | import LRU from 'lru-cache'; | 
|  |  | 
|  | import {GetResult} from '../../types/cache.interfaces'; | 
|  | import {logger} from '../logger'; | 
|  |  | 
|  | import {BaseCache} from './base'; | 
|  |  | 
|  | // With thanks to https://gist.github.com/kethinov/6658166 | 
|  | function getAllFiles(root: string, dir?: string) { | 
|  | const actualDir = dir || root; | 
|  | return fs.readdirSync(actualDir).reduce((files: Array<string>, file: string) => { | 
|  | const fullPath = path.join(actualDir, file); | 
|  | const name = path.relative(root, fullPath); | 
|  | const isDirectory = fs.statSync(fullPath).isDirectory(); | 
|  | return isDirectory ? [...files, ...getAllFiles(root, fullPath)] : [...files, {name, fullPath}]; | 
|  | }, []); | 
|  | } | 
|  |  | 
|  | export class OnDiskCache extends BaseCache { | 
|  | readonly path: string; | 
|  | readonly cacheMb: number; | 
|  | private readonly cache: LRU; | 
|  |  | 
|  | constructor(cacheName: string, path: string, cacheMb: number) { | 
|  | super(cacheName, `OnDiskCache(${path}, ${cacheMb}mb)`, 'disk'); | 
|  | this.path = path; | 
|  | this.cacheMb = cacheMb; | 
|  | this.cache = new LRU({ | 
|  | max: cacheMb * 1024 * 1024, | 
|  | length: n => n.size, | 
|  | noDisposeOnSet: true, | 
|  | dispose: (key, n) => { | 
|  | fs.unlink(n.path, () => {}); | 
|  | }, | 
|  | }); | 
|  | fs.mkdirSync(path, {recursive: true}); | 
|  | const info = getAllFiles(path).map(({name, fullPath}) => { | 
|  | const stat = fs.statSync(fullPath); | 
|  | return { | 
|  | key: name, | 
|  | sort: stat.ctimeMs, | 
|  | data: { | 
|  | path: fullPath, | 
|  | size: stat.size, | 
|  | }, | 
|  | }; | 
|  | }); | 
|  | // Sort oldest first | 
|  | info.sort((x, y) => x.sort - y.sort); | 
|  | for (const i of info) { | 
|  | this.cache.set(i.key, i.data); | 
|  | } | 
|  | } | 
|  |  | 
|  | override statString(): string { | 
|  | return ( | 
|  | `${super.statString()}, LRU has ${this.cache.itemCount} item(s) ` + | 
|  | `totalling ${this.cache.length} bytes on disk` | 
|  | ); | 
|  | } | 
|  |  | 
|  | override async getInternal(key: string): Promise<GetResult> { | 
|  | const cached = this.cache.get(key); | 
|  | if (!cached) return {hit: false}; | 
|  |  | 
|  | try { | 
|  | const data = await fs.readFile(cached.path); | 
|  | return {hit: true, data: data}; | 
|  | } catch (err) { | 
|  | logger.error(`error reading '${key}' from disk cache: `, err); | 
|  | return {hit: false}; | 
|  | } | 
|  | } | 
|  |  | 
|  | async putInternal(key: string, value: Buffer): Promise<void> { | 
|  | const info = { | 
|  | path: path.join(this.path, key), | 
|  | size: value.length, | 
|  | }; | 
|  | await fs.writeFile(info.path, value); | 
|  | return this.cache.set(key, info); | 
|  | } | 
|  | } |