@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,315 @@
1
+ /**
2
+ * GitHubRateLimiter - Specialized rate limiter for GitHub API calls
3
+ *
4
+ * Features:
5
+ * - Respects GitHub's authenticated (5000/hour) and unauthenticated (60/hour) limits
6
+ * - Client-side queuing when approaching limits
7
+ * - Request prioritization for critical operations
8
+ * - Comprehensive logging for quota management
9
+ * - Early termination when exact matches are found
10
+ */
11
+ import { RateLimiter } from './RateLimiter.js';
12
+ import { GITHUB_API_RATE_LIMITS } from '../config/portfolio-constants.js';
13
+ import { TokenManager } from '../security/tokenManager.js';
14
+ import { logger } from './logger.js';
15
+ import { SecurityMonitor } from '../security/securityMonitor.js';
16
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
17
+ export class GitHubRateLimiter {
18
+ rateLimiter;
19
+ requestQueue = [];
20
+ processing = false;
21
+ lastRateLimitInfo;
22
+ isAuthenticated = false;
23
+ constructor() {
24
+ // Initialize with conservative limits - will update based on auth status
25
+ this.updateLimitsForAuthStatus();
26
+ this.setupPeriodicStatusCheck();
27
+ }
28
+ /**
29
+ * Update rate limits based on current authentication status
30
+ */
31
+ async updateLimitsForAuthStatus() {
32
+ try {
33
+ const token = await TokenManager.getGitHubTokenAsync();
34
+ const newIsAuthenticated = !!token;
35
+ // Only recreate rate limiter if auth status changed
36
+ if (newIsAuthenticated !== this.isAuthenticated) {
37
+ this.isAuthenticated = newIsAuthenticated;
38
+ const limit = this.isAuthenticated
39
+ ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT
40
+ : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT;
41
+ // Apply buffer to stay below actual limits
42
+ const bufferedLimit = Math.floor(limit * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE);
43
+ const config = {
44
+ maxRequests: bufferedLimit,
45
+ windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,
46
+ minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS
47
+ };
48
+ this.rateLimiter = new RateLimiter(config);
49
+ logger.info('GitHub rate limiter updated', {
50
+ authenticated: this.isAuthenticated,
51
+ limit: bufferedLimit,
52
+ originalLimit: limit,
53
+ bufferPercentage: GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE
54
+ });
55
+ }
56
+ }
57
+ catch (error) {
58
+ logger.warn('Failed to check authentication status for rate limiting', { error });
59
+ // Fall back to unauthenticated limits
60
+ this.isAuthenticated = false;
61
+ this.rateLimiter = new RateLimiter({
62
+ maxRequests: Math.floor(GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE),
63
+ windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,
64
+ minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS
65
+ });
66
+ }
67
+ }
68
+ /**
69
+ * Setup periodic check for rate limit status
70
+ */
71
+ setupPeriodicStatusCheck() {
72
+ // Check auth status every 5 minutes
73
+ setInterval(() => {
74
+ this.updateLimitsForAuthStatus().catch(error => {
75
+ logger.warn('Periodic auth status check failed', { error });
76
+ });
77
+ }, 5 * 60 * 1000);
78
+ }
79
+ /**
80
+ * Queue a GitHub API request with rate limiting
81
+ * @param operation Description of the operation
82
+ * @param apiCall Function that makes the actual API call
83
+ * @param priority Request priority (high, normal, low)
84
+ * @returns Promise that resolves with the API response
85
+ */
86
+ async queueRequest(operation, apiCall, priority = 'normal') {
87
+ // SECURITY FIX (DMCP-SEC-004): Normalize Unicode in operation name to prevent injection attacks
88
+ const normalizedOperation = UnicodeValidator.normalize(operation);
89
+ if (!normalizedOperation.isValid) {
90
+ SecurityMonitor.logSecurityEvent({
91
+ type: 'UNICODE_VALIDATION_ERROR',
92
+ severity: 'MEDIUM',
93
+ source: 'GitHubRateLimiter.queueRequest',
94
+ details: `Invalid Unicode in operation name: ${normalizedOperation.detectedIssues?.[0] || 'unknown error'}`
95
+ });
96
+ // Use a safe fallback for the operation name
97
+ operation = 'github-api-request';
98
+ }
99
+ else {
100
+ operation = normalizedOperation.normalizedContent;
101
+ }
102
+ const requestId = `${operation}-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
103
+ return new Promise((resolve, reject) => {
104
+ const request = {
105
+ id: requestId,
106
+ operation,
107
+ priority,
108
+ timestamp: Date.now(),
109
+ resolve: async (value) => {
110
+ try {
111
+ logger.debug('Executing GitHub API request', {
112
+ operation,
113
+ requestId,
114
+ queueWaitTime: Date.now() - request.timestamp
115
+ });
116
+ const result = await apiCall();
117
+ resolve(result);
118
+ // Log successful API usage for quota tracking
119
+ this.logApiUsage(operation, 'success');
120
+ }
121
+ catch (error) {
122
+ // Check if this is a rate limit error from GitHub
123
+ if (this.isGitHubRateLimitError(error)) {
124
+ this.handleGitHubRateLimit(error);
125
+ }
126
+ reject(error);
127
+ this.logApiUsage(operation, 'error', error);
128
+ }
129
+ },
130
+ reject
131
+ };
132
+ // Add to queue with priority ordering
133
+ this.addToQueue(request);
134
+ this.processQueue();
135
+ });
136
+ }
137
+ /**
138
+ * Add request to queue with priority ordering
139
+ */
140
+ addToQueue(request) {
141
+ // Insert based on priority: high > normal > low
142
+ // Within same priority, maintain FIFO order
143
+ const priorityOrder = { high: 0, normal: 1, low: 2 };
144
+ let insertIndex = this.requestQueue.length;
145
+ for (let i = 0; i < this.requestQueue.length; i++) {
146
+ if (priorityOrder[request.priority] < priorityOrder[this.requestQueue[i].priority]) {
147
+ insertIndex = i;
148
+ break;
149
+ }
150
+ }
151
+ this.requestQueue.splice(insertIndex, 0, request);
152
+ logger.debug('GitHub API request queued', {
153
+ operation: request.operation,
154
+ priority: request.priority,
155
+ queuePosition: insertIndex,
156
+ totalQueued: this.requestQueue.length
157
+ });
158
+ }
159
+ /**
160
+ * Process the request queue
161
+ */
162
+ async processQueue() {
163
+ if (this.processing || this.requestQueue.length === 0) {
164
+ return;
165
+ }
166
+ this.processing = true;
167
+ try {
168
+ while (this.requestQueue.length > 0) {
169
+ // Update auth status periodically
170
+ if (Math.random() < 0.1) { // 10% chance
171
+ await this.updateLimitsForAuthStatus();
172
+ }
173
+ const rateLimitStatus = this.rateLimiter.checkLimit();
174
+ if (!rateLimitStatus.allowed) {
175
+ // Log rate limit wait
176
+ logger.info('GitHub API rate limit reached, waiting', {
177
+ retryAfterMs: rateLimitStatus.retryAfterMs,
178
+ remainingTokens: rateLimitStatus.remainingTokens,
179
+ queueLength: this.requestQueue.length,
180
+ resetTime: rateLimitStatus.resetTime
181
+ });
182
+ // Wait for the specified time
183
+ await new Promise(resolve => setTimeout(resolve, rateLimitStatus.retryAfterMs || 1000));
184
+ continue;
185
+ }
186
+ // Process the next request
187
+ const request = this.requestQueue.shift();
188
+ this.rateLimiter.consumeToken();
189
+ // Execute the request
190
+ request.resolve(null); // This will trigger the actual API call
191
+ }
192
+ }
193
+ finally {
194
+ this.processing = false;
195
+ }
196
+ }
197
+ /**
198
+ * Get current rate limit status
199
+ */
200
+ getStatus() {
201
+ const baseStatus = this.rateLimiter.getStatus();
202
+ return {
203
+ ...baseStatus,
204
+ queueLength: this.requestQueue.length,
205
+ currentLimit: this.isAuthenticated
206
+ ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT
207
+ : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT,
208
+ rateLimitInfo: this.lastRateLimitInfo
209
+ };
210
+ }
211
+ /**
212
+ * Check if an error is a GitHub rate limit error
213
+ */
214
+ isGitHubRateLimitError(error) {
215
+ return error?.status === 429 ||
216
+ error?.response?.status === 429 ||
217
+ (typeof error?.message === 'string' && error.message.toLowerCase().includes('rate limit'));
218
+ }
219
+ /**
220
+ * Handle GitHub rate limit error response
221
+ */
222
+ handleGitHubRateLimit(error) {
223
+ let resetTime;
224
+ let remainingRequests = 0;
225
+ // Parse rate limit headers if available
226
+ if (error?.response?.headers) {
227
+ const headers = error.response.headers;
228
+ const resetTimestamp = parseInt(headers['x-ratelimit-reset'] || '0');
229
+ const remaining = parseInt(headers['x-ratelimit-remaining'] || '0');
230
+ const limit = parseInt(headers['x-ratelimit-limit'] || '0');
231
+ if (resetTimestamp > 0) {
232
+ resetTime = new Date(resetTimestamp * 1000);
233
+ }
234
+ this.lastRateLimitInfo = {
235
+ limit,
236
+ remaining,
237
+ reset: resetTime || new Date(Date.now() + 60 * 60 * 1000), // Default to 1 hour
238
+ used: limit - remaining
239
+ };
240
+ remainingRequests = remaining;
241
+ }
242
+ logger.warn('GitHub API rate limit hit from server', {
243
+ remaining: remainingRequests,
244
+ resetTime,
245
+ queueLength: this.requestQueue.length,
246
+ errorMessage: error?.message
247
+ });
248
+ // Log as a security event for monitoring
249
+ SecurityMonitor.logSecurityEvent({
250
+ type: 'RATE_LIMIT_EXCEEDED',
251
+ severity: 'MEDIUM',
252
+ source: 'GitHubRateLimiter.handleGitHubRateLimit',
253
+ details: `GitHub API rate limit exceeded. Remaining: ${remainingRequests}, Queue: ${this.requestQueue.length}`,
254
+ metadata: {
255
+ rateLimitInfo: this.lastRateLimitInfo,
256
+ authenticated: this.isAuthenticated
257
+ }
258
+ });
259
+ }
260
+ /**
261
+ * Log API usage for monitoring and diagnostics
262
+ */
263
+ logApiUsage(operation, result, error) {
264
+ const status = this.getStatus();
265
+ logger.debug('GitHub API usage logged', {
266
+ operation,
267
+ result,
268
+ remainingTokens: status.remainingTokens,
269
+ queueLength: status.queueLength,
270
+ authenticated: this.isAuthenticated,
271
+ error: error?.message
272
+ });
273
+ // Log warning if getting close to rate limits
274
+ if (status.remainingTokens < 100 && this.isAuthenticated) {
275
+ logger.warn('Approaching GitHub API rate limit', {
276
+ operation,
277
+ remainingTokens: status.remainingTokens,
278
+ currentLimit: status.currentLimit,
279
+ recommendation: 'Consider reducing API usage frequency'
280
+ });
281
+ }
282
+ else if (status.remainingTokens < 10 && !this.isAuthenticated) {
283
+ logger.warn('Approaching GitHub API rate limit (unauthenticated)', {
284
+ operation,
285
+ remainingTokens: status.remainingTokens,
286
+ currentLimit: status.currentLimit,
287
+ recommendation: 'Consider authenticating for higher rate limits'
288
+ });
289
+ }
290
+ }
291
+ /**
292
+ * Clear the request queue (for testing or emergency situations)
293
+ */
294
+ clearQueue() {
295
+ const clearedCount = this.requestQueue.length;
296
+ // Reject all pending requests
297
+ this.requestQueue.forEach(request => {
298
+ request.reject(new Error('Request queue cleared'));
299
+ });
300
+ this.requestQueue = [];
301
+ logger.info('GitHub API request queue cleared', { clearedCount });
302
+ }
303
+ /**
304
+ * Reset the rate limiter (for testing)
305
+ */
306
+ reset() {
307
+ this.rateLimiter.reset();
308
+ this.clearQueue();
309
+ this.processing = false;
310
+ logger.info('GitHub rate limiter reset');
311
+ }
312
+ }
313
+ // Singleton instance for global use
314
+ export const githubRateLimiter = new GitHubRateLimiter();
315
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"GitHubRateLimiter.js","sourceRoot":"","sources":["../../src/utils/GitHubRateLimiter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * GitHubRateLimiter - Specialized rate limiter for GitHub API calls\n * \n * Features:\n * - Respects GitHub's authenticated (5000/hour) and unauthenticated (60/hour) limits\n * - Client-side queuing when approaching limits\n * - Request prioritization for critical operations\n * - Comprehensive logging for quota management\n * - Early termination when exact matches are found\n */\n\nimport { RateLimiter, RateLimiterConfig, RateLimitStatus } from './RateLimiter.js';\nimport { GITHUB_API_RATE_LIMITS } from '../config/portfolio-constants.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { logger } from './logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\nexport interface GitHubRateLimitInfo {\n  limit: number;\n  remaining: number;\n  reset: Date;\n  used: number;\n}\n\nexport interface GitHubApiRequest {\n  id: string;\n  operation: string;\n  priority: 'high' | 'normal' | 'low';\n  timestamp: number;\n  resolve: (value: any) => void;\n  reject: (error: any) => void;\n}\n\nexport interface GitHubRateStatus extends RateLimitStatus {\n  queueLength: number;\n  currentLimit: number;\n  rateLimitInfo?: GitHubRateLimitInfo;\n}\n\nexport class GitHubRateLimiter {\n  private rateLimiter!: RateLimiter;\n  private requestQueue: GitHubApiRequest[] = [];\n  private processing = false;\n  private lastRateLimitInfo?: GitHubRateLimitInfo;\n  private isAuthenticated = false;\n\n  constructor() {\n    // Initialize with conservative limits - will update based on auth status\n    this.updateLimitsForAuthStatus();\n    this.setupPeriodicStatusCheck();\n  }\n\n  /**\n   * Update rate limits based on current authentication status\n   */\n  private async updateLimitsForAuthStatus(): Promise<void> {\n    try {\n      const token = await TokenManager.getGitHubTokenAsync();\n      const newIsAuthenticated = !!token;\n      \n      // Only recreate rate limiter if auth status changed\n      if (newIsAuthenticated !== this.isAuthenticated) {\n        this.isAuthenticated = newIsAuthenticated;\n        \n        const limit = this.isAuthenticated \n          ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT \n          : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT;\n          \n        // Apply buffer to stay below actual limits\n        const bufferedLimit = Math.floor(limit * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE);\n        \n        const config: RateLimiterConfig = {\n          maxRequests: bufferedLimit,\n          windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,\n          minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS\n        };\n        \n        this.rateLimiter = new RateLimiter(config);\n        \n        logger.info('GitHub rate limiter updated', {\n          authenticated: this.isAuthenticated,\n          limit: bufferedLimit,\n          originalLimit: limit,\n          bufferPercentage: GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE\n        });\n      }\n    } catch (error) {\n      logger.warn('Failed to check authentication status for rate limiting', { error });\n      // Fall back to unauthenticated limits\n      this.isAuthenticated = false;\n      this.rateLimiter = new RateLimiter({\n        maxRequests: Math.floor(GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE),\n        windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,\n        minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS\n      });\n    }\n  }\n\n  /**\n   * Setup periodic check for rate limit status\n   */\n  private setupPeriodicStatusCheck(): void {\n    // Check auth status every 5 minutes\n    setInterval(() => {\n      this.updateLimitsForAuthStatus().catch(error => {\n        logger.warn('Periodic auth status check failed', { error });\n      });\n    }, 5 * 60 * 1000);\n  }\n\n  /**\n   * Queue a GitHub API request with rate limiting\n   * @param operation Description of the operation\n   * @param apiCall Function that makes the actual API call\n   * @param priority Request priority (high, normal, low)\n   * @returns Promise that resolves with the API response\n   */\n  async queueRequest<T>(\n    operation: string,\n    apiCall: () => Promise<T>,\n    priority: 'high' | 'normal' | 'low' = 'normal'\n  ): Promise<T> {\n    // SECURITY FIX (DMCP-SEC-004): Normalize Unicode in operation name to prevent injection attacks\n    const normalizedOperation = UnicodeValidator.normalize(operation);\n    if (!normalizedOperation.isValid) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'MEDIUM',\n        source: 'GitHubRateLimiter.queueRequest',\n        details: `Invalid Unicode in operation name: ${normalizedOperation.detectedIssues?.[0] || 'unknown error'}`\n      });\n      // Use a safe fallback for the operation name\n      operation = 'github-api-request';\n    } else {\n      operation = normalizedOperation.normalizedContent;\n    }\n    \n    const requestId = `${operation}-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n    \n    return new Promise<T>((resolve, reject) => {\n      const request: GitHubApiRequest = {\n        id: requestId,\n        operation,\n        priority,\n        timestamp: Date.now(),\n        resolve: async (value) => {\n          try {\n            logger.debug('Executing GitHub API request', {\n              operation,\n              requestId,\n              queueWaitTime: Date.now() - request.timestamp\n            });\n            \n            const result = await apiCall();\n            resolve(result);\n            \n            // Log successful API usage for quota tracking\n            this.logApiUsage(operation, 'success');\n            \n          } catch (error) {\n            // Check if this is a rate limit error from GitHub\n            if (this.isGitHubRateLimitError(error)) {\n              this.handleGitHubRateLimit(error);\n            }\n            reject(error);\n            this.logApiUsage(operation, 'error', error);\n          }\n        },\n        reject\n      };\n\n      // Add to queue with priority ordering\n      this.addToQueue(request);\n      this.processQueue();\n    });\n  }\n\n  /**\n   * Add request to queue with priority ordering\n   */\n  private addToQueue(request: GitHubApiRequest): void {\n    // Insert based on priority: high > normal > low\n    // Within same priority, maintain FIFO order\n    const priorityOrder = { high: 0, normal: 1, low: 2 };\n    \n    let insertIndex = this.requestQueue.length;\n    for (let i = 0; i < this.requestQueue.length; i++) {\n      if (priorityOrder[request.priority] < priorityOrder[this.requestQueue[i].priority]) {\n        insertIndex = i;\n        break;\n      }\n    }\n    \n    this.requestQueue.splice(insertIndex, 0, request);\n    \n    logger.debug('GitHub API request queued', {\n      operation: request.operation,\n      priority: request.priority,\n      queuePosition: insertIndex,\n      totalQueued: this.requestQueue.length\n    });\n  }\n\n  /**\n   * Process the request queue\n   */\n  private async processQueue(): Promise<void> {\n    if (this.processing || this.requestQueue.length === 0) {\n      return;\n    }\n\n    this.processing = true;\n\n    try {\n      while (this.requestQueue.length > 0) {\n        // Update auth status periodically\n        if (Math.random() < 0.1) { // 10% chance\n          await this.updateLimitsForAuthStatus();\n        }\n\n        const rateLimitStatus = this.rateLimiter.checkLimit();\n        \n        if (!rateLimitStatus.allowed) {\n          // Log rate limit wait\n          logger.info('GitHub API rate limit reached, waiting', {\n            retryAfterMs: rateLimitStatus.retryAfterMs,\n            remainingTokens: rateLimitStatus.remainingTokens,\n            queueLength: this.requestQueue.length,\n            resetTime: rateLimitStatus.resetTime\n          });\n\n          // Wait for the specified time\n          await new Promise(resolve => setTimeout(resolve, rateLimitStatus.retryAfterMs || 1000));\n          continue;\n        }\n\n        // Process the next request\n        const request = this.requestQueue.shift()!;\n        this.rateLimiter.consumeToken();\n\n        // Execute the request\n        request.resolve(null); // This will trigger the actual API call\n      }\n    } finally {\n      this.processing = false;\n    }\n  }\n\n  /**\n   * Get current rate limit status\n   */\n  getStatus(): GitHubRateStatus {\n    const baseStatus = this.rateLimiter.getStatus();\n    \n    return {\n      ...baseStatus,\n      queueLength: this.requestQueue.length,\n      currentLimit: this.isAuthenticated \n        ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT \n        : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT,\n      rateLimitInfo: this.lastRateLimitInfo\n    };\n  }\n\n  /**\n   * Check if an error is a GitHub rate limit error\n   */\n  private isGitHubRateLimitError(error: any): boolean {\n    return error?.status === 429 || \n           error?.response?.status === 429 ||\n           (typeof error?.message === 'string' && error.message.toLowerCase().includes('rate limit'));\n  }\n\n  /**\n   * Handle GitHub rate limit error response\n   */\n  private handleGitHubRateLimit(error: any): void {\n    let resetTime: Date | undefined;\n    let remainingRequests = 0;\n    \n    // Parse rate limit headers if available\n    if (error?.response?.headers) {\n      const headers = error.response.headers;\n      const resetTimestamp = parseInt(headers['x-ratelimit-reset'] || '0');\n      const remaining = parseInt(headers['x-ratelimit-remaining'] || '0');\n      const limit = parseInt(headers['x-ratelimit-limit'] || '0');\n      \n      if (resetTimestamp > 0) {\n        resetTime = new Date(resetTimestamp * 1000);\n      }\n      \n      this.lastRateLimitInfo = {\n        limit,\n        remaining,\n        reset: resetTime || new Date(Date.now() + 60 * 60 * 1000), // Default to 1 hour\n        used: limit - remaining\n      };\n      \n      remainingRequests = remaining;\n    }\n\n    logger.warn('GitHub API rate limit hit from server', {\n      remaining: remainingRequests,\n      resetTime,\n      queueLength: this.requestQueue.length,\n      errorMessage: error?.message\n    });\n\n    // Log as a security event for monitoring\n    SecurityMonitor.logSecurityEvent({\n      type: 'RATE_LIMIT_EXCEEDED',\n      severity: 'MEDIUM',\n      source: 'GitHubRateLimiter.handleGitHubRateLimit',\n      details: `GitHub API rate limit exceeded. Remaining: ${remainingRequests}, Queue: ${this.requestQueue.length}`,\n      metadata: {\n        rateLimitInfo: this.lastRateLimitInfo,\n        authenticated: this.isAuthenticated\n      }\n    });\n  }\n\n  /**\n   * Log API usage for monitoring and diagnostics\n   */\n  private logApiUsage(operation: string, result: 'success' | 'error', error?: any): void {\n    const status = this.getStatus();\n    \n    logger.debug('GitHub API usage logged', {\n      operation,\n      result,\n      remainingTokens: status.remainingTokens,\n      queueLength: status.queueLength,\n      authenticated: this.isAuthenticated,\n      error: error?.message\n    });\n\n    // Log warning if getting close to rate limits\n    if (status.remainingTokens < 100 && this.isAuthenticated) {\n      logger.warn('Approaching GitHub API rate limit', {\n        operation,\n        remainingTokens: status.remainingTokens,\n        currentLimit: status.currentLimit,\n        recommendation: 'Consider reducing API usage frequency'\n      });\n    } else if (status.remainingTokens < 10 && !this.isAuthenticated) {\n      logger.warn('Approaching GitHub API rate limit (unauthenticated)', {\n        operation,\n        remainingTokens: status.remainingTokens,\n        currentLimit: status.currentLimit,\n        recommendation: 'Consider authenticating for higher rate limits'\n      });\n    }\n  }\n\n  /**\n   * Clear the request queue (for testing or emergency situations)\n   */\n  clearQueue(): void {\n    const clearedCount = this.requestQueue.length;\n    \n    // Reject all pending requests\n    this.requestQueue.forEach(request => {\n      request.reject(new Error('Request queue cleared'));\n    });\n    \n    this.requestQueue = [];\n    \n    logger.info('GitHub API request queue cleared', { clearedCount });\n  }\n\n  /**\n   * Reset the rate limiter (for testing)\n   */\n  reset(): void {\n    this.rateLimiter.reset();\n    this.clearQueue();\n    this.processing = false;\n    logger.info('GitHub rate limiter reset');\n  }\n}\n\n// Singleton instance for global use\nexport const githubRateLimiter = new GitHubRateLimiter();"]}
@@ -0,0 +1,134 @@
1
+ /**
2
+ * Comprehensive Performance Monitoring System
3
+ * Tracks search times, memory usage, cache performance, and system metrics
4
+ */
5
+ export interface PerformanceMetrics {
6
+ searchTimes: number[];
7
+ memoryUsage: MemoryUsage[];
8
+ cacheStats: CachePerformance;
9
+ systemStats: SystemStats;
10
+ timestamp: Date;
11
+ }
12
+ export interface MemoryUsage {
13
+ heapUsed: number;
14
+ heapTotal: number;
15
+ rss: number;
16
+ external: number;
17
+ timestamp: Date;
18
+ }
19
+ export interface CachePerformance {
20
+ hitRate: number;
21
+ avgHitTime: number;
22
+ avgMissTime: number;
23
+ totalHits: number;
24
+ totalMisses: number;
25
+ evictions: number;
26
+ }
27
+ export interface SystemStats {
28
+ cpuUsage: number;
29
+ loadAverage: number[];
30
+ freeMemory: number;
31
+ totalMemory: number;
32
+ uptime: number;
33
+ }
34
+ export interface SearchMetrics {
35
+ query: string;
36
+ duration: number;
37
+ resultCount: number;
38
+ sources: string[];
39
+ cacheHit: boolean;
40
+ memoryBefore: number;
41
+ memoryAfter: number;
42
+ timestamp: Date;
43
+ }
44
+ export interface SlowQuery {
45
+ query: string;
46
+ duration: number;
47
+ threshold: number;
48
+ sources: string[];
49
+ resultCount: number;
50
+ memoryUsage: number;
51
+ timestamp: Date;
52
+ }
53
+ export declare class PerformanceMonitor {
54
+ private static instance;
55
+ private searchMetrics;
56
+ private slowQueries;
57
+ private memorySnapshots;
58
+ private cacheMetrics;
59
+ private readonly maxMetricsHistory;
60
+ private readonly slowQueryThreshold;
61
+ private readonly memorySnapshotInterval;
62
+ private readonly maxSlowQueries;
63
+ private memoryMonitorInterval?;
64
+ private isMonitoring;
65
+ private constructor();
66
+ static getInstance(): PerformanceMonitor;
67
+ /**
68
+ * Start performance monitoring
69
+ */
70
+ startMonitoring(): void;
71
+ /**
72
+ * Stop performance monitoring
73
+ */
74
+ stopMonitoring(): void;
75
+ /**
76
+ * Record search performance metrics
77
+ */
78
+ recordSearch(metrics: SearchMetrics): void;
79
+ /**
80
+ * Record cache performance metrics
81
+ */
82
+ recordCachePerformance(cacheName: string, stats: CachePerformance): void;
83
+ /**
84
+ * Get comprehensive performance metrics
85
+ */
86
+ getMetrics(): PerformanceMetrics;
87
+ /**
88
+ * Get search performance statistics
89
+ */
90
+ getSearchStats(): {
91
+ totalSearches: number;
92
+ averageTime: number;
93
+ medianTime: number;
94
+ p95Time: number;
95
+ p99Time: number;
96
+ slowQueries: number;
97
+ cacheHitRate: number;
98
+ };
99
+ /**
100
+ * Get memory usage statistics
101
+ */
102
+ getMemoryStats(): {
103
+ currentUsage: MemoryUsage;
104
+ peakUsage: MemoryUsage;
105
+ averageUsage: MemoryUsage;
106
+ growthRate: number;
107
+ };
108
+ /**
109
+ * Get slow queries with analysis
110
+ */
111
+ getSlowQueries(limit?: number): SlowQuery[];
112
+ /**
113
+ * Analyze performance trends
114
+ */
115
+ analyzeTrends(): {
116
+ performanceTrend: 'improving' | 'degrading' | 'stable';
117
+ memoryTrend: 'growing' | 'shrinking' | 'stable';
118
+ recommendations: string[];
119
+ };
120
+ /**
121
+ * Reset all performance metrics
122
+ */
123
+ reset(): void;
124
+ /**
125
+ * Export metrics for external analysis
126
+ */
127
+ exportMetrics(): string;
128
+ private recordSlowQuery;
129
+ private startMemoryMonitoring;
130
+ private takeMemorySnapshot;
131
+ private aggregateCacheStats;
132
+ private getSystemStats;
133
+ }
134
+ //# sourceMappingURL=PerformanceMonitor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PerformanceMonitor.d.ts","sourceRoot":"","sources":["../../src/utils/PerformanceMonitor.ts"],"names":[],"mappings":"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"}