@dollhousemcp/mcp-server 1.5.2 → 1.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +56 -0
- package/README.md +494 -111
- package/data/agents/code-reviewer.md +8 -1
- package/data/agents/research-assistant.md +8 -1
- package/data/agents/task-manager.md +8 -1
- package/data/ensembles/business-advisor.md +8 -1
- package/data/ensembles/creative-studio.md +8 -1
- package/data/ensembles/development-team.md +8 -1
- package/data/ensembles/security-analysis-team.md +8 -1
- package/data/memories/conversation-history.md +8 -1
- package/data/memories/learning-progress.md +8 -1
- package/data/memories/project-context.md +8 -1
- package/data/personas/business-consultant.md +8 -1
- package/data/personas/creative-writer.md +8 -1
- package/data/personas/debug-detective.md +8 -1
- package/data/personas/eli5-explainer.md +8 -1
- package/data/personas/security-analyst.md +8 -1
- package/data/personas/technical-analyst.md +8 -1
- package/data/skills/code-review.md +8 -1
- package/data/skills/creative-writing.md +8 -1
- package/data/skills/data-analysis.md +8 -1
- package/data/skills/penetration-testing.md +8 -1
- package/data/skills/research.md +8 -1
- package/data/skills/threat-modeling.md +8 -1
- package/data/skills/translation.md +8 -1
- package/data/templates/code-documentation.md +8 -1
- package/data/templates/email-professional.md +8 -1
- package/data/templates/meeting-notes.md +8 -1
- package/data/templates/penetration-test-report.md +8 -1
- package/data/templates/project-brief.md +8 -1
- package/data/templates/report-executive.md +8 -1
- package/data/templates/security-vulnerability-report.md +8 -1
- package/data/templates/threat-assessment-report.md +8 -1
- package/dist/auth/GitHubAuthManager.d.ts +6 -1
- package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
- package/dist/auth/GitHubAuthManager.js +45 -18
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
- package/dist/cache/CollectionCache.d.ts.map +1 -1
- package/dist/cache/CollectionCache.js +13 -3
- package/dist/cache/CollectionIndexCache.d.ts +77 -0
- package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
- package/dist/cache/CollectionIndexCache.js +349 -0
- package/dist/cache/LRUCache.d.ts +93 -0
- package/dist/cache/LRUCache.d.ts.map +1 -0
- package/dist/cache/LRUCache.js +299 -0
- package/dist/cache/index.d.ts +1 -0
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +2 -1
- package/dist/collection/CollectionBrowser.d.ts +21 -1
- package/dist/collection/CollectionBrowser.d.ts.map +1 -1
- package/dist/collection/CollectionBrowser.js +130 -10
- package/dist/collection/CollectionIndexManager.d.ts +151 -0
- package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
- package/dist/collection/CollectionIndexManager.js +499 -0
- package/dist/collection/CollectionSearch.d.ts +55 -0
- package/dist/collection/CollectionSearch.d.ts.map +1 -1
- package/dist/collection/CollectionSearch.js +338 -13
- package/dist/collection/CollectionSeeder.d.ts.map +1 -1
- package/dist/collection/CollectionSeeder.js +38 -1
- package/dist/collection/ElementInstaller.d.ts +31 -0
- package/dist/collection/ElementInstaller.d.ts.map +1 -1
- package/dist/collection/ElementInstaller.js +77 -15
- package/dist/collection/PersonaSubmitter.d.ts +1 -1
- package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
- package/dist/collection/PersonaSubmitter.js +2 -2
- package/dist/collection/index.d.ts +1 -0
- package/dist/collection/index.d.ts.map +1 -1
- package/dist/collection/index.js +2 -1
- package/dist/config/ConfigManager.d.ts +78 -0
- package/dist/config/ConfigManager.d.ts.map +1 -0
- package/dist/config/ConfigManager.js +216 -0
- package/dist/config/element-types.d.ts +135 -0
- package/dist/config/element-types.d.ts.map +1 -0
- package/dist/config/element-types.js +108 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.d.ts.map +1 -1
- package/dist/config/index.js +3 -1
- package/dist/config/portfolio-constants.d.ts +83 -0
- package/dist/config/portfolio-constants.d.ts.map +1 -0
- package/dist/config/portfolio-constants.js +99 -0
- package/dist/elements/BaseElement.d.ts +14 -2
- package/dist/elements/BaseElement.d.ts.map +1 -1
- package/dist/elements/BaseElement.js +88 -6
- package/dist/elements/agents/Agent.d.ts +10 -1
- package/dist/elements/agents/Agent.d.ts.map +1 -1
- package/dist/elements/agents/Agent.js +66 -19
- package/dist/elements/agents/AgentManager.d.ts +2 -0
- package/dist/elements/agents/AgentManager.d.ts.map +1 -1
- package/dist/elements/agents/AgentManager.js +12 -10
- package/dist/elements/skills/Skill.d.ts +10 -1
- package/dist/elements/skills/Skill.d.ts.map +1 -1
- package/dist/elements/skills/Skill.js +40 -3
- package/dist/elements/skills/SkillManager.d.ts +1 -0
- package/dist/elements/skills/SkillManager.d.ts.map +1 -1
- package/dist/elements/skills/SkillManager.js +10 -4
- package/dist/elements/templates/Template.d.ts +10 -1
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +35 -18
- package/dist/elements/templates/TemplateManager.d.ts +1 -1
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
- package/dist/elements/templates/TemplateManager.js +6 -5
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/index.barrel.d.ts +1 -2
- package/dist/index.barrel.d.ts.map +1 -1
- package/dist/index.barrel.js +2 -4
- package/dist/index.d.ts +143 -25
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1883 -310
- package/dist/persona/PersonaElement.d.ts +10 -0
- package/dist/persona/PersonaElement.d.ts.map +1 -1
- package/dist/persona/PersonaElement.js +55 -32
- package/dist/persona/PersonaElementManager.d.ts.map +1 -1
- package/dist/persona/PersonaElementManager.js +13 -11
- package/dist/persona/PersonaLoader.d.ts.map +1 -1
- package/dist/persona/PersonaLoader.js +8 -2
- package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaImporter.js +24 -5
- package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
- package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaSharer.js +198 -22
- package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
- package/dist/portfolio/DefaultElementProvider.js +499 -7
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
- package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
- package/dist/portfolio/MigrationManager.d.ts.map +1 -1
- package/dist/portfolio/MigrationManager.js +136 -3
- package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
- package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioIndexManager.js +478 -0
- package/dist/portfolio/PortfolioManager.d.ts +5 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioManager.js +61 -20
- package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioRepoManager.js +337 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
- package/dist/portfolio/UnifiedIndexManager.js +1434 -0
- package/dist/portfolio/index.d.ts +15 -0
- package/dist/portfolio/index.d.ts.map +1 -0
- package/dist/portfolio/index.js +15 -0
- package/dist/portfolio/types.d.ts +7 -0
- package/dist/portfolio/types.d.ts.map +1 -1
- package/dist/portfolio/types.js +6 -1
- package/dist/security/InputValidator.d.ts.map +1 -1
- package/dist/security/InputValidator.js +50 -48
- package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
- package/dist/security/audit/SecurityAuditor.js +17 -9
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +19 -3
- package/dist/security/contentValidator.d.ts +2 -0
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +115 -4
- package/dist/security/secureYamlParser.d.ts +1 -0
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +29 -7
- package/dist/security/securityMonitor.d.ts +1 -1
- package/dist/security/securityMonitor.d.ts.map +1 -1
- package/dist/security/securityMonitor.js +1 -1
- package/dist/security/tokenManager.d.ts +1 -1
- package/dist/security/tokenManager.d.ts.map +1 -1
- package/dist/security/tokenManager.js +30 -10
- package/dist/server/ServerSetup.d.ts +22 -2
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +77 -12
- package/dist/server/tools/AuthTools.d.ts.map +1 -1
- package/dist/server/tools/AuthTools.js +33 -1
- package/dist/server/tools/BuildInfoTools.d.ts +25 -0
- package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
- package/dist/server/tools/BuildInfoTools.js +36 -0
- package/dist/server/tools/CollectionTools.d.ts.map +1 -1
- package/dist/server/tools/CollectionTools.js +55 -46
- package/dist/server/tools/ConfigTools.d.ts.map +1 -1
- package/dist/server/tools/ConfigTools.js +29 -1
- package/dist/server/tools/PersonaTools.d.ts +4 -2
- package/dist/server/tools/PersonaTools.d.ts.map +1 -1
- package/dist/server/tools/PersonaTools.js +5 -152
- package/dist/server/tools/PortfolioTools.d.ts +12 -0
- package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
- package/dist/server/tools/PortfolioTools.js +221 -0
- package/dist/server/tools/index.d.ts +3 -1
- package/dist/server/tools/index.d.ts.map +1 -1
- package/dist/server/tools/index.js +4 -2
- package/dist/server/types.d.ts +40 -5
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/services/BuildInfoService.d.ts +84 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -0
- package/dist/services/BuildInfoService.js +271 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
- package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
- package/dist/tools/portfolio/types.d.ts +41 -0
- package/dist/tools/portfolio/types.d.ts.map +1 -0
- package/dist/tools/portfolio/types.js +15 -0
- package/dist/types/collection.d.ts +51 -0
- package/dist/types/collection.d.ts.map +1 -1
- package/dist/types/collection.js +1 -1
- package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
- package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
- package/dist/utils/EarlyTerminationSearch.js +164 -0
- package/dist/utils/ErrorHandler.d.ts +86 -0
- package/dist/utils/ErrorHandler.d.ts.map +1 -0
- package/dist/utils/ErrorHandler.js +201 -0
- package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
- package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
- package/dist/utils/FileDiscoveryUtil.js +169 -0
- package/dist/utils/GitHubRateLimiter.d.ts +88 -0
- package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
- package/dist/utils/GitHubRateLimiter.js +315 -0
- package/dist/utils/PerformanceMonitor.d.ts +134 -0
- package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
- package/dist/utils/PerformanceMonitor.js +347 -0
- package/dist/utils/RateLimiter.d.ts.map +1 -0
- package/dist/utils/RateLimiter.js +172 -0
- package/dist/utils/SecureDownloader.d.ts +241 -0
- package/dist/utils/SecureDownloader.d.ts.map +1 -0
- package/dist/utils/SecureDownloader.js +759 -0
- package/dist/utils/ToolCache.d.ts +82 -0
- package/dist/utils/ToolCache.d.ts.map +1 -0
- package/dist/utils/ToolCache.js +196 -0
- package/dist/utils/errorCodes.d.ts +136 -0
- package/dist/utils/errorCodes.d.ts.map +1 -0
- package/dist/utils/errorCodes.js +87 -0
- package/dist/utils/index.d.ts +3 -0
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +4 -1
- package/dist/utils/installation.d.ts +1 -1
- package/dist/utils/installation.d.ts.map +1 -1
- package/dist/utils/installation.js +9 -8
- package/dist/utils/searchUtils.d.ts +31 -0
- package/dist/utils/searchUtils.d.ts.map +1 -1
- package/dist/utils/searchUtils.js +62 -1
- package/package.json +17 -7
- package/dist/config/updateConfig.d.ts +0 -84
- package/dist/config/updateConfig.d.ts.map +0 -1
- package/dist/config/updateConfig.js +0 -148
- package/dist/server/tools/UpdateTools.d.ts +0 -10
- package/dist/server/tools/UpdateTools.d.ts.map +0 -1
- package/dist/server/tools/UpdateTools.js +0 -85
- package/dist/update/BackupManager.d.ts +0 -63
- package/dist/update/BackupManager.d.ts.map +0 -1
- package/dist/update/BackupManager.js +0 -370
- package/dist/update/DependencyChecker.d.ts +0 -41
- package/dist/update/DependencyChecker.d.ts.map +0 -1
- package/dist/update/DependencyChecker.js +0 -132
- package/dist/update/RateLimiter.d.ts.map +0 -1
- package/dist/update/RateLimiter.js +0 -172
- package/dist/update/SignatureVerifier.d.ts +0 -71
- package/dist/update/SignatureVerifier.d.ts.map +0 -1
- package/dist/update/SignatureVerifier.js +0 -214
- package/dist/update/UpdateChecker.d.ts +0 -132
- package/dist/update/UpdateChecker.d.ts.map +0 -1
- package/dist/update/UpdateChecker.js +0 -506
- package/dist/update/UpdateManager.d.ts +0 -60
- package/dist/update/UpdateManager.d.ts.map +0 -1
- package/dist/update/UpdateManager.js +0 -730
- package/dist/update/VersionManager.d.ts +0 -31
- package/dist/update/VersionManager.d.ts.map +0 -1
- package/dist/update/VersionManager.js +0 -181
- package/dist/update/index.d.ts +0 -9
- package/dist/update/index.d.ts.map +0 -1
- package/dist/update/index.js +0 -9
- /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
|
@@ -0,0 +1,499 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Collection Index Manager with Background Refresh and Robust Caching
|
|
3
|
+
*
|
|
4
|
+
* This manager implements:
|
|
5
|
+
* - 1-hour TTL with local file caching
|
|
6
|
+
* - Background refresh without blocking operations
|
|
7
|
+
* - Exponential backoff retry logic
|
|
8
|
+
* - Configurable timeouts via environment variables
|
|
9
|
+
* - Return stale cache while refreshing in background
|
|
10
|
+
* - Comprehensive error handling for production use
|
|
11
|
+
*/
|
|
12
|
+
import * as fs from 'fs/promises';
|
|
13
|
+
import * as path from 'path';
|
|
14
|
+
import * as os from 'os';
|
|
15
|
+
import { logger } from '../utils/logger.js';
|
|
16
|
+
export class CollectionIndexManager {
|
|
17
|
+
INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';
|
|
18
|
+
TTL_MS;
|
|
19
|
+
FETCH_TIMEOUT_MS;
|
|
20
|
+
MAX_RETRIES;
|
|
21
|
+
BASE_RETRY_DELAY_MS;
|
|
22
|
+
MAX_RETRY_DELAY_MS;
|
|
23
|
+
CACHE_FILE;
|
|
24
|
+
cachedIndex = null;
|
|
25
|
+
backgroundRefreshPromise = null;
|
|
26
|
+
isRefreshing = false;
|
|
27
|
+
circuitBreakerFailures = 0;
|
|
28
|
+
circuitBreakerLastFailure = 0;
|
|
29
|
+
CIRCUIT_BREAKER_THRESHOLD = 5;
|
|
30
|
+
CIRCUIT_BREAKER_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes
|
|
31
|
+
REFRESH_THRESHOLD = 0.8; // Refresh when 80% of TTL has passed
|
|
32
|
+
JITTER_FACTOR = 0.25; // ±25% randomness for jitter
|
|
33
|
+
// Default configuration constants
|
|
34
|
+
DEFAULT_TTL_MS = 60 * 60 * 1000; // 1 hour
|
|
35
|
+
DEFAULT_MAX_RETRIES = 3;
|
|
36
|
+
DEFAULT_BASE_RETRY_DELAY_MS = 1000;
|
|
37
|
+
DEFAULT_MAX_RETRY_DELAY_MS = 30000;
|
|
38
|
+
DEFAULT_FETCH_TIMEOUT_MS = 5000; // 5 seconds
|
|
39
|
+
CHECKSUM_LENGTH = 8;
|
|
40
|
+
JSON_INDENT = 2;
|
|
41
|
+
constructor(config = {}) {
|
|
42
|
+
// Configuration with environment variable overrides
|
|
43
|
+
this.TTL_MS = config.ttlMs || this.DEFAULT_TTL_MS;
|
|
44
|
+
this.FETCH_TIMEOUT_MS = this.parseFetchTimeout(config.fetchTimeoutMs);
|
|
45
|
+
this.MAX_RETRIES = config.maxRetries || this.DEFAULT_MAX_RETRIES;
|
|
46
|
+
this.BASE_RETRY_DELAY_MS = config.baseRetryDelayMs || this.DEFAULT_BASE_RETRY_DELAY_MS;
|
|
47
|
+
this.MAX_RETRY_DELAY_MS = config.maxRetryDelayMs || this.DEFAULT_MAX_RETRY_DELAY_MS;
|
|
48
|
+
// Cache directory - use ~/.dollhouse/cache/collection-index.json as specified
|
|
49
|
+
const cacheDir = config.cacheDir || path.join(os.homedir(), '.dollhouse', 'cache');
|
|
50
|
+
this.CACHE_FILE = path.join(cacheDir, 'collection-index.json');
|
|
51
|
+
logger.debug('CollectionIndexManager initialized', {
|
|
52
|
+
ttlMs: this.TTL_MS,
|
|
53
|
+
fetchTimeoutMs: this.FETCH_TIMEOUT_MS,
|
|
54
|
+
cacheFile: this.CACHE_FILE,
|
|
55
|
+
maxRetries: this.MAX_RETRIES
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Parse fetch timeout from config or environment variable
|
|
60
|
+
*/
|
|
61
|
+
parseFetchTimeout(configValue) {
|
|
62
|
+
// Check environment variable first
|
|
63
|
+
const envTimeout = process.env.COLLECTION_FETCH_TIMEOUT;
|
|
64
|
+
if (envTimeout) {
|
|
65
|
+
const parsed = parseInt(envTimeout, 10);
|
|
66
|
+
if (!isNaN(parsed) && parsed > 0) {
|
|
67
|
+
logger.debug(`Using COLLECTION_FETCH_TIMEOUT from environment: ${parsed}ms`);
|
|
68
|
+
return parsed;
|
|
69
|
+
}
|
|
70
|
+
logger.warn(`Invalid COLLECTION_FETCH_TIMEOUT value: ${envTimeout}, using default`);
|
|
71
|
+
}
|
|
72
|
+
// Fall back to config value or default
|
|
73
|
+
return configValue || this.DEFAULT_FETCH_TIMEOUT_MS;
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Get collection index with stale-while-revalidate pattern
|
|
77
|
+
* Returns cached data immediately if available, refreshes in background
|
|
78
|
+
*/
|
|
79
|
+
async getIndex() {
|
|
80
|
+
try {
|
|
81
|
+
// Load from memory cache first
|
|
82
|
+
if (!this.cachedIndex) {
|
|
83
|
+
await this.loadFromDisk();
|
|
84
|
+
}
|
|
85
|
+
// Check if we should return stale cache while refreshing
|
|
86
|
+
const shouldRefresh = this.shouldRefreshCache();
|
|
87
|
+
if (this.cachedIndex && !this.isCacheExpired()) {
|
|
88
|
+
// Cache is valid, return immediately
|
|
89
|
+
logger.debug('Returning valid cached collection index');
|
|
90
|
+
// Start background refresh if needed but not already running
|
|
91
|
+
if (shouldRefresh && !this.isRefreshing) {
|
|
92
|
+
this.startBackgroundRefresh();
|
|
93
|
+
}
|
|
94
|
+
return this.cachedIndex.data;
|
|
95
|
+
}
|
|
96
|
+
// If we have stale cache, return it while refreshing in background
|
|
97
|
+
if (this.cachedIndex && this.isCacheExpired()) {
|
|
98
|
+
logger.debug('Returning stale cache while refreshing in background');
|
|
99
|
+
// Start background refresh if not already running
|
|
100
|
+
if (!this.isRefreshing) {
|
|
101
|
+
this.startBackgroundRefresh();
|
|
102
|
+
}
|
|
103
|
+
return this.cachedIndex.data;
|
|
104
|
+
}
|
|
105
|
+
// No cache available, must fetch synchronously
|
|
106
|
+
logger.debug('No cache available, fetching collection index synchronously');
|
|
107
|
+
const freshIndex = await this.fetchWithRetry();
|
|
108
|
+
await this.updateCache(freshIndex);
|
|
109
|
+
return freshIndex.data;
|
|
110
|
+
}
|
|
111
|
+
catch (error) {
|
|
112
|
+
logger.error('Failed to get collection index', { error: this.getErrorMessage(error) });
|
|
113
|
+
// If we have any cached data (even expired), return it as last resort
|
|
114
|
+
if (this.cachedIndex) {
|
|
115
|
+
logger.warn('Returning expired cache as last resort');
|
|
116
|
+
return this.cachedIndex.data;
|
|
117
|
+
}
|
|
118
|
+
throw new Error(`Collection index not available: ${this.getErrorMessage(error)}`);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Force refresh the collection index
|
|
123
|
+
*/
|
|
124
|
+
async forceRefresh() {
|
|
125
|
+
logger.debug('Force refreshing collection index');
|
|
126
|
+
try {
|
|
127
|
+
const freshIndex = await this.fetchWithRetry();
|
|
128
|
+
await this.updateCache(freshIndex);
|
|
129
|
+
return freshIndex.data;
|
|
130
|
+
}
|
|
131
|
+
catch (error) {
|
|
132
|
+
logger.error('Force refresh failed', { error: this.getErrorMessage(error) });
|
|
133
|
+
// If we have cached data, return it
|
|
134
|
+
if (this.cachedIndex) {
|
|
135
|
+
logger.warn('Force refresh failed, returning cached data');
|
|
136
|
+
return this.cachedIndex.data;
|
|
137
|
+
}
|
|
138
|
+
throw error;
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Check if cache should be refreshed (within TTL but getting close to expiry)
|
|
143
|
+
*/
|
|
144
|
+
shouldRefreshCache() {
|
|
145
|
+
if (!this.cachedIndex)
|
|
146
|
+
return true;
|
|
147
|
+
const age = Date.now() - this.cachedIndex.timestamp;
|
|
148
|
+
const refreshThreshold = this.TTL_MS * this.REFRESH_THRESHOLD;
|
|
149
|
+
return age > refreshThreshold;
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Check if cache is expired
|
|
153
|
+
*/
|
|
154
|
+
isCacheExpired() {
|
|
155
|
+
if (!this.cachedIndex)
|
|
156
|
+
return true;
|
|
157
|
+
const age = Date.now() - this.cachedIndex.timestamp;
|
|
158
|
+
return age > this.TTL_MS;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Start background refresh without blocking
|
|
162
|
+
*/
|
|
163
|
+
startBackgroundRefresh() {
|
|
164
|
+
if (this.isRefreshing) {
|
|
165
|
+
logger.debug('Background refresh already in progress');
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
// Check circuit breaker
|
|
169
|
+
if (this.isCircuitBreakerOpen()) {
|
|
170
|
+
logger.debug('Circuit breaker open, skipping background refresh');
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
this.isRefreshing = true;
|
|
174
|
+
this.backgroundRefreshPromise = this.performBackgroundRefresh()
|
|
175
|
+
.catch(error => {
|
|
176
|
+
logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });
|
|
177
|
+
this.recordCircuitBreakerFailure();
|
|
178
|
+
})
|
|
179
|
+
.finally(() => {
|
|
180
|
+
this.isRefreshing = false;
|
|
181
|
+
this.backgroundRefreshPromise = null;
|
|
182
|
+
});
|
|
183
|
+
}
|
|
184
|
+
/**
|
|
185
|
+
* Perform background refresh
|
|
186
|
+
*/
|
|
187
|
+
async performBackgroundRefresh() {
|
|
188
|
+
logger.debug('Starting background refresh of collection index');
|
|
189
|
+
try {
|
|
190
|
+
const freshIndex = await this.fetchWithRetry();
|
|
191
|
+
await this.updateCache(freshIndex);
|
|
192
|
+
// Reset circuit breaker on success
|
|
193
|
+
this.circuitBreakerFailures = 0;
|
|
194
|
+
logger.debug('Background refresh completed successfully');
|
|
195
|
+
}
|
|
196
|
+
catch (error) {
|
|
197
|
+
logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });
|
|
198
|
+
throw error;
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Fetch collection index with retry logic and exponential backoff
|
|
203
|
+
*/
|
|
204
|
+
async fetchWithRetry() {
|
|
205
|
+
let lastError = null;
|
|
206
|
+
for (let attempt = 0; attempt <= this.MAX_RETRIES; attempt++) {
|
|
207
|
+
try {
|
|
208
|
+
if (attempt > 0) {
|
|
209
|
+
const delay = this.calculateRetryDelay(attempt);
|
|
210
|
+
logger.debug(`Retrying fetch in ${delay}ms (attempt ${attempt + 1}/${this.MAX_RETRIES + 1})`);
|
|
211
|
+
await this.sleep(delay);
|
|
212
|
+
}
|
|
213
|
+
return await this.fetchCollectionIndex();
|
|
214
|
+
}
|
|
215
|
+
catch (error) {
|
|
216
|
+
lastError = error instanceof Error ? error : new Error(String(error));
|
|
217
|
+
logger.debug(`Fetch attempt ${attempt + 1} failed`, {
|
|
218
|
+
error: this.getErrorMessage(lastError),
|
|
219
|
+
willRetry: attempt < this.MAX_RETRIES
|
|
220
|
+
});
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
throw lastError || new Error('All fetch attempts failed');
|
|
224
|
+
}
|
|
225
|
+
/**
|
|
226
|
+
* Calculate retry delay with exponential backoff and jitter
|
|
227
|
+
*/
|
|
228
|
+
calculateRetryDelay(attempt) {
|
|
229
|
+
// Exponential backoff: baseDelay * (2 ^ attempt)
|
|
230
|
+
const exponentialDelay = this.BASE_RETRY_DELAY_MS * Math.pow(2, attempt - 1);
|
|
231
|
+
// Cap at maximum delay
|
|
232
|
+
const cappedDelay = Math.min(exponentialDelay, this.MAX_RETRY_DELAY_MS);
|
|
233
|
+
// Add jitter to prevent thundering herd
|
|
234
|
+
return this.addJitter(cappedDelay);
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* Add jitter (±25% randomness) to a delay to prevent thundering herd problems
|
|
238
|
+
*/
|
|
239
|
+
addJitter(delay) {
|
|
240
|
+
const jitter = delay * this.JITTER_FACTOR * (Math.random() - 0.5);
|
|
241
|
+
return Math.max(0, delay + jitter);
|
|
242
|
+
}
|
|
243
|
+
/**
|
|
244
|
+
* Fetch collection index from GitHub
|
|
245
|
+
*/
|
|
246
|
+
async fetchCollectionIndex() {
|
|
247
|
+
const controller = new AbortController();
|
|
248
|
+
const timeoutId = setTimeout(() => controller.abort(), this.FETCH_TIMEOUT_MS);
|
|
249
|
+
try {
|
|
250
|
+
// Build headers for conditional requests
|
|
251
|
+
const headers = {
|
|
252
|
+
'Accept': 'application/json',
|
|
253
|
+
'User-Agent': 'DollhouseMCP/1.0',
|
|
254
|
+
'Cache-Control': 'no-cache'
|
|
255
|
+
};
|
|
256
|
+
// Add conditional headers if we have cached data
|
|
257
|
+
if (this.cachedIndex?.etag) {
|
|
258
|
+
headers['If-None-Match'] = this.cachedIndex.etag;
|
|
259
|
+
}
|
|
260
|
+
if (this.cachedIndex?.lastModified) {
|
|
261
|
+
headers['If-Modified-Since'] = this.cachedIndex.lastModified;
|
|
262
|
+
}
|
|
263
|
+
logger.debug('Fetching collection index', {
|
|
264
|
+
url: this.INDEX_URL,
|
|
265
|
+
timeout: this.FETCH_TIMEOUT_MS,
|
|
266
|
+
hasEtag: !!this.cachedIndex?.etag
|
|
267
|
+
});
|
|
268
|
+
const response = await fetch(this.INDEX_URL, {
|
|
269
|
+
headers,
|
|
270
|
+
signal: controller.signal
|
|
271
|
+
});
|
|
272
|
+
clearTimeout(timeoutId);
|
|
273
|
+
// Handle 304 Not Modified
|
|
274
|
+
if (response.status === 304 && this.cachedIndex) {
|
|
275
|
+
logger.debug('Collection index not modified (304), updating cache timestamp');
|
|
276
|
+
this.cachedIndex.timestamp = Date.now();
|
|
277
|
+
await this.saveToDisk();
|
|
278
|
+
return {
|
|
279
|
+
data: this.cachedIndex.data,
|
|
280
|
+
etag: this.cachedIndex.etag,
|
|
281
|
+
lastModified: this.cachedIndex.lastModified
|
|
282
|
+
};
|
|
283
|
+
}
|
|
284
|
+
if (!response.ok) {
|
|
285
|
+
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
|
|
286
|
+
}
|
|
287
|
+
const indexData = await response.json();
|
|
288
|
+
// Validate the index structure
|
|
289
|
+
this.validateIndexStructure(indexData);
|
|
290
|
+
const etag = response.headers.get('etag') || undefined;
|
|
291
|
+
const lastModified = response.headers.get('last-modified') || undefined;
|
|
292
|
+
logger.debug('Collection index fetched successfully', {
|
|
293
|
+
totalElements: indexData.total_elements,
|
|
294
|
+
version: indexData.version,
|
|
295
|
+
hasEtag: !!etag
|
|
296
|
+
});
|
|
297
|
+
return { data: indexData, etag, lastModified };
|
|
298
|
+
}
|
|
299
|
+
catch (error) {
|
|
300
|
+
clearTimeout(timeoutId);
|
|
301
|
+
if (error instanceof Error && error.name === 'AbortError') {
|
|
302
|
+
throw new Error(`Fetch timeout after ${this.FETCH_TIMEOUT_MS}ms`);
|
|
303
|
+
}
|
|
304
|
+
throw error;
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
/**
|
|
308
|
+
* Validate collection index structure
|
|
309
|
+
*/
|
|
310
|
+
validateIndexStructure(index) {
|
|
311
|
+
if (!index || typeof index !== 'object') {
|
|
312
|
+
throw new Error('Invalid index: not an object');
|
|
313
|
+
}
|
|
314
|
+
if (typeof index.version !== 'string') {
|
|
315
|
+
throw new Error('Invalid index: missing or invalid version');
|
|
316
|
+
}
|
|
317
|
+
if (typeof index.generated !== 'string') {
|
|
318
|
+
throw new Error('Invalid index: missing or invalid generated timestamp');
|
|
319
|
+
}
|
|
320
|
+
if (typeof index.total_elements !== 'number') {
|
|
321
|
+
throw new Error('Invalid index: missing or invalid total_elements');
|
|
322
|
+
}
|
|
323
|
+
if (!index.index || typeof index.index !== 'object') {
|
|
324
|
+
throw new Error('Invalid index: missing or invalid index object');
|
|
325
|
+
}
|
|
326
|
+
if (!index.metadata || typeof index.metadata !== 'object') {
|
|
327
|
+
throw new Error('Invalid index: missing or invalid metadata');
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
/**
|
|
331
|
+
* Update cache with new data
|
|
332
|
+
*/
|
|
333
|
+
async updateCache(fetchResult) {
|
|
334
|
+
const checksum = this.calculateChecksum(fetchResult.data);
|
|
335
|
+
this.cachedIndex = {
|
|
336
|
+
data: fetchResult.data,
|
|
337
|
+
timestamp: Date.now(),
|
|
338
|
+
etag: fetchResult.etag,
|
|
339
|
+
lastModified: fetchResult.lastModified,
|
|
340
|
+
version: fetchResult.data.version,
|
|
341
|
+
checksum
|
|
342
|
+
};
|
|
343
|
+
await this.saveToDisk();
|
|
344
|
+
logger.debug('Collection index cache updated', {
|
|
345
|
+
version: fetchResult.data.version,
|
|
346
|
+
totalElements: fetchResult.data.total_elements,
|
|
347
|
+
checksum
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
/**
|
|
351
|
+
* Calculate checksum for data integrity verification
|
|
352
|
+
*/
|
|
353
|
+
calculateChecksum(data) {
|
|
354
|
+
// Simple checksum based on key properties
|
|
355
|
+
const checksumData = {
|
|
356
|
+
version: data.version,
|
|
357
|
+
generated: data.generated,
|
|
358
|
+
total_elements: data.total_elements
|
|
359
|
+
};
|
|
360
|
+
return Buffer.from(JSON.stringify(checksumData)).toString('base64').substring(0, this.CHECKSUM_LENGTH);
|
|
361
|
+
}
|
|
362
|
+
/**
|
|
363
|
+
* Load cache from disk
|
|
364
|
+
*/
|
|
365
|
+
async loadFromDisk() {
|
|
366
|
+
try {
|
|
367
|
+
const data = await fs.readFile(this.CACHE_FILE, 'utf8');
|
|
368
|
+
const cached = JSON.parse(data);
|
|
369
|
+
// Validate cache structure
|
|
370
|
+
if (!cached.data || !cached.timestamp || !cached.version) {
|
|
371
|
+
logger.debug('Invalid cache structure, ignoring');
|
|
372
|
+
return;
|
|
373
|
+
}
|
|
374
|
+
// Verify checksum if available
|
|
375
|
+
if (cached.checksum) {
|
|
376
|
+
const expectedChecksum = this.calculateChecksum(cached.data);
|
|
377
|
+
if (cached.checksum !== expectedChecksum) {
|
|
378
|
+
logger.debug('Cache checksum mismatch, ignoring cached data');
|
|
379
|
+
return;
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
this.cachedIndex = cached;
|
|
383
|
+
const age = Date.now() - cached.timestamp;
|
|
384
|
+
const isExpired = age > this.TTL_MS;
|
|
385
|
+
logger.debug('Loaded collection index from disk cache', {
|
|
386
|
+
version: cached.version,
|
|
387
|
+
age: Math.round(age / 1000),
|
|
388
|
+
isExpired,
|
|
389
|
+
totalElements: cached.data.total_elements
|
|
390
|
+
});
|
|
391
|
+
}
|
|
392
|
+
catch (error) {
|
|
393
|
+
if (error.code !== 'ENOENT') {
|
|
394
|
+
logger.debug('Failed to load cache from disk', { error: this.getErrorMessage(error) });
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
}
|
|
398
|
+
/**
|
|
399
|
+
* Save cache to disk
|
|
400
|
+
*/
|
|
401
|
+
async saveToDisk() {
|
|
402
|
+
if (!this.cachedIndex)
|
|
403
|
+
return;
|
|
404
|
+
try {
|
|
405
|
+
// Ensure cache directory exists
|
|
406
|
+
await fs.mkdir(path.dirname(this.CACHE_FILE), { recursive: true });
|
|
407
|
+
const cacheData = JSON.stringify(this.cachedIndex, null, this.JSON_INDENT);
|
|
408
|
+
await fs.writeFile(this.CACHE_FILE, cacheData, 'utf8');
|
|
409
|
+
logger.debug('Collection index cache saved to disk');
|
|
410
|
+
}
|
|
411
|
+
catch (error) {
|
|
412
|
+
logger.debug('Failed to save cache to disk', { error: this.getErrorMessage(error) });
|
|
413
|
+
// Don't throw - cache persistence failures shouldn't break functionality
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
/**
|
|
417
|
+
* Circuit breaker logic
|
|
418
|
+
*/
|
|
419
|
+
isCircuitBreakerOpen() {
|
|
420
|
+
if (this.circuitBreakerFailures < this.CIRCUIT_BREAKER_THRESHOLD) {
|
|
421
|
+
return false;
|
|
422
|
+
}
|
|
423
|
+
const timeSinceLastFailure = Date.now() - this.circuitBreakerLastFailure;
|
|
424
|
+
return timeSinceLastFailure < this.CIRCUIT_BREAKER_TIMEOUT_MS;
|
|
425
|
+
}
|
|
426
|
+
recordCircuitBreakerFailure() {
|
|
427
|
+
this.circuitBreakerFailures++;
|
|
428
|
+
this.circuitBreakerLastFailure = Date.now();
|
|
429
|
+
if (this.circuitBreakerFailures >= this.CIRCUIT_BREAKER_THRESHOLD) {
|
|
430
|
+
logger.warn('Circuit breaker opened due to repeated failures', {
|
|
431
|
+
failures: this.circuitBreakerFailures,
|
|
432
|
+
timeoutMs: this.CIRCUIT_BREAKER_TIMEOUT_MS
|
|
433
|
+
});
|
|
434
|
+
}
|
|
435
|
+
}
|
|
436
|
+
/**
|
|
437
|
+
* Utility methods
|
|
438
|
+
*/
|
|
439
|
+
sleep(ms) {
|
|
440
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
441
|
+
}
|
|
442
|
+
getErrorMessage(error) {
|
|
443
|
+
if (error instanceof Error) {
|
|
444
|
+
return error.message;
|
|
445
|
+
}
|
|
446
|
+
return String(error);
|
|
447
|
+
}
|
|
448
|
+
/**
|
|
449
|
+
* Get cache statistics for monitoring
|
|
450
|
+
*/
|
|
451
|
+
getCacheStats() {
|
|
452
|
+
if (!this.cachedIndex) {
|
|
453
|
+
return {
|
|
454
|
+
isValid: false,
|
|
455
|
+
age: 0,
|
|
456
|
+
hasCache: false,
|
|
457
|
+
isRefreshing: this.isRefreshing,
|
|
458
|
+
circuitBreakerFailures: this.circuitBreakerFailures,
|
|
459
|
+
circuitBreakerOpen: this.isCircuitBreakerOpen()
|
|
460
|
+
};
|
|
461
|
+
}
|
|
462
|
+
const age = Date.now() - this.cachedIndex.timestamp;
|
|
463
|
+
return {
|
|
464
|
+
isValid: !this.isCacheExpired(),
|
|
465
|
+
age: Math.round(age / 1000), // age in seconds
|
|
466
|
+
hasCache: true,
|
|
467
|
+
version: this.cachedIndex.version,
|
|
468
|
+
totalElements: this.cachedIndex.data.total_elements,
|
|
469
|
+
isRefreshing: this.isRefreshing,
|
|
470
|
+
circuitBreakerFailures: this.circuitBreakerFailures,
|
|
471
|
+
circuitBreakerOpen: this.isCircuitBreakerOpen()
|
|
472
|
+
};
|
|
473
|
+
}
|
|
474
|
+
/**
|
|
475
|
+
* Clear all cache data
|
|
476
|
+
*/
|
|
477
|
+
async clearCache() {
|
|
478
|
+
this.cachedIndex = null;
|
|
479
|
+
this.circuitBreakerFailures = 0;
|
|
480
|
+
try {
|
|
481
|
+
await fs.unlink(this.CACHE_FILE);
|
|
482
|
+
logger.debug('Collection index cache file deleted');
|
|
483
|
+
}
|
|
484
|
+
catch (error) {
|
|
485
|
+
if (error.code !== 'ENOENT') {
|
|
486
|
+
logger.debug('Failed to delete cache file', { error: this.getErrorMessage(error) });
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
/**
|
|
491
|
+
* Wait for any ongoing background refresh to complete
|
|
492
|
+
*/
|
|
493
|
+
async waitForBackgroundRefresh() {
|
|
494
|
+
if (this.backgroundRefreshPromise) {
|
|
495
|
+
await this.backgroundRefreshPromise;
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionIndexManager.js","sourceRoot":"","sources":["../../src/collection/CollectionIndexManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Collection Index Manager with Background Refresh and Robust Caching\n * \n * This manager implements:\n * - 1-hour TTL with local file caching\n * - Background refresh without blocking operations\n * - Exponential backoff retry logic\n * - Configurable timeouts via environment variables\n * - Return stale cache while refreshing in background\n * - Comprehensive error handling for production use\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { CollectionIndex } from '../types/collection';\nimport { logger } from '../utils/logger.js';\n\nexport interface CollectionIndexCacheEntry {\n  data: CollectionIndex;\n  timestamp: number;\n  etag?: string;\n  lastModified?: string;\n  version: string;\n  checksum: string;\n}\n\nexport interface CollectionIndexManagerConfig {\n  ttlMs?: number;\n  fetchTimeoutMs?: number;\n  maxRetries?: number;\n  baseRetryDelayMs?: number;\n  maxRetryDelayMs?: number;\n  cacheDir?: string;\n}\n\nexport class CollectionIndexManager {\n  private readonly INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';\n  private readonly TTL_MS: number;\n  private readonly FETCH_TIMEOUT_MS: number;\n  private readonly MAX_RETRIES: number;\n  private readonly BASE_RETRY_DELAY_MS: number;\n  private readonly MAX_RETRY_DELAY_MS: number;\n  private readonly CACHE_FILE: string;\n  \n  private cachedIndex: CollectionIndexCacheEntry | null = null;\n  private backgroundRefreshPromise: Promise<void> | null = null;\n  private isRefreshing = false;\n  private circuitBreakerFailures = 0;\n  private circuitBreakerLastFailure = 0;\n  private readonly CIRCUIT_BREAKER_THRESHOLD = 5;\n  private readonly CIRCUIT_BREAKER_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\n  private readonly REFRESH_THRESHOLD = 0.8; // Refresh when 80% of TTL has passed\n  private readonly JITTER_FACTOR = 0.25; // ±25% randomness for jitter\n  \n  // Default configuration constants\n  private readonly DEFAULT_TTL_MS = 60 * 60 * 1000; // 1 hour\n  private readonly DEFAULT_MAX_RETRIES = 3;\n  private readonly DEFAULT_BASE_RETRY_DELAY_MS = 1000;\n  private readonly DEFAULT_MAX_RETRY_DELAY_MS = 30000;\n  private readonly DEFAULT_FETCH_TIMEOUT_MS = 5000; // 5 seconds\n  private readonly CHECKSUM_LENGTH = 8;\n  private readonly JSON_INDENT = 2;\n  \n  constructor(config: CollectionIndexManagerConfig = {}) {\n    // Configuration with environment variable overrides\n    this.TTL_MS = config.ttlMs || this.DEFAULT_TTL_MS;\n    this.FETCH_TIMEOUT_MS = this.parseFetchTimeout(config.fetchTimeoutMs);\n    this.MAX_RETRIES = config.maxRetries || this.DEFAULT_MAX_RETRIES;\n    this.BASE_RETRY_DELAY_MS = config.baseRetryDelayMs || this.DEFAULT_BASE_RETRY_DELAY_MS;\n    this.MAX_RETRY_DELAY_MS = config.maxRetryDelayMs || this.DEFAULT_MAX_RETRY_DELAY_MS;\n    \n    // Cache directory - use ~/.dollhouse/cache/collection-index.json as specified\n    const cacheDir = config.cacheDir || path.join(os.homedir(), '.dollhouse', 'cache');\n    this.CACHE_FILE = path.join(cacheDir, 'collection-index.json');\n    \n    logger.debug('CollectionIndexManager initialized', {\n      ttlMs: this.TTL_MS,\n      fetchTimeoutMs: this.FETCH_TIMEOUT_MS,\n      cacheFile: this.CACHE_FILE,\n      maxRetries: this.MAX_RETRIES\n    });\n  }\n  \n  /**\n   * Parse fetch timeout from config or environment variable\n   */\n  private parseFetchTimeout(configValue?: number): number {\n    // Check environment variable first\n    const envTimeout = process.env.COLLECTION_FETCH_TIMEOUT;\n    if (envTimeout) {\n      const parsed = parseInt(envTimeout, 10);\n      if (!isNaN(parsed) && parsed > 0) {\n        logger.debug(`Using COLLECTION_FETCH_TIMEOUT from environment: ${parsed}ms`);\n        return parsed;\n      }\n      logger.warn(`Invalid COLLECTION_FETCH_TIMEOUT value: ${envTimeout}, using default`);\n    }\n    \n    // Fall back to config value or default\n    return configValue || this.DEFAULT_FETCH_TIMEOUT_MS;\n  }\n  \n  /**\n   * Get collection index with stale-while-revalidate pattern\n   * Returns cached data immediately if available, refreshes in background\n   */\n  async getIndex(): Promise<CollectionIndex> {\n    try {\n      // Load from memory cache first\n      if (!this.cachedIndex) {\n        await this.loadFromDisk();\n      }\n      \n      // Check if we should return stale cache while refreshing\n      const shouldRefresh = this.shouldRefreshCache();\n      \n      if (this.cachedIndex && !this.isCacheExpired()) {\n        // Cache is valid, return immediately\n        logger.debug('Returning valid cached collection index');\n        \n        // Start background refresh if needed but not already running\n        if (shouldRefresh && !this.isRefreshing) {\n          this.startBackgroundRefresh();\n        }\n        \n        return this.cachedIndex.data;\n      }\n      \n      // If we have stale cache, return it while refreshing in background\n      if (this.cachedIndex && this.isCacheExpired()) {\n        logger.debug('Returning stale cache while refreshing in background');\n        \n        // Start background refresh if not already running\n        if (!this.isRefreshing) {\n          this.startBackgroundRefresh();\n        }\n        \n        return this.cachedIndex.data;\n      }\n      \n      // No cache available, must fetch synchronously\n      logger.debug('No cache available, fetching collection index synchronously');\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      return freshIndex.data;\n      \n    } catch (error) {\n      logger.error('Failed to get collection index', { error: this.getErrorMessage(error) });\n      \n      // If we have any cached data (even expired), return it as last resort\n      if (this.cachedIndex) {\n        logger.warn('Returning expired cache as last resort');\n        return this.cachedIndex.data;\n      }\n      \n      throw new Error(`Collection index not available: ${this.getErrorMessage(error)}`);\n    }\n  }\n  \n  /**\n   * Force refresh the collection index\n   */\n  async forceRefresh(): Promise<CollectionIndex> {\n    logger.debug('Force refreshing collection index');\n    \n    try {\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      return freshIndex.data;\n    } catch (error) {\n      logger.error('Force refresh failed', { error: this.getErrorMessage(error) });\n      \n      // If we have cached data, return it\n      if (this.cachedIndex) {\n        logger.warn('Force refresh failed, returning cached data');\n        return this.cachedIndex.data;\n      }\n      \n      throw error;\n    }\n  }\n  \n  /**\n   * Check if cache should be refreshed (within TTL but getting close to expiry)\n   */\n  private shouldRefreshCache(): boolean {\n    if (!this.cachedIndex) return true;\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    const refreshThreshold = this.TTL_MS * this.REFRESH_THRESHOLD;\n    \n    return age > refreshThreshold;\n  }\n  \n  /**\n   * Check if cache is expired\n   */\n  private isCacheExpired(): boolean {\n    if (!this.cachedIndex) return true;\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    return age > this.TTL_MS;\n  }\n  \n  /**\n   * Start background refresh without blocking\n   */\n  private startBackgroundRefresh(): void {\n    if (this.isRefreshing) {\n      logger.debug('Background refresh already in progress');\n      return;\n    }\n    \n    // Check circuit breaker\n    if (this.isCircuitBreakerOpen()) {\n      logger.debug('Circuit breaker open, skipping background refresh');\n      return;\n    }\n    \n    this.isRefreshing = true;\n    \n    this.backgroundRefreshPromise = this.performBackgroundRefresh()\n      .catch(error => {\n        logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });\n        this.recordCircuitBreakerFailure();\n      })\n      .finally(() => {\n        this.isRefreshing = false;\n        this.backgroundRefreshPromise = null;\n      });\n  }\n  \n  /**\n   * Perform background refresh\n   */\n  private async performBackgroundRefresh(): Promise<void> {\n    logger.debug('Starting background refresh of collection index');\n    \n    try {\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      \n      // Reset circuit breaker on success\n      this.circuitBreakerFailures = 0;\n      \n      logger.debug('Background refresh completed successfully');\n    } catch (error) {\n      logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });\n      throw error;\n    }\n  }\n  \n  /**\n   * Fetch collection index with retry logic and exponential backoff\n   */\n  private async fetchWithRetry(): Promise<{ data: CollectionIndex; etag?: string; lastModified?: string }> {\n    let lastError: Error | null = null;\n    \n    for (let attempt = 0; attempt <= this.MAX_RETRIES; attempt++) {\n      try {\n        if (attempt > 0) {\n          const delay = this.calculateRetryDelay(attempt);\n          logger.debug(`Retrying fetch in ${delay}ms (attempt ${attempt + 1}/${this.MAX_RETRIES + 1})`);\n          await this.sleep(delay);\n        }\n        \n        return await this.fetchCollectionIndex();\n      } catch (error) {\n        lastError = error instanceof Error ? error : new Error(String(error));\n        logger.debug(`Fetch attempt ${attempt + 1} failed`, { \n          error: this.getErrorMessage(lastError),\n          willRetry: attempt < this.MAX_RETRIES\n        });\n      }\n    }\n    \n    throw lastError || new Error('All fetch attempts failed');\n  }\n  \n  /**\n   * Calculate retry delay with exponential backoff and jitter\n   */\n  private calculateRetryDelay(attempt: number): number {\n    // Exponential backoff: baseDelay * (2 ^ attempt)\n    const exponentialDelay = this.BASE_RETRY_DELAY_MS * Math.pow(2, attempt - 1);\n    \n    // Cap at maximum delay\n    const cappedDelay = Math.min(exponentialDelay, this.MAX_RETRY_DELAY_MS);\n    \n    // Add jitter to prevent thundering herd\n    return this.addJitter(cappedDelay);\n  }\n  \n  /**\n   * Add jitter (±25% randomness) to a delay to prevent thundering herd problems\n   */\n  private addJitter(delay: number): number {\n    const jitter = delay * this.JITTER_FACTOR * (Math.random() - 0.5);\n    return Math.max(0, delay + jitter);\n  }\n  \n  /**\n   * Fetch collection index from GitHub\n   */\n  private async fetchCollectionIndex(): Promise<{ data: CollectionIndex; etag?: string; lastModified?: string }> {\n    const controller = new AbortController();\n    const timeoutId = setTimeout(() => controller.abort(), this.FETCH_TIMEOUT_MS);\n    \n    try {\n      // Build headers for conditional requests\n      const headers: Record<string, string> = {\n        'Accept': 'application/json',\n        'User-Agent': 'DollhouseMCP/1.0',\n        'Cache-Control': 'no-cache'\n      };\n      \n      // Add conditional headers if we have cached data\n      if (this.cachedIndex?.etag) {\n        headers['If-None-Match'] = this.cachedIndex.etag;\n      }\n      if (this.cachedIndex?.lastModified) {\n        headers['If-Modified-Since'] = this.cachedIndex.lastModified;\n      }\n      \n      logger.debug('Fetching collection index', { \n        url: this.INDEX_URL,\n        timeout: this.FETCH_TIMEOUT_MS,\n        hasEtag: !!this.cachedIndex?.etag\n      });\n      \n      const response = await fetch(this.INDEX_URL, {\n        headers,\n        signal: controller.signal\n      });\n      \n      clearTimeout(timeoutId);\n      \n      // Handle 304 Not Modified\n      if (response.status === 304 && this.cachedIndex) {\n        logger.debug('Collection index not modified (304), updating cache timestamp');\n        this.cachedIndex.timestamp = Date.now();\n        await this.saveToDisk();\n        return {\n          data: this.cachedIndex.data,\n          etag: this.cachedIndex.etag,\n          lastModified: this.cachedIndex.lastModified\n        };\n      }\n      \n      if (!response.ok) {\n        throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n      }\n      \n      const indexData = await response.json() as CollectionIndex;\n      \n      // Validate the index structure\n      this.validateIndexStructure(indexData);\n      \n      const etag = response.headers.get('etag') || undefined;\n      const lastModified = response.headers.get('last-modified') || undefined;\n      \n      logger.debug('Collection index fetched successfully', {\n        totalElements: indexData.total_elements,\n        version: indexData.version,\n        hasEtag: !!etag\n      });\n      \n      return { data: indexData, etag, lastModified };\n      \n    } catch (error) {\n      clearTimeout(timeoutId);\n      \n      if (error instanceof Error && error.name === 'AbortError') {\n        throw new Error(`Fetch timeout after ${this.FETCH_TIMEOUT_MS}ms`);\n      }\n      \n      throw error;\n    }\n  }\n  \n  /**\n   * Validate collection index structure\n   */\n  private validateIndexStructure(index: any): asserts index is CollectionIndex {\n    if (!index || typeof index !== 'object') {\n      throw new Error('Invalid index: not an object');\n    }\n    \n    if (typeof index.version !== 'string') {\n      throw new Error('Invalid index: missing or invalid version');\n    }\n    \n    if (typeof index.generated !== 'string') {\n      throw new Error('Invalid index: missing or invalid generated timestamp');\n    }\n    \n    if (typeof index.total_elements !== 'number') {\n      throw new Error('Invalid index: missing or invalid total_elements');\n    }\n    \n    if (!index.index || typeof index.index !== 'object') {\n      throw new Error('Invalid index: missing or invalid index object');\n    }\n    \n    if (!index.metadata || typeof index.metadata !== 'object') {\n      throw new Error('Invalid index: missing or invalid metadata');\n    }\n  }\n  \n  /**\n   * Update cache with new data\n   */\n  private async updateCache(fetchResult: { data: CollectionIndex; etag?: string; lastModified?: string }): Promise<void> {\n    const checksum = this.calculateChecksum(fetchResult.data);\n    \n    this.cachedIndex = {\n      data: fetchResult.data,\n      timestamp: Date.now(),\n      etag: fetchResult.etag,\n      lastModified: fetchResult.lastModified,\n      version: fetchResult.data.version,\n      checksum\n    };\n    \n    await this.saveToDisk();\n    \n    logger.debug('Collection index cache updated', {\n      version: fetchResult.data.version,\n      totalElements: fetchResult.data.total_elements,\n      checksum\n    });\n  }\n  \n  /**\n   * Calculate checksum for data integrity verification\n   */\n  private calculateChecksum(data: CollectionIndex): string {\n    // Simple checksum based on key properties\n    const checksumData = {\n      version: data.version,\n      generated: data.generated,\n      total_elements: data.total_elements\n    };\n    return Buffer.from(JSON.stringify(checksumData)).toString('base64').substring(0, this.CHECKSUM_LENGTH);\n  }\n  \n  /**\n   * Load cache from disk\n   */\n  private async loadFromDisk(): Promise<void> {\n    try {\n      const data = await fs.readFile(this.CACHE_FILE, 'utf8');\n      const cached = JSON.parse(data) as CollectionIndexCacheEntry;\n      \n      // Validate cache structure\n      if (!cached.data || !cached.timestamp || !cached.version) {\n        logger.debug('Invalid cache structure, ignoring');\n        return;\n      }\n      \n      // Verify checksum if available\n      if (cached.checksum) {\n        const expectedChecksum = this.calculateChecksum(cached.data);\n        if (cached.checksum !== expectedChecksum) {\n          logger.debug('Cache checksum mismatch, ignoring cached data');\n          return;\n        }\n      }\n      \n      this.cachedIndex = cached;\n      \n      const age = Date.now() - cached.timestamp;\n      const isExpired = age > this.TTL_MS;\n      \n      logger.debug('Loaded collection index from disk cache', {\n        version: cached.version,\n        age: Math.round(age / 1000),\n        isExpired,\n        totalElements: cached.data.total_elements\n      });\n      \n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to load cache from disk', { error: this.getErrorMessage(error) });\n      }\n    }\n  }\n  \n  /**\n   * Save cache to disk\n   */\n  private async saveToDisk(): Promise<void> {\n    if (!this.cachedIndex) return;\n    \n    try {\n      // Ensure cache directory exists\n      await fs.mkdir(path.dirname(this.CACHE_FILE), { recursive: true });\n      \n      const cacheData = JSON.stringify(this.cachedIndex, null, this.JSON_INDENT);\n      await fs.writeFile(this.CACHE_FILE, cacheData, 'utf8');\n      \n      logger.debug('Collection index cache saved to disk');\n    } catch (error) {\n      logger.debug('Failed to save cache to disk', { error: this.getErrorMessage(error) });\n      // Don't throw - cache persistence failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Circuit breaker logic\n   */\n  private isCircuitBreakerOpen(): boolean {\n    if (this.circuitBreakerFailures < this.CIRCUIT_BREAKER_THRESHOLD) {\n      return false;\n    }\n    \n    const timeSinceLastFailure = Date.now() - this.circuitBreakerLastFailure;\n    return timeSinceLastFailure < this.CIRCUIT_BREAKER_TIMEOUT_MS;\n  }\n  \n  private recordCircuitBreakerFailure(): void {\n    this.circuitBreakerFailures++;\n    this.circuitBreakerLastFailure = Date.now();\n    \n    if (this.circuitBreakerFailures >= this.CIRCUIT_BREAKER_THRESHOLD) {\n      logger.warn('Circuit breaker opened due to repeated failures', {\n        failures: this.circuitBreakerFailures,\n        timeoutMs: this.CIRCUIT_BREAKER_TIMEOUT_MS\n      });\n    }\n  }\n  \n  /**\n   * Utility methods\n   */\n  private sleep(ms: number): Promise<void> {\n    return new Promise(resolve => setTimeout(resolve, ms));\n  }\n  \n  private getErrorMessage(error: unknown): string {\n    if (error instanceof Error) {\n      return error.message;\n    }\n    return String(error);\n  }\n  \n  /**\n   * Get cache statistics for monitoring\n   */\n  getCacheStats(): {\n    isValid: boolean;\n    age: number;\n    hasCache: boolean;\n    version?: string;\n    totalElements?: number;\n    isRefreshing: boolean;\n    circuitBreakerFailures: number;\n    circuitBreakerOpen: boolean;\n  } {\n    if (!this.cachedIndex) {\n      return {\n        isValid: false,\n        age: 0,\n        hasCache: false,\n        isRefreshing: this.isRefreshing,\n        circuitBreakerFailures: this.circuitBreakerFailures,\n        circuitBreakerOpen: this.isCircuitBreakerOpen()\n      };\n    }\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    \n    return {\n      isValid: !this.isCacheExpired(),\n      age: Math.round(age / 1000), // age in seconds\n      hasCache: true,\n      version: this.cachedIndex.version,\n      totalElements: this.cachedIndex.data.total_elements,\n      isRefreshing: this.isRefreshing,\n      circuitBreakerFailures: this.circuitBreakerFailures,\n      circuitBreakerOpen: this.isCircuitBreakerOpen()\n    };\n  }\n  \n  /**\n   * Clear all cache data\n   */\n  async clearCache(): Promise<void> {\n    this.cachedIndex = null;\n    this.circuitBreakerFailures = 0;\n    \n    try {\n      await fs.unlink(this.CACHE_FILE);\n      logger.debug('Collection index cache file deleted');\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to delete cache file', { error: this.getErrorMessage(error) });\n      }\n    }\n  }\n  \n  /**\n   * Wait for any ongoing background refresh to complete\n   */\n  async waitForBackgroundRefresh(): Promise<void> {\n    if (this.backgroundRefreshPromise) {\n      await this.backgroundRefreshPromise;\n    }\n  }\n}"]}
|
|
@@ -3,11 +3,18 @@
|
|
|
3
3
|
*/
|
|
4
4
|
import { GitHubClient } from './GitHubClient.js';
|
|
5
5
|
import { CollectionCache } from '../cache/CollectionCache.js';
|
|
6
|
+
import { SearchResults, SearchOptions } from '../types/collection.js';
|
|
6
7
|
export declare class CollectionSearch {
|
|
7
8
|
private githubClient;
|
|
8
9
|
private collectionCache;
|
|
10
|
+
private indexCache;
|
|
9
11
|
private searchBaseUrl;
|
|
10
12
|
constructor(githubClient: GitHubClient, collectionCache?: CollectionCache);
|
|
13
|
+
/**
|
|
14
|
+
* Enhanced search using collection index with pagination and filtering
|
|
15
|
+
* Falls back to API search and cache when index is unavailable
|
|
16
|
+
*/
|
|
17
|
+
searchCollectionWithOptions(query: string, options?: SearchOptions): Promise<SearchResults>;
|
|
11
18
|
/**
|
|
12
19
|
* Search collection for content matching query
|
|
13
20
|
* Falls back to cached data when GitHub API is not available or not authenticated
|
|
@@ -21,6 +28,10 @@ export declare class CollectionSearch {
|
|
|
21
28
|
* Search seed data for matching items with fuzzy matching
|
|
22
29
|
*/
|
|
23
30
|
private searchSeedData;
|
|
31
|
+
/**
|
|
32
|
+
* Fuzzy matching algorithm for partial string matches
|
|
33
|
+
*/
|
|
34
|
+
private fuzzyMatch;
|
|
24
35
|
/**
|
|
25
36
|
* Convert cache items to GitHub API format for consistent response structure
|
|
26
37
|
*/
|
|
@@ -33,5 +44,49 @@ export declare class CollectionSearch {
|
|
|
33
44
|
* Format search results
|
|
34
45
|
*/
|
|
35
46
|
formatSearchResults(items: any[], query: string, personaIndicator?: string): string;
|
|
47
|
+
/**
|
|
48
|
+
* Search from collection index with full featured search and pagination
|
|
49
|
+
*/
|
|
50
|
+
private searchFromIndex;
|
|
51
|
+
/**
|
|
52
|
+
* Flatten index entries from all categories into a single array
|
|
53
|
+
*/
|
|
54
|
+
private flattenIndexEntries;
|
|
55
|
+
/**
|
|
56
|
+
* Perform search matching on index entries
|
|
57
|
+
*/
|
|
58
|
+
private performIndexSearch;
|
|
59
|
+
/**
|
|
60
|
+
* Sort search results by relevance, name, or date
|
|
61
|
+
*/
|
|
62
|
+
private sortSearchResults;
|
|
63
|
+
/**
|
|
64
|
+
* Calculate relevance score for search results
|
|
65
|
+
*/
|
|
66
|
+
private calculateRelevanceScore;
|
|
67
|
+
/**
|
|
68
|
+
* Convert legacy search results to new SearchResults format
|
|
69
|
+
*/
|
|
70
|
+
private convertLegacyResults;
|
|
71
|
+
/**
|
|
72
|
+
* Extract element type from file path
|
|
73
|
+
*/
|
|
74
|
+
private extractTypeFromPath;
|
|
75
|
+
/**
|
|
76
|
+
* Extract category from file path
|
|
77
|
+
*/
|
|
78
|
+
private extractCategoryFromPath;
|
|
79
|
+
/**
|
|
80
|
+
* Create empty search results for error cases
|
|
81
|
+
*/
|
|
82
|
+
private createEmptySearchResults;
|
|
83
|
+
/**
|
|
84
|
+
* Enhanced format for search results with pagination info
|
|
85
|
+
*/
|
|
86
|
+
formatSearchResultsWithPagination(results: SearchResults, personaIndicator?: string): string;
|
|
87
|
+
/**
|
|
88
|
+
* Get cache statistics for debugging
|
|
89
|
+
*/
|
|
90
|
+
getCacheStats(): Promise<any>;
|
|
36
91
|
}
|
|
37
92
|
//# sourceMappingURL=CollectionSearch.d.ts.map
|