93 lines
3.8 KiB
JavaScript
93 lines
3.8 KiB
JavaScript
"use strict";
|
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const p_defer_1 = __importDefault(require("p-defer"));
|
|
function mapAgeCleaner(map, property = 'maxAge') {
|
|
let processingKey;
|
|
let processingTimer;
|
|
let processingDeferred;
|
|
const cleanup = () => __awaiter(this, void 0, void 0, function* () {
|
|
if (processingKey !== undefined) {
|
|
// If we are already processing an item, we can safely exit
|
|
return;
|
|
}
|
|
const setupTimer = (item) => __awaiter(this, void 0, void 0, function* () {
|
|
processingDeferred = p_defer_1.default();
|
|
const delay = item[1][property] - Date.now();
|
|
if (delay <= 0) {
|
|
// Remove the item immediately if the delay is equal to or below 0
|
|
map.delete(item[0]);
|
|
processingDeferred.resolve();
|
|
return;
|
|
}
|
|
// Keep track of the current processed key
|
|
processingKey = item[0];
|
|
processingTimer = setTimeout(() => {
|
|
// Remove the item when the timeout fires
|
|
map.delete(item[0]);
|
|
if (processingDeferred) {
|
|
processingDeferred.resolve();
|
|
}
|
|
}, delay);
|
|
// tslint:disable-next-line:strict-type-predicates
|
|
if (typeof processingTimer.unref === 'function') {
|
|
// Don't hold up the process from exiting
|
|
processingTimer.unref();
|
|
}
|
|
return processingDeferred.promise;
|
|
});
|
|
try {
|
|
for (const entry of map) {
|
|
yield setupTimer(entry);
|
|
}
|
|
}
|
|
catch (_a) {
|
|
// Do nothing if an error occurs, this means the timer was cleaned up and we should stop processing
|
|
}
|
|
processingKey = undefined;
|
|
});
|
|
const reset = () => {
|
|
processingKey = undefined;
|
|
if (processingTimer !== undefined) {
|
|
clearTimeout(processingTimer);
|
|
processingTimer = undefined;
|
|
}
|
|
if (processingDeferred !== undefined) { // tslint:disable-line:early-exit
|
|
processingDeferred.reject(undefined);
|
|
processingDeferred = undefined;
|
|
}
|
|
};
|
|
const originalSet = map.set.bind(map);
|
|
map.set = (key, value) => {
|
|
if (map.has(key)) {
|
|
// If the key already exist, remove it so we can add it back at the end of the map.
|
|
map.delete(key);
|
|
}
|
|
// Call the original `map.set`
|
|
const result = originalSet(key, value);
|
|
// If we are already processing a key and the key added is the current processed key, stop processing it
|
|
if (processingKey && processingKey === key) {
|
|
reset();
|
|
}
|
|
// Always run the cleanup method in case it wasn't started yet
|
|
cleanup(); // tslint:disable-line:no-floating-promises
|
|
return result;
|
|
};
|
|
cleanup(); // tslint:disable-line:no-floating-promises
|
|
return map;
|
|
}
|
|
exports.default = mapAgeCleaner;
|
|
// Add support for CJS
|
|
module.exports = mapAgeCleaner;
|
|
module.exports.default = mapAgeCleaner;
|