"use strict"; var __create = Object.create; var __defProp = Object.defineProperty; var __getOwnPropDesc = Object.getOwnPropertyDescriptor; var __getOwnPropNames = Object.getOwnPropertyNames; var __getProtoOf = Object.getPrototypeOf; var __hasOwnProp = Object.prototype.hasOwnProperty; var __export = (target, all) => { for (var name in all) __defProp(target, name, { get: all[name], enumerable: true }); }; var __copyProps = (to, from, except, desc) => { if (from && typeof from === "object" || typeof from === "function") { for (let key of __getOwnPropNames(from)) if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); } return to; }; var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps( // If the importer is in node compatibility mode or this is not an ESM // file that has been converted to a CommonJS file using a Babel- // compatible transform (i.e. "__esModule" has not been set), then set // "default" to the CommonJS "module.exports" for node compatibility. isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod )); var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); // src/index.ts var index_exports = {}; __export(index_exports, { FlatCache: () => FlatCache, FlatCacheEvents: () => FlatCacheEvents, clearAll: () => clearAll, clearCacheById: () => clearCacheById, create: () => create, createFromFile: () => createFromFile, default: () => FlatCacheDefault }); module.exports = __toCommonJS(index_exports); var import_node_path = __toESM(require("path"), 1); var import_node_fs = __toESM(require("fs"), 1); var import_cacheable = require("cacheable"); var import_flatted = require("flatted"); var import_hookified = require("hookified"); var FlatCacheEvents = /* @__PURE__ */ ((FlatCacheEvents2) => { FlatCacheEvents2["SAVE"] = "save"; FlatCacheEvents2["LOAD"] = "load"; FlatCacheEvents2["DELETE"] = "delete"; FlatCacheEvents2["CLEAR"] = "clear"; FlatCacheEvents2["DESTROY"] = "destroy"; FlatCacheEvents2["ERROR"] = "error"; FlatCacheEvents2["EXPIRED"] = "expired"; return FlatCacheEvents2; })(FlatCacheEvents || {}); var FlatCache = class extends import_hookified.Hookified { _cache = new import_cacheable.CacheableMemory(); _cacheDir = ".cache"; _cacheId = "cache1"; _persistInterval = 0; _persistTimer; _changesSinceLastSave = false; _parse = import_flatted.parse; _stringify = import_flatted.stringify; constructor(options) { super(); if (options) { this._cache = new import_cacheable.CacheableMemory({ ttl: options.ttl, useClone: options.useClone, lruSize: options.lruSize, checkInterval: options.expirationInterval }); } if (options?.cacheDir) { this._cacheDir = options.cacheDir; } if (options?.cacheId) { this._cacheId = options.cacheId; } if (options?.persistInterval) { this._persistInterval = options.persistInterval; this.startAutoPersist(); } if (options?.deserialize) { this._parse = options.deserialize; } if (options?.serialize) { this._stringify = options.serialize; } } /** * The cache object * @property cache * @type {CacheableMemory} */ get cache() { return this._cache; } /** * The cache directory * @property cacheDir * @type {String} * @default '.cache' */ get cacheDir() { return this._cacheDir; } /** * Set the cache directory * @property cacheDir * @type {String} * @default '.cache' */ set cacheDir(value) { this._cacheDir = value; } /** * The cache id * @property cacheId * @type {String} * @default 'cache1' */ get cacheId() { return this._cacheId; } /** * Set the cache id * @property cacheId * @type {String} * @default 'cache1' */ set cacheId(value) { this._cacheId = value; } /** * The flag to indicate if there are changes since the last save * @property changesSinceLastSave * @type {Boolean} * @default false */ get changesSinceLastSave() { return this._changesSinceLastSave; } /** * The interval to persist the cache to disk. 0 means no timed persistence * @property persistInterval * @type {Number} * @default 0 */ get persistInterval() { return this._persistInterval; } /** * Set the interval to persist the cache to disk. 0 means no timed persistence * @property persistInterval * @type {Number} * @default 0 */ set persistInterval(value) { this._persistInterval = value; } /** * Load a cache identified by the given Id. If the element does not exists, then initialize an empty * cache storage. If specified `cacheDir` will be used as the directory to persist the data to. If omitted * then the cache module directory `.cacheDir` will be used instead * * @method load * @param cacheId {String} the id of the cache, would also be used as the name of the file cache * @param cacheDir {String} directory for the cache entry */ // eslint-disable-next-line unicorn/prevent-abbreviations load(cacheId, cacheDir) { try { const filePath = import_node_path.default.resolve(`${cacheDir ?? this._cacheDir}/${cacheId ?? this._cacheId}`); this.loadFile(filePath); this.emit("load" /* LOAD */); } catch (error) { this.emit("error" /* ERROR */, error); } } /** * Load the cache from the provided file * @method loadFile * @param {String} pathToFile the path to the file containing the info for the cache */ loadFile(pathToFile) { if (import_node_fs.default.existsSync(pathToFile)) { const data = import_node_fs.default.readFileSync(pathToFile, "utf8"); const items = this._parse(data); for (const key of Object.keys(items)) { this._cache.set(items[key].key, items[key].value, { expire: items[key].expires }); } this._changesSinceLastSave = true; } } /** * Returns the entire persisted object * @method all * @returns {*} */ all() { const result = {}; const items = Array.from(this._cache.items); for (const item of items) { result[item.key] = item.value; } return result; } /** * Returns an array with all the items in the cache { key, value, ttl } * @method items * @returns {Array} */ get items() { return Array.from(this._cache.items); } /** * Returns the path to the file where the cache is persisted * @method cacheFilePath * @returns {String} */ get cacheFilePath() { return import_node_path.default.resolve(`${this._cacheDir}/${this._cacheId}`); } /** * Returns the path to the cache directory * @method cacheDirPath * @returns {String} */ get cacheDirPath() { return import_node_path.default.resolve(this._cacheDir); } /** * Returns an array with all the keys in the cache * @method keys * @returns {Array} */ keys() { return Array.from(this._cache.keys); } /** * (Legacy) set key method. This method will be deprecated in the future * @method setKey * @param key {string} the key to set * @param value {object} the value of the key. Could be any object that can be serialized with JSON.stringify */ setKey(key, value, ttl) { this.set(key, value, ttl); } /** * Sets a key to a given value * @method set * @param key {string} the key to set * @param value {object} the value of the key. Could be any object that can be serialized with JSON.stringify * @param [ttl] {number} the time to live in milliseconds */ set(key, value, ttl) { this._cache.set(key, value, ttl); this._changesSinceLastSave = true; } /** * (Legacy) Remove a given key from the cache. This method will be deprecated in the future * @method removeKey * @param key {String} the key to remove from the object */ removeKey(key) { this.delete(key); } /** * Remove a given key from the cache * @method delete * @param key {String} the key to remove from the object */ delete(key) { this._cache.delete(key); this._changesSinceLastSave = true; this.emit("delete" /* DELETE */, key); } /** * (Legacy) Return the value of the provided key. This method will be deprecated in the future * @method getKey * @param key {String} the name of the key to retrieve * @returns {*} at T the value from the key */ getKey(key) { return this.get(key); } /** * Return the value of the provided key * @method get * @param key {String} the name of the key to retrieve * @returns {*} at T the value from the key */ get(key) { return this._cache.get(key); } /** * Clear the cache and save the state to disk * @method clear */ clear() { try { this._cache.clear(); this._changesSinceLastSave = true; this.save(); this.emit("clear" /* CLEAR */); } catch (error) { this.emit("error" /* ERROR */, error); } } /** * Save the state of the cache identified by the docId to disk * as a JSON structure * @method save */ save(force = false) { try { if (this._changesSinceLastSave || force) { const filePath = this.cacheFilePath; const items = Array.from(this._cache.items); const data = this._stringify(items); if (!import_node_fs.default.existsSync(this._cacheDir)) { import_node_fs.default.mkdirSync(this._cacheDir, { recursive: true }); } import_node_fs.default.writeFileSync(filePath, data); this._changesSinceLastSave = false; this.emit("save" /* SAVE */); } } catch (error) { this.emit("error" /* ERROR */, error); } } /** * Remove the file where the cache is persisted * @method removeCacheFile * @return {Boolean} true or false if the file was successfully deleted */ removeCacheFile() { try { if (import_node_fs.default.existsSync(this.cacheFilePath)) { import_node_fs.default.rmSync(this.cacheFilePath); return true; } } catch (error) { this.emit("error" /* ERROR */, error); } return false; } /** * Destroy the cache. This will remove the directory, file, and memory cache * @method destroy * @param [includeCacheDir=false] {Boolean} if true, the cache directory will be removed * @return {undefined} */ destroy(includeCacheDirectory = false) { try { this._cache.clear(); this.stopAutoPersist(); if (includeCacheDirectory) { import_node_fs.default.rmSync(this.cacheDirPath, { recursive: true, force: true }); } else { import_node_fs.default.rmSync(this.cacheFilePath, { recursive: true, force: true }); } this._changesSinceLastSave = false; this.emit("destroy" /* DESTROY */); } catch (error) { this.emit("error" /* ERROR */, error); } } /** * Start the auto persist interval * @method startAutoPersist */ startAutoPersist() { if (this._persistInterval > 0) { if (this._persistTimer) { clearInterval(this._persistTimer); this._persistTimer = void 0; } this._persistTimer = setInterval(() => { this.save(); }, this._persistInterval); } } /** * Stop the auto persist interval * @method stopAutoPersist */ stopAutoPersist() { if (this._persistTimer) { clearInterval(this._persistTimer); this._persistTimer = void 0; } } }; var FlatCacheDefault = class { static create = create; static createFromFile = createFromFile; static clearCacheById = clearCacheById; static clearAll = clearAll; }; function create(options) { const cache = new FlatCache(options); cache.load(); return cache; } function createFromFile(filePath, options) { const cache = new FlatCache(options); cache.loadFile(filePath); return cache; } function clearCacheById(cacheId, cacheDirectory) { const cache = new FlatCache({ cacheId, cacheDir: cacheDirectory }); cache.destroy(); } function clearAll(cacheDirectory) { import_node_fs.default.rmSync(cacheDirectory ?? ".cache", { recursive: true, force: true }); } // Annotate the CommonJS export names for ESM import in node: 0 && (module.exports = { FlatCache, FlatCacheEvents, clearAll, clearCacheById, create, createFromFile });