@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
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"CollectionSearch.d.ts","sourceRoot":"","sources":["../../src/collection/CollectionSearch.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,eAAe,EAAkB,MAAM,6BAA6B,CAAC;
|
|
1
|
+
{"version":3,"file":"CollectionSearch.d.ts","sourceRoot":"","sources":["../../src/collection/CollectionSearch.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,eAAe,EAAkB,MAAM,6BAA6B,CAAC;AAM9E,OAAO,EAAc,aAAa,EAAE,aAAa,EAAmB,MAAM,wBAAwB,CAAC;AAEnG,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,UAAU,CAAuB;IACzC,OAAO,CAAC,aAAa,CAAwC;gBAEjD,YAAY,EAAE,YAAY,EAAE,eAAe,CAAC,EAAE,eAAe;IAMzE;;;OAGG;IACG,2BAA2B,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,GAAE,aAAkB,GAAG,OAAO,CAAC,aAAa,CAAC;IAiCrG;;;OAGG;IACG,gBAAgB,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;IAwCrD;;OAEG;YACW,eAAe;IA2C7B;;OAEG;IACH,OAAO,CAAC,cAAc;IAmCtB;;OAEG;IACH,OAAO,CAAC,UAAU;IAmBlB;;OAEG;IACH,OAAO,CAAC,+BAA+B;IAcvC;;OAEG;YACW,0BAA0B;IAiBxC;;OAEG;IACH,mBAAmB,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,gBAAgB,GAAE,MAAW,GAAG,MAAM;IAkCvF;;OAEG;YACW,eAAe;IAuC7B;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAU3B;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAuB1B;;OAEG;IACH,OAAO,CAAC,iBAAiB;IAwBzB;;OAEG;IACH,OAAO,CAAC,uBAAuB;IAkC/B;;OAEG;IACH,OAAO,CAAC,oBAAoB;IAkC5B;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAQ3B;;OAEG;IACH,OAAO,CAAC,uBAAuB;IAQ/B;;OAEG;IACH,OAAO,CAAC,wBAAwB;IAYhC;;OAEG;IACH,iCAAiC,CAAC,OAAO,EAAE,aAAa,EAAE,gBAAgB,GAAE,MAAW,GAAG,MAAM;IA8ChG;;OAEG;IACG,aAAa,IAAI,OAAO,CAAC,GAAG,CAAC;CASpC"}
|
|
@@ -2,33 +2,74 @@
|
|
|
2
2
|
* Search for content in the collection
|
|
3
3
|
*/
|
|
4
4
|
import { CollectionCache } from '../cache/CollectionCache.js';
|
|
5
|
+
import { CollectionIndexCache } from '../cache/CollectionIndexCache.js';
|
|
5
6
|
import { CollectionSeeder } from './CollectionSeeder.js';
|
|
6
7
|
import { logger } from '../utils/logger.js';
|
|
7
|
-
import { normalizeSearchTerm, validateSearchQuery } from '../utils/searchUtils.js';
|
|
8
|
+
import { normalizeSearchTerm, validateSearchQuery, isSearchMatch, debugNormalization } from '../utils/searchUtils.js';
|
|
9
|
+
import { ErrorHandler } from '../utils/ErrorHandler.js';
|
|
8
10
|
export class CollectionSearch {
|
|
9
11
|
githubClient;
|
|
10
12
|
collectionCache;
|
|
13
|
+
indexCache;
|
|
11
14
|
searchBaseUrl = 'https://api.github.com/search/code';
|
|
12
15
|
constructor(githubClient, collectionCache) {
|
|
13
16
|
this.githubClient = githubClient;
|
|
14
17
|
this.collectionCache = collectionCache || new CollectionCache();
|
|
18
|
+
this.indexCache = new CollectionIndexCache(githubClient);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Enhanced search using collection index with pagination and filtering
|
|
22
|
+
* Falls back to API search and cache when index is unavailable
|
|
23
|
+
*/
|
|
24
|
+
async searchCollectionWithOptions(query, options = {}) {
|
|
25
|
+
const startTime = Date.now();
|
|
26
|
+
logger.debug(`CollectionSearch.searchCollectionWithOptions called with query: "${query}"`, options);
|
|
27
|
+
// Validate search query for security
|
|
28
|
+
try {
|
|
29
|
+
validateSearchQuery(query, 1000);
|
|
30
|
+
}
|
|
31
|
+
catch (error) {
|
|
32
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
33
|
+
logger.error('Search query validation failed:', { query, error: errorMessage });
|
|
34
|
+
ErrorHandler.logError('CollectionSearch.searchWithOptions.validateQuery', error, { query });
|
|
35
|
+
return this.createEmptySearchResults(query, options);
|
|
36
|
+
}
|
|
37
|
+
try {
|
|
38
|
+
// Try index-based search first
|
|
39
|
+
const indexResults = await this.searchFromIndex(query, options);
|
|
40
|
+
const searchTime = Date.now() - startTime;
|
|
41
|
+
logger.debug(`Index search completed in ${searchTime}ms with ${indexResults.items.length} results`);
|
|
42
|
+
return { ...indexResults, searchTime };
|
|
43
|
+
}
|
|
44
|
+
catch (error) {
|
|
45
|
+
logger.debug('Index search failed, falling back to legacy search:', error);
|
|
46
|
+
// Fallback to legacy search
|
|
47
|
+
const legacyResults = await this.searchCollection(query);
|
|
48
|
+
const searchTime = Date.now() - startTime;
|
|
49
|
+
// Convert legacy results to new format
|
|
50
|
+
return this.convertLegacyResults(legacyResults, query, options, searchTime);
|
|
51
|
+
}
|
|
15
52
|
}
|
|
16
53
|
/**
|
|
17
54
|
* Search collection for content matching query
|
|
18
55
|
* Falls back to cached data when GitHub API is not available or not authenticated
|
|
19
56
|
*/
|
|
20
57
|
async searchCollection(query) {
|
|
58
|
+
logger.debug(`CollectionSearch.searchCollection called with query: "${query}"`);
|
|
21
59
|
// Validate search query for security
|
|
22
60
|
try {
|
|
23
61
|
validateSearchQuery(query, 1000);
|
|
24
62
|
}
|
|
25
63
|
catch (error) {
|
|
26
|
-
|
|
64
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
65
|
+
logger.error('Search query validation failed:', { query, error: errorMessage });
|
|
66
|
+
ErrorHandler.logError('CollectionSearch.search.validateQuery', error, { query });
|
|
27
67
|
return [];
|
|
28
68
|
}
|
|
29
69
|
try {
|
|
30
70
|
// First, try GitHub API search if authenticated
|
|
31
71
|
const searchUrl = `${this.searchBaseUrl}?q=${encodeURIComponent(query)}+repo:DollhouseMCP/collection+path:library+extension:md`;
|
|
72
|
+
logger.debug(`Attempting GitHub API search with URL: ${searchUrl}`);
|
|
32
73
|
const data = await this.githubClient.fetchFromGitHub(searchUrl, false); // Don't require auth for search
|
|
33
74
|
if (data.items && Array.isArray(data.items)) {
|
|
34
75
|
logger.debug(`Found ${data.items.length} items via GitHub API search`);
|
|
@@ -36,10 +77,13 @@ export class CollectionSearch {
|
|
|
36
77
|
await this.updateCacheFromGitHubItems(data.items);
|
|
37
78
|
return data.items;
|
|
38
79
|
}
|
|
80
|
+
logger.debug('GitHub API search returned no items, falling back to cache');
|
|
39
81
|
return [];
|
|
40
82
|
}
|
|
41
83
|
catch (error) {
|
|
42
|
-
|
|
84
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
85
|
+
logger.debug(`GitHub API search failed: ${errorMessage}. Falling back to cached search.`);
|
|
86
|
+
ErrorHandler.logError('CollectionSearch.search.githubApi', error, { query });
|
|
43
87
|
// Fallback to cached search
|
|
44
88
|
return this.searchFromCache(query);
|
|
45
89
|
}
|
|
@@ -48,6 +92,7 @@ export class CollectionSearch {
|
|
|
48
92
|
* Search cached collection items
|
|
49
93
|
*/
|
|
50
94
|
async searchFromCache(query) {
|
|
95
|
+
logger.debug(`Searching cache for query: "${query}"`);
|
|
51
96
|
try {
|
|
52
97
|
// Try to load from cache first
|
|
53
98
|
const cachedItems = await this.collectionCache.searchCache(query);
|
|
@@ -55,19 +100,29 @@ export class CollectionSearch {
|
|
|
55
100
|
logger.debug(`Found ${cachedItems.length} items from cache`);
|
|
56
101
|
return this.convertCacheItemsToGitHubFormat(cachedItems);
|
|
57
102
|
}
|
|
103
|
+
logger.debug('Cache search returned no results, trying seed data');
|
|
58
104
|
// If cache is empty or no results, use seed data
|
|
59
105
|
const seedItems = this.searchSeedData(query);
|
|
60
106
|
if (seedItems.length > 0) {
|
|
61
107
|
logger.debug(`Found ${seedItems.length} items from seed data`);
|
|
62
108
|
// Save seed data to cache for future use
|
|
63
|
-
|
|
109
|
+
try {
|
|
110
|
+
await this.collectionCache.saveCache(CollectionSeeder.getSeedData());
|
|
111
|
+
logger.debug('Saved seed data to cache');
|
|
112
|
+
}
|
|
113
|
+
catch (cacheError) {
|
|
114
|
+
const cacheErrorMessage = cacheError instanceof Error ? cacheError.message : String(cacheError);
|
|
115
|
+
logger.debug(`Failed to save seed data to cache: ${cacheErrorMessage}`);
|
|
116
|
+
}
|
|
64
117
|
return this.convertCacheItemsToGitHubFormat(seedItems);
|
|
65
118
|
}
|
|
66
119
|
logger.debug('No items found in cache or seed data');
|
|
67
120
|
return [];
|
|
68
121
|
}
|
|
69
122
|
catch (error) {
|
|
70
|
-
|
|
123
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
124
|
+
logger.debug(`Cache search failed: ${errorMessage}`);
|
|
125
|
+
ErrorHandler.logError('CollectionSearch.search.cache', error, { query });
|
|
71
126
|
// Last resort: search seed data without cache
|
|
72
127
|
const seedItems = this.searchSeedData(query);
|
|
73
128
|
logger.debug(`Fallback to seed data found ${seedItems.length} items`);
|
|
@@ -79,13 +134,50 @@ export class CollectionSearch {
|
|
|
79
134
|
*/
|
|
80
135
|
searchSeedData(query) {
|
|
81
136
|
const seedData = CollectionSeeder.getSeedData();
|
|
82
|
-
const
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
137
|
+
const normDebug = debugNormalization(query);
|
|
138
|
+
logger.debug(`Searching seed data - Original: "${normDebug.original}", Normalized: "${normDebug.normalized}", Partial: "${normDebug.partialMatch}"`);
|
|
139
|
+
logger.debug(`Searching against ${seedData.length} seed items`);
|
|
140
|
+
const matches = seedData.filter(item => {
|
|
141
|
+
// Use the improved matching function that tries multiple strategies
|
|
142
|
+
const nameMatches = isSearchMatch(query, item.name);
|
|
143
|
+
const pathMatches = isSearchMatch(query, item.path);
|
|
144
|
+
const isMatch = nameMatches || pathMatches;
|
|
145
|
+
if (isMatch) {
|
|
146
|
+
logger.debug(`✓ Match found: ${item.name} (${item.path}) matches query "${query}"`);
|
|
147
|
+
}
|
|
148
|
+
return isMatch;
|
|
88
149
|
});
|
|
150
|
+
// If no matches found, let's debug what we have
|
|
151
|
+
if (matches.length === 0) {
|
|
152
|
+
logger.debug('No matches found. Available seed data:');
|
|
153
|
+
seedData.slice(0, 10).forEach(item => {
|
|
154
|
+
logger.debug(` - ${item.name} (${item.path})`);
|
|
155
|
+
});
|
|
156
|
+
if (seedData.length > 10) {
|
|
157
|
+
logger.debug(` ... and ${seedData.length - 10} more items`);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
logger.debug(`Found ${matches.length} matches in seed data`);
|
|
161
|
+
return matches;
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Fuzzy matching algorithm for partial string matches
|
|
165
|
+
*/
|
|
166
|
+
fuzzyMatch(term, target) {
|
|
167
|
+
// Simple fuzzy matching: check if all characters of term appear in order in target
|
|
168
|
+
if (term.length === 0)
|
|
169
|
+
return true;
|
|
170
|
+
if (target.length === 0)
|
|
171
|
+
return false;
|
|
172
|
+
let termIndex = 0;
|
|
173
|
+
let targetIndex = 0;
|
|
174
|
+
while (termIndex < term.length && targetIndex < target.length) {
|
|
175
|
+
if (term[termIndex] === target[targetIndex]) {
|
|
176
|
+
termIndex++;
|
|
177
|
+
}
|
|
178
|
+
targetIndex++;
|
|
179
|
+
}
|
|
180
|
+
return termIndex === term.length;
|
|
89
181
|
}
|
|
90
182
|
/**
|
|
91
183
|
* Convert cache items to GitHub API format for consistent response structure
|
|
@@ -118,7 +210,7 @@ export class CollectionSearch {
|
|
|
118
210
|
logger.debug(`Updated cache with ${cacheItems.length} items from GitHub API`);
|
|
119
211
|
}
|
|
120
212
|
catch (error) {
|
|
121
|
-
|
|
213
|
+
ErrorHandler.logError('CollectionSearch.updateCacheInBackground', error);
|
|
122
214
|
// Don't throw - cache update failures shouldn't break functionality
|
|
123
215
|
}
|
|
124
216
|
}
|
|
@@ -148,5 +240,238 @@ export class CollectionSearch {
|
|
|
148
240
|
});
|
|
149
241
|
return textParts.join('');
|
|
150
242
|
}
|
|
243
|
+
/**
|
|
244
|
+
* Search from collection index with full featured search and pagination
|
|
245
|
+
*/
|
|
246
|
+
async searchFromIndex(query, options) {
|
|
247
|
+
const index = await this.indexCache.getIndex();
|
|
248
|
+
const allEntries = this.flattenIndexEntries(index);
|
|
249
|
+
// Filter by element type if specified
|
|
250
|
+
let filteredEntries = allEntries;
|
|
251
|
+
if (options.elementType) {
|
|
252
|
+
filteredEntries = allEntries.filter(entry => entry.type === options.elementType);
|
|
253
|
+
}
|
|
254
|
+
// Filter by category if specified
|
|
255
|
+
if (options.category) {
|
|
256
|
+
filteredEntries = filteredEntries.filter(entry => entry.category === options.category);
|
|
257
|
+
}
|
|
258
|
+
// Search matching
|
|
259
|
+
const matchedEntries = this.performIndexSearch(query, filteredEntries);
|
|
260
|
+
// Sort results
|
|
261
|
+
const sortedEntries = this.sortSearchResults(matchedEntries, options.sortBy || 'relevance', query);
|
|
262
|
+
// Apply pagination
|
|
263
|
+
const page = options.page || 1;
|
|
264
|
+
const pageSize = options.pageSize || 25;
|
|
265
|
+
const startIndex = (page - 1) * pageSize;
|
|
266
|
+
const endIndex = startIndex + pageSize;
|
|
267
|
+
const paginatedEntries = sortedEntries.slice(startIndex, endIndex);
|
|
268
|
+
return {
|
|
269
|
+
items: paginatedEntries,
|
|
270
|
+
total: sortedEntries.length,
|
|
271
|
+
page,
|
|
272
|
+
pageSize,
|
|
273
|
+
hasMore: endIndex < sortedEntries.length,
|
|
274
|
+
query,
|
|
275
|
+
searchTime: 0 // Will be set by caller
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Flatten index entries from all categories into a single array
|
|
280
|
+
*/
|
|
281
|
+
flattenIndexEntries(index) {
|
|
282
|
+
const entries = [];
|
|
283
|
+
for (const [elementType, typeEntries] of Object.entries(index.index)) {
|
|
284
|
+
entries.push(...typeEntries);
|
|
285
|
+
}
|
|
286
|
+
return entries;
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* Perform search matching on index entries
|
|
290
|
+
*/
|
|
291
|
+
performIndexSearch(query, entries) {
|
|
292
|
+
const normalizedQuery = normalizeSearchTerm(query);
|
|
293
|
+
const queryWords = normalizedQuery.split(/\s+/).filter(word => word.length > 0);
|
|
294
|
+
return entries.filter(entry => {
|
|
295
|
+
// Search in multiple fields
|
|
296
|
+
const searchableText = [
|
|
297
|
+
entry.name,
|
|
298
|
+
entry.description,
|
|
299
|
+
entry.path,
|
|
300
|
+
...entry.tags
|
|
301
|
+
].join(' ').toLowerCase();
|
|
302
|
+
// Use existing search utilities for consistency
|
|
303
|
+
const nameMatch = isSearchMatch(query, entry.name);
|
|
304
|
+
const descMatch = isSearchMatch(query, entry.description);
|
|
305
|
+
const pathMatch = isSearchMatch(query, entry.path);
|
|
306
|
+
const tagMatch = entry.tags.some(tag => isSearchMatch(query, tag));
|
|
307
|
+
return nameMatch || descMatch || pathMatch || tagMatch;
|
|
308
|
+
});
|
|
309
|
+
}
|
|
310
|
+
/**
|
|
311
|
+
* Sort search results by relevance, name, or date
|
|
312
|
+
*/
|
|
313
|
+
sortSearchResults(entries, sortBy, query) {
|
|
314
|
+
const sorted = [...entries];
|
|
315
|
+
switch (sortBy) {
|
|
316
|
+
case 'name':
|
|
317
|
+
sorted.sort((a, b) => a.name.localeCompare(b.name));
|
|
318
|
+
break;
|
|
319
|
+
case 'date':
|
|
320
|
+
sorted.sort((a, b) => new Date(b.created).getTime() - new Date(a.created).getTime());
|
|
321
|
+
break;
|
|
322
|
+
case 'relevance':
|
|
323
|
+
default:
|
|
324
|
+
// Calculate relevance scores
|
|
325
|
+
sorted.sort((a, b) => {
|
|
326
|
+
const scoreA = this.calculateRelevanceScore(query, a);
|
|
327
|
+
const scoreB = this.calculateRelevanceScore(query, b);
|
|
328
|
+
return scoreB - scoreA;
|
|
329
|
+
});
|
|
330
|
+
break;
|
|
331
|
+
}
|
|
332
|
+
return sorted;
|
|
333
|
+
}
|
|
334
|
+
/**
|
|
335
|
+
* Calculate relevance score for search results
|
|
336
|
+
*/
|
|
337
|
+
calculateRelevanceScore(query, entry) {
|
|
338
|
+
const normalizedQuery = normalizeSearchTerm(query);
|
|
339
|
+
let score = 0;
|
|
340
|
+
// Exact name match gets highest score
|
|
341
|
+
if (normalizeSearchTerm(entry.name).includes(normalizedQuery)) {
|
|
342
|
+
score += 100;
|
|
343
|
+
}
|
|
344
|
+
// Description match
|
|
345
|
+
if (normalizeSearchTerm(entry.description).includes(normalizedQuery)) {
|
|
346
|
+
score += 50;
|
|
347
|
+
}
|
|
348
|
+
// Tag matches
|
|
349
|
+
const matchingTags = entry.tags.filter(tag => normalizeSearchTerm(tag).includes(normalizedQuery));
|
|
350
|
+
score += matchingTags.length * 25;
|
|
351
|
+
// Path match (lower priority)
|
|
352
|
+
if (normalizeSearchTerm(entry.path).includes(normalizedQuery)) {
|
|
353
|
+
score += 10;
|
|
354
|
+
}
|
|
355
|
+
// Bonus for recent content
|
|
356
|
+
const daysSinceCreated = (Date.now() - new Date(entry.created).getTime()) / (1000 * 60 * 60 * 24);
|
|
357
|
+
if (daysSinceCreated < 30) {
|
|
358
|
+
score += 5;
|
|
359
|
+
}
|
|
360
|
+
return score;
|
|
361
|
+
}
|
|
362
|
+
/**
|
|
363
|
+
* Convert legacy search results to new SearchResults format
|
|
364
|
+
*/
|
|
365
|
+
convertLegacyResults(legacyResults, query, options, searchTime) {
|
|
366
|
+
// Convert GitHub API format to IndexEntry format
|
|
367
|
+
const entries = legacyResults.map(item => ({
|
|
368
|
+
path: item.path,
|
|
369
|
+
type: this.extractTypeFromPath(item.path),
|
|
370
|
+
name: item.name?.replace('.md', '') || 'Unknown',
|
|
371
|
+
description: 'No description available',
|
|
372
|
+
version: '1.0.0',
|
|
373
|
+
author: 'Unknown',
|
|
374
|
+
tags: [],
|
|
375
|
+
sha: item.sha || '',
|
|
376
|
+
category: this.extractCategoryFromPath(item.path),
|
|
377
|
+
created: new Date().toISOString(),
|
|
378
|
+
license: 'Unknown'
|
|
379
|
+
}));
|
|
380
|
+
// Apply pagination
|
|
381
|
+
const page = options.page || 1;
|
|
382
|
+
const pageSize = options.pageSize || 25;
|
|
383
|
+
const startIndex = (page - 1) * pageSize;
|
|
384
|
+
const endIndex = startIndex + pageSize;
|
|
385
|
+
const paginatedEntries = entries.slice(startIndex, endIndex);
|
|
386
|
+
return {
|
|
387
|
+
items: paginatedEntries,
|
|
388
|
+
total: entries.length,
|
|
389
|
+
page,
|
|
390
|
+
pageSize,
|
|
391
|
+
hasMore: endIndex < entries.length,
|
|
392
|
+
query,
|
|
393
|
+
searchTime
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
/**
|
|
397
|
+
* Extract element type from file path
|
|
398
|
+
*/
|
|
399
|
+
extractTypeFromPath(path) {
|
|
400
|
+
const parts = path.split('/');
|
|
401
|
+
if (parts.length >= 2 && parts[0] === 'library') {
|
|
402
|
+
return parts[1];
|
|
403
|
+
}
|
|
404
|
+
return 'unknown';
|
|
405
|
+
}
|
|
406
|
+
/**
|
|
407
|
+
* Extract category from file path
|
|
408
|
+
*/
|
|
409
|
+
extractCategoryFromPath(path) {
|
|
410
|
+
const parts = path.split('/');
|
|
411
|
+
if (parts.length >= 3 && parts[0] === 'library') {
|
|
412
|
+
return parts[2];
|
|
413
|
+
}
|
|
414
|
+
return 'uncategorized';
|
|
415
|
+
}
|
|
416
|
+
/**
|
|
417
|
+
* Create empty search results for error cases
|
|
418
|
+
*/
|
|
419
|
+
createEmptySearchResults(query, options) {
|
|
420
|
+
return {
|
|
421
|
+
items: [],
|
|
422
|
+
total: 0,
|
|
423
|
+
page: options.page || 1,
|
|
424
|
+
pageSize: options.pageSize || 25,
|
|
425
|
+
hasMore: false,
|
|
426
|
+
query,
|
|
427
|
+
searchTime: 0
|
|
428
|
+
};
|
|
429
|
+
}
|
|
430
|
+
/**
|
|
431
|
+
* Enhanced format for search results with pagination info
|
|
432
|
+
*/
|
|
433
|
+
formatSearchResultsWithPagination(results, personaIndicator = '') {
|
|
434
|
+
if (results.total === 0) {
|
|
435
|
+
return `${personaIndicator}🔍 No content found for query: "${results.query}"`;
|
|
436
|
+
}
|
|
437
|
+
const startItem = (results.page - 1) * results.pageSize + 1;
|
|
438
|
+
const endItem = Math.min(results.page * results.pageSize, results.total);
|
|
439
|
+
const textParts = [
|
|
440
|
+
`${personaIndicator}🔍 **Search Results for "${results.query}"**\n`,
|
|
441
|
+
`📊 Showing ${startItem}-${endItem} of ${results.total} results (Page ${results.page})\n`,
|
|
442
|
+
`⚡ Search time: ${results.searchTime}ms\n\n`
|
|
443
|
+
];
|
|
444
|
+
results.items.forEach((item) => {
|
|
445
|
+
const contentIcons = {
|
|
446
|
+
'personas': '🎭',
|
|
447
|
+
'skills': '🛠️',
|
|
448
|
+
'agents': '🤖',
|
|
449
|
+
'prompts': '💬',
|
|
450
|
+
'templates': '📄',
|
|
451
|
+
'tools': '🔧',
|
|
452
|
+
'ensembles': '🎼',
|
|
453
|
+
'memories': '🧠'
|
|
454
|
+
};
|
|
455
|
+
const icon = contentIcons[item.type] || '📄';
|
|
456
|
+
textParts.push(` ${icon} **${item.name}** (${item.type})\n`, ` 📝 ${item.description}\n`, ` 🏷️ Tags: ${item.tags.join(', ')}\n`, ` 📂 Path: ${item.path}\n`, ` 📥 Install: \`install_content "${item.path}"\`\n`, ` 👁️ Details: \`get_collection_content "${item.path}"\`\n\n`);
|
|
457
|
+
});
|
|
458
|
+
// Add pagination info
|
|
459
|
+
if (results.hasMore) {
|
|
460
|
+
const nextPage = results.page + 1;
|
|
461
|
+
textParts.push(`📄 More results available. Use page ${nextPage} to see next ${results.pageSize} items.\n`);
|
|
462
|
+
}
|
|
463
|
+
return textParts.join('');
|
|
464
|
+
}
|
|
465
|
+
/**
|
|
466
|
+
* Get cache statistics for debugging
|
|
467
|
+
*/
|
|
468
|
+
async getCacheStats() {
|
|
469
|
+
const indexStats = this.indexCache.getCacheStats();
|
|
470
|
+
const cacheStats = await this.collectionCache.getCacheStats();
|
|
471
|
+
return {
|
|
472
|
+
index: indexStats,
|
|
473
|
+
collection: cacheStats
|
|
474
|
+
};
|
|
475
|
+
}
|
|
151
476
|
}
|
|
152
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionSearch.js","sourceRoot":"","sources":["../../src/collection/CollectionSearch.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Search for content in the collection\n */\n\nimport { GitHubClient } from './GitHubClient.js';\nimport { CollectionCache, CollectionItem } from '../cache/CollectionCache.js';\nimport { CollectionSeeder } from './CollectionSeeder.js';\nimport { logger } from '../utils/logger.js';\nimport { normalizeSearchTerm, validateSearchQuery } from '../utils/searchUtils.js';\n\nexport class CollectionSearch {\n  private githubClient: GitHubClient;\n  private collectionCache: CollectionCache;\n  private searchBaseUrl = 'https://api.github.com/search/code';\n  \n  constructor(githubClient: GitHubClient, collectionCache?: CollectionCache) {\n    this.githubClient = githubClient;\n    this.collectionCache = collectionCache || new CollectionCache();\n  }\n  \n  /**\n   * Search collection for content matching query\n   * Falls back to cached data when GitHub API is not available or not authenticated\n   */\n  async searchCollection(query: string): Promise<any[]> {\n    // Validate search query for security\n    try {\n      validateSearchQuery(query, 1000);\n    } catch (error) {\n      logger.warn(`Invalid search query: ${error}`);\n      return [];\n    }\n    \n    try {\n      // First, try GitHub API search if authenticated\n      const searchUrl = `${this.searchBaseUrl}?q=${encodeURIComponent(query)}+repo:DollhouseMCP/collection+path:library+extension:md`;\n      const data = await this.githubClient.fetchFromGitHub(searchUrl, false); // Don't require auth for search\n      \n      if (data.items && Array.isArray(data.items)) {\n        logger.debug(`Found ${data.items.length} items via GitHub API search`);\n        \n        // Update cache with fresh data from API\n        await this.updateCacheFromGitHubItems(data.items);\n        \n        return data.items;\n      }\n      \n      return [];\n    } catch (error) {\n      logger.debug(`GitHub API search failed, falling back to cache: ${error}`);\n      \n      // Fallback to cached search\n      return this.searchFromCache(query);\n    }\n  }\n  \n  /**\n   * Search cached collection items\n   */\n  private async searchFromCache(query: string): Promise<any[]> {\n    try {\n      // Try to load from cache first\n      const cachedItems = await this.collectionCache.searchCache(query);\n      \n      if (cachedItems.length > 0) {\n        logger.debug(`Found ${cachedItems.length} items from cache`);\n        return this.convertCacheItemsToGitHubFormat(cachedItems);\n      }\n      \n      // If cache is empty or no results, use seed data\n      const seedItems = this.searchSeedData(query);\n      if (seedItems.length > 0) {\n        logger.debug(`Found ${seedItems.length} items from seed data`);\n        // Save seed data to cache for future use\n        await this.collectionCache.saveCache(CollectionSeeder.getSeedData());\n        return this.convertCacheItemsToGitHubFormat(seedItems);\n      }\n      \n      logger.debug('No items found in cache or seed data');\n      return [];\n    } catch (error) {\n      logger.error(`Cache search failed: ${error}`);\n      \n      // Last resort: search seed data without cache\n      const seedItems = this.searchSeedData(query);\n      logger.debug(`Fallback to seed data found ${seedItems.length} items`);\n      return this.convertCacheItemsToGitHubFormat(seedItems);\n    }\n  }\n  \n  /**\n   * Search seed data for matching items with fuzzy matching\n   */\n  private searchSeedData(query: string): CollectionItem[] {\n    const seedData = CollectionSeeder.getSeedData();\n    const normalizedQuery = normalizeSearchTerm(query);\n    \n    return seedData.filter(item => {\n      const normalizedName = normalizeSearchTerm(item.name);\n      const normalizedPath = normalizeSearchTerm(item.path);\n      \n      return normalizedName.includes(normalizedQuery) || \n             normalizedPath.includes(normalizedQuery);\n    });\n  }\n  \n  \n  /**\n   * Convert cache items to GitHub API format for consistent response structure\n   */\n  private convertCacheItemsToGitHubFormat(cacheItems: CollectionItem[]): any[] {\n    return cacheItems.map(item => ({\n      name: item.name,\n      path: item.path,\n      sha: item.sha,\n      url: `https://api.github.com/repos/DollhouseMCP/collection/contents/${item.path}`,\n      html_url: `https://github.com/DollhouseMCP/collection/blob/main/${item.path}`,\n      repository: {\n        name: 'collection',\n        full_name: 'DollhouseMCP/collection'\n      }\n    }));\n  }\n  \n  /**\n   * Update cache with fresh data from GitHub API items\n   */\n  private async updateCacheFromGitHubItems(githubItems: any[]): Promise<void> {\n    try {\n      const cacheItems: CollectionItem[] = githubItems.map(item => ({\n        name: item.name,\n        path: item.path,\n        sha: item.sha,\n        last_modified: new Date().toISOString()\n      }));\n      \n      await this.collectionCache.saveCache(cacheItems);\n      logger.debug(`Updated cache with ${cacheItems.length} items from GitHub API`);\n    } catch (error) {\n      logger.debug(`Failed to update cache: ${error}`);\n      // Don't throw - cache update failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Format search results\n   */\n  formatSearchResults(items: any[], query: string, personaIndicator: string = ''): string {\n    if (items.length === 0) {\n      return `${personaIndicator}🔍 No content found for query: \"${query}\"`;\n    }\n    \n    const textParts = [`${personaIndicator}🔍 **Search Results for \"${query}\"** (${items.length} found)\\n\\n`];\n    \n    items.forEach((item: any) => {\n      // Extract content type from path (library/personas/creative/writer.md -> personas)\n      const pathParts = item.path.split('/');\n      const contentType = pathParts[1] || 'content';\n      \n      const contentIcons: { [key: string]: string } = {\n        'personas': '🎭',\n        'skills': '🛠️',\n        'agents': '🤖',\n        'prompts': '💬',\n        'templates': '📄',\n        'tools': '🔧',\n        'ensembles': '🎼'\n      };\n      const icon = contentIcons[contentType] || '📄';\n      \n      textParts.push(\n        `   ${icon} **${item.name.replace('.md', '')}**\\n`,\n        `      📂 Path: ${item.path}\\n`,\n        `      📥 Install: \\`install_content \"${item.path}\"\\`\\n`,\n        `      👁️ Details: \\`get_collection_content \"${item.path}\"\\`\\n\\n`\n      );\n    });\n    \n    return textParts.join('');\n  }\n}"]}
|
|
477
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionSearch.js","sourceRoot":"","sources":["../../src/collection/CollectionSearch.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Search for content in the collection\n */\n\nimport { GitHubClient } from './GitHubClient.js';\nimport { CollectionCache, CollectionItem } from '../cache/CollectionCache.js';\nimport { CollectionIndexCache } from '../cache/CollectionIndexCache.js';\nimport { CollectionSeeder } from './CollectionSeeder.js';\nimport { logger } from '../utils/logger.js';\nimport { normalizeSearchTerm, validateSearchQuery, isSearchMatch, debugNormalization } from '../utils/searchUtils.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { IndexEntry, SearchResults, SearchOptions, CollectionIndex } from '../types/collection.js';\n\nexport class CollectionSearch {\n  private githubClient: GitHubClient;\n  private collectionCache: CollectionCache;\n  private indexCache: CollectionIndexCache;\n  private searchBaseUrl = 'https://api.github.com/search/code';\n  \n  constructor(githubClient: GitHubClient, collectionCache?: CollectionCache) {\n    this.githubClient = githubClient;\n    this.collectionCache = collectionCache || new CollectionCache();\n    this.indexCache = new CollectionIndexCache(githubClient);\n  }\n  \n  /**\n   * Enhanced search using collection index with pagination and filtering\n   * Falls back to API search and cache when index is unavailable\n   */\n  async searchCollectionWithOptions(query: string, options: SearchOptions = {}): Promise<SearchResults> {\n    const startTime = Date.now();\n    logger.debug(`CollectionSearch.searchCollectionWithOptions called with query: \"${query}\"`, options);\n    \n    // Validate search query for security\n    try {\n      validateSearchQuery(query, 1000);\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      logger.error('Search query validation failed:', { query, error: errorMessage });\n      ErrorHandler.logError('CollectionSearch.searchWithOptions.validateQuery', error, { query });\n      return this.createEmptySearchResults(query, options);\n    }\n    \n    try {\n      // Try index-based search first\n      const indexResults = await this.searchFromIndex(query, options);\n      const searchTime = Date.now() - startTime;\n      \n      logger.debug(`Index search completed in ${searchTime}ms with ${indexResults.items.length} results`);\n      return { ...indexResults, searchTime };\n    } catch (error) {\n      logger.debug('Index search failed, falling back to legacy search:', error);\n      \n      // Fallback to legacy search\n      const legacyResults = await this.searchCollection(query);\n      const searchTime = Date.now() - startTime;\n      \n      // Convert legacy results to new format\n      return this.convertLegacyResults(legacyResults, query, options, searchTime);\n    }\n  }\n\n  /**\n   * Search collection for content matching query\n   * Falls back to cached data when GitHub API is not available or not authenticated\n   */\n  async searchCollection(query: string): Promise<any[]> {\n    logger.debug(`CollectionSearch.searchCollection called with query: \"${query}\"`);\n    \n    // Validate search query for security\n    try {\n      validateSearchQuery(query, 1000);\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      logger.error('Search query validation failed:', { query, error: errorMessage });\n      ErrorHandler.logError('CollectionSearch.search.validateQuery', error, { query });\n      return [];\n    }\n    \n    try {\n      // First, try GitHub API search if authenticated\n      const searchUrl = `${this.searchBaseUrl}?q=${encodeURIComponent(query)}+repo:DollhouseMCP/collection+path:library+extension:md`;\n      logger.debug(`Attempting GitHub API search with URL: ${searchUrl}`);\n      const data = await this.githubClient.fetchFromGitHub(searchUrl, false); // Don't require auth for search\n      \n      if (data.items && Array.isArray(data.items)) {\n        logger.debug(`Found ${data.items.length} items via GitHub API search`);\n        \n        // Update cache with fresh data from API\n        await this.updateCacheFromGitHubItems(data.items);\n        \n        return data.items;\n      }\n      \n      logger.debug('GitHub API search returned no items, falling back to cache');\n      return [];\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      logger.debug(`GitHub API search failed: ${errorMessage}. Falling back to cached search.`);\n      ErrorHandler.logError('CollectionSearch.search.githubApi', error, { query });\n      \n      // Fallback to cached search\n      return this.searchFromCache(query);\n    }\n  }\n  \n  /**\n   * Search cached collection items\n   */\n  private async searchFromCache(query: string): Promise<any[]> {\n    logger.debug(`Searching cache for query: \"${query}\"`);\n    \n    try {\n      // Try to load from cache first\n      const cachedItems = await this.collectionCache.searchCache(query);\n      \n      if (cachedItems.length > 0) {\n        logger.debug(`Found ${cachedItems.length} items from cache`);\n        return this.convertCacheItemsToGitHubFormat(cachedItems);\n      }\n      \n      logger.debug('Cache search returned no results, trying seed data');\n      \n      // If cache is empty or no results, use seed data\n      const seedItems = this.searchSeedData(query);\n      if (seedItems.length > 0) {\n        logger.debug(`Found ${seedItems.length} items from seed data`);\n        // Save seed data to cache for future use\n        try {\n          await this.collectionCache.saveCache(CollectionSeeder.getSeedData());\n          logger.debug('Saved seed data to cache');\n        } catch (cacheError) {\n          const cacheErrorMessage = cacheError instanceof Error ? cacheError.message : String(cacheError);\n          logger.debug(`Failed to save seed data to cache: ${cacheErrorMessage}`);\n        }\n        return this.convertCacheItemsToGitHubFormat(seedItems);\n      }\n      \n      logger.debug('No items found in cache or seed data');\n      return [];\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      logger.debug(`Cache search failed: ${errorMessage}`);\n      ErrorHandler.logError('CollectionSearch.search.cache', error, { query });\n      \n      // Last resort: search seed data without cache\n      const seedItems = this.searchSeedData(query);\n      logger.debug(`Fallback to seed data found ${seedItems.length} items`);\n      return this.convertCacheItemsToGitHubFormat(seedItems);\n    }\n  }\n  \n  /**\n   * Search seed data for matching items with fuzzy matching\n   */\n  private searchSeedData(query: string): CollectionItem[] {\n    const seedData = CollectionSeeder.getSeedData();\n    const normDebug = debugNormalization(query);\n    logger.debug(`Searching seed data - Original: \"${normDebug.original}\", Normalized: \"${normDebug.normalized}\", Partial: \"${normDebug.partialMatch}\"`);\n    logger.debug(`Searching against ${seedData.length} seed items`);\n    \n    const matches = seedData.filter(item => {\n      // Use the improved matching function that tries multiple strategies\n      const nameMatches = isSearchMatch(query, item.name);\n      const pathMatches = isSearchMatch(query, item.path);\n      \n      const isMatch = nameMatches || pathMatches;\n      \n      if (isMatch) {\n        logger.debug(`✓ Match found: ${item.name} (${item.path}) matches query \"${query}\"`);\n      }\n      \n      return isMatch;\n    });\n    \n    // If no matches found, let's debug what we have\n    if (matches.length === 0) {\n      logger.debug('No matches found. Available seed data:');\n      seedData.slice(0, 10).forEach(item => {\n        logger.debug(`  - ${item.name} (${item.path})`);\n      });\n      if (seedData.length > 10) {\n        logger.debug(`  ... and ${seedData.length - 10} more items`);\n      }\n    }\n    \n    logger.debug(`Found ${matches.length} matches in seed data`);\n    return matches;\n  }\n  \n  /**\n   * Fuzzy matching algorithm for partial string matches\n   */\n  private fuzzyMatch(term: string, target: string): boolean {\n    // Simple fuzzy matching: check if all characters of term appear in order in target\n    if (term.length === 0) return true;\n    if (target.length === 0) return false;\n    \n    let termIndex = 0;\n    let targetIndex = 0;\n    \n    while (termIndex < term.length && targetIndex < target.length) {\n      if (term[termIndex] === target[targetIndex]) {\n        termIndex++;\n      }\n      targetIndex++;\n    }\n    \n    return termIndex === term.length;\n  }\n  \n  \n  /**\n   * Convert cache items to GitHub API format for consistent response structure\n   */\n  private convertCacheItemsToGitHubFormat(cacheItems: CollectionItem[]): any[] {\n    return cacheItems.map(item => ({\n      name: item.name,\n      path: item.path,\n      sha: item.sha,\n      url: `https://api.github.com/repos/DollhouseMCP/collection/contents/${item.path}`,\n      html_url: `https://github.com/DollhouseMCP/collection/blob/main/${item.path}`,\n      repository: {\n        name: 'collection',\n        full_name: 'DollhouseMCP/collection'\n      }\n    }));\n  }\n  \n  /**\n   * Update cache with fresh data from GitHub API items\n   */\n  private async updateCacheFromGitHubItems(githubItems: any[]): Promise<void> {\n    try {\n      const cacheItems: CollectionItem[] = githubItems.map(item => ({\n        name: item.name,\n        path: item.path,\n        sha: item.sha,\n        last_modified: new Date().toISOString()\n      }));\n      \n      await this.collectionCache.saveCache(cacheItems);\n      logger.debug(`Updated cache with ${cacheItems.length} items from GitHub API`);\n    } catch (error) {\n      ErrorHandler.logError('CollectionSearch.updateCacheInBackground', error);\n      // Don't throw - cache update failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Format search results\n   */\n  formatSearchResults(items: any[], query: string, personaIndicator: string = ''): string {\n    if (items.length === 0) {\n      return `${personaIndicator}🔍 No content found for query: \"${query}\"`;\n    }\n    \n    const textParts = [`${personaIndicator}🔍 **Search Results for \"${query}\"** (${items.length} found)\\n\\n`];\n    \n    items.forEach((item: any) => {\n      // Extract content type from path (library/personas/creative/writer.md -> personas)\n      const pathParts = item.path.split('/');\n      const contentType = pathParts[1] || 'content';\n      \n      const contentIcons: { [key: string]: string } = {\n        'personas': '🎭',\n        'skills': '🛠️',\n        'agents': '🤖',\n        'prompts': '💬',\n        'templates': '📄',\n        'tools': '🔧',\n        'ensembles': '🎼'\n      };\n      const icon = contentIcons[contentType] || '📄';\n      \n      textParts.push(\n        `   ${icon} **${item.name.replace('.md', '')}**\\n`,\n        `      📂 Path: ${item.path}\\n`,\n        `      📥 Install: \\`install_content \"${item.path}\"\\`\\n`,\n        `      👁️ Details: \\`get_collection_content \"${item.path}\"\\`\\n\\n`\n      );\n    });\n    \n    return textParts.join('');\n  }\n\n  /**\n   * Search from collection index with full featured search and pagination\n   */\n  private async searchFromIndex(query: string, options: SearchOptions): Promise<SearchResults> {\n    const index = await this.indexCache.getIndex();\n    const allEntries = this.flattenIndexEntries(index);\n    \n    // Filter by element type if specified\n    let filteredEntries = allEntries;\n    if (options.elementType) {\n      filteredEntries = allEntries.filter(entry => entry.type === options.elementType);\n    }\n    \n    // Filter by category if specified\n    if (options.category) {\n      filteredEntries = filteredEntries.filter(entry => entry.category === options.category);\n    }\n    \n    // Search matching\n    const matchedEntries = this.performIndexSearch(query, filteredEntries);\n    \n    // Sort results\n    const sortedEntries = this.sortSearchResults(matchedEntries, options.sortBy || 'relevance', query);\n    \n    // Apply pagination\n    const page = options.page || 1;\n    const pageSize = options.pageSize || 25;\n    const startIndex = (page - 1) * pageSize;\n    const endIndex = startIndex + pageSize;\n    const paginatedEntries = sortedEntries.slice(startIndex, endIndex);\n    \n    return {\n      items: paginatedEntries,\n      total: sortedEntries.length,\n      page,\n      pageSize,\n      hasMore: endIndex < sortedEntries.length,\n      query,\n      searchTime: 0 // Will be set by caller\n    };\n  }\n\n  /**\n   * Flatten index entries from all categories into a single array\n   */\n  private flattenIndexEntries(index: CollectionIndex): IndexEntry[] {\n    const entries: IndexEntry[] = [];\n    \n    for (const [elementType, typeEntries] of Object.entries(index.index)) {\n      entries.push(...typeEntries);\n    }\n    \n    return entries;\n  }\n\n  /**\n   * Perform search matching on index entries\n   */\n  private performIndexSearch(query: string, entries: IndexEntry[]): IndexEntry[] {\n    const normalizedQuery = normalizeSearchTerm(query);\n    const queryWords = normalizedQuery.split(/\\s+/).filter(word => word.length > 0);\n    \n    return entries.filter(entry => {\n      // Search in multiple fields\n      const searchableText = [\n        entry.name,\n        entry.description,\n        entry.path,\n        ...entry.tags\n      ].join(' ').toLowerCase();\n      \n      // Use existing search utilities for consistency\n      const nameMatch = isSearchMatch(query, entry.name);\n      const descMatch = isSearchMatch(query, entry.description);\n      const pathMatch = isSearchMatch(query, entry.path);\n      const tagMatch = entry.tags.some(tag => isSearchMatch(query, tag));\n      \n      return nameMatch || descMatch || pathMatch || tagMatch;\n    });\n  }\n\n  /**\n   * Sort search results by relevance, name, or date\n   */\n  private sortSearchResults(entries: IndexEntry[], sortBy: 'relevance' | 'name' | 'date', query: string): IndexEntry[] {\n    const sorted = [...entries];\n    \n    switch (sortBy) {\n      case 'name':\n        sorted.sort((a, b) => a.name.localeCompare(b.name));\n        break;\n      case 'date':\n        sorted.sort((a, b) => new Date(b.created).getTime() - new Date(a.created).getTime());\n        break;\n      case 'relevance':\n      default:\n        // Calculate relevance scores\n        sorted.sort((a, b) => {\n          const scoreA = this.calculateRelevanceScore(query, a);\n          const scoreB = this.calculateRelevanceScore(query, b);\n          return scoreB - scoreA;\n        });\n        break;\n    }\n    \n    return sorted;\n  }\n\n  /**\n   * Calculate relevance score for search results\n   */\n  private calculateRelevanceScore(query: string, entry: IndexEntry): number {\n    const normalizedQuery = normalizeSearchTerm(query);\n    let score = 0;\n    \n    // Exact name match gets highest score\n    if (normalizeSearchTerm(entry.name).includes(normalizedQuery)) {\n      score += 100;\n    }\n    \n    // Description match\n    if (normalizeSearchTerm(entry.description).includes(normalizedQuery)) {\n      score += 50;\n    }\n    \n    // Tag matches\n    const matchingTags = entry.tags.filter(tag => \n      normalizeSearchTerm(tag).includes(normalizedQuery)\n    );\n    score += matchingTags.length * 25;\n    \n    // Path match (lower priority)\n    if (normalizeSearchTerm(entry.path).includes(normalizedQuery)) {\n      score += 10;\n    }\n    \n    // Bonus for recent content\n    const daysSinceCreated = (Date.now() - new Date(entry.created).getTime()) / (1000 * 60 * 60 * 24);\n    if (daysSinceCreated < 30) {\n      score += 5;\n    }\n    \n    return score;\n  }\n\n  /**\n   * Convert legacy search results to new SearchResults format\n   */\n  private convertLegacyResults(legacyResults: any[], query: string, options: SearchOptions, searchTime: number): SearchResults {\n    // Convert GitHub API format to IndexEntry format\n    const entries: IndexEntry[] = legacyResults.map(item => ({\n      path: item.path,\n      type: this.extractTypeFromPath(item.path),\n      name: item.name?.replace('.md', '') || 'Unknown',\n      description: 'No description available',\n      version: '1.0.0',\n      author: 'Unknown',\n      tags: [],\n      sha: item.sha || '',\n      category: this.extractCategoryFromPath(item.path),\n      created: new Date().toISOString(),\n      license: 'Unknown'\n    }));\n    \n    // Apply pagination\n    const page = options.page || 1;\n    const pageSize = options.pageSize || 25;\n    const startIndex = (page - 1) * pageSize;\n    const endIndex = startIndex + pageSize;\n    const paginatedEntries = entries.slice(startIndex, endIndex);\n    \n    return {\n      items: paginatedEntries,\n      total: entries.length,\n      page,\n      pageSize,\n      hasMore: endIndex < entries.length,\n      query,\n      searchTime\n    };\n  }\n\n  /**\n   * Extract element type from file path\n   */\n  private extractTypeFromPath(path: string): string {\n    const parts = path.split('/');\n    if (parts.length >= 2 && parts[0] === 'library') {\n      return parts[1];\n    }\n    return 'unknown';\n  }\n\n  /**\n   * Extract category from file path\n   */\n  private extractCategoryFromPath(path: string): string {\n    const parts = path.split('/');\n    if (parts.length >= 3 && parts[0] === 'library') {\n      return parts[2];\n    }\n    return 'uncategorized';\n  }\n\n  /**\n   * Create empty search results for error cases\n   */\n  private createEmptySearchResults(query: string, options: SearchOptions): SearchResults {\n    return {\n      items: [],\n      total: 0,\n      page: options.page || 1,\n      pageSize: options.pageSize || 25,\n      hasMore: false,\n      query,\n      searchTime: 0\n    };\n  }\n\n  /**\n   * Enhanced format for search results with pagination info\n   */\n  formatSearchResultsWithPagination(results: SearchResults, personaIndicator: string = ''): string {\n    if (results.total === 0) {\n      return `${personaIndicator}🔍 No content found for query: \"${results.query}\"`;\n    }\n    \n    const startItem = (results.page - 1) * results.pageSize + 1;\n    const endItem = Math.min(results.page * results.pageSize, results.total);\n    \n    const textParts = [\n      `${personaIndicator}🔍 **Search Results for \"${results.query}\"**\\n`,\n      `📊 Showing ${startItem}-${endItem} of ${results.total} results (Page ${results.page})\\n`,\n      `⚡ Search time: ${results.searchTime}ms\\n\\n`\n    ];\n    \n    results.items.forEach((item: IndexEntry) => {\n      const contentIcons: { [key: string]: string } = {\n        'personas': '🎭',\n        'skills': '🛠️',\n        'agents': '🤖',\n        'prompts': '💬',\n        'templates': '📄',\n        'tools': '🔧',\n        'ensembles': '🎼',\n        'memories': '🧠'\n      };\n      const icon = contentIcons[item.type] || '📄';\n      \n      textParts.push(\n        `   ${icon} **${item.name}** (${item.type})\\n`,\n        `      📝 ${item.description}\\n`,\n        `      🏷️ Tags: ${item.tags.join(', ')}\\n`,\n        `      📂 Path: ${item.path}\\n`,\n        `      📥 Install: \\`install_content \"${item.path}\"\\`\\n`,\n        `      👁️ Details: \\`get_collection_content \"${item.path}\"\\`\\n\\n`\n      );\n    });\n    \n    // Add pagination info\n    if (results.hasMore) {\n      const nextPage = results.page + 1;\n      textParts.push(`📄 More results available. Use page ${nextPage} to see next ${results.pageSize} items.\\n`);\n    }\n    \n    return textParts.join('');\n  }\n\n  /**\n   * Get cache statistics for debugging\n   */\n  async getCacheStats(): Promise<any> {\n    const indexStats = this.indexCache.getCacheStats();\n    const cacheStats = await this.collectionCache.getCacheStats();\n    \n    return {\n      index: indexStats,\n      collection: cacheStats\n    };\n  }\n}"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"CollectionSeeder.d.ts","sourceRoot":"","sources":["../../src/collection/CollectionSeeder.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAE7D;;;GAGG;AACH,qBAAa,gBAAgB;IAE3B,OAAO,CAAC,MAAM,CAAC,cAAc,CAAiC;IAE9D;;;;OAIG;IACH,MAAM,CAAC,WAAW,IAAI,cAAc,EAAE;
|
|
1
|
+
{"version":3,"file":"CollectionSeeder.d.ts","sourceRoot":"","sources":["../../src/collection/CollectionSeeder.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAE7D;;;GAGG;AACH,qBAAa,gBAAgB;IAE3B,OAAO,CAAC,MAAM,CAAC,cAAc,CAAiC;IAE9D;;;;OAIG;IACH,MAAM,CAAC,WAAW,IAAI,cAAc,EAAE;IAoOtC;;OAEG;IACH,MAAM,CAAC,YAAY;;;;;;IAiBnB;;OAEG;IACH,MAAM,CAAC,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAI9C;;OAEG;IACH,MAAM,CAAC,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,GAAG,SAAS;CAG7D"}
|