mirror of
https://github.com/danny-avila/LibreChat.git
synced 2025-09-22 06:00:56 +02:00
🔄 refactor: Consolidate Tokenizer; Fix Jest Open Handles (#5175)
* refactor: consolidate tokenizer to singleton * fix: remove legacy tokenizer code, add Tokenizer singleton tests * ci: fix jest open handles
This commit is contained in:
parent
bf0a84e45a
commit
c26b54c74d
11 changed files with 202 additions and 221 deletions
47
api/cache/getLogStores.js
vendored
47
api/cache/getLogStores.js
vendored
|
@ -5,7 +5,7 @@ const { math, isEnabled } = require('~/server/utils');
|
|||
const keyvRedis = require('./keyvRedis');
|
||||
const keyvMongo = require('./keyvMongo');
|
||||
|
||||
const { BAN_DURATION, USE_REDIS, DEBUG_MEMORY_CACHE } = process.env ?? {};
|
||||
const { BAN_DURATION, USE_REDIS, DEBUG_MEMORY_CACHE, CI } = process.env ?? {};
|
||||
|
||||
const duration = math(BAN_DURATION, 7200000);
|
||||
const isRedisEnabled = isEnabled(USE_REDIS);
|
||||
|
@ -95,10 +95,8 @@ const namespaces = {
|
|||
* @returns {Keyv[]}
|
||||
*/
|
||||
function getTTLStores() {
|
||||
return Object.values(namespaces).filter((store) =>
|
||||
store instanceof Keyv &&
|
||||
typeof store.opts?.ttl === 'number' &&
|
||||
store.opts.ttl > 0,
|
||||
return Object.values(namespaces).filter(
|
||||
(store) => store instanceof Keyv && typeof store.opts?.ttl === 'number' && store.opts.ttl > 0,
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -125,23 +123,28 @@ async function clearExpiredFromCache(cache) {
|
|||
for (const key of keys) {
|
||||
try {
|
||||
const raw = cache.opts.store.get(key);
|
||||
if (!raw) {continue;}
|
||||
if (!raw) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const data = cache.opts.deserialize(raw);
|
||||
// Check if the entry is older than TTL
|
||||
if (data?.expires && data.expires <= expiryTime) {
|
||||
const deleted = await cache.opts.store.delete(key);
|
||||
if (!deleted) {
|
||||
debugMemoryCache && console.warn(`[Cache] Error deleting entry: ${key} from ${cache.opts.namespace}`);
|
||||
debugMemoryCache &&
|
||||
console.warn(`[Cache] Error deleting entry: ${key} from ${cache.opts.namespace}`);
|
||||
continue;
|
||||
}
|
||||
cleared++;
|
||||
}
|
||||
} catch (error) {
|
||||
debugMemoryCache && console.log(`[Cache] Error processing entry from ${cache.opts.namespace}:`, error);
|
||||
debugMemoryCache &&
|
||||
console.log(`[Cache] Error processing entry from ${cache.opts.namespace}:`, error);
|
||||
const deleted = await cache.opts.store.delete(key);
|
||||
if (!deleted) {
|
||||
debugMemoryCache && console.warn(`[Cache] Error deleting entry: ${key} from ${cache.opts.namespace}`);
|
||||
debugMemoryCache &&
|
||||
console.warn(`[Cache] Error deleting entry: ${key} from ${cache.opts.namespace}`);
|
||||
continue;
|
||||
}
|
||||
cleared++;
|
||||
|
@ -149,7 +152,10 @@ async function clearExpiredFromCache(cache) {
|
|||
}
|
||||
|
||||
if (cleared > 0) {
|
||||
debugMemoryCache && console.log(`[Cache] Cleared ${cleared} entries older than ${ttl}ms from ${cache.opts.namespace}`);
|
||||
debugMemoryCache &&
|
||||
console.log(
|
||||
`[Cache] Cleared ${cleared} entries older than ${ttl}ms from ${cache.opts.namespace}`,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -157,7 +163,7 @@ const auditCache = () => {
|
|||
const ttlStores = getTTLStores();
|
||||
console.log('[Cache] Starting audit');
|
||||
|
||||
ttlStores.forEach(store => {
|
||||
ttlStores.forEach((store) => {
|
||||
if (!store?.opts?.store?.entries) {
|
||||
return;
|
||||
}
|
||||
|
@ -166,21 +172,20 @@ const auditCache = () => {
|
|||
count: store.opts.store.size,
|
||||
ttl: store.opts.ttl,
|
||||
keys: Array.from(store.opts.store.keys()),
|
||||
entriesWithTimestamps: Array.from(store.opts.store.entries())
|
||||
.map(([key, value]) => ({
|
||||
key,
|
||||
value,
|
||||
})),
|
||||
entriesWithTimestamps: Array.from(store.opts.store.entries()).map(([key, value]) => ({
|
||||
key,
|
||||
value,
|
||||
})),
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* Clears expired entries from all TTL-enabled stores
|
||||
*/
|
||||
* Clears expired entries from all TTL-enabled stores
|
||||
*/
|
||||
async function clearAllExpiredFromCache() {
|
||||
const ttlStores = getTTLStores();
|
||||
await Promise.all(ttlStores.map(store => clearExpiredFromCache(store)));
|
||||
await Promise.all(ttlStores.map((store) => clearExpiredFromCache(store)));
|
||||
|
||||
// Force garbage collection if available (Node.js with --expose-gc flag)
|
||||
if (global.gc) {
|
||||
|
@ -188,7 +193,7 @@ async function clearAllExpiredFromCache() {
|
|||
}
|
||||
}
|
||||
|
||||
if (!isRedisEnabled) {
|
||||
if (!isRedisEnabled && !isEnabled(CI)) {
|
||||
/** @type {Set<NodeJS.Timeout>} */
|
||||
const cleanupIntervals = new Set();
|
||||
|
||||
|
@ -221,7 +226,7 @@ if (!isRedisEnabled) {
|
|||
|
||||
const dispose = () => {
|
||||
debugMemoryCache && console.log('[Cache] Cleaning up and shutting down...');
|
||||
cleanupIntervals.forEach(interval => clearInterval(interval));
|
||||
cleanupIntervals.forEach((interval) => clearInterval(interval));
|
||||
cleanupIntervals.clear();
|
||||
|
||||
// One final cleanup before exit
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue