@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,478 @@
1
+ /**
2
+ * Portfolio Index Manager - Maps element names to file paths
3
+ *
4
+ * Solves critical issues:
5
+ * 1. submit_content can't find elements by metadata name (e.g., "Safe Roundtrip Tester" -> "safe-roundtrip-tester.md")
6
+ * 2. search_collection doesn't search local portfolio content
7
+ *
8
+ * Features:
9
+ * - In-memory index mapping metadata.name → file path
10
+ * - Keywords/tags → file paths mapping
11
+ * - Element type → file paths mapping
12
+ * - Fast O(1) lookups with Maps
13
+ * - Lazy loading with 5-minute TTL cache
14
+ * - Unicode normalization for security
15
+ * - Error handling and logging
16
+ */
17
+ import * as fs from 'fs/promises';
18
+ import * as path from 'path';
19
+ import { logger } from '../utils/logger.js';
20
+ import { ElementType } from './types.js';
21
+ import { PortfolioManager } from './PortfolioManager.js';
22
+ import { SecureYamlParser } from '../security/secureYamlParser.js';
23
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
24
+ import { SecurityMonitor } from '../security/securityMonitor.js';
25
+ import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
26
+ export class PortfolioIndexManager {
27
+ static instance = null;
28
+ static instanceLock = false;
29
+ index = null;
30
+ lastBuilt = null;
31
+ TTL_MS = 5 * 60 * 1000; // 5 minutes
32
+ isBuilding = false;
33
+ buildPromise = null;
34
+ constructor() {
35
+ logger.debug('PortfolioIndexManager created');
36
+ }
37
+ static getInstance() {
38
+ if (!this.instance) {
39
+ if (this.instanceLock) {
40
+ throw new Error('PortfolioIndexManager instance is being created by another thread');
41
+ }
42
+ try {
43
+ this.instanceLock = true;
44
+ this.instance = new PortfolioIndexManager();
45
+ }
46
+ finally {
47
+ this.instanceLock = false;
48
+ }
49
+ }
50
+ return this.instance;
51
+ }
52
+ /**
53
+ * Get the current index, building it if necessary
54
+ */
55
+ async getIndex() {
56
+ // Check if we need to rebuild
57
+ if (this.needsRebuild()) {
58
+ await this.buildIndex();
59
+ }
60
+ return this.index;
61
+ }
62
+ /**
63
+ * Search the portfolio index by name with fuzzy matching
64
+ */
65
+ async findByName(name, options = {}) {
66
+ const index = await this.getIndex();
67
+ // Normalize input for security
68
+ const normalizedName = UnicodeValidator.normalize(name);
69
+ if (!normalizedName.isValid) {
70
+ logger.warn('Invalid Unicode in search name', {
71
+ issues: normalizedName.detectedIssues
72
+ });
73
+ return null;
74
+ }
75
+ const safeName = normalizedName.normalizedContent;
76
+ // Try exact match first (case insensitive)
77
+ const exactMatch = index.byName.get(safeName.toLowerCase());
78
+ if (exactMatch) {
79
+ logger.debug('Found exact name match', { name: safeName, filePath: exactMatch.filePath });
80
+ return exactMatch;
81
+ }
82
+ // Try filename match
83
+ const filenameMatch = index.byFilename.get(safeName.toLowerCase());
84
+ if (filenameMatch) {
85
+ logger.debug('Found filename match', { name: safeName, filePath: filenameMatch.filePath });
86
+ return filenameMatch;
87
+ }
88
+ // Try fuzzy matching if enabled
89
+ if (options.fuzzyMatch !== false) {
90
+ const fuzzyMatch = this.findFuzzyMatch(safeName, index, options);
91
+ if (fuzzyMatch) {
92
+ logger.debug('Found fuzzy match', {
93
+ name: safeName,
94
+ matchName: fuzzyMatch.metadata.name,
95
+ filePath: fuzzyMatch.filePath
96
+ });
97
+ return fuzzyMatch;
98
+ }
99
+ }
100
+ logger.debug('No match found for name', { name: safeName });
101
+ return null;
102
+ }
103
+ /**
104
+ * Search the portfolio with comprehensive text search
105
+ */
106
+ async search(query, options = {}) {
107
+ const index = await this.getIndex();
108
+ // Normalize query for security
109
+ const normalizedQuery = UnicodeValidator.normalize(query);
110
+ if (!normalizedQuery.isValid) {
111
+ logger.warn('Invalid Unicode in search query', {
112
+ issues: normalizedQuery.detectedIssues
113
+ });
114
+ return [];
115
+ }
116
+ const safeQuery = normalizedQuery.normalizedContent.toLowerCase().trim();
117
+ const queryTokens = safeQuery.split(/\s+/).filter(token => token.length > 0);
118
+ if (queryTokens.length === 0) {
119
+ return [];
120
+ }
121
+ const results = [];
122
+ const seenPaths = new Set();
123
+ const maxResults = options.maxResults || 20;
124
+ // Helper to add unique results
125
+ const addResult = (entry, matchType, score = 1) => {
126
+ if (!seenPaths.has(entry.filePath) && results.length < maxResults) {
127
+ // Filter by element type if specified
128
+ if (options.elementType && entry.elementType !== options.elementType) {
129
+ return;
130
+ }
131
+ seenPaths.add(entry.filePath);
132
+ results.push({ entry, matchType, score });
133
+ }
134
+ };
135
+ // 1. Search by name (highest priority)
136
+ for (const [name, entry] of index.byName) {
137
+ if (this.matchesQuery(name, queryTokens)) {
138
+ addResult(entry, 'name', 3);
139
+ }
140
+ }
141
+ // 2. Search by filename
142
+ for (const [filename, entry] of index.byFilename) {
143
+ if (this.matchesQuery(filename, queryTokens)) {
144
+ addResult(entry, 'filename', 2.5);
145
+ }
146
+ }
147
+ // 3. Search by keywords
148
+ if (options.includeKeywords !== false) {
149
+ for (const [keyword, entries] of index.byKeyword) {
150
+ if (this.matchesQuery(keyword, queryTokens)) {
151
+ for (const entry of entries) {
152
+ addResult(entry, 'keyword', 2);
153
+ }
154
+ }
155
+ }
156
+ }
157
+ // 4. Search by tags
158
+ if (options.includeTags !== false) {
159
+ for (const [tag, entries] of index.byTag) {
160
+ if (this.matchesQuery(tag, queryTokens)) {
161
+ for (const entry of entries) {
162
+ addResult(entry, 'tag', 2);
163
+ }
164
+ }
165
+ }
166
+ }
167
+ // 5. Search by triggers
168
+ if (options.includeTriggers !== false) {
169
+ for (const [trigger, entries] of index.byTrigger) {
170
+ if (this.matchesQuery(trigger, queryTokens)) {
171
+ for (const entry of entries) {
172
+ addResult(entry, 'trigger', 1.8);
173
+ }
174
+ }
175
+ }
176
+ }
177
+ // 6. Search by description
178
+ if (options.includeDescriptions !== false) {
179
+ for (const [_, entry] of index.byName) {
180
+ if (entry.metadata.description &&
181
+ this.matchesQuery(entry.metadata.description.toLowerCase(), queryTokens)) {
182
+ addResult(entry, 'description', 1.5);
183
+ }
184
+ }
185
+ }
186
+ // Sort by score (descending)
187
+ results.sort((a, b) => b.score - a.score);
188
+ logger.debug('Portfolio search completed', {
189
+ query: safeQuery,
190
+ resultCount: results.length,
191
+ totalIndexed: index.byName.size
192
+ });
193
+ return results;
194
+ }
195
+ /**
196
+ * Get all elements of a specific type
197
+ */
198
+ async getElementsByType(elementType) {
199
+ const index = await this.getIndex();
200
+ return index.byType.get(elementType) || [];
201
+ }
202
+ /**
203
+ * Get statistics about the index
204
+ */
205
+ async getStats() {
206
+ const index = await this.getIndex();
207
+ const stats = {
208
+ totalElements: index.byName.size,
209
+ elementsByType: {},
210
+ lastBuilt: this.lastBuilt,
211
+ isStale: this.needsRebuild()
212
+ };
213
+ for (const elementType of Object.values(ElementType)) {
214
+ stats.elementsByType[elementType] = (index.byType.get(elementType) || []).length;
215
+ }
216
+ return stats;
217
+ }
218
+ /**
219
+ * Force rebuild the index
220
+ */
221
+ async rebuildIndex() {
222
+ this.index = null;
223
+ this.lastBuilt = null;
224
+ await this.buildIndex();
225
+ }
226
+ /**
227
+ * Check if the index needs rebuilding
228
+ */
229
+ needsRebuild() {
230
+ if (!this.index || !this.lastBuilt) {
231
+ return true;
232
+ }
233
+ const age = Date.now() - this.lastBuilt.getTime();
234
+ return age > this.TTL_MS;
235
+ }
236
+ /**
237
+ * Build the index by scanning all portfolio directories
238
+ */
239
+ async buildIndex() {
240
+ // Prevent concurrent builds
241
+ if (this.isBuilding) {
242
+ if (this.buildPromise) {
243
+ await this.buildPromise;
244
+ }
245
+ return;
246
+ }
247
+ this.isBuilding = true;
248
+ this.buildPromise = this.performBuild();
249
+ try {
250
+ await this.buildPromise;
251
+ }
252
+ finally {
253
+ this.isBuilding = false;
254
+ this.buildPromise = null;
255
+ }
256
+ }
257
+ /**
258
+ * Perform the actual index building
259
+ */
260
+ async performBuild() {
261
+ const startTime = Date.now();
262
+ logger.info('Building portfolio index...');
263
+ try {
264
+ const portfolioManager = PortfolioManager.getInstance();
265
+ // Initialize empty index
266
+ const newIndex = {
267
+ byName: new Map(),
268
+ byFilename: new Map(),
269
+ byType: new Map(),
270
+ byKeyword: new Map(),
271
+ byTag: new Map(),
272
+ byTrigger: new Map()
273
+ };
274
+ // Initialize type maps
275
+ for (const elementType of Object.values(ElementType)) {
276
+ newIndex.byType.set(elementType, []);
277
+ }
278
+ let totalFiles = 0;
279
+ let processedFiles = 0;
280
+ // Scan each element type
281
+ for (const elementType of Object.values(ElementType)) {
282
+ try {
283
+ const elementDir = portfolioManager.getElementDir(elementType);
284
+ // Check if directory exists
285
+ try {
286
+ await fs.access(elementDir);
287
+ }
288
+ catch {
289
+ logger.debug(`Element directory doesn't exist: ${elementDir}`);
290
+ continue;
291
+ }
292
+ const files = await fs.readdir(elementDir);
293
+ const mdFiles = files.filter(file => file.endsWith('.md'));
294
+ totalFiles += mdFiles.length;
295
+ for (const file of mdFiles) {
296
+ try {
297
+ const filePath = path.join(elementDir, file);
298
+ const entry = await this.createIndexEntry(filePath, elementType);
299
+ if (entry) {
300
+ this.addToIndex(newIndex, entry);
301
+ processedFiles++;
302
+ }
303
+ }
304
+ catch (error) {
305
+ logger.warn(`Failed to index file: ${file}`, {
306
+ elementType,
307
+ error: error instanceof Error ? error.message : String(error)
308
+ });
309
+ }
310
+ }
311
+ }
312
+ catch (error) {
313
+ logger.error(`Failed to scan element type: ${elementType}`, {
314
+ error: error instanceof Error ? error.message : String(error)
315
+ });
316
+ }
317
+ }
318
+ // Update instance state
319
+ this.index = newIndex;
320
+ this.lastBuilt = new Date();
321
+ const duration = Date.now() - startTime;
322
+ logger.info('Portfolio index built successfully', {
323
+ totalFiles,
324
+ processedFiles,
325
+ duration: `${duration}ms`,
326
+ uniqueNames: newIndex.byName.size,
327
+ uniqueKeywords: newIndex.byKeyword.size,
328
+ uniqueTags: newIndex.byTag.size
329
+ });
330
+ // Log security event for audit trail
331
+ SecurityMonitor.logSecurityEvent({
332
+ type: 'PORTFOLIO_INITIALIZATION',
333
+ severity: 'LOW',
334
+ source: 'PortfolioIndexManager.performBuild',
335
+ details: `Portfolio index rebuilt with ${processedFiles} elements in ${duration}ms`
336
+ });
337
+ }
338
+ catch (error) {
339
+ ErrorHandler.logError('PortfolioIndexManager.performBuild', error);
340
+ throw ErrorHandler.wrapError(error, 'Failed to build portfolio index', ErrorCategory.SYSTEM_ERROR);
341
+ }
342
+ }
343
+ /**
344
+ * Create an index entry from a file
345
+ */
346
+ async createIndexEntry(filePath, elementType) {
347
+ try {
348
+ // Get file stats
349
+ const stats = await fs.stat(filePath);
350
+ // Read file content
351
+ const content = await fs.readFile(filePath, 'utf-8');
352
+ // Parse frontmatter securely
353
+ const parsed = SecureYamlParser.parse(content);
354
+ // Extract base filename
355
+ const filename = path.basename(filePath, '.md');
356
+ // Build metadata with defaults
357
+ const metadata = {
358
+ name: parsed.data.name || filename,
359
+ description: parsed.data.description,
360
+ version: parsed.data.version,
361
+ author: parsed.data.author,
362
+ tags: Array.isArray(parsed.data.tags) ? parsed.data.tags : [],
363
+ keywords: Array.isArray(parsed.data.keywords) ? parsed.data.keywords : [],
364
+ triggers: Array.isArray(parsed.data.triggers) ? parsed.data.triggers : [],
365
+ category: parsed.data.category,
366
+ created: parsed.data.created || parsed.data.created_date,
367
+ updated: parsed.data.updated || parsed.data.updated_date
368
+ };
369
+ const entry = {
370
+ filePath,
371
+ elementType,
372
+ metadata,
373
+ lastModified: stats.mtime,
374
+ filename
375
+ };
376
+ return entry;
377
+ }
378
+ catch (error) {
379
+ logger.debug(`Failed to create index entry for: ${filePath}`, {
380
+ error: error instanceof Error ? error.message : String(error)
381
+ });
382
+ return null;
383
+ }
384
+ }
385
+ /**
386
+ * Add entry to all relevant index maps
387
+ */
388
+ addToIndex(index, entry) {
389
+ // Normalize keys for case-insensitive lookup
390
+ const normalizedName = entry.metadata.name.toLowerCase();
391
+ const normalizedFilename = entry.filename.toLowerCase();
392
+ // Add to name map
393
+ index.byName.set(normalizedName, entry);
394
+ // Add to filename map
395
+ index.byFilename.set(normalizedFilename, entry);
396
+ // Add to type map
397
+ const typeEntries = index.byType.get(entry.elementType) || [];
398
+ typeEntries.push(entry);
399
+ index.byType.set(entry.elementType, typeEntries);
400
+ // Add keywords
401
+ for (const keyword of entry.metadata.keywords || []) {
402
+ const normalizedKeyword = keyword.toLowerCase();
403
+ const keywordEntries = index.byKeyword.get(normalizedKeyword) || [];
404
+ keywordEntries.push(entry);
405
+ index.byKeyword.set(normalizedKeyword, keywordEntries);
406
+ }
407
+ // Add tags
408
+ for (const tag of entry.metadata.tags || []) {
409
+ const normalizedTag = tag.toLowerCase();
410
+ const tagEntries = index.byTag.get(normalizedTag) || [];
411
+ tagEntries.push(entry);
412
+ index.byTag.set(normalizedTag, tagEntries);
413
+ }
414
+ // Add triggers
415
+ for (const trigger of entry.metadata.triggers || []) {
416
+ const normalizedTrigger = trigger.toLowerCase();
417
+ const triggerEntries = index.byTrigger.get(normalizedTrigger) || [];
418
+ triggerEntries.push(entry);
419
+ index.byTrigger.set(normalizedTrigger, triggerEntries);
420
+ }
421
+ }
422
+ /**
423
+ * Find fuzzy matches for a name
424
+ */
425
+ findFuzzyMatch(searchName, index, options) {
426
+ const search = searchName.toLowerCase();
427
+ let bestMatch = null;
428
+ let bestScore = 0;
429
+ // Search names with partial matching
430
+ for (const [name, entry] of index.byName) {
431
+ if (options.elementType && entry.elementType !== options.elementType) {
432
+ continue;
433
+ }
434
+ const score = this.calculateSimilarity(search, name);
435
+ if (score > bestScore && score > 0.3) { // Minimum similarity threshold
436
+ bestScore = score;
437
+ bestMatch = entry;
438
+ }
439
+ }
440
+ // Also check filenames
441
+ for (const [filename, entry] of index.byFilename) {
442
+ if (options.elementType && entry.elementType !== options.elementType) {
443
+ continue;
444
+ }
445
+ const score = this.calculateSimilarity(search, filename);
446
+ if (score > bestScore && score > 0.3) {
447
+ bestScore = score;
448
+ bestMatch = entry;
449
+ }
450
+ }
451
+ return bestMatch;
452
+ }
453
+ /**
454
+ * Calculate similarity between two strings
455
+ */
456
+ calculateSimilarity(a, b) {
457
+ // Simple similarity based on substring containment and length
458
+ if (a === b)
459
+ return 1.0;
460
+ if (a.includes(b) || b.includes(a))
461
+ return 0.8;
462
+ // Check for word overlap
463
+ const wordsA = a.split(/\s+/);
464
+ const wordsB = b.split(/\s+/);
465
+ const commonWords = wordsA.filter(word => wordsB.includes(word));
466
+ if (commonWords.length > 0) {
467
+ return commonWords.length / Math.max(wordsA.length, wordsB.length);
468
+ }
469
+ return 0;
470
+ }
471
+ /**
472
+ * Check if any query tokens match the text
473
+ */
474
+ matchesQuery(text, queryTokens) {
475
+ return queryTokens.some(token => text.includes(token));
476
+ }
477
+ }
478
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioIndexManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioIndexManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Portfolio Index Manager - Maps element names to file paths\n * \n * Solves critical issues:\n * 1. submit_content can't find elements by metadata name (e.g., \"Safe Roundtrip Tester\" -> \"safe-roundtrip-tester.md\")\n * 2. search_collection doesn't search local portfolio content\n * \n * Features:\n * - In-memory index mapping metadata.name → file path\n * - Keywords/tags → file paths mapping\n * - Element type → file paths mapping\n * - Fast O(1) lookups with Maps\n * - Lazy loading with 5-minute TTL cache\n * - Unicode normalization for security\n * - Error handling and logging\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from '../utils/logger.js';\nimport { ElementType } from './types.js';\nimport { PortfolioManager } from './PortfolioManager.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\n\nexport interface IndexEntry {\n  filePath: string;\n  elementType: ElementType;\n  metadata: {\n    name: string;\n    description?: string;\n    version?: string;\n    author?: string;\n    tags?: string[];\n    keywords?: string[];\n    triggers?: string[];\n    category?: string;\n    created?: string;\n    updated?: string;\n  };\n  lastModified: Date;\n  filename: string; // Base filename without extension\n}\n\nexport interface PortfolioIndex {\n  byName: Map<string, IndexEntry>;\n  byFilename: Map<string, IndexEntry>;\n  byType: Map<ElementType, IndexEntry[]>;\n  byKeyword: Map<string, IndexEntry[]>;\n  byTag: Map<string, IndexEntry[]>;\n  byTrigger: Map<string, IndexEntry[]>;\n}\n\nexport interface SearchOptions {\n  elementType?: ElementType;\n  fuzzyMatch?: boolean;\n  maxResults?: number;\n  includeKeywords?: boolean;\n  includeTags?: boolean;\n  includeTriggers?: boolean;\n  includeDescriptions?: boolean;\n}\n\nexport interface SearchResult {\n  entry: IndexEntry;\n  matchType: 'name' | 'filename' | 'keyword' | 'tag' | 'trigger' | 'description';\n  score: number; // For future ranking\n}\n\nexport class PortfolioIndexManager {\n  private static instance: PortfolioIndexManager | null = null;\n  private static instanceLock = false;\n  \n  private index: PortfolioIndex | null = null;\n  private lastBuilt: Date | null = null;\n  private readonly TTL_MS = 5 * 60 * 1000; // 5 minutes\n  private isBuilding = false;\n  private buildPromise: Promise<void> | null = null;\n\n  private constructor() {\n    logger.debug('PortfolioIndexManager created');\n  }\n\n  public static getInstance(): PortfolioIndexManager {\n    if (!this.instance) {\n      if (this.instanceLock) {\n        throw new Error('PortfolioIndexManager instance is being created by another thread');\n      }\n      \n      try {\n        this.instanceLock = true;\n        this.instance = new PortfolioIndexManager();\n      } finally {\n        this.instanceLock = false;\n      }\n    }\n    return this.instance;\n  }\n\n  /**\n   * Get the current index, building it if necessary\n   */\n  public async getIndex(): Promise<PortfolioIndex> {\n    // Check if we need to rebuild\n    if (this.needsRebuild()) {\n      await this.buildIndex();\n    }\n    \n    return this.index!;\n  }\n\n  /**\n   * Search the portfolio index by name with fuzzy matching\n   */\n  public async findByName(name: string, options: SearchOptions = {}): Promise<IndexEntry | null> {\n    const index = await this.getIndex();\n    \n    // Normalize input for security\n    const normalizedName = UnicodeValidator.normalize(name);\n    if (!normalizedName.isValid) {\n      logger.warn('Invalid Unicode in search name', {\n        issues: normalizedName.detectedIssues\n      });\n      return null;\n    }\n    \n    const safeName = normalizedName.normalizedContent;\n    \n    // Try exact match first (case insensitive)\n    const exactMatch = index.byName.get(safeName.toLowerCase());\n    if (exactMatch) {\n      logger.debug('Found exact name match', { name: safeName, filePath: exactMatch.filePath });\n      return exactMatch;\n    }\n    \n    // Try filename match\n    const filenameMatch = index.byFilename.get(safeName.toLowerCase());\n    if (filenameMatch) {\n      logger.debug('Found filename match', { name: safeName, filePath: filenameMatch.filePath });\n      return filenameMatch;\n    }\n    \n    // Try fuzzy matching if enabled\n    if (options.fuzzyMatch !== false) {\n      const fuzzyMatch = this.findFuzzyMatch(safeName, index, options);\n      if (fuzzyMatch) {\n        logger.debug('Found fuzzy match', { \n          name: safeName, \n          matchName: fuzzyMatch.metadata.name,\n          filePath: fuzzyMatch.filePath \n        });\n        return fuzzyMatch;\n      }\n    }\n    \n    logger.debug('No match found for name', { name: safeName });\n    return null;\n  }\n\n  /**\n   * Search the portfolio with comprehensive text search\n   */\n  public async search(query: string, options: SearchOptions = {}): Promise<SearchResult[]> {\n    const index = await this.getIndex();\n    \n    // Normalize query for security\n    const normalizedQuery = UnicodeValidator.normalize(query);\n    if (!normalizedQuery.isValid) {\n      logger.warn('Invalid Unicode in search query', {\n        issues: normalizedQuery.detectedIssues\n      });\n      return [];\n    }\n    \n    const safeQuery = normalizedQuery.normalizedContent.toLowerCase().trim();\n    const queryTokens = safeQuery.split(/\\s+/).filter(token => token.length > 0);\n    \n    if (queryTokens.length === 0) {\n      return [];\n    }\n    \n    const results: SearchResult[] = [];\n    const seenPaths = new Set<string>();\n    const maxResults = options.maxResults || 20;\n    \n    // Helper to add unique results\n    const addResult = (entry: IndexEntry, matchType: SearchResult['matchType'], score: number = 1) => {\n      if (!seenPaths.has(entry.filePath) && results.length < maxResults) {\n        // Filter by element type if specified\n        if (options.elementType && entry.elementType !== options.elementType) {\n          return;\n        }\n        \n        seenPaths.add(entry.filePath);\n        results.push({ entry, matchType, score });\n      }\n    };\n    \n    // 1. Search by name (highest priority)\n    for (const [name, entry] of index.byName) {\n      if (this.matchesQuery(name, queryTokens)) {\n        addResult(entry, 'name', 3);\n      }\n    }\n    \n    // 2. Search by filename\n    for (const [filename, entry] of index.byFilename) {\n      if (this.matchesQuery(filename, queryTokens)) {\n        addResult(entry, 'filename', 2.5);\n      }\n    }\n    \n    // 3. Search by keywords\n    if (options.includeKeywords !== false) {\n      for (const [keyword, entries] of index.byKeyword) {\n        if (this.matchesQuery(keyword, queryTokens)) {\n          for (const entry of entries) {\n            addResult(entry, 'keyword', 2);\n          }\n        }\n      }\n    }\n    \n    // 4. Search by tags\n    if (options.includeTags !== false) {\n      for (const [tag, entries] of index.byTag) {\n        if (this.matchesQuery(tag, queryTokens)) {\n          for (const entry of entries) {\n            addResult(entry, 'tag', 2);\n          }\n        }\n      }\n    }\n    \n    // 5. Search by triggers\n    if (options.includeTriggers !== false) {\n      for (const [trigger, entries] of index.byTrigger) {\n        if (this.matchesQuery(trigger, queryTokens)) {\n          for (const entry of entries) {\n            addResult(entry, 'trigger', 1.8);\n          }\n        }\n      }\n    }\n    \n    // 6. Search by description\n    if (options.includeDescriptions !== false) {\n      for (const [_, entry] of index.byName) {\n        if (entry.metadata.description && \n            this.matchesQuery(entry.metadata.description.toLowerCase(), queryTokens)) {\n          addResult(entry, 'description', 1.5);\n        }\n      }\n    }\n    \n    // Sort by score (descending)\n    results.sort((a, b) => b.score - a.score);\n    \n    logger.debug('Portfolio search completed', {\n      query: safeQuery,\n      resultCount: results.length,\n      totalIndexed: index.byName.size\n    });\n    \n    return results;\n  }\n\n  /**\n   * Get all elements of a specific type\n   */\n  public async getElementsByType(elementType: ElementType): Promise<IndexEntry[]> {\n    const index = await this.getIndex();\n    return index.byType.get(elementType) || [];\n  }\n\n  /**\n   * Get statistics about the index\n   */\n  public async getStats(): Promise<{\n    totalElements: number;\n    elementsByType: Record<ElementType, number>;\n    lastBuilt: Date | null;\n    isStale: boolean;\n  }> {\n    const index = await this.getIndex();\n    const stats = {\n      totalElements: index.byName.size,\n      elementsByType: {} as Record<ElementType, number>,\n      lastBuilt: this.lastBuilt,\n      isStale: this.needsRebuild()\n    };\n    \n    for (const elementType of Object.values(ElementType)) {\n      stats.elementsByType[elementType] = (index.byType.get(elementType) || []).length;\n    }\n    \n    return stats;\n  }\n\n  /**\n   * Force rebuild the index\n   */\n  public async rebuildIndex(): Promise<void> {\n    this.index = null;\n    this.lastBuilt = null;\n    await this.buildIndex();\n  }\n\n  /**\n   * Check if the index needs rebuilding\n   */\n  private needsRebuild(): boolean {\n    if (!this.index || !this.lastBuilt) {\n      return true;\n    }\n    \n    const age = Date.now() - this.lastBuilt.getTime();\n    return age > this.TTL_MS;\n  }\n\n  /**\n   * Build the index by scanning all portfolio directories\n   */\n  private async buildIndex(): Promise<void> {\n    // Prevent concurrent builds\n    if (this.isBuilding) {\n      if (this.buildPromise) {\n        await this.buildPromise;\n      }\n      return;\n    }\n    \n    this.isBuilding = true;\n    \n    this.buildPromise = this.performBuild();\n    \n    try {\n      await this.buildPromise;\n    } finally {\n      this.isBuilding = false;\n      this.buildPromise = null;\n    }\n  }\n\n  /**\n   * Perform the actual index building\n   */\n  private async performBuild(): Promise<void> {\n    const startTime = Date.now();\n    logger.info('Building portfolio index...');\n    \n    try {\n      const portfolioManager = PortfolioManager.getInstance();\n      \n      // Initialize empty index\n      const newIndex: PortfolioIndex = {\n        byName: new Map(),\n        byFilename: new Map(),\n        byType: new Map(),\n        byKeyword: new Map(),\n        byTag: new Map(),\n        byTrigger: new Map()\n      };\n      \n      // Initialize type maps\n      for (const elementType of Object.values(ElementType)) {\n        newIndex.byType.set(elementType, []);\n      }\n      \n      let totalFiles = 0;\n      let processedFiles = 0;\n      \n      // Scan each element type\n      for (const elementType of Object.values(ElementType)) {\n        try {\n          const elementDir = portfolioManager.getElementDir(elementType);\n          \n          // Check if directory exists\n          try {\n            await fs.access(elementDir);\n          } catch {\n            logger.debug(`Element directory doesn't exist: ${elementDir}`);\n            continue;\n          }\n          \n          const files = await fs.readdir(elementDir);\n          const mdFiles = files.filter(file => file.endsWith('.md'));\n          totalFiles += mdFiles.length;\n          \n          for (const file of mdFiles) {\n            try {\n              const filePath = path.join(elementDir, file);\n              const entry = await this.createIndexEntry(filePath, elementType);\n              \n              if (entry) {\n                this.addToIndex(newIndex, entry);\n                processedFiles++;\n              }\n            } catch (error) {\n              logger.warn(`Failed to index file: ${file}`, {\n                elementType,\n                error: error instanceof Error ? error.message : String(error)\n              });\n            }\n          }\n        } catch (error) {\n          logger.error(`Failed to scan element type: ${elementType}`, {\n            error: error instanceof Error ? error.message : String(error)\n          });\n        }\n      }\n      \n      // Update instance state\n      this.index = newIndex;\n      this.lastBuilt = new Date();\n      \n      const duration = Date.now() - startTime;\n      logger.info('Portfolio index built successfully', {\n        totalFiles,\n        processedFiles,\n        duration: `${duration}ms`,\n        uniqueNames: newIndex.byName.size,\n        uniqueKeywords: newIndex.byKeyword.size,\n        uniqueTags: newIndex.byTag.size\n      });\n      \n      // Log security event for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_INITIALIZATION',\n        severity: 'LOW',\n        source: 'PortfolioIndexManager.performBuild',\n        details: `Portfolio index rebuilt with ${processedFiles} elements in ${duration}ms`\n      });\n      \n    } catch (error) {\n      ErrorHandler.logError('PortfolioIndexManager.performBuild', error);\n      throw ErrorHandler.wrapError(error, 'Failed to build portfolio index', ErrorCategory.SYSTEM_ERROR);\n    }\n  }\n\n  /**\n   * Create an index entry from a file\n   */\n  private async createIndexEntry(filePath: string, elementType: ElementType): Promise<IndexEntry | null> {\n    try {\n      // Get file stats\n      const stats = await fs.stat(filePath);\n      \n      // Read file content\n      const content = await fs.readFile(filePath, 'utf-8');\n      \n      // Parse frontmatter securely\n      const parsed = SecureYamlParser.parse(content);\n      \n      // Extract base filename\n      const filename = path.basename(filePath, '.md');\n      \n      // Build metadata with defaults\n      const metadata = {\n        name: parsed.data.name || filename,\n        description: parsed.data.description,\n        version: parsed.data.version,\n        author: parsed.data.author,\n        tags: Array.isArray(parsed.data.tags) ? parsed.data.tags : [],\n        keywords: Array.isArray(parsed.data.keywords) ? parsed.data.keywords : [],\n        triggers: Array.isArray(parsed.data.triggers) ? parsed.data.triggers : [],\n        category: parsed.data.category,\n        created: parsed.data.created || parsed.data.created_date,\n        updated: parsed.data.updated || parsed.data.updated_date\n      };\n      \n      const entry: IndexEntry = {\n        filePath,\n        elementType,\n        metadata,\n        lastModified: stats.mtime,\n        filename\n      };\n      \n      return entry;\n      \n    } catch (error) {\n      logger.debug(`Failed to create index entry for: ${filePath}`, {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Add entry to all relevant index maps\n   */\n  private addToIndex(index: PortfolioIndex, entry: IndexEntry): void {\n    // Normalize keys for case-insensitive lookup\n    const normalizedName = entry.metadata.name.toLowerCase();\n    const normalizedFilename = entry.filename.toLowerCase();\n    \n    // Add to name map\n    index.byName.set(normalizedName, entry);\n    \n    // Add to filename map\n    index.byFilename.set(normalizedFilename, entry);\n    \n    // Add to type map\n    const typeEntries = index.byType.get(entry.elementType) || [];\n    typeEntries.push(entry);\n    index.byType.set(entry.elementType, typeEntries);\n    \n    // Add keywords\n    for (const keyword of entry.metadata.keywords || []) {\n      const normalizedKeyword = keyword.toLowerCase();\n      const keywordEntries = index.byKeyword.get(normalizedKeyword) || [];\n      keywordEntries.push(entry);\n      index.byKeyword.set(normalizedKeyword, keywordEntries);\n    }\n    \n    // Add tags\n    for (const tag of entry.metadata.tags || []) {\n      const normalizedTag = tag.toLowerCase();\n      const tagEntries = index.byTag.get(normalizedTag) || [];\n      tagEntries.push(entry);\n      index.byTag.set(normalizedTag, tagEntries);\n    }\n    \n    // Add triggers\n    for (const trigger of entry.metadata.triggers || []) {\n      const normalizedTrigger = trigger.toLowerCase();\n      const triggerEntries = index.byTrigger.get(normalizedTrigger) || [];\n      triggerEntries.push(entry);\n      index.byTrigger.set(normalizedTrigger, triggerEntries);\n    }\n  }\n\n  /**\n   * Find fuzzy matches for a name\n   */\n  private findFuzzyMatch(searchName: string, index: PortfolioIndex, options: SearchOptions): IndexEntry | null {\n    const search = searchName.toLowerCase();\n    let bestMatch: IndexEntry | null = null;\n    let bestScore = 0;\n    \n    // Search names with partial matching\n    for (const [name, entry] of index.byName) {\n      if (options.elementType && entry.elementType !== options.elementType) {\n        continue;\n      }\n      \n      const score = this.calculateSimilarity(search, name);\n      if (score > bestScore && score > 0.3) { // Minimum similarity threshold\n        bestScore = score;\n        bestMatch = entry;\n      }\n    }\n    \n    // Also check filenames\n    for (const [filename, entry] of index.byFilename) {\n      if (options.elementType && entry.elementType !== options.elementType) {\n        continue;\n      }\n      \n      const score = this.calculateSimilarity(search, filename);\n      if (score > bestScore && score > 0.3) {\n        bestScore = score;\n        bestMatch = entry;\n      }\n    }\n    \n    return bestMatch;\n  }\n\n  /**\n   * Calculate similarity between two strings\n   */\n  private calculateSimilarity(a: string, b: string): number {\n    // Simple similarity based on substring containment and length\n    if (a === b) return 1.0;\n    if (a.includes(b) || b.includes(a)) return 0.8;\n    \n    // Check for word overlap\n    const wordsA = a.split(/\\s+/);\n    const wordsB = b.split(/\\s+/);\n    const commonWords = wordsA.filter(word => wordsB.includes(word));\n    \n    if (commonWords.length > 0) {\n      return commonWords.length / Math.max(wordsA.length, wordsB.length);\n    }\n    \n    return 0;\n  }\n\n  /**\n   * Check if any query tokens match the text\n   */\n  private matchesQuery(text: string, queryTokens: string[]): boolean {\n    return queryTokens.some(token => text.includes(token));\n  }\n}"]}
@@ -33,6 +33,11 @@ export declare class PortfolioManager {
33
33
  * Check if portfolio directory exists
34
34
  */
35
35
  exists(): Promise<boolean>;
36
+ /**
37
+ * Check if a filename appears to be a test element
38
+ * SAFETY: Pattern-based filtering only, no content parsing
39
+ */
40
+ isTestElement(filename: string): boolean;
36
41
  /**
37
42
  * List all elements of a specific type
38
43
  */
@@ -1 +1 @@
1
- {"version":3,"file":"PortfolioManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAQ1D,OAAO,EAAE,WAAW,EAAE,CAAC;AACvB,YAAY,EAAE,eAAe,EAAE,CAAC;AAEhC,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAmB;IAC1C,OAAO,CAAC,MAAM,CAAC,YAAY,CAAS;IACpC,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAS;IAC1C,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAA8B;IAClE,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO;WA4BO,WAAW,CAAC,MAAM,CAAC,EAAE,eAAe,GAAG,gBAAgB;IAiBrE;;OAEG;IACI,UAAU,IAAI,MAAM;IAI3B;;OAEG;IACI,aAAa,CAAC,IAAI,EAAE,WAAW,GAAG,MAAM;IAI/C;;;OAGG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBxC;;OAEG;YACW,qBAAqB;IAiDnC;;OAEG;IACU,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC;IASvC;;OAEG;IACU,YAAY,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IA4C/D;;OAEG;IACI,cAAc,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM;IA0ClE;;OAEG;IACU,aAAa,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IASjF;;OAEG;IACI,oBAAoB,IAAI,MAAM;IAIrC;;OAEG;IACU,iBAAiB,IAAI,OAAO,CAAC,OAAO,CAAC;IAUlD;;OAEG;IACU,aAAa,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAWlE;;;OAGG;YACW,8BAA8B;CA+D7C"}
1
+ {"version":3,"file":"PortfolioManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAS1D,OAAO,EAAE,WAAW,EAAE,CAAC;AACvB,YAAY,EAAE,eAAe,EAAE,CAAC;AAEhC,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAmB;IAC1C,OAAO,CAAC,MAAM,CAAC,YAAY,CAAS;IACpC,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAS;IAC1C,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAA8B;IAClE,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO;WA4BO,WAAW,CAAC,MAAM,CAAC,EAAE,eAAe,GAAG,gBAAgB;IAiBrE;;OAEG;IACI,UAAU,IAAI,MAAM;IAI3B;;OAEG;IACI,aAAa,CAAC,IAAI,EAAE,WAAW,GAAG,MAAM;IAI/C;;;OAGG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBxC;;OAEG;YACW,qBAAqB;IAiDnC;;OAEG;IACU,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC;IASvC;;;OAGG;IACI,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAgD/C;;OAEG;IACU,YAAY,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAoC/D;;OAEG;IACI,cAAc,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM;IA0ClE;;OAEG;IACU,aAAa,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IASjF;;OAEG;IACI,oBAAoB,IAAI,MAAM;IAIrC;;OAEG;IACU,iBAAiB,IAAI,OAAO,CAAC,OAAO,CAAC;IAUlD;;OAEG;IACU,aAAa,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAWlE;;;OAGG;YACW,8BAA8B;CA+D7C"}