Skip to main content
Glama

Feishu MCP Server

cache.ts5.57 kB
import { Config } from './config.js'; import { Logger } from './logger.js'; /** * 缓存项接口 */ interface CacheItem<T> { data: T; timestamp: number; expiresAt: number; } /** * 缓存管理器类 * 提供内存缓存功能,支持TTL和最大容量限制 */ export class CacheManager { private static instance: CacheManager; private cache: Map<string, CacheItem<any>>; private readonly config: Config; /** * 私有构造函数,用于单例模式 */ private constructor() { this.cache = new Map(); this.config = Config.getInstance(); // 定期清理过期缓存 setInterval(() => { this.cleanExpiredCache(); }, 60000); // 每分钟清理一次过期缓存 } /** * 获取缓存管理器实例 * @returns 缓存管理器实例 */ public static getInstance(): CacheManager { if (!CacheManager.instance) { CacheManager.instance = new CacheManager(); } return CacheManager.instance; } /** * 设置缓存 * @param key 缓存键 * @param data 缓存数据 * @param ttl 缓存生存时间(秒),默认使用配置中的TTL * @returns 是否成功设置缓存 */ public set<T>(key: string, data: T, ttl?: number): boolean { if (!this.config.cache.enabled) { return false; } // 如果缓存已达到最大容量,清理最早的条目 if (this.cache.size >= this.config.cache.maxSize) { this.cleanOldestCache(); } const now = Date.now(); const actualTtl = ttl || this.config.cache.ttl; this.cache.set(key, { data, timestamp: now, expiresAt: now + (actualTtl * 1000) }); Logger.debug(`缓存设置: ${key} (TTL: ${actualTtl}秒)`); return true; } /** * 获取缓存 * @param key 缓存键 * @returns 缓存数据,如果未找到或已过期则返回null */ public get<T>(key: string): T | null { if (!this.config.cache.enabled) { return null; } const cacheItem = this.cache.get(key); if (!cacheItem) { Logger.debug(`缓存未命中: ${key}`); return null; } // 检查是否过期 if (Date.now() > cacheItem.expiresAt) { Logger.debug(`缓存已过期: ${key}`); this.cache.delete(key); return null; } Logger.debug(`缓存命中: ${key}`); return cacheItem.data as T; } /** * 删除缓存 * @param key 缓存键 * @returns 是否成功删除 */ public delete(key: string): boolean { if (!this.config.cache.enabled) { return false; } const result = this.cache.delete(key); if (result) { Logger.debug(`缓存删除: ${key}`); } return result; } /** * 清空所有缓存 */ public clear(): void { if (!this.config.cache.enabled) { return; } const size = this.cache.size; this.cache.clear(); Logger.debug(`清空全部缓存,删除了 ${size} 条记录`); } /** * 根据前缀清除缓存 * @param prefix 缓存键前缀 * @returns 清除的缓存数量 */ public clearByPrefix(prefix: string): number { if (!this.config.cache.enabled) { return 0; } let count = 0; for (const key of this.cache.keys()) { if (key.startsWith(prefix)) { this.cache.delete(key); count++; } } if (count > 0) { Logger.debug(`按前缀清除缓存: ${prefix}, 删除了 ${count} 条记录`); } return count; } /** * 清理过期缓存 * @returns 清理的缓存数量 */ private cleanExpiredCache(): number { if (!this.config.cache.enabled) { return 0; } const now = Date.now(); let count = 0; for (const [key, item] of this.cache.entries()) { if (now > item.expiresAt) { this.cache.delete(key); count++; } } if (count > 0) { Logger.debug(`清理过期缓存,删除了 ${count} 条记录`); } return count; } /** * 清理最旧的缓存 * @param count 要清理的条目数,默认为1 */ private cleanOldestCache(count: number = 1): void { if (!this.config.cache.enabled || this.cache.size === 0) { return; } // 按时间戳排序 const entries = Array.from(this.cache.entries()) .sort((a, b) => a[1].timestamp - b[1].timestamp); // 删除最早的几条记录 const toDelete = Math.min(count, entries.length); for (let i = 0; i < toDelete; i++) { this.cache.delete(entries[i][0]); } Logger.debug(`清理最旧缓存,删除了 ${toDelete} 条记录`); } /** * 获取缓存统计信息 * @returns 缓存统计信息对象 */ public getStats(): { size: number; enabled: boolean; maxSize: number; ttl: number } { return { size: this.cache.size, enabled: this.config.cache.enabled, maxSize: this.config.cache.maxSize, ttl: this.config.cache.ttl }; } /** * 缓存Wiki到文档ID的转换结果 * @param wikiToken Wiki Token * @param documentId 文档ID * @returns 是否成功设置缓存 */ public cacheWikiToDocId(wikiToken: string, documentId: string): boolean { return this.set(`wiki:${wikiToken}`, documentId); } /** * 获取缓存的Wiki转换结果 * @param wikiToken Wiki Token * @returns 文档ID,如果未找到或已过期则返回null */ public getWikiToDocId(wikiToken: string): string | null { return this.get<string>(`wiki:${wikiToken}`); } }

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/cso1z/Feishu-MCP'

If you have feedback or need assistance with the MCP directory API, please join our Discord server