@dollhousemcp/mcp-server 1.5.2 → 1.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +56 -0
- package/README.md +494 -111
- package/data/agents/code-reviewer.md +8 -1
- package/data/agents/research-assistant.md +8 -1
- package/data/agents/task-manager.md +8 -1
- package/data/ensembles/business-advisor.md +8 -1
- package/data/ensembles/creative-studio.md +8 -1
- package/data/ensembles/development-team.md +8 -1
- package/data/ensembles/security-analysis-team.md +8 -1
- package/data/memories/conversation-history.md +8 -1
- package/data/memories/learning-progress.md +8 -1
- package/data/memories/project-context.md +8 -1
- package/data/personas/business-consultant.md +8 -1
- package/data/personas/creative-writer.md +8 -1
- package/data/personas/debug-detective.md +8 -1
- package/data/personas/eli5-explainer.md +8 -1
- package/data/personas/security-analyst.md +8 -1
- package/data/personas/technical-analyst.md +8 -1
- package/data/skills/code-review.md +8 -1
- package/data/skills/creative-writing.md +8 -1
- package/data/skills/data-analysis.md +8 -1
- package/data/skills/penetration-testing.md +8 -1
- package/data/skills/research.md +8 -1
- package/data/skills/threat-modeling.md +8 -1
- package/data/skills/translation.md +8 -1
- package/data/templates/code-documentation.md +8 -1
- package/data/templates/email-professional.md +8 -1
- package/data/templates/meeting-notes.md +8 -1
- package/data/templates/penetration-test-report.md +8 -1
- package/data/templates/project-brief.md +8 -1
- package/data/templates/report-executive.md +8 -1
- package/data/templates/security-vulnerability-report.md +8 -1
- package/data/templates/threat-assessment-report.md +8 -1
- package/dist/auth/GitHubAuthManager.d.ts +6 -1
- package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
- package/dist/auth/GitHubAuthManager.js +45 -18
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
- package/dist/cache/CollectionCache.d.ts.map +1 -1
- package/dist/cache/CollectionCache.js +13 -3
- package/dist/cache/CollectionIndexCache.d.ts +77 -0
- package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
- package/dist/cache/CollectionIndexCache.js +349 -0
- package/dist/cache/LRUCache.d.ts +93 -0
- package/dist/cache/LRUCache.d.ts.map +1 -0
- package/dist/cache/LRUCache.js +299 -0
- package/dist/cache/index.d.ts +1 -0
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +2 -1
- package/dist/collection/CollectionBrowser.d.ts +21 -1
- package/dist/collection/CollectionBrowser.d.ts.map +1 -1
- package/dist/collection/CollectionBrowser.js +130 -10
- package/dist/collection/CollectionIndexManager.d.ts +151 -0
- package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
- package/dist/collection/CollectionIndexManager.js +499 -0
- package/dist/collection/CollectionSearch.d.ts +55 -0
- package/dist/collection/CollectionSearch.d.ts.map +1 -1
- package/dist/collection/CollectionSearch.js +338 -13
- package/dist/collection/CollectionSeeder.d.ts.map +1 -1
- package/dist/collection/CollectionSeeder.js +38 -1
- package/dist/collection/ElementInstaller.d.ts +31 -0
- package/dist/collection/ElementInstaller.d.ts.map +1 -1
- package/dist/collection/ElementInstaller.js +77 -15
- package/dist/collection/PersonaSubmitter.d.ts +1 -1
- package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
- package/dist/collection/PersonaSubmitter.js +2 -2
- package/dist/collection/index.d.ts +1 -0
- package/dist/collection/index.d.ts.map +1 -1
- package/dist/collection/index.js +2 -1
- package/dist/config/ConfigManager.d.ts +78 -0
- package/dist/config/ConfigManager.d.ts.map +1 -0
- package/dist/config/ConfigManager.js +216 -0
- package/dist/config/element-types.d.ts +135 -0
- package/dist/config/element-types.d.ts.map +1 -0
- package/dist/config/element-types.js +108 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.d.ts.map +1 -1
- package/dist/config/index.js +3 -1
- package/dist/config/portfolio-constants.d.ts +83 -0
- package/dist/config/portfolio-constants.d.ts.map +1 -0
- package/dist/config/portfolio-constants.js +99 -0
- package/dist/elements/BaseElement.d.ts +14 -2
- package/dist/elements/BaseElement.d.ts.map +1 -1
- package/dist/elements/BaseElement.js +88 -6
- package/dist/elements/agents/Agent.d.ts +10 -1
- package/dist/elements/agents/Agent.d.ts.map +1 -1
- package/dist/elements/agents/Agent.js +66 -19
- package/dist/elements/agents/AgentManager.d.ts +2 -0
- package/dist/elements/agents/AgentManager.d.ts.map +1 -1
- package/dist/elements/agents/AgentManager.js +12 -10
- package/dist/elements/skills/Skill.d.ts +10 -1
- package/dist/elements/skills/Skill.d.ts.map +1 -1
- package/dist/elements/skills/Skill.js +40 -3
- package/dist/elements/skills/SkillManager.d.ts +1 -0
- package/dist/elements/skills/SkillManager.d.ts.map +1 -1
- package/dist/elements/skills/SkillManager.js +10 -4
- package/dist/elements/templates/Template.d.ts +10 -1
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +35 -18
- package/dist/elements/templates/TemplateManager.d.ts +1 -1
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
- package/dist/elements/templates/TemplateManager.js +6 -5
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/index.barrel.d.ts +1 -2
- package/dist/index.barrel.d.ts.map +1 -1
- package/dist/index.barrel.js +2 -4
- package/dist/index.d.ts +143 -25
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1883 -310
- package/dist/persona/PersonaElement.d.ts +10 -0
- package/dist/persona/PersonaElement.d.ts.map +1 -1
- package/dist/persona/PersonaElement.js +55 -32
- package/dist/persona/PersonaElementManager.d.ts.map +1 -1
- package/dist/persona/PersonaElementManager.js +13 -11
- package/dist/persona/PersonaLoader.d.ts.map +1 -1
- package/dist/persona/PersonaLoader.js +8 -2
- package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaImporter.js +24 -5
- package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
- package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaSharer.js +198 -22
- package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
- package/dist/portfolio/DefaultElementProvider.js +499 -7
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
- package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
- package/dist/portfolio/MigrationManager.d.ts.map +1 -1
- package/dist/portfolio/MigrationManager.js +136 -3
- package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
- package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioIndexManager.js +478 -0
- package/dist/portfolio/PortfolioManager.d.ts +5 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioManager.js +61 -20
- package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioRepoManager.js +337 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
- package/dist/portfolio/UnifiedIndexManager.js +1434 -0
- package/dist/portfolio/index.d.ts +15 -0
- package/dist/portfolio/index.d.ts.map +1 -0
- package/dist/portfolio/index.js +15 -0
- package/dist/portfolio/types.d.ts +7 -0
- package/dist/portfolio/types.d.ts.map +1 -1
- package/dist/portfolio/types.js +6 -1
- package/dist/security/InputValidator.d.ts.map +1 -1
- package/dist/security/InputValidator.js +50 -48
- package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
- package/dist/security/audit/SecurityAuditor.js +17 -9
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +19 -3
- package/dist/security/contentValidator.d.ts +2 -0
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +115 -4
- package/dist/security/secureYamlParser.d.ts +1 -0
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +29 -7
- package/dist/security/securityMonitor.d.ts +1 -1
- package/dist/security/securityMonitor.d.ts.map +1 -1
- package/dist/security/securityMonitor.js +1 -1
- package/dist/security/tokenManager.d.ts +1 -1
- package/dist/security/tokenManager.d.ts.map +1 -1
- package/dist/security/tokenManager.js +30 -10
- package/dist/server/ServerSetup.d.ts +22 -2
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +77 -12
- package/dist/server/tools/AuthTools.d.ts.map +1 -1
- package/dist/server/tools/AuthTools.js +33 -1
- package/dist/server/tools/BuildInfoTools.d.ts +25 -0
- package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
- package/dist/server/tools/BuildInfoTools.js +36 -0
- package/dist/server/tools/CollectionTools.d.ts.map +1 -1
- package/dist/server/tools/CollectionTools.js +55 -46
- package/dist/server/tools/ConfigTools.d.ts.map +1 -1
- package/dist/server/tools/ConfigTools.js +29 -1
- package/dist/server/tools/PersonaTools.d.ts +4 -2
- package/dist/server/tools/PersonaTools.d.ts.map +1 -1
- package/dist/server/tools/PersonaTools.js +5 -152
- package/dist/server/tools/PortfolioTools.d.ts +12 -0
- package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
- package/dist/server/tools/PortfolioTools.js +221 -0
- package/dist/server/tools/index.d.ts +3 -1
- package/dist/server/tools/index.d.ts.map +1 -1
- package/dist/server/tools/index.js +4 -2
- package/dist/server/types.d.ts +40 -5
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/services/BuildInfoService.d.ts +84 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -0
- package/dist/services/BuildInfoService.js +271 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
- package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
- package/dist/tools/portfolio/types.d.ts +41 -0
- package/dist/tools/portfolio/types.d.ts.map +1 -0
- package/dist/tools/portfolio/types.js +15 -0
- package/dist/types/collection.d.ts +51 -0
- package/dist/types/collection.d.ts.map +1 -1
- package/dist/types/collection.js +1 -1
- package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
- package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
- package/dist/utils/EarlyTerminationSearch.js +164 -0
- package/dist/utils/ErrorHandler.d.ts +86 -0
- package/dist/utils/ErrorHandler.d.ts.map +1 -0
- package/dist/utils/ErrorHandler.js +201 -0
- package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
- package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
- package/dist/utils/FileDiscoveryUtil.js +169 -0
- package/dist/utils/GitHubRateLimiter.d.ts +88 -0
- package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
- package/dist/utils/GitHubRateLimiter.js +315 -0
- package/dist/utils/PerformanceMonitor.d.ts +134 -0
- package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
- package/dist/utils/PerformanceMonitor.js +347 -0
- package/dist/utils/RateLimiter.d.ts.map +1 -0
- package/dist/utils/RateLimiter.js +172 -0
- package/dist/utils/SecureDownloader.d.ts +241 -0
- package/dist/utils/SecureDownloader.d.ts.map +1 -0
- package/dist/utils/SecureDownloader.js +759 -0
- package/dist/utils/ToolCache.d.ts +82 -0
- package/dist/utils/ToolCache.d.ts.map +1 -0
- package/dist/utils/ToolCache.js +196 -0
- package/dist/utils/errorCodes.d.ts +136 -0
- package/dist/utils/errorCodes.d.ts.map +1 -0
- package/dist/utils/errorCodes.js +87 -0
- package/dist/utils/index.d.ts +3 -0
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +4 -1
- package/dist/utils/installation.d.ts +1 -1
- package/dist/utils/installation.d.ts.map +1 -1
- package/dist/utils/installation.js +9 -8
- package/dist/utils/searchUtils.d.ts +31 -0
- package/dist/utils/searchUtils.d.ts.map +1 -1
- package/dist/utils/searchUtils.js +62 -1
- package/package.json +17 -7
- package/dist/config/updateConfig.d.ts +0 -84
- package/dist/config/updateConfig.d.ts.map +0 -1
- package/dist/config/updateConfig.js +0 -148
- package/dist/server/tools/UpdateTools.d.ts +0 -10
- package/dist/server/tools/UpdateTools.d.ts.map +0 -1
- package/dist/server/tools/UpdateTools.js +0 -85
- package/dist/update/BackupManager.d.ts +0 -63
- package/dist/update/BackupManager.d.ts.map +0 -1
- package/dist/update/BackupManager.js +0 -370
- package/dist/update/DependencyChecker.d.ts +0 -41
- package/dist/update/DependencyChecker.d.ts.map +0 -1
- package/dist/update/DependencyChecker.js +0 -132
- package/dist/update/RateLimiter.d.ts.map +0 -1
- package/dist/update/RateLimiter.js +0 -172
- package/dist/update/SignatureVerifier.d.ts +0 -71
- package/dist/update/SignatureVerifier.d.ts.map +0 -1
- package/dist/update/SignatureVerifier.js +0 -214
- package/dist/update/UpdateChecker.d.ts +0 -132
- package/dist/update/UpdateChecker.d.ts.map +0 -1
- package/dist/update/UpdateChecker.js +0 -506
- package/dist/update/UpdateManager.d.ts +0 -60
- package/dist/update/UpdateManager.d.ts.map +0 -1
- package/dist/update/UpdateManager.js +0 -730
- package/dist/update/VersionManager.d.ts +0 -31
- package/dist/update/VersionManager.d.ts.map +0 -1
- package/dist/update/VersionManager.js +0 -181
- package/dist/update/index.d.ts +0 -9
- package/dist/update/index.d.ts.map +0 -1
- package/dist/update/index.js +0 -9
- /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
|
@@ -0,0 +1,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;
|
|
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"}
|