@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.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /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":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAC7D,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AAEjE,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AACzC,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,gBAAgB,EAAE,MAAM,4CAA4C,CAAC;AAC9E,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACvE,OAAO,EAAE,QAAQ,EAAE,MAAM,sBAAsB,CAAC;AAoChD,MAAM,OAAO,sBAAsB;IACzB,MAAM,CAAC,QAAQ,GAAkC,IAAI,CAAC;IACtD,MAAM,CAAC,YAAY,GAAG,KAAK,CAAC;IAE5B,KAAK,GAAgC,IAAI,CAAC;IAC1C,SAAS,GAAgB,IAAI,CAAC;IACrB,GAAG,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,aAAa;IAC5C,gBAAgB,GAAG,KAAK,CAAC;IACzB,eAAe,GAAgB,IAAI,CAAC;IAC3B,iBAAiB,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,yBAAyB;IAErE,YAAY,CAAe;IAC3B,oBAAoB,CAAuB;IAC3C,QAAQ,CAAW;IACnB,gBAAgB,CAAwB;IAEhD;QACE,IAAI,CAAC,QAAQ,GAAG,IAAI,QAAQ,EAAE,CAAC,CAAC,wBAAwB;QACxD,IAAI,CAAC,gBAAgB,GAAG,IAAI,GAAG,EAAE,CAAC;QAClC,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC3E,IAAI,CAAC,oBAAoB,GAAG,IAAI,oBAAoB,EAAE,CAAC;QAEvD,MAAM,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;IACjD,CAAC;IAED;;OAEG;IACI,MAAM,CAAC,WAAW;QACvB,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACnB,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;gBACtB,MAAM,IAAI,KAAK,CAAC,oEAAoE,CAAC,CAAC;YACxF,CAAC;YAED,IAAI,CAAC;gBACH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;gBACzB,IAAI,CAAC,QAAQ,GAAG,IAAI,sBAAsB,EAAE,CAAC;YAC/C,CAAC;oBAAS,CAAC;gBACT,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;YAC5B,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED;;OAEG;IACI,KAAK,CAAC,QAAQ,CAAC,KAAK,GAAG,KAAK;QACjC,IAAI,CAAC;YACH,8BAA8B;YAC9B,IAAI,KAAK,IAAI,IAAI,CAAC,gBAAgB,EAAE,EAAE,CAAC;gBACrC,OAAO,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;YACjC,CAAC;YAED,4CAA4C;YAC5C,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;gBACtC,MAAM,CAAC,KAAK,CAAC,yCAAyC,EAAE;oBACtD,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,QAAQ;oBAC7B,aAAa,EAAE,IAAI,CAAC,KAAK,CAAC,aAAa;oBACvC,GAAG,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS;iBACxE,CAAC,CAAC;gBACH,OAAO,IAAI,CAAC,KAAK,CAAC;YACpB,CAAC;YAED,0DAA0D;YAC1D,IAAI,CAAC;gBACH,OAAO,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;YACjC,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,IAAI,CAAC,wEAAwE,EAAE;oBACpF,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;iBAC9D,CAAC,CAAC;gBAEH,kCAAkC;gBAClC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;oBACf,MAAM,CAAC,IAAI,CAAC,oDAAoD,EAAE;wBAChE,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,QAAQ;wBAC7B,GAAG,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS;qBACxE,CAAC,CAAC;oBACH,OAAO,IAAI,CAAC,KAAK,CAAC;gBACpB,CAAC;gBAED,oCAAoC;gBACpC,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACjC,CAAC;QAEH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,YAAY,CAAC,QAAQ,CAAC,iCAAiC,EAAE,KAAK,CAAC,CAAC;YAEhE,oCAAoC;YACpC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,OAAO,IAAI,CAAC,KAAK,CAAC;YACpB,CAAC;YAED,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACjC,CAAC;IACH,CAAC;IAED;;OAEG;IACI,qBAAqB,CAAC,MAAc;QACzC,MAAM,CAAC,IAAI,CAAC,uDAAuD,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;QAEjF,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC7B,IAAI,CAAC,eAAe,GAAG,IAAI,IAAI,EAAE,CAAC;QAElC,qCAAqC;QACrC,eAAe,CAAC,gBAAgB,CAAC;YAC/B,IAAI,EAAE,8BAA8B;YACpC,QAAQ,EAAE,KAAK;YACf,MAAM,EAAE,8CAA8C;YACtD,OAAO,EAAE,wCAAwC,MAAM,EAAE;YACzD,QAAQ,EAAE,EAAE,MAAM,EAAE;SACrB,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACI,UAAU;QACf,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QAClB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAC9B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;QAC5B,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;QAEtB,MAAM,CAAC,IAAI,CAAC,gCAAgC,CAAC,CAAC;IAChD,CAAC;IAED;;OAEG;IACI,aAAa;QAOlB,OAAO;YACL,aAAa,EAAE,IAAI,CAAC,KAAK,KAAK,IAAI;YAClC,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,OAAO,EAAE,CAAC,IAAI,CAAC,YAAY,EAAE;YAC7B,gBAAgB,EAAE,IAAI,CAAC,gBAAgB;YACvC,aAAa,EAAE,IAAI,CAAC,KAAK,EAAE,aAAa,IAAI,CAAC;SAC9C,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,UAAU;QACtB,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,MAAM,CAAC,IAAI,CAAC,0CAA0C,CAAC,CAAC;QAExD,IAAI,CAAC;YACH,iCAAiC;YACjC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAChD,MAAM,UAAU,GAAG,qBAAqB,CAAC;YAEzC,uCAAuC;YACvC,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;YAClF,IAAI,CAAC,UAAU,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,4CAA4C,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;gBACxE,OAAO,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;YACrD,CAAC;YAED,4CAA4C;YAC5C,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,sBAAsB,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;YAEtE,eAAe;YACf,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;YACnB,IAAI,CAAC,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC;YAC5B,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;YAC9B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;YAE5B,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YACxC,MAAM,CAAC,IAAI,CAAC,6CAA6C,EAAE;gBACzD,QAAQ;gBACR,aAAa,EAAE,KAAK,CAAC,aAAa;gBAClC,QAAQ,EAAE,GAAG,QAAQ,IAAI;gBACzB,kBAAkB,EAAE,KAAK,CAAC,aAAa,EAAE,SAAS;aACnD,CAAC,CAAC;YAEH,qBAAqB;YACrB,eAAe,CAAC,gBAAgB,CAAC;gBAC/B,IAAI,EAAE,yBAAyB;gBAC/B,QAAQ,EAAE,KAAK;gBACf,MAAM,EAAE,mCAAmC;gBAC3C,OAAO,EAAE,iCAAiC,KAAK,CAAC,aAAa,gBAAgB,QAAQ,IAAI;gBACzF,QAAQ,EAAE,EAAE,QAAQ,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,CAAC,aAAa,EAAE;aACrE,CAAC,CAAC;YAEH,OAAO,KAAK,CAAC;QAEf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YACxC,YAAY,CAAC,QAAQ,CAAC,mCAAmC,EAAE,KAAK,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YAChF,MAAM,YAAY,CAAC,SAAS,CAAC,KAAK,EAAE,wCAAwC,EAAE,aAAa,CAAC,aAAa,CAAC,CAAC;QAC7G,CAAC;IACH,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,sBAAsB,CAAC,QAAgB,EAAE,UAAkB;QACvE,6DAA6D;QAC7D,IAAI,CAAC;YACH,OAAO,MAAM,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;QAC3D,CAAC;QAAC,OAAO,YAAY,EAAE,CAAC;YACtB,MAAM,CAAC,KAAK,CAAC,gDAAgD,EAAE;gBAC7D,KAAK,EAAE,YAAY,YAAY,KAAK,CAAC,CAAC,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC;aACnF,CAAC,CAAC;YAEH,OAAO,MAAM,IAAI,CAAC,aAAa,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;QACxD,CAAC;IACH,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,gBAAgB,CAAC,QAAgB,EAAE,UAAkB;QACjE,MAAM,KAAK,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;KA4Cb,CAAC;QAEF,MAAM,SAAS,GAAG,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC;QAExD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,gCAAgC,EAAE,IAAI,CAAC,CAAC;QAEjG,oDAAoD;QACpD,iFAAiF;QACjF,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;IAC7D,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,aAAa,CAAC,QAAgB,EAAE,UAAkB;QAC9D,MAAM,kBAAkB,GAAG,gBAAgB,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,iBAAiB,CAAC;QAElF,wCAAwC;QACxC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CACtD,gCAAgC,kBAAkB,IAAI,UAAU,EAAE,CACnE,CAAC;QAEF,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAC1D,gCAAgC,kBAAkB,IAAI,UAAU,eAAe,CAChF,CAAC;QAEF,mBAAmB;QACnB,MAAM,KAAK,GAAyB;YAClC,QAAQ,EAAE,kBAAkB;YAC5B,UAAU;YACV,WAAW,EAAE,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC;YACzD,QAAQ,EAAE,IAAI,GAAG,EAAE;YACnB,aAAa,EAAE,CAAC;YAChB,GAAG,EAAE,YAAY,CAAC,GAAG;SACtB,CAAC;QAEF,+BAA+B;QAC/B,KAAK,MAAM,WAAW,IAAI,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QACtC,CAAC;QAED,sCAAsC;QACtC,KAAK,MAAM,WAAW,IAAI,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,uBAAuB,CAAC,kBAAkB,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC;gBAChG,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;gBACzC,KAAK,CAAC,aAAa,IAAI,OAAO,CAAC,MAAM,CAAC;YACxC,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,IAAI,CAAC,mBAAmB,WAAW,wBAAwB,EAAE;oBAClE,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;iBAC9D,CAAC,CAAC;gBACH,oCAAoC;YACtC,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,uBAAuB,CACnC,QAAgB,EAChB,UAAkB,EAClB,WAAwB;QAExB,IAAI,CAAC;YACH,wBAAwB;YACxB,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CACtD,gCAAgC,QAAQ,IAAI,UAAU,aAAa,WAAW,EAAE,CACjF,CAAC;YAEF,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAC7B,OAAO,EAAE,CAAC;YACZ,CAAC;YAED,MAAM,OAAO,GAAuB,EAAE,CAAC;YACvC,MAAM,aAAa,GAAG,CAAC,CAAC,CAAC,4BAA4B;YAErD,kDAAkD;YAClD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC,IAAI,aAAa,EAAE,CAAC;gBACxD,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,CAAC;gBACnD,MAAM,aAAa,GAAG,KAAK;qBACxB,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;qBACjE,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,QAAQ,EAAE,UAAU,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC,CAAC;gBAErF,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;gBAE7D,KAAK,MAAM,MAAM,IAAI,YAAY,EAAE,CAAC;oBAClC,IAAI,MAAM,CAAC,MAAM,KAAK,WAAW,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBAClD,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;oBAC7B,CAAC;gBACH,CAAC;gBAED,mDAAmD;gBACnD,IAAI,CAAC,GAAG,aAAa,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC;oBACxC,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,CAAC;gBACzD,CAAC;YACH,CAAC;YAED,OAAO,OAAO,CAAC;QAEjB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,4BAA4B;YAC5B,IAAI,KAAK,YAAY,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC5D,OAAO,EAAE,CAAC;YACZ,CAAC;YACD,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,sBAAsB,CAClC,QAAgB,EAChB,UAAkB,EAClB,WAAwB,EACxB,QAAa;QAEb,IAAI,CAAC;YACH,0DAA0D;YAC1D,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;YAEjE,MAAM,KAAK,GAAqB;gBAC9B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,IAAI;gBACJ,WAAW;gBACX,GAAG,EAAE,QAAQ,CAAC,GAAG;gBACjB,OAAO,EAAE,QAAQ,CAAC,QAAQ;gBAC1B,WAAW,EAAE,QAAQ,CAAC,YAAY;gBAClC,YAAY,EAAE,IAAI,IAAI,EAAE,EAAE,6DAA6D;gBACvF,IAAI,EAAE,QAAQ,CAAC,IAAI,IAAI,CAAC;aACzB,CAAC;YAEF,+CAA+C;YAC/C,+EAA+E;YAC/E,IAAI,QAAQ,CAAC,IAAI,IAAI,QAAQ,CAAC,IAAI,GAAG,KAAK,EAAE,CAAC,CAAC,wBAAwB;gBACpE,IAAI,CAAC;oBACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;oBAC/E,MAAM,QAAQ,GAAG,IAAI,CAAC,wBAAwB,CAAC,OAAO,CAAC,CAAC;oBAExD,IAAI,QAAQ,CAAC,IAAI;wBAAE,KAAK,CAAC,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;oBAC9C,IAAI,QAAQ,CAAC,WAAW;wBAAE,KAAK,CAAC,WAAW,GAAG,QAAQ,CAAC,WAAW,CAAC;oBACnE,IAAI,QAAQ,CAAC,OAAO;wBAAE,KAAK,CAAC,OAAO,GAAG,QAAQ,CAAC,OAAO,CAAC;oBACvD,IAAI,QAAQ,CAAC,MAAM;wBAAE,KAAK,CAAC,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC;gBACtD,CAAC;gBAAC,OAAO,aAAa,EAAE,CAAC;oBACvB,gDAAgD;oBAChD,MAAM,CAAC,KAAK,CAAC,mCAAmC,EAAE;wBAChD,IAAI,EAAE,QAAQ,CAAC,IAAI;wBACnB,KAAK,EAAE,aAAa,YAAY,KAAK,CAAC,CAAC,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,aAAa,CAAC;qBACtF,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YAED,OAAO,KAAK,CAAC;QAEf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,qCAAqC,EAAE;gBAClD,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;aAC9D,CAAC,CAAC;YACH,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACK,wBAAwB,CAAC,OAAe;QAM9C,MAAM,QAAQ,GAAQ,EAAE,CAAC;QAEzB,8DAA8D;QAC9D,MAAM,gBAAgB,GAAG,OAAO,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;QAChE,IAAI,gBAAgB,EAAE,CAAC;YACrB,MAAM,WAAW,GAAG,gBAAgB,CAAC,CAAC,CAAC,CAAC;YAExC,MAAM,SAAS,GAAG,WAAW,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;YACvD,IAAI,SAAS;gBAAE,QAAQ,CAAC,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAEnD,MAAM,SAAS,GAAG,WAAW,CAAC,KAAK,CAAC,wBAAwB,CAAC,CAAC;YAC9D,IAAI,SAAS;gBAAE,QAAQ,CAAC,WAAW,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAE1D,MAAM,YAAY,GAAG,WAAW,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;YAC7D,IAAI,YAAY;gBAAE,QAAQ,CAAC,OAAO,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAE5D,MAAM,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAC3D,IAAI,WAAW;gBAAE,QAAQ,CAAC,MAAM,GAAG,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;QAC3D,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,iBAAiB;QAC7B,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,6BAA6B,EAAE,IAAI,CAAC,CAAC;YAC9F,OAAO,QAAQ,CAAC,KAAK,CAAC;QACxB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,IAAI,KAAK,CAAC,iFAAiF,CAAC,CAAC;QACrG,CAAC;IACH,CAAC;IAED;;OAEG;IACK,YAAY;QAClB,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACnC,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;QAClD,OAAO,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC;IACxB,CAAC;IAED;;OAEG;IACK,gBAAgB;QACtB,2BAA2B;QAC3B,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACnC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,gCAAgC;QAChC,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YAClD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;YAC9D,IAAI,SAAS,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACvC,MAAM,CAAC,KAAK,CAAC,0CAA0C,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;gBACxE,OAAO,IAAI,CAAC;YACd,CAAC;iBAAM,CAAC;gBACN,0CAA0C;gBAC1C,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;gBAC9B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;YAC9B,CAAC;QACH,CAAC;QAED,YAAY;QACZ,OAAO,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC;IAC9B,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,QAAiB,EAAE,UAAmB;QAC7D,MAAM,KAAK,GAAyB;YAClC,QAAQ,EAAE,QAAQ,IAAI,SAAS;YAC/B,UAAU,EAAE,UAAU,IAAI,qBAAqB;YAC/C,WAAW,EAAE,IAAI,IAAI,EAAE;YACvB,QAAQ,EAAE,IAAI,GAAG,EAAE;YACnB,aAAa,EAAE,CAAC;YAChB,GAAG,EAAE,EAAE;SACR,CAAC;QAEF,qCAAqC;QACrC,KAAK,MAAM,WAAW,IAAI,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QACtC,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC","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;AAIzC,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;IA2E9E;;OAEG;YACW,cAAc;IAiC5B;;OAEG;YACW,YAAY;IAuB1B;;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"}
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"}