@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,475 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GitHub Portfolio Indexer - Fetches and indexes user's GitHub portfolio for fast searching
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - Singleton pattern for efficient resource usage
|
|
6
|
+
* - Smart caching with TTL and invalidation after user actions
|
|
7
|
+
* - GraphQL/REST API integration for efficient fetching
|
|
8
|
+
* - Rate limiting and authentication handling
|
|
9
|
+
* - Fallback strategy for resilient operation
|
|
10
|
+
* - Performance optimized for 1000+ portfolio elements
|
|
11
|
+
*/
|
|
12
|
+
import { GitHubClient } from '../collection/GitHubClient.js';
|
|
13
|
+
import { PortfolioRepoManager } from './PortfolioRepoManager.js';
|
|
14
|
+
import { ElementType } from './types.js';
|
|
15
|
+
import { logger } from '../utils/logger.js';
|
|
16
|
+
import { SecurityMonitor } from '../security/securityMonitor.js';
|
|
17
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
18
|
+
import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
|
|
19
|
+
import { APICache } from '../cache/APICache.js';
|
|
20
|
+
export class GitHubPortfolioIndexer {
|
|
21
|
+
static instance = null;
|
|
22
|
+
static instanceLock = false;
|
|
23
|
+
cache = null;
|
|
24
|
+
lastFetch = null;
|
|
25
|
+
ttl = 15 * 60 * 1000; // 15 minutes
|
|
26
|
+
recentUserAction = false;
|
|
27
|
+
actionTimestamp = null;
|
|
28
|
+
actionGracePeriod = 2 * 60 * 1000; // 2 minutes after action
|
|
29
|
+
githubClient;
|
|
30
|
+
portfolioRepoManager;
|
|
31
|
+
apiCache;
|
|
32
|
+
rateLimitTracker;
|
|
33
|
+
constructor() {
|
|
34
|
+
this.apiCache = new APICache(); // Uses default settings
|
|
35
|
+
this.rateLimitTracker = new Map();
|
|
36
|
+
this.githubClient = new GitHubClient(this.apiCache, this.rateLimitTracker);
|
|
37
|
+
this.portfolioRepoManager = new PortfolioRepoManager();
|
|
38
|
+
logger.debug('GitHubPortfolioIndexer created');
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Singleton pattern with thread safety
|
|
42
|
+
*/
|
|
43
|
+
static getInstance() {
|
|
44
|
+
if (!this.instance) {
|
|
45
|
+
if (this.instanceLock) {
|
|
46
|
+
throw new Error('GitHubPortfolioIndexer instance is being created by another thread');
|
|
47
|
+
}
|
|
48
|
+
try {
|
|
49
|
+
this.instanceLock = true;
|
|
50
|
+
this.instance = new GitHubPortfolioIndexer();
|
|
51
|
+
}
|
|
52
|
+
finally {
|
|
53
|
+
this.instanceLock = false;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
return this.instance;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Main method to get GitHub portfolio index
|
|
60
|
+
*/
|
|
61
|
+
async getIndex(force = false) {
|
|
62
|
+
try {
|
|
63
|
+
// Check if we need fresh data
|
|
64
|
+
if (force || this.shouldFetchFresh()) {
|
|
65
|
+
return await this.fetchFresh();
|
|
66
|
+
}
|
|
67
|
+
// Return cached data if available and valid
|
|
68
|
+
if (this.cache && this.isCacheValid()) {
|
|
69
|
+
logger.debug('Returning cached GitHub portfolio index', {
|
|
70
|
+
username: this.cache.username,
|
|
71
|
+
totalElements: this.cache.totalElements,
|
|
72
|
+
age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'
|
|
73
|
+
});
|
|
74
|
+
return this.cache;
|
|
75
|
+
}
|
|
76
|
+
// Try to fetch fresh, fall back to stale cache on failure
|
|
77
|
+
try {
|
|
78
|
+
return await this.fetchFresh();
|
|
79
|
+
}
|
|
80
|
+
catch (error) {
|
|
81
|
+
logger.warn('Failed to fetch fresh GitHub portfolio index, checking for stale cache', {
|
|
82
|
+
error: error instanceof Error ? error.message : String(error)
|
|
83
|
+
});
|
|
84
|
+
// Return stale cache if available
|
|
85
|
+
if (this.cache) {
|
|
86
|
+
logger.info('Returning stale GitHub portfolio cache as fallback', {
|
|
87
|
+
username: this.cache.username,
|
|
88
|
+
age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'
|
|
89
|
+
});
|
|
90
|
+
return this.cache;
|
|
91
|
+
}
|
|
92
|
+
// Return empty index as last resort
|
|
93
|
+
return this.createEmptyIndex();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
catch (error) {
|
|
97
|
+
ErrorHandler.logError('GitHubPortfolioIndexer.getIndex', error);
|
|
98
|
+
// Return stale cache or empty index
|
|
99
|
+
if (this.cache) {
|
|
100
|
+
return this.cache;
|
|
101
|
+
}
|
|
102
|
+
return this.createEmptyIndex();
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Invalidate cache after user actions
|
|
107
|
+
*/
|
|
108
|
+
invalidateAfterAction(action) {
|
|
109
|
+
logger.info('Invalidating GitHub portfolio cache after user action', { action });
|
|
110
|
+
this.recentUserAction = true;
|
|
111
|
+
this.actionTimestamp = new Date();
|
|
112
|
+
// Log security event for audit trail
|
|
113
|
+
SecurityMonitor.logSecurityEvent({
|
|
114
|
+
type: 'PORTFOLIO_CACHE_INVALIDATION',
|
|
115
|
+
severity: 'LOW',
|
|
116
|
+
source: 'GitHubPortfolioIndexer.invalidateAfterAction',
|
|
117
|
+
details: `Cache invalidated after user action: ${action}`,
|
|
118
|
+
metadata: { action }
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Clear all cached data
|
|
123
|
+
*/
|
|
124
|
+
clearCache() {
|
|
125
|
+
this.cache = null;
|
|
126
|
+
this.lastFetch = null;
|
|
127
|
+
this.recentUserAction = false;
|
|
128
|
+
this.actionTimestamp = null;
|
|
129
|
+
this.apiCache.clear();
|
|
130
|
+
logger.info('GitHub portfolio cache cleared');
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Get cache statistics
|
|
134
|
+
*/
|
|
135
|
+
getCacheStats() {
|
|
136
|
+
return {
|
|
137
|
+
hasCachedData: this.cache !== null,
|
|
138
|
+
lastFetch: this.lastFetch,
|
|
139
|
+
isStale: !this.isCacheValid(),
|
|
140
|
+
recentUserAction: this.recentUserAction,
|
|
141
|
+
totalElements: this.cache?.totalElements || 0
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Fetch fresh data from GitHub
|
|
146
|
+
*/
|
|
147
|
+
async fetchFresh() {
|
|
148
|
+
const startTime = Date.now();
|
|
149
|
+
logger.info('Fetching fresh GitHub portfolio index...');
|
|
150
|
+
try {
|
|
151
|
+
// Get GitHub username from token
|
|
152
|
+
const username = await this.getGitHubUsername();
|
|
153
|
+
const repository = 'dollhouse-portfolio';
|
|
154
|
+
// Check if portfolio repository exists
|
|
155
|
+
const repoExists = await this.portfolioRepoManager.checkPortfolioExists(username);
|
|
156
|
+
if (!repoExists) {
|
|
157
|
+
logger.info('GitHub portfolio repository does not exist', { username });
|
|
158
|
+
return this.createEmptyIndex(username, repository);
|
|
159
|
+
}
|
|
160
|
+
// Fetch repository content using GitHub API
|
|
161
|
+
const index = await this.fetchRepositoryContent(username, repository);
|
|
162
|
+
// Update cache
|
|
163
|
+
this.cache = index;
|
|
164
|
+
this.lastFetch = new Date();
|
|
165
|
+
this.recentUserAction = false;
|
|
166
|
+
this.actionTimestamp = null;
|
|
167
|
+
const duration = Date.now() - startTime;
|
|
168
|
+
logger.info('GitHub portfolio index fetched successfully', {
|
|
169
|
+
username,
|
|
170
|
+
totalElements: index.totalElements,
|
|
171
|
+
duration: `${duration}ms`,
|
|
172
|
+
rateLimitRemaining: index.rateLimitInfo?.remaining
|
|
173
|
+
});
|
|
174
|
+
// Log security event
|
|
175
|
+
SecurityMonitor.logSecurityEvent({
|
|
176
|
+
type: 'PORTFOLIO_FETCH_SUCCESS',
|
|
177
|
+
severity: 'LOW',
|
|
178
|
+
source: 'GitHubPortfolioIndexer.fetchFresh',
|
|
179
|
+
details: `Fetched GitHub portfolio with ${index.totalElements} elements in ${duration}ms`,
|
|
180
|
+
metadata: { username, duration, totalElements: index.totalElements }
|
|
181
|
+
});
|
|
182
|
+
return index;
|
|
183
|
+
}
|
|
184
|
+
catch (error) {
|
|
185
|
+
const duration = Date.now() - startTime;
|
|
186
|
+
ErrorHandler.logError('GitHubPortfolioIndexer.fetchFresh', error, { duration });
|
|
187
|
+
throw ErrorHandler.wrapError(error, 'Failed to fetch GitHub portfolio index', ErrorCategory.NETWORK_ERROR);
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Fetch repository content from GitHub API
|
|
192
|
+
*/
|
|
193
|
+
async fetchRepositoryContent(username, repository) {
|
|
194
|
+
// Try GraphQL first for better performance, fallback to REST
|
|
195
|
+
try {
|
|
196
|
+
return await this.fetchWithGraphQL(username, repository);
|
|
197
|
+
}
|
|
198
|
+
catch (graphqlError) {
|
|
199
|
+
logger.debug('GraphQL fetch failed, falling back to REST API', {
|
|
200
|
+
error: graphqlError instanceof Error ? graphqlError.message : String(graphqlError)
|
|
201
|
+
});
|
|
202
|
+
return await this.fetchWithREST(username, repository);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Fetch using GraphQL for better performance
|
|
207
|
+
*/
|
|
208
|
+
async fetchWithGraphQL(username, repository) {
|
|
209
|
+
const query = `
|
|
210
|
+
query GetPortfolioContent($owner: String!, $name: String!) {
|
|
211
|
+
repository(owner: $owner, name: $name) {
|
|
212
|
+
defaultBranchRef {
|
|
213
|
+
target {
|
|
214
|
+
... on Commit {
|
|
215
|
+
oid
|
|
216
|
+
history(first: 1) {
|
|
217
|
+
nodes {
|
|
218
|
+
committedDate
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
object(expression: "HEAD:") {
|
|
225
|
+
... on Tree {
|
|
226
|
+
entries {
|
|
227
|
+
name
|
|
228
|
+
type
|
|
229
|
+
object {
|
|
230
|
+
... on Tree {
|
|
231
|
+
entries {
|
|
232
|
+
name
|
|
233
|
+
type
|
|
234
|
+
oid
|
|
235
|
+
object {
|
|
236
|
+
... on Blob {
|
|
237
|
+
byteSize
|
|
238
|
+
text
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
rateLimit {
|
|
249
|
+
remaining
|
|
250
|
+
resetAt
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
`;
|
|
254
|
+
const variables = { owner: username, name: repository };
|
|
255
|
+
const response = await this.githubClient.fetchFromGitHub('https://api.github.com/graphql', true);
|
|
256
|
+
// Note: This is a simplified GraphQL implementation
|
|
257
|
+
// In a real implementation, you would send POST request with query and variables
|
|
258
|
+
throw new Error('GraphQL implementation not yet complete');
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Fetch using REST API with pagination
|
|
262
|
+
*/
|
|
263
|
+
async fetchWithREST(username, repository) {
|
|
264
|
+
const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;
|
|
265
|
+
// Get repository info and latest commit
|
|
266
|
+
const repoInfo = await this.githubClient.fetchFromGitHub(`https://api.github.com/repos/${normalizedUsername}/${repository}`);
|
|
267
|
+
const latestCommit = await this.githubClient.fetchFromGitHub(`https://api.github.com/repos/${normalizedUsername}/${repository}/commits/HEAD`);
|
|
268
|
+
// Initialize index
|
|
269
|
+
const index = {
|
|
270
|
+
username: normalizedUsername,
|
|
271
|
+
repository,
|
|
272
|
+
lastUpdated: new Date(latestCommit.commit.committer.date),
|
|
273
|
+
elements: new Map(),
|
|
274
|
+
totalElements: 0,
|
|
275
|
+
sha: latestCommit.sha
|
|
276
|
+
};
|
|
277
|
+
// Initialize element type maps
|
|
278
|
+
for (const elementType of Object.values(ElementType)) {
|
|
279
|
+
index.elements.set(elementType, []);
|
|
280
|
+
}
|
|
281
|
+
// Fetch content for each element type
|
|
282
|
+
for (const elementType of Object.values(ElementType)) {
|
|
283
|
+
try {
|
|
284
|
+
const entries = await this.fetchElementTypeContent(normalizedUsername, repository, elementType);
|
|
285
|
+
index.elements.set(elementType, entries);
|
|
286
|
+
index.totalElements += entries.length;
|
|
287
|
+
}
|
|
288
|
+
catch (error) {
|
|
289
|
+
logger.warn(`Failed to fetch ${elementType} from GitHub portfolio`, {
|
|
290
|
+
error: error instanceof Error ? error.message : String(error)
|
|
291
|
+
});
|
|
292
|
+
// Continue with other element types
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
return index;
|
|
296
|
+
}
|
|
297
|
+
/**
|
|
298
|
+
* Fetch content for a specific element type
|
|
299
|
+
*/
|
|
300
|
+
async fetchElementTypeContent(username, repository, elementType) {
|
|
301
|
+
try {
|
|
302
|
+
// Get directory listing
|
|
303
|
+
const contents = await this.githubClient.fetchFromGitHub(`https://api.github.com/repos/${username}/${repository}/contents/${elementType}`);
|
|
304
|
+
if (!Array.isArray(contents)) {
|
|
305
|
+
return [];
|
|
306
|
+
}
|
|
307
|
+
const entries = [];
|
|
308
|
+
const maxConcurrent = 5; // Limit concurrent requests
|
|
309
|
+
// Process files in batches to avoid rate limiting
|
|
310
|
+
for (let i = 0; i < contents.length; i += maxConcurrent) {
|
|
311
|
+
const batch = contents.slice(i, i + maxConcurrent);
|
|
312
|
+
const batchPromises = batch
|
|
313
|
+
.filter(item => item.type === 'file' && item.name.endsWith('.md'))
|
|
314
|
+
.map(item => this.createGitHubIndexEntry(username, repository, elementType, item));
|
|
315
|
+
const batchResults = await Promise.allSettled(batchPromises);
|
|
316
|
+
for (const result of batchResults) {
|
|
317
|
+
if (result.status === 'fulfilled' && result.value) {
|
|
318
|
+
entries.push(result.value);
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
// Add delay between batches to respect rate limits
|
|
322
|
+
if (i + maxConcurrent < contents.length) {
|
|
323
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
return entries;
|
|
327
|
+
}
|
|
328
|
+
catch (error) {
|
|
329
|
+
// Directory might not exist
|
|
330
|
+
if (error instanceof Error && error.message.includes('404')) {
|
|
331
|
+
return [];
|
|
332
|
+
}
|
|
333
|
+
throw error;
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
/**
|
|
337
|
+
* Create GitHub index entry from API response
|
|
338
|
+
*/
|
|
339
|
+
async createGitHubIndexEntry(username, repository, elementType, fileInfo) {
|
|
340
|
+
try {
|
|
341
|
+
// Parse metadata from filename or fetch content if needed
|
|
342
|
+
const name = fileInfo.name.replace('.md', '').replace(/-/g, ' ');
|
|
343
|
+
const entry = {
|
|
344
|
+
path: fileInfo.path,
|
|
345
|
+
name,
|
|
346
|
+
elementType,
|
|
347
|
+
sha: fileInfo.sha,
|
|
348
|
+
htmlUrl: fileInfo.html_url,
|
|
349
|
+
downloadUrl: fileInfo.download_url,
|
|
350
|
+
lastModified: new Date(), // GitHub API doesn't provide file modification time directly
|
|
351
|
+
size: fileInfo.size || 0
|
|
352
|
+
};
|
|
353
|
+
// Optionally fetch content to extract metadata
|
|
354
|
+
// This is expensive, so only do it for small files or when specifically needed
|
|
355
|
+
if (fileInfo.size && fileInfo.size < 10000) { // Only for files < 10KB
|
|
356
|
+
try {
|
|
357
|
+
const content = await this.githubClient.fetchFromGitHub(fileInfo.download_url);
|
|
358
|
+
const metadata = this.parseMetadataFromContent(content);
|
|
359
|
+
if (metadata.name)
|
|
360
|
+
entry.name = metadata.name;
|
|
361
|
+
if (metadata.description)
|
|
362
|
+
entry.description = metadata.description;
|
|
363
|
+
if (metadata.version)
|
|
364
|
+
entry.version = metadata.version;
|
|
365
|
+
if (metadata.author)
|
|
366
|
+
entry.author = metadata.author;
|
|
367
|
+
}
|
|
368
|
+
catch (metadataError) {
|
|
369
|
+
// Non-critical error, continue without metadata
|
|
370
|
+
logger.debug('Failed to fetch metadata for file', {
|
|
371
|
+
path: fileInfo.path,
|
|
372
|
+
error: metadataError instanceof Error ? metadataError.message : String(metadataError)
|
|
373
|
+
});
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
return entry;
|
|
377
|
+
}
|
|
378
|
+
catch (error) {
|
|
379
|
+
logger.debug('Failed to create GitHub index entry', {
|
|
380
|
+
path: fileInfo.path,
|
|
381
|
+
error: error instanceof Error ? error.message : String(error)
|
|
382
|
+
});
|
|
383
|
+
return null;
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
/**
|
|
387
|
+
* Parse metadata from file content (frontmatter)
|
|
388
|
+
*/
|
|
389
|
+
parseMetadataFromContent(content) {
|
|
390
|
+
const metadata = {};
|
|
391
|
+
// Simple frontmatter parsing (could use a proper YAML parser)
|
|
392
|
+
const frontmatterMatch = content.match(/^---\n([\s\S]*?)\n---/);
|
|
393
|
+
if (frontmatterMatch) {
|
|
394
|
+
const frontmatter = frontmatterMatch[1];
|
|
395
|
+
const nameMatch = frontmatter.match(/^name:\s*(.+)$/m);
|
|
396
|
+
if (nameMatch)
|
|
397
|
+
metadata.name = nameMatch[1].trim();
|
|
398
|
+
const descMatch = frontmatter.match(/^description:\s*(.+)$/m);
|
|
399
|
+
if (descMatch)
|
|
400
|
+
metadata.description = descMatch[1].trim();
|
|
401
|
+
const versionMatch = frontmatter.match(/^version:\s*(.+)$/m);
|
|
402
|
+
if (versionMatch)
|
|
403
|
+
metadata.version = versionMatch[1].trim();
|
|
404
|
+
const authorMatch = frontmatter.match(/^author:\s*(.+)$/m);
|
|
405
|
+
if (authorMatch)
|
|
406
|
+
metadata.author = authorMatch[1].trim();
|
|
407
|
+
}
|
|
408
|
+
return metadata;
|
|
409
|
+
}
|
|
410
|
+
/**
|
|
411
|
+
* Get GitHub username from authenticated token
|
|
412
|
+
*/
|
|
413
|
+
async getGitHubUsername() {
|
|
414
|
+
try {
|
|
415
|
+
const userInfo = await this.githubClient.fetchFromGitHub('https://api.github.com/user', true);
|
|
416
|
+
return userInfo.login;
|
|
417
|
+
}
|
|
418
|
+
catch (error) {
|
|
419
|
+
throw new Error('Failed to get GitHub username. Please ensure you are authenticated with GitHub.');
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
/**
|
|
423
|
+
* Check if cache is valid
|
|
424
|
+
*/
|
|
425
|
+
isCacheValid() {
|
|
426
|
+
if (!this.cache || !this.lastFetch) {
|
|
427
|
+
return false;
|
|
428
|
+
}
|
|
429
|
+
const age = Date.now() - this.lastFetch.getTime();
|
|
430
|
+
return age < this.ttl;
|
|
431
|
+
}
|
|
432
|
+
/**
|
|
433
|
+
* Determine if we should fetch fresh data
|
|
434
|
+
*/
|
|
435
|
+
shouldFetchFresh() {
|
|
436
|
+
// Always fetch if no cache
|
|
437
|
+
if (!this.cache || !this.lastFetch) {
|
|
438
|
+
return true;
|
|
439
|
+
}
|
|
440
|
+
// Check for recent user actions
|
|
441
|
+
if (this.recentUserAction && this.actionTimestamp) {
|
|
442
|
+
const actionAge = Date.now() - this.actionTimestamp.getTime();
|
|
443
|
+
if (actionAge < this.actionGracePeriod) {
|
|
444
|
+
logger.debug('Fetching fresh due to recent user action', { actionAge });
|
|
445
|
+
return true;
|
|
446
|
+
}
|
|
447
|
+
else {
|
|
448
|
+
// Grace period expired, clear action flag
|
|
449
|
+
this.recentUserAction = false;
|
|
450
|
+
this.actionTimestamp = null;
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
// Check TTL
|
|
454
|
+
return !this.isCacheValid();
|
|
455
|
+
}
|
|
456
|
+
/**
|
|
457
|
+
* Create empty index when no portfolio exists
|
|
458
|
+
*/
|
|
459
|
+
createEmptyIndex(username, repository) {
|
|
460
|
+
const index = {
|
|
461
|
+
username: username || 'unknown',
|
|
462
|
+
repository: repository || 'dollhouse-portfolio',
|
|
463
|
+
lastUpdated: new Date(),
|
|
464
|
+
elements: new Map(),
|
|
465
|
+
totalElements: 0,
|
|
466
|
+
sha: ''
|
|
467
|
+
};
|
|
468
|
+
// Initialize empty element type maps
|
|
469
|
+
for (const elementType of Object.values(ElementType)) {
|
|
470
|
+
index.elements.set(elementType, []);
|
|
471
|
+
}
|
|
472
|
+
return index;
|
|
473
|
+
}
|
|
474
|
+
}
|
|
475
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"GitHubPortfolioIndexer.js","sourceRoot":"","sources":["../../src/portfolio/GitHubPortfolioIndexer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * GitHub Portfolio Indexer - Fetches and indexes user's GitHub portfolio for fast searching\n * \n * Features:\n * - Singleton pattern for efficient resource usage\n * - Smart caching with TTL and invalidation after user actions\n * - GraphQL/REST API integration for efficient fetching\n * - Rate limiting and authentication handling\n * - Fallback strategy for resilient operation\n * - Performance optimized for 1000+ portfolio elements\n */\n\nimport { GitHubClient } from '../collection/GitHubClient.js';\nimport { PortfolioRepoManager } from './PortfolioRepoManager.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { ElementType } from './types.js';\nimport { logger } from '../utils/logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { APICache } from '../cache/APICache.js';\n\nexport interface GitHubIndexEntry {\n  path: string;\n  name: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  elementType: ElementType;\n  sha: string; // File SHA for change detection\n  htmlUrl: string; // Link to GitHub\n  downloadUrl: string;\n  lastModified: Date;\n  size: number;\n}\n\nexport interface GitHubPortfolioIndex {\n  username: string;\n  repository: string;\n  lastUpdated: Date;\n  elements: Map<ElementType, GitHubIndexEntry[]>;\n  totalElements: number;\n  sha: string; // Latest commit SHA\n  rateLimitInfo?: {\n    remaining: number;\n    resetTime: Date;\n  };\n}\n\nexport interface GitHubFetchOptions {\n  force?: boolean;\n  maxElements?: number;\n  elementTypes?: ElementType[];\n  useGraphQL?: boolean;\n}\n\nexport class GitHubPortfolioIndexer {\n  private static instance: GitHubPortfolioIndexer | null = null;\n  private static instanceLock = false;\n  \n  private cache: GitHubPortfolioIndex | null = null;\n  private lastFetch: Date | null = null;\n  private readonly ttl = 15 * 60 * 1000; // 15 minutes\n  private recentUserAction = false;\n  private actionTimestamp: Date | null = null;\n  private readonly actionGracePeriod = 2 * 60 * 1000; // 2 minutes after action\n  \n  private githubClient: GitHubClient;\n  private portfolioRepoManager: PortfolioRepoManager;\n  private apiCache: APICache;\n  private rateLimitTracker: Map<string, number[]>;\n  \n  private constructor() {\n    this.apiCache = new APICache(); // Uses default settings\n    this.rateLimitTracker = new Map();\n    this.githubClient = new GitHubClient(this.apiCache, this.rateLimitTracker);\n    this.portfolioRepoManager = new PortfolioRepoManager();\n    \n    logger.debug('GitHubPortfolioIndexer created');\n  }\n\n  /**\n   * Singleton pattern with thread safety\n   */\n  public static getInstance(): GitHubPortfolioIndexer {\n    if (!this.instance) {\n      if (this.instanceLock) {\n        throw new Error('GitHubPortfolioIndexer instance is being created by another thread');\n      }\n      \n      try {\n        this.instanceLock = true;\n        this.instance = new GitHubPortfolioIndexer();\n      } finally {\n        this.instanceLock = false;\n      }\n    }\n    return this.instance;\n  }\n\n  /**\n   * Main method to get GitHub portfolio index\n   */\n  public async getIndex(force = false): Promise<GitHubPortfolioIndex> {\n    try {\n      // Check if we need fresh data\n      if (force || this.shouldFetchFresh()) {\n        return await this.fetchFresh();\n      }\n      \n      // Return cached data if available and valid\n      if (this.cache && this.isCacheValid()) {\n        logger.debug('Returning cached GitHub portfolio index', {\n          username: this.cache.username,\n          totalElements: this.cache.totalElements,\n          age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n        });\n        return this.cache;\n      }\n      \n      // Try to fetch fresh, fall back to stale cache on failure\n      try {\n        return await this.fetchFresh();\n      } catch (error) {\n        logger.warn('Failed to fetch fresh GitHub portfolio index, checking for stale cache', {\n          error: error instanceof Error ? error.message : String(error)\n        });\n        \n        // Return stale cache if available\n        if (this.cache) {\n          logger.info('Returning stale GitHub portfolio cache as fallback', {\n            username: this.cache.username,\n            age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n          });\n          return this.cache;\n        }\n        \n        // Return empty index as last resort\n        return this.createEmptyIndex();\n      }\n      \n    } catch (error) {\n      ErrorHandler.logError('GitHubPortfolioIndexer.getIndex', error);\n      \n      // Return stale cache or empty index\n      if (this.cache) {\n        return this.cache;\n      }\n      \n      return this.createEmptyIndex();\n    }\n  }\n\n  /**\n   * Invalidate cache after user actions\n   */\n  public invalidateAfterAction(action: string): void {\n    logger.info('Invalidating GitHub portfolio cache after user action', { action });\n    \n    this.recentUserAction = true;\n    this.actionTimestamp = new Date();\n    \n    // Log security event for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_CACHE_INVALIDATION',\n      severity: 'LOW',\n      source: 'GitHubPortfolioIndexer.invalidateAfterAction',\n      details: `Cache invalidated after user action: ${action}`,\n      metadata: { action }\n    });\n  }\n\n  /**\n   * Clear all cached data\n   */\n  public clearCache(): void {\n    this.cache = null;\n    this.lastFetch = null;\n    this.recentUserAction = false;\n    this.actionTimestamp = null;\n    this.apiCache.clear();\n    \n    logger.info('GitHub portfolio cache cleared');\n  }\n\n  /**\n   * Get cache statistics\n   */\n  public getCacheStats(): {\n    hasCachedData: boolean;\n    lastFetch: Date | null;\n    isStale: boolean;\n    recentUserAction: boolean;\n    totalElements: number;\n  } {\n    return {\n      hasCachedData: this.cache !== null,\n      lastFetch: this.lastFetch,\n      isStale: !this.isCacheValid(),\n      recentUserAction: this.recentUserAction,\n      totalElements: this.cache?.totalElements || 0\n    };\n  }\n\n  /**\n   * Fetch fresh data from GitHub\n   */\n  private async fetchFresh(): Promise<GitHubPortfolioIndex> {\n    const startTime = Date.now();\n    logger.info('Fetching fresh GitHub portfolio index...');\n    \n    try {\n      // Get GitHub username from token\n      const username = await this.getGitHubUsername();\n      const repository = 'dollhouse-portfolio';\n      \n      // Check if portfolio repository exists\n      const repoExists = await this.portfolioRepoManager.checkPortfolioExists(username);\n      if (!repoExists) {\n        logger.info('GitHub portfolio repository does not exist', { username });\n        return this.createEmptyIndex(username, repository);\n      }\n      \n      // Fetch repository content using GitHub API\n      const index = await this.fetchRepositoryContent(username, repository);\n      \n      // Update cache\n      this.cache = index;\n      this.lastFetch = new Date();\n      this.recentUserAction = false;\n      this.actionTimestamp = null;\n      \n      const duration = Date.now() - startTime;\n      logger.info('GitHub portfolio index fetched successfully', {\n        username,\n        totalElements: index.totalElements,\n        duration: `${duration}ms`,\n        rateLimitRemaining: index.rateLimitInfo?.remaining\n      });\n      \n      // Log security event\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_FETCH_SUCCESS',\n        severity: 'LOW',\n        source: 'GitHubPortfolioIndexer.fetchFresh',\n        details: `Fetched GitHub portfolio with ${index.totalElements} elements in ${duration}ms`,\n        metadata: { username, duration, totalElements: index.totalElements }\n      });\n      \n      return index;\n      \n    } catch (error) {\n      const duration = Date.now() - startTime;\n      ErrorHandler.logError('GitHubPortfolioIndexer.fetchFresh', error, { duration });\n      throw ErrorHandler.wrapError(error, 'Failed to fetch GitHub portfolio index', ErrorCategory.NETWORK_ERROR);\n    }\n  }\n\n  /**\n   * Fetch repository content from GitHub API\n   */\n  private async fetchRepositoryContent(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    // Try GraphQL first for better performance, fallback to REST\n    try {\n      return await this.fetchWithGraphQL(username, repository);\n    } catch (graphqlError) {\n      logger.debug('GraphQL fetch failed, falling back to REST API', {\n        error: graphqlError instanceof Error ? graphqlError.message : String(graphqlError)\n      });\n      \n      return await this.fetchWithREST(username, repository);\n    }\n  }\n\n  /**\n   * Fetch using GraphQL for better performance\n   */\n  private async fetchWithGraphQL(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const query = `\n      query GetPortfolioContent($owner: String!, $name: String!) {\n        repository(owner: $owner, name: $name) {\n          defaultBranchRef {\n            target {\n              ... on Commit {\n                oid\n                history(first: 1) {\n                  nodes {\n                    committedDate\n                  }\n                }\n              }\n            }\n          }\n          object(expression: \"HEAD:\") {\n            ... on Tree {\n              entries {\n                name\n                type\n                object {\n                  ... on Tree {\n                    entries {\n                      name\n                      type\n                      oid\n                      object {\n                        ... on Blob {\n                          byteSize\n                          text\n                        }\n                      }\n                    }\n                  }\n                }\n              }\n            }\n          }\n        }\n        rateLimit {\n          remaining\n          resetAt\n        }\n      }\n    `;\n    \n    const variables = { owner: username, name: repository };\n    \n    const response = await this.githubClient.fetchFromGitHub('https://api.github.com/graphql', true);\n    \n    // Note: This is a simplified GraphQL implementation\n    // In a real implementation, you would send POST request with query and variables\n    throw new Error('GraphQL implementation not yet complete');\n  }\n\n  /**\n   * Fetch using REST API with pagination\n   */\n  private async fetchWithREST(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    \n    // Get repository info and latest commit\n    const repoInfo = await this.githubClient.fetchFromGitHub(\n      `https://api.github.com/repos/${normalizedUsername}/${repository}`\n    );\n    \n    const latestCommit = await this.githubClient.fetchFromGitHub(\n      `https://api.github.com/repos/${normalizedUsername}/${repository}/commits/HEAD`\n    );\n    \n    // Initialize index\n    const index: GitHubPortfolioIndex = {\n      username: normalizedUsername,\n      repository,\n      lastUpdated: new Date(latestCommit.commit.committer.date),\n      elements: new Map(),\n      totalElements: 0,\n      sha: latestCommit.sha\n    };\n    \n    // Initialize element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    // Fetch content for each element type\n    for (const elementType of Object.values(ElementType)) {\n      try {\n        const entries = await this.fetchElementTypeContent(normalizedUsername, repository, elementType);\n        index.elements.set(elementType, entries);\n        index.totalElements += entries.length;\n      } catch (error) {\n        logger.warn(`Failed to fetch ${elementType} from GitHub portfolio`, {\n          error: error instanceof Error ? error.message : String(error)\n        });\n        // Continue with other element types\n      }\n    }\n    \n    return index;\n  }\n\n  /**\n   * Fetch content for a specific element type\n   */\n  private async fetchElementTypeContent(\n    username: string,\n    repository: string,\n    elementType: ElementType\n  ): Promise<GitHubIndexEntry[]> {\n    try {\n      // Get directory listing\n      const contents = await this.githubClient.fetchFromGitHub(\n        `https://api.github.com/repos/${username}/${repository}/contents/${elementType}`\n      );\n      \n      if (!Array.isArray(contents)) {\n        return [];\n      }\n      \n      const entries: GitHubIndexEntry[] = [];\n      const maxConcurrent = 5; // Limit concurrent requests\n      \n      // Process files in batches to avoid rate limiting\n      for (let i = 0; i < contents.length; i += maxConcurrent) {\n        const batch = contents.slice(i, i + maxConcurrent);\n        const batchPromises = batch\n          .filter(item => item.type === 'file' && item.name.endsWith('.md'))\n          .map(item => this.createGitHubIndexEntry(username, repository, elementType, item));\n        \n        const batchResults = await Promise.allSettled(batchPromises);\n        \n        for (const result of batchResults) {\n          if (result.status === 'fulfilled' && result.value) {\n            entries.push(result.value);\n          }\n        }\n        \n        // Add delay between batches to respect rate limits\n        if (i + maxConcurrent < contents.length) {\n          await new Promise(resolve => setTimeout(resolve, 100));\n        }\n      }\n      \n      return entries;\n      \n    } catch (error) {\n      // Directory might not exist\n      if (error instanceof Error && error.message.includes('404')) {\n        return [];\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Create GitHub index entry from API response\n   */\n  private async createGitHubIndexEntry(\n    username: string,\n    repository: string,\n    elementType: ElementType,\n    fileInfo: any\n  ): Promise<GitHubIndexEntry | null> {\n    try {\n      // Parse metadata from filename or fetch content if needed\n      const name = fileInfo.name.replace('.md', '').replace(/-/g, ' ');\n      \n      const entry: GitHubIndexEntry = {\n        path: fileInfo.path,\n        name,\n        elementType,\n        sha: fileInfo.sha,\n        htmlUrl: fileInfo.html_url,\n        downloadUrl: fileInfo.download_url,\n        lastModified: new Date(), // GitHub API doesn't provide file modification time directly\n        size: fileInfo.size || 0\n      };\n      \n      // Optionally fetch content to extract metadata\n      // This is expensive, so only do it for small files or when specifically needed\n      if (fileInfo.size && fileInfo.size < 10000) { // Only for files < 10KB\n        try {\n          const content = await this.githubClient.fetchFromGitHub(fileInfo.download_url);\n          const metadata = this.parseMetadataFromContent(content);\n          \n          if (metadata.name) entry.name = metadata.name;\n          if (metadata.description) entry.description = metadata.description;\n          if (metadata.version) entry.version = metadata.version;\n          if (metadata.author) entry.author = metadata.author;\n        } catch (metadataError) {\n          // Non-critical error, continue without metadata\n          logger.debug('Failed to fetch metadata for file', {\n            path: fileInfo.path,\n            error: metadataError instanceof Error ? metadataError.message : String(metadataError)\n          });\n        }\n      }\n      \n      return entry;\n      \n    } catch (error) {\n      logger.debug('Failed to create GitHub index entry', {\n        path: fileInfo.path,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Parse metadata from file content (frontmatter)\n   */\n  private parseMetadataFromContent(content: string): {\n    name?: string;\n    description?: string;\n    version?: string;\n    author?: string;\n  } {\n    const metadata: any = {};\n    \n    // Simple frontmatter parsing (could use a proper YAML parser)\n    const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n    if (frontmatterMatch) {\n      const frontmatter = frontmatterMatch[1];\n      \n      const nameMatch = frontmatter.match(/^name:\\s*(.+)$/m);\n      if (nameMatch) metadata.name = nameMatch[1].trim();\n      \n      const descMatch = frontmatter.match(/^description:\\s*(.+)$/m);\n      if (descMatch) metadata.description = descMatch[1].trim();\n      \n      const versionMatch = frontmatter.match(/^version:\\s*(.+)$/m);\n      if (versionMatch) metadata.version = versionMatch[1].trim();\n      \n      const authorMatch = frontmatter.match(/^author:\\s*(.+)$/m);\n      if (authorMatch) metadata.author = authorMatch[1].trim();\n    }\n    \n    return metadata;\n  }\n\n  /**\n   * Get GitHub username from authenticated token\n   */\n  private async getGitHubUsername(): Promise<string> {\n    try {\n      const userInfo = await this.githubClient.fetchFromGitHub('https://api.github.com/user', true);\n      return userInfo.login;\n    } catch (error) {\n      throw new Error('Failed to get GitHub username. Please ensure you are authenticated with GitHub.');\n    }\n  }\n\n  /**\n   * Check if cache is valid\n   */\n  private isCacheValid(): boolean {\n    if (!this.cache || !this.lastFetch) {\n      return false;\n    }\n    \n    const age = Date.now() - this.lastFetch.getTime();\n    return age < this.ttl;\n  }\n\n  /**\n   * Determine if we should fetch fresh data\n   */\n  private shouldFetchFresh(): boolean {\n    // Always fetch if no cache\n    if (!this.cache || !this.lastFetch) {\n      return true;\n    }\n    \n    // Check for recent user actions\n    if (this.recentUserAction && this.actionTimestamp) {\n      const actionAge = Date.now() - this.actionTimestamp.getTime();\n      if (actionAge < this.actionGracePeriod) {\n        logger.debug('Fetching fresh due to recent user action', { actionAge });\n        return true;\n      } else {\n        // Grace period expired, clear action flag\n        this.recentUserAction = false;\n        this.actionTimestamp = null;\n      }\n    }\n    \n    // Check TTL\n    return !this.isCacheValid();\n  }\n\n  /**\n   * Create empty index when no portfolio exists\n   */\n  private createEmptyIndex(username?: string, repository?: string): GitHubPortfolioIndex {\n    const index: GitHubPortfolioIndex = {\n      username: username || 'unknown',\n      repository: repository || 'dollhouse-portfolio',\n      lastUpdated: new Date(),\n      elements: new Map(),\n      totalElements: 0,\n      sha: ''\n    };\n    \n    // Initialize empty element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    return index;\n  }\n}"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"MigrationManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/MigrationManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAIH,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AACzD,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"MigrationManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/MigrationManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAIH,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AACzD,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAOzC,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,OAAO,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,QAAQ,EAAE,OAAO,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,gBAAgB,CAAmB;gBAE/B,gBAAgB,EAAE,gBAAgB;IAI9C;;OAEG;IACU,cAAc,IAAI,OAAO,CAAC,OAAO,CAAC;IAQ/C;;OAEG;IACU,OAAO,CAAC,OAAO,CAAC,EAAE;QAAE,MAAM,CAAC,EAAE,OAAO,CAAA;KAAE,GAAG,OAAO,CAAC,eAAe,CAAC;IA8I9E;;OAEG;YACW,cAAc;IA6F5B;;OAEG;YACW,YAAY;IAwC1B;;OAEG;IACU,kBAAkB,IAAI,OAAO,CAAC;QACzC,iBAAiB,EAAE,OAAO,CAAC;QAC3B,kBAAkB,EAAE,MAAM,CAAC;QAC3B,eAAe,EAAE,OAAO,CAAC;QACzB,cAAc,EAAE,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;KAC7C,CAAC;CAsBH"}
|