@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -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;AAK9E,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,aAAa,CAAwC;gBAEjD,YAAY,EAAE,YAAY,EAAE,eAAe,CAAC,EAAE,eAAe;IAKzE;;;OAGG;IACG,gBAAgB,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;IAgCrD;;OAEG;YACW,eAAe;IA+B7B;;OAEG;IACH,OAAO,CAAC,cAAc;IActB;;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;CAiCxF"}
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
- logger.warn(`Invalid search query: ${error}`);
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
- logger.debug(`GitHub API search failed, falling back to cache: ${error}`);
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
- await this.collectionCache.saveCache(CollectionSeeder.getSeedData());
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
- logger.error(`Cache search failed: ${error}`);
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 normalizedQuery = normalizeSearchTerm(query);
83
- return seedData.filter(item => {
84
- const normalizedName = normalizeSearchTerm(item.name);
85
- const normalizedPath = normalizeSearchTerm(item.path);
86
- return normalizedName.includes(normalizedQuery) ||
87
- normalizedPath.includes(normalizedQuery);
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
- logger.debug(`Failed to update cache: ${error}`);
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":"AAAA;;GAEG;AAGH,OAAO,EAAE,eAAe,EAAkB,MAAM,6BAA6B,CAAC;AAC9E,OAAO,EAAE,oBAAoB,EAAE,MAAM,kCAAkC,CAAC;AACxE,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AACzD,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAE,mBAAmB,EAAE,mBAAmB,EAAE,aAAa,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AACtH,OAAO,EAAE,YAAY,EAAiB,MAAM,0BAA0B,CAAC;AAGvE,MAAM,OAAO,gBAAgB;IACnB,YAAY,CAAe;IAC3B,eAAe,CAAkB;IACjC,UAAU,CAAuB;IACjC,aAAa,GAAG,oCAAoC,CAAC;IAE7D,YAAY,YAA0B,EAAE,eAAiC;QACvE,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,eAAe,GAAG,eAAe,IAAI,IAAI,eAAe,EAAE,CAAC;QAChE,IAAI,CAAC,UAAU,GAAG,IAAI,oBAAoB,CAAC,YAAY,CAAC,CAAC;IAC3D,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,2BAA2B,CAAC,KAAa,EAAE,UAAyB,EAAE;QAC1E,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,MAAM,CAAC,KAAK,CAAC,oEAAoE,KAAK,GAAG,EAAE,OAAO,CAAC,CAAC;QAEpG,qCAAqC;QACrC,IAAI,CAAC;YACH,mBAAmB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC5E,MAAM,CAAC,KAAK,CAAC,iCAAiC,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAChF,YAAY,CAAC,QAAQ,CAAC,kDAAkD,EAAE,KAAK,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAC5F,OAAO,IAAI,CAAC,wBAAwB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;QACvD,CAAC;QAED,IAAI,CAAC;YACH,+BAA+B;YAC/B,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;YAChE,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YAE1C,MAAM,CAAC,KAAK,CAAC,6BAA6B,UAAU,WAAW,YAAY,CAAC,KAAK,CAAC,MAAM,UAAU,CAAC,CAAC;YACpG,OAAO,EAAE,GAAG,YAAY,EAAE,UAAU,EAAE,CAAC;QACzC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,qDAAqD,EAAE,KAAK,CAAC,CAAC;YAE3E,4BAA4B;YAC5B,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;YACzD,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;YAE1C,uCAAuC;YACvC,OAAO,IAAI,CAAC,oBAAoB,CAAC,aAAa,EAAE,KAAK,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;QAC9E,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,gBAAgB,CAAC,KAAa;QAClC,MAAM,CAAC,KAAK,CAAC,yDAAyD,KAAK,GAAG,CAAC,CAAC;QAEhF,qCAAqC;QACrC,IAAI,CAAC;YACH,mBAAmB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC5E,MAAM,CAAC,KAAK,CAAC,iCAAiC,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC,CAAC;YAChF,YAAY,CAAC,QAAQ,CAAC,uCAAuC,EAAE,KAAK,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YACjF,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,IAAI,CAAC;YACH,gDAAgD;YAChD,MAAM,SAAS,GAAG,GAAG,IAAI,CAAC,aAAa,MAAM,kBAAkB,CAAC,KAAK,CAAC,yDAAyD,CAAC;YAChI,MAAM,CAAC,KAAK,CAAC,0CAA0C,SAAS,EAAE,CAAC,CAAC;YACpE,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC,CAAC,gCAAgC;YAExG,IAAI,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC5C,MAAM,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,KAAK,CAAC,MAAM,8BAA8B,CAAC,CAAC;gBAEvE,wCAAwC;gBACxC,MAAM,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAElD,OAAO,IAAI,CAAC,KAAK,CAAC;YACpB,CAAC;YAED,MAAM,CAAC,KAAK,CAAC,4DAA4D,CAAC,CAAC;YAC3E,OAAO,EAAE,CAAC;QACZ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC5E,MAAM,CAAC,KAAK,CAAC,6BAA6B,YAAY,kCAAkC,CAAC,CAAC;YAC1F,YAAY,CAAC,QAAQ,CAAC,mCAAmC,EAAE,KAAK,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAE7E,4BAA4B;YAC5B,OAAO,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;QACrC,CAAC;IACH,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,eAAe,CAAC,KAAa;QACzC,MAAM,CAAC,KAAK,CAAC,+BAA+B,KAAK,GAAG,CAAC,CAAC;QAEtD,IAAI,CAAC;YACH,+BAA+B;YAC/B,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAElE,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC3B,MAAM,CAAC,KAAK,CAAC,SAAS,WAAW,CAAC,MAAM,mBAAmB,CAAC,CAAC;gBAC7D,OAAO,IAAI,CAAC,+BAA+B,CAAC,WAAW,CAAC,CAAC;YAC3D,CAAC;YAED,MAAM,CAAC,KAAK,CAAC,oDAAoD,CAAC,CAAC;YAEnE,iDAAiD;YACjD,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;YAC7C,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACzB,MAAM,CAAC,KAAK,CAAC,SAAS,SAAS,CAAC,MAAM,uBAAuB,CAAC,CAAC;gBAC/D,yCAAyC;gBACzC,IAAI,CAAC;oBACH,MAAM,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,CAAC;oBACrE,MAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;gBAC3C,CAAC;gBAAC,OAAO,UAAU,EAAE,CAAC;oBACpB,MAAM,iBAAiB,GAAG,UAAU,YAAY,KAAK,CAAC,CAAC,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;oBAChG,MAAM,CAAC,KAAK,CAAC,sCAAsC,iBAAiB,EAAE,CAAC,CAAC;gBAC1E,CAAC;gBACD,OAAO,IAAI,CAAC,+BAA+B,CAAC,SAAS,CAAC,CAAC;YACzD,CAAC;YAED,MAAM,CAAC,KAAK,CAAC,sCAAsC,CAAC,CAAC;YACrD,OAAO,EAAE,CAAC;QACZ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC5E,MAAM,CAAC,KAAK,CAAC,wBAAwB,YAAY,EAAE,CAAC,CAAC;YACrD,YAAY,CAAC,QAAQ,CAAC,+BAA+B,EAAE,KAAK,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAEzE,8CAA8C;YAC9C,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;YAC7C,MAAM,CAAC,KAAK,CAAC,+BAA+B,SAAS,CAAC,MAAM,QAAQ,CAAC,CAAC;YACtE,OAAO,IAAI,CAAC,+BAA+B,CAAC,SAAS,CAAC,CAAC;QACzD,CAAC;IACH,CAAC;IAED;;OAEG;IACK,cAAc,CAAC,KAAa;QAClC,MAAM,QAAQ,GAAG,gBAAgB,CAAC,WAAW,EAAE,CAAC;QAChD,MAAM,SAAS,GAAG,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAC5C,MAAM,CAAC,KAAK,CAAC,oCAAoC,SAAS,CAAC,QAAQ,mBAAmB,SAAS,CAAC,UAAU,gBAAgB,SAAS,CAAC,YAAY,GAAG,CAAC,CAAC;QACrJ,MAAM,CAAC,KAAK,CAAC,qBAAqB,QAAQ,CAAC,MAAM,aAAa,CAAC,CAAC;QAEhE,MAAM,OAAO,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE;YACrC,oEAAoE;YACpE,MAAM,WAAW,GAAG,aAAa,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YACpD,MAAM,WAAW,GAAG,aAAa,CAAC,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YAEpD,MAAM,OAAO,GAAG,WAAW,IAAI,WAAW,CAAC;YAE3C,IAAI,OAAO,EAAE,CAAC;gBACZ,MAAM,CAAC,KAAK,CAAC,kBAAkB,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,oBAAoB,KAAK,GAAG,CAAC,CAAC;YACtF,CAAC;YAED,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CAAC;QAEH,gDAAgD;QAChD,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACzB,MAAM,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC;YACvD,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBACnC,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,CAAC,IAAI,KAAK,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;YACH,IAAI,QAAQ,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;gBACzB,MAAM,CAAC,KAAK,CAAC,aAAa,QAAQ,CAAC,MAAM,GAAG,EAAE,aAAa,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;QAED,MAAM,CAAC,KAAK,CAAC,SAAS,OAAO,CAAC,MAAM,uBAAuB,CAAC,CAAC;QAC7D,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;OAEG;IACK,UAAU,CAAC,IAAY,EAAE,MAAc;QAC7C,mFAAmF;QACnF,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QACnC,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC;QAEtC,IAAI,SAAS,GAAG,CAAC,CAAC;QAClB,IAAI,WAAW,GAAG,CAAC,CAAC;QAEpB,OAAO,SAAS,GAAG,IAAI,CAAC,MAAM,IAAI,WAAW,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;YAC9D,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;gBAC5C,SAAS,EAAE,CAAC;YACd,CAAC;YACD,WAAW,EAAE,CAAC;QAChB,CAAC;QAED,OAAO,SAAS,KAAK,IAAI,CAAC,MAAM,CAAC;IACnC,CAAC;IAGD;;OAEG;IACK,+BAA+B,CAAC,UAA4B;QAClE,OAAO,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAC7B,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,GAAG,EAAE,iEAAiE,IAAI,CAAC,IAAI,EAAE;YACjF,QAAQ,EAAE,wDAAwD,IAAI,CAAC,IAAI,EAAE;YAC7E,UAAU,EAAE;gBACV,IAAI,EAAE,YAAY;gBAClB,SAAS,EAAE,yBAAyB;aACrC;SACF,CAAC,CAAC,CAAC;IACN,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,0BAA0B,CAAC,WAAkB;QACzD,IAAI,CAAC;YACH,MAAM,UAAU,GAAqB,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAC5D,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACxC,CAAC,CAAC,CAAC;YAEJ,MAAM,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;YACjD,MAAM,CAAC,KAAK,CAAC,sBAAsB,UAAU,CAAC,MAAM,wBAAwB,CAAC,CAAC;QAChF,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,YAAY,CAAC,QAAQ,CAAC,0CAA0C,EAAE,KAAK,CAAC,CAAC;YACzE,oEAAoE;QACtE,CAAC;IACH,CAAC;IAED;;OAEG;IACH,mBAAmB,CAAC,KAAY,EAAE,KAAa,EAAE,mBAA2B,EAAE;QAC5E,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvB,OAAO,GAAG,gBAAgB,mCAAmC,KAAK,GAAG,CAAC;QACxE,CAAC;QAED,MAAM,SAAS,GAAG,CAAC,GAAG,gBAAgB,4BAA4B,KAAK,QAAQ,KAAK,CAAC,MAAM,aAAa,CAAC,CAAC;QAE1G,KAAK,CAAC,OAAO,CAAC,CAAC,IAAS,EAAE,EAAE;YAC1B,mFAAmF;YACnF,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACvC,MAAM,WAAW,GAAG,SAAS,CAAC,CAAC,CAAC,IAAI,SAAS,CAAC;YAE9C,MAAM,YAAY,GAA8B;gBAC9C,UAAU,EAAE,IAAI;gBAChB,QAAQ,EAAE,KAAK;gBACf,QAAQ,EAAE,IAAI;gBACd,SAAS,EAAE,IAAI;gBACf,WAAW,EAAE,IAAI;gBACjB,OAAO,EAAE,IAAI;gBACb,WAAW,EAAE,IAAI;aAClB,CAAC;YACF,MAAM,IAAI,GAAG,YAAY,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC;YAE/C,SAAS,CAAC,IAAI,CACZ,MAAM,IAAI,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,EAClD,kBAAkB,IAAI,CAAC,IAAI,IAAI,EAC/B,wCAAwC,IAAI,CAAC,IAAI,OAAO,EACxD,gDAAgD,IAAI,CAAC,IAAI,SAAS,CACnE,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,OAAO,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,eAAe,CAAC,KAAa,EAAE,OAAsB;QACjE,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;QAC/C,MAAM,UAAU,GAAG,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;QAEnD,sCAAsC;QACtC,IAAI,eAAe,GAAG,UAAU,CAAC;QACjC,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;YACxB,eAAe,GAAG,UAAU,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,KAAK,OAAO,CAAC,WAAW,CAAC,CAAC;QACnF,CAAC;QAED,kCAAkC;QAClC,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrB,eAAe,GAAG,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,QAAQ,KAAK,OAAO,CAAC,QAAQ,CAAC,CAAC;QACzF,CAAC;QAED,kBAAkB;QAClB,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,eAAe,CAAC,CAAC;QAEvE,eAAe;QACf,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,cAAc,EAAE,OAAO,CAAC,MAAM,IAAI,WAAW,EAAE,KAAK,CAAC,CAAC;QAEnG,mBAAmB;QACnB,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QAC/B,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,EAAE,CAAC;QACxC,MAAM,UAAU,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,QAAQ,CAAC;QACzC,MAAM,QAAQ,GAAG,UAAU,GAAG,QAAQ,CAAC;QACvC,MAAM,gBAAgB,GAAG,aAAa,CAAC,KAAK,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QAEnE,OAAO;YACL,KAAK,EAAE,gBAAgB;YACvB,KAAK,EAAE,aAAa,CAAC,MAAM;YAC3B,IAAI;YACJ,QAAQ;YACR,OAAO,EAAE,QAAQ,GAAG,aAAa,CAAC,MAAM;YACxC,KAAK;YACL,UAAU,EAAE,CAAC,CAAC,wBAAwB;SACvC,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,mBAAmB,CAAC,KAAsB;QAChD,MAAM,OAAO,GAAiB,EAAE,CAAC;QAEjC,KAAK,MAAM,CAAC,WAAW,EAAE,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC;YACrE,OAAO,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,CAAC;QAC/B,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;OAEG;IACK,kBAAkB,CAAC,KAAa,EAAE,OAAqB;QAC7D,MAAM,eAAe,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;QACnD,MAAM,UAAU,GAAG,eAAe,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAEhF,OAAO,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;YAC5B,4BAA4B;YAC5B,MAAM,cAAc,GAAG;gBACrB,KAAK,CAAC,IAAI;gBACV,KAAK,CAAC,WAAW;gBACjB,KAAK,CAAC,IAAI;gBACV,GAAG,KAAK,CAAC,IAAI;aACd,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC;YAE1B,gDAAgD;YAChD,MAAM,SAAS,GAAG,aAAa,CAAC,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;YACnD,MAAM,SAAS,GAAG,aAAa,CAAC,KAAK,EAAE,KAAK,CAAC,WAAW,CAAC,CAAC;YAC1D,MAAM,SAAS,GAAG,aAAa,CAAC,KAAK,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;YACnD,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,CAAC;YAEnE,OAAO,SAAS,IAAI,SAAS,IAAI,SAAS,IAAI,QAAQ,CAAC;QACzD,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,iBAAiB,CAAC,OAAqB,EAAE,MAAqC,EAAE,KAAa;QACnG,MAAM,MAAM,GAAG,CAAC,GAAG,OAAO,CAAC,CAAC;QAE5B,QAAQ,MAAM,EAAE,CAAC;YACf,KAAK,MAAM;gBACT,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;gBACpD,MAAM;YACR,KAAK,MAAM;gBACT,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;gBACrF,MAAM;YACR,KAAK,WAAW,CAAC;YACjB;gBACE,6BAA6B;gBAC7B,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;oBACnB,MAAM,MAAM,GAAG,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;oBACtD,MAAM,MAAM,GAAG,IAAI,CAAC,uBAAuB,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;oBACtD,OAAO,MAAM,GAAG,MAAM,CAAC;gBACzB,CAAC,CAAC,CAAC;gBACH,MAAM;QACV,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,uBAAuB,CAAC,KAAa,EAAE,KAAiB;QAC9D,MAAM,eAAe,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;QACnD,IAAI,KAAK,GAAG,CAAC,CAAC;QAEd,sCAAsC;QACtC,IAAI,mBAAmB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;YAC9D,KAAK,IAAI,GAAG,CAAC;QACf,CAAC;QAED,oBAAoB;QACpB,IAAI,mBAAmB,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;YACrE,KAAK,IAAI,EAAE,CAAC;QACd,CAAC;QAED,cAAc;QACd,MAAM,YAAY,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAC3C,mBAAmB,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,eAAe,CAAC,CACnD,CAAC;QACF,KAAK,IAAI,YAAY,CAAC,MAAM,GAAG,EAAE,CAAC;QAElC,8BAA8B;QAC9B,IAAI,mBAAmB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE,CAAC;YAC9D,KAAK,IAAI,EAAE,CAAC;QACd,CAAC;QAED,2BAA2B;QAC3B,MAAM,gBAAgB,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QAClG,IAAI,gBAAgB,GAAG,EAAE,EAAE,CAAC;YAC1B,KAAK,IAAI,CAAC,CAAC;QACb,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACK,oBAAoB,CAAC,aAAoB,EAAE,KAAa,EAAE,OAAsB,EAAE,UAAkB;QAC1G,iDAAiD;QACjD,MAAM,OAAO,GAAiB,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACvD,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,IAAI,EAAE,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC;YACzC,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,SAAS;YAChD,WAAW,EAAE,0BAA0B;YACvC,OAAO,EAAE,OAAO;YAChB,MAAM,EAAE,SAAS;YACjB,IAAI,EAAE,EAAE;YACR,GAAG,EAAE,IAAI,CAAC,GAAG,IAAI,EAAE;YACnB,QAAQ,EAAE,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,IAAI,CAAC;YACjD,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACjC,OAAO,EAAE,SAAS;SACnB,CAAC,CAAC,CAAC;QAEJ,mBAAmB;QACnB,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QAC/B,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,EAAE,CAAC;QACxC,MAAM,UAAU,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,QAAQ,CAAC;QACzC,MAAM,QAAQ,GAAG,UAAU,GAAG,QAAQ,CAAC;QACvC,MAAM,gBAAgB,GAAG,OAAO,CAAC,KAAK,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QAE7D,OAAO;YACL,KAAK,EAAE,gBAAgB;YACvB,KAAK,EAAE,OAAO,CAAC,MAAM;YACrB,IAAI;YACJ,QAAQ;YACR,OAAO,EAAE,QAAQ,GAAG,OAAO,CAAC,MAAM;YAClC,KAAK;YACL,UAAU;SACX,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,mBAAmB,CAAC,IAAY;QACtC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC9B,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,SAAS,EAAE,CAAC;YAChD,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;OAEG;IACK,uBAAuB,CAAC,IAAY;QAC1C,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAC9B,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,KAAK,SAAS,EAAE,CAAC;YAChD,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QACD,OAAO,eAAe,CAAC;IACzB,CAAC;IAED;;OAEG;IACK,wBAAwB,CAAC,KAAa,EAAE,OAAsB;QACpE,OAAO;YACL,KAAK,EAAE,EAAE;YACT,KAAK,EAAE,CAAC;YACR,IAAI,EAAE,OAAO,CAAC,IAAI,IAAI,CAAC;YACvB,QAAQ,EAAE,OAAO,CAAC,QAAQ,IAAI,EAAE;YAChC,OAAO,EAAE,KAAK;YACd,KAAK;YACL,UAAU,EAAE,CAAC;SACd,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,iCAAiC,CAAC,OAAsB,EAAE,mBAA2B,EAAE;QACrF,IAAI,OAAO,CAAC,KAAK,KAAK,CAAC,EAAE,CAAC;YACxB,OAAO,GAAG,gBAAgB,mCAAmC,OAAO,CAAC,KAAK,GAAG,CAAC;QAChF,CAAC;QAED,MAAM,SAAS,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,CAAC,GAAG,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;QAC5D,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,OAAO,CAAC,QAAQ,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;QAEzE,MAAM,SAAS,GAAG;YAChB,GAAG,gBAAgB,4BAA4B,OAAO,CAAC,KAAK,OAAO;YACnE,cAAc,SAAS,IAAI,OAAO,OAAO,OAAO,CAAC,KAAK,kBAAkB,OAAO,CAAC,IAAI,KAAK;YACzF,kBAAkB,OAAO,CAAC,UAAU,QAAQ;SAC7C,CAAC;QAEF,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAgB,EAAE,EAAE;YACzC,MAAM,YAAY,GAA8B;gBAC9C,UAAU,EAAE,IAAI;gBAChB,QAAQ,EAAE,KAAK;gBACf,QAAQ,EAAE,IAAI;gBACd,SAAS,EAAE,IAAI;gBACf,WAAW,EAAE,IAAI;gBACjB,OAAO,EAAE,IAAI;gBACb,WAAW,EAAE,IAAI;gBACjB,UAAU,EAAE,IAAI;aACjB,CAAC;YACF,MAAM,IAAI,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;YAE7C,SAAS,CAAC,IAAI,CACZ,MAAM,IAAI,MAAM,IAAI,CAAC,IAAI,OAAO,IAAI,CAAC,IAAI,KAAK,EAC9C,YAAY,IAAI,CAAC,WAAW,IAAI,EAChC,mBAAmB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAC3C,kBAAkB,IAAI,CAAC,IAAI,IAAI,EAC/B,wCAAwC,IAAI,CAAC,IAAI,OAAO,EACxD,gDAAgD,IAAI,CAAC,IAAI,SAAS,CACnE,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,sBAAsB;QACtB,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,GAAG,CAAC,CAAC;YAClC,SAAS,CAAC,IAAI,CAAC,uCAAuC,QAAQ,gBAAgB,OAAO,CAAC,QAAQ,WAAW,CAAC,CAAC;QAC7G,CAAC;QAED,OAAO,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa;QACjB,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC;QACnD,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,CAAC;QAE9D,OAAO;YACL,KAAK,EAAE,UAAU;YACjB,UAAU,EAAE,UAAU;SACvB,CAAC;IACJ,CAAC;CACF","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;IA8LtC;;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"}
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"}