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;
 |