A fast and modern in-memory cache library with TTL and LRU for javascript runtimes.
- β‘ Fast β Optimized for high-throughput.
- π§ In-Memory β Lightweight and efficient key-value store for temporary data.
- β± TTL Support β Automatically expire items after a configurable time (Time To Live).
- β»οΈ LRU Eviction β Least Recently Used eviction strategy to cap memory usage.
- π§Ή Auto Cleanup β Background cleanup of expired items at regular intervals.
- π§© Custom Expiry Logic β Use fixed timestamps, max-ages or dynamic expiry functions per entry.
- π° Custom Time Source β Override time logic (e.g., for time units like seconds).
- π Event Hooks β Event handling of get, miss, delete, expire, lru, etc.
- π Peeking & Expired Access β Read expired entries or inspect values without triggering eviction.
- π Iterator Support β Fully iterable using for...of over entries.
- π§ͺ Cross-runtime Compatibility β Works seamlessly in Node.js, Deno, and Bun.
- π¦ Zero Dependencies β Clean, modern codebase with no external runtime dependencies.
Node.js / Bun (npm / pnpm / yarn)
bun add @bepalo/cache
# or
pnpm add @bepalo/cache
# or
npm install @bepalo/cache
# or
yarn add @bepalo/cacheDeno
Import directly using the URL:
import { Cache } from "npm:@bepalo/cache";
// or
import { Cache } from "jsr:@bepalo/cache";These benchmarks were done on the following system with 1,000,000 iterations,
10,000 warmup iterations, 500,000 LRU limit and UUIDv4 keys. Checkout benchmark
System Info:
CPU: AMD Ryzen 7 5700U (8 cores, 16 threads)
RAM: 16β―GB DDR4
OS: Pop!_OS 22.04 (Linux 6.8)
Node.js: v22.6.0Bun benchmark results
Benchmarking @bepalo/cache (N=1,000,000 LRU-Limit=500,000 K=UUIDv4)
| Operation | ns/operation | operations/s | 
|---|---|---|
| cache.get: hit | 116.564 | 8,578,950 | 
| cache.get: miss | 94.295 | 10,605,059 | 
| cache.get: miss, empty | 18.543 | 53,929,990 | 
| cache.set: new | 486.156 | 2,056,952 | 
| cache.set: override | 498.203 | 2,007,212 | 
| cache.update: | 331.407 | 3,017,433 | 
| cache.deleteExpired: all | 649.958 | 1,538,560 | 
| cache.deleteExpired: none | 228.615 | 4,374,163 | 
Comparing with native Map
| Operation | ns/operation | operations/s | 
|---|---|---|
| Map.get: hit | 9.427 | 106,073,598 | 
| Map.get: miss | 7.259 | 137,769,231 | 
| Map.get: miss, empty | 7.935 | 126,018,988 | 
| Map.set: | 279.194 | 3,581,743 | 
| Map.set: update | 173.293 | 5,770,580 | 
| Map.delete: all | 191.224 | 5,229,465 | 
| Map.delete: none | 11.472 | 87,171,224 | 
Deno benchmark results
Benchmarking @bepalo/cache (N=1,000,000 LRU-Limit=500,000 K=UUIDv4)
| Operation | ns/operation | operations/s | 
|---|---|---|
| cache.get: hit | 131.629 | 7,597,089 | 
| cache.get: miss | 178.165 | 5,612,785 | 
| cache.get: miss, empty | 10.161 | 98,418,223 | 
| cache.set: new | 482.446 | 2,072,770 | 
| cache.set: override | 641.308 | 1,559,313 | 
| cache.update: | 311.356 | 3,211,761 | 
| cache.deleteExpired: all | 588.244 | 1,699,973 | 
| cache.deleteExpired: none | 77.864 | 12,842,870 | 
Comparing with native Map
| Operation | ns/operation | operations/s | 
|---|---|---|
| Map.get: hit | 8.294 | 120,570,922 | 
| Map.get: miss | 9.033 | 110,710,977 | 
| Map.get: miss, empty | 5.933 | 168,554,446 | 
| Map.set: | 312.715 | 3,197,799 | 
| Map.set: update | 179.718 | 5,564,266 | 
| Map.delete: all | 218.321 | 4,580,405 | 
| Map.delete: none | 10.184 | 98,192,050 | 
Node benchmark results
Benchmarking @bepalo/cache (N=1,000,000 LRU-Limit=500,000 K=UUIDv4)
| Operation | ns/operation | operations/s | 
|---|---|---|
| cache.get: hit | 245.625 | 4,071,248 | 
| cache.get: miss | 234.214 | 4,269,607 | 
| cache.get: miss, empty | 29.907 | 33,436,985 | 
| cache.set: new | 854.164 | 1,170,734 | 
| cache.set: override | 1,138.106 | 878,652 | 
| cache.update: | 523.995 | 1,908,415 | 
| cache.deleteExpired: all | 788.144 | 1,268,803 | 
| cache.deleteExpired: none | 163.846 | 6,103,304 | 
Comparing with native Map
| Operation | ns/operation | operations/s | 
|---|---|---|
| Map.get: hit | 191.249 | 5,228,772 | 
| Map.get: miss | 187.621 | 5,329,892 | 
| Map.get: miss, empty | 9.039 | 110,635,966 | 
| Map.set: | 291.251 | 3,433,466 | 
| Map.set: update | 239.813 | 4,169,920 | 
| Map.delete: all | 383.207 | 2,609,554 | 
| Map.delete: none | 9.990 | 100,100,731 | 
import { Cache } from "@bepalo/cache";
const cache = new Cache();
cache.set("hello", "world");
console.log(cache.get("hello")?.value); // => "world"
cache.delete("hello");
console.log(cache.get("hello")); // => undefinedconst cache = new Cache({
  defaultMaxAge: 1000, // 1 second TTL
});
cache.set("foo", 123);
setTimeout(() => {
  console.log(cache.get("foo")); // => undefined (expired)
}, 1500);const cache = new Cache({
  lruMaxSize: 2,
});
cache.set("a", 1);
cache.set("b", 2);
cache.set("c", 3); // "a" gets evicted (least recently used)
console.log(cache.has("a")); // => false
console.log(cache.has("b")); // => trueconst cache = new Cache({
  defaultExp: () => Date.now() + 5000, // set default expiry using a function
  cleanupInterval: 500, // auto-clean every 500ms
});
cache.set("temp", "value", { maxAge: 100 }); // expires in 100ms
setTimeout(() => {
  console.log(cache.has("temp")); // => false
}, 1000);const cache = new Cache({
  now: () => Date.now() / 1000, // now will return time in seconds
  defaultMaxAge: 60 // treated as 60 seconds
  cleanupInterval: 5, // auto-clean every 5sec
});
cache.set("temp", "value", { maxAge: 3 }); // expires in 3sec
setTimeout(() => {
  console.log(cache.has("temp")); // => false
}, 4000);const cache = new Cache({
  deleteExpiredOnGet: true,
  onGetHit: (cache, key, entry) => {
    console.log(`Hit: ${key}`);
  },
  onGetMiss: (cache, key, reason) => {
    console.log(`Miss: ${key} (${reason})`);
  },
  onDelete: (cache, key, entry, reason) => {
    console.log(`Deleted: ${key} (${reason})`);
  },
  onDeleteExpired: (count) => {
    console.log(`Expired entries removed: ${count}`);
  },
});
cache.set("x", 42, { maxAge: 10 });
cache.get("y"); // triggers `onGetMiss`
setTimeout(() => {
  cache.get("x"); // triggers `onGetHit`, `onDelete`, `onDeleteExpired`
}, 100);Code
import { Cache } from ".";
const timestampRef = performance.now();
const timestamp = () => `${(performance.now() - timestampRef).toFixed()}ms: `;
const log = console.log;
const cache = new Cache<string, string>({
  // now: () => Date.now(),
  // defaultExp: () => Date.now() + 5000,
  defaultMaxAge: 5000,
  // cleanupInterval: 5000,
  // expiryBucketSize: 5000,
  lruMaxSize: 3,
  // getExpired: true,
  // deleteExpiredOnGet: true,
  onGetHit: async (cache, key, entry) => log(timestamp(), "cache-hit", key),
  onGetMiss: async (cache, key, reason) => log(timestamp(), reason, key),
  onDelete: async (cache, key, entry, reason) => log(timestamp(), "Evict", reason, key),
  onDeleteExpired: async (count) => count > 0 && log(timestamp(), `Expired ${count} entries.`),
});
cache.set("item-1", "'sample entry 1'");
cache.set("item-2", "'sample entry 2'", { exp: Date.now() + 2000 });
cache.set("item-3", "'sample entry 3'", { maxAge: 3000 });
cache.set("item-4", "'sample entry 4'", { maxAge: 2500 });
setTimeout(() => log(timestamp(), "1 get", cache.get("item-4")?.value), 1000);
setTimeout(() => log(timestamp(), "2 get", cache.get("item-4", { expired: true })?.value), 2000);
setTimeout(() => log(timestamp(), "3 peek", cache.peek("item-4")?.value), 3000);
setTimeout(() => log(timestamp(), "4 peek", cache.peek("item-4", { expired: true })?.value), 3000);
setTimeout(
  () => log(timestamp(), "5 get", cache.get("item-4", { deleteExpired: true })?.value),
  3000,
);
setTimeout(() => log(timestamp(), "6 get", cache.get("item-4", { expired: true })?.value), 4000);
for (const [key, entry] of cache) {
  log(timestamp(), key, entry);
}Output
1ms:  Evict LRU item-1
3ms:  item-2 {
  value: "'sample entry 2'",
  exp: 1752603303022,
}
4ms:  item-3 {
  value: "'sample entry 3'",
  exp: 1752603304022,
}
4ms:  item-4 {
  value: "'sample entry 4'",
  exp: 1752603303522,
}
1003ms:  cache-hit item-4
1003ms:  1 get 'sample entry 4'
2003ms:  cache-hit item-4
2003ms:  2 get 'sample entry 4'
3003ms:  3 peek undefined
3003ms:  4 peek 'sample entry 4'
3004ms:  cache-hit item-4
3004ms:  Evict deleted item-4
3003ms:  5 get undefined
4003ms:  missing item-4
4003ms:  6 get undefinedIf you like this library and want to support then please give a star on GitHub.
Fund me so I can give more attention to the products and services you liked.

