@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -0,0 +1,1523 @@
1
+ /**
2
+ * Tool for submitting content to GitHub portfolio repositories
3
+ * Replaces the broken issue-based submission with direct repository saves
4
+ *
5
+ * FIXES IMPLEMENTED (PR #503):
6
+ * 1. TYPE SAFETY FIX #1 (Issue #497): Changed apiCache from 'any' to proper APICache type
7
+ * 2. TYPE SAFETY FIX #2 (Issue #497): Replaced complex type casting with PortfolioElementAdapter
8
+ * 3. PERFORMANCE (PR #496 recommendation): Using FileDiscoveryUtil for optimized file search
9
+ */
10
+ import { GitHubAuthManager } from '../../auth/GitHubAuthManager.js';
11
+ import { PortfolioRepoManager } from '../../portfolio/PortfolioRepoManager.js';
12
+ import { TokenManager } from '../../security/tokenManager.js';
13
+ import { ContentValidator } from '../../security/contentValidator.js';
14
+ import { PortfolioManager } from '../../portfolio/PortfolioManager.js';
15
+ import { PortfolioIndexManager } from '../../portfolio/PortfolioIndexManager.js';
16
+ import { ElementType } from '../../portfolio/types.js';
17
+ import { logger } from '../../utils/logger.js';
18
+ import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
19
+ import { SecurityMonitor } from '../../security/securityMonitor.js';
20
+ import { PortfolioElementAdapter } from './PortfolioElementAdapter.js';
21
+ import { FileDiscoveryUtil } from '../../utils/FileDiscoveryUtil.js';
22
+ import { ErrorHandler } from '../../utils/ErrorHandler.js';
23
+ import { FILE_SIZE_LIMITS, RETRY_CONFIG, SEARCH_CONFIG, getValidatedTimeout, calculateRetryDelay } from '../../config/portfolio-constants.js';
24
+ import { githubRateLimiter } from '../../utils/GitHubRateLimiter.js';
25
+ import { EarlyTerminationSearch } from '../../utils/EarlyTerminationSearch.js';
26
+ import * as path from 'path';
27
+ import * as fs from 'fs/promises';
28
+ export class SubmitToPortfolioTool {
29
+ authManager;
30
+ portfolioManager;
31
+ contentValidator;
32
+ constructor(apiCache) {
33
+ // TYPE SAFETY FIX #1: Proper typing for apiCache parameter
34
+ // Previously: constructor(apiCache: any)
35
+ // Now: constructor(apiCache: APICache) with proper import
36
+ this.authManager = new GitHubAuthManager(apiCache);
37
+ this.portfolioManager = new PortfolioRepoManager();
38
+ this.contentValidator = new ContentValidator();
39
+ }
40
+ /**
41
+ * Validates and normalizes input parameters to prevent Unicode attacks and ensure data safety
42
+ * @param params The input parameters from the user
43
+ * @returns Validation result with normalized name or error response
44
+ */
45
+ async validateAndNormalizeParams(params) {
46
+ // Normalize user input to prevent Unicode attacks (DMCP-SEC-004)
47
+ const normalizedName = UnicodeValidator.normalize(params.name);
48
+ if (!normalizedName.isValid) {
49
+ SecurityMonitor.logSecurityEvent({
50
+ type: 'UNICODE_VALIDATION_ERROR',
51
+ severity: 'MEDIUM',
52
+ source: 'SubmitToPortfolioTool.execute',
53
+ details: `Invalid Unicode in element name: ${normalizedName.detectedIssues?.[0] || 'unknown error'}`
54
+ });
55
+ return {
56
+ success: false,
57
+ error: {
58
+ success: false,
59
+ message: `Invalid characters in element name: ${normalizedName.detectedIssues?.[0] || 'unknown error'}`,
60
+ error: 'INVALID_INPUT'
61
+ }
62
+ };
63
+ }
64
+ return {
65
+ success: true,
66
+ safeName: normalizedName.normalizedContent
67
+ };
68
+ }
69
+ /**
70
+ * Checks if the user is authenticated with GitHub
71
+ * @returns Authentication check result with status or error response
72
+ */
73
+ async checkAuthentication() {
74
+ const authStatus = await this.authManager.getAuthStatus();
75
+ if (!authStatus.isAuthenticated) {
76
+ // Log authentication required (using existing event type)
77
+ logger.warn('User attempted portfolio submission without authentication');
78
+ return {
79
+ success: false,
80
+ error: {
81
+ success: false,
82
+ message: 'Not authenticated. Please authenticate first using the GitHub OAuth flow.\n\n' +
83
+ 'Visit: https://docs.anthropic.com/en/docs/claude-code/oauth-setup\n' +
84
+ 'Or run: gh auth login --web',
85
+ error: 'NOT_AUTHENTICATED'
86
+ }
87
+ };
88
+ }
89
+ return {
90
+ success: true,
91
+ authStatus
92
+ };
93
+ }
94
+ /**
95
+ * Discovers content locally with smart type detection
96
+ * @param safeName The normalized name to search for
97
+ * @param explicitType Optional explicit element type provided by user
98
+ * @param originalName Original user-provided name for error messages
99
+ * @returns Content discovery result with element type and path or error response
100
+ */
101
+ async discoverContentWithTypeDetection(safeName, explicitType, originalName) {
102
+ let elementType = explicitType;
103
+ let localPath = null;
104
+ if (elementType) {
105
+ // Type explicitly provided - search in that specific directory only
106
+ localPath = await this.findLocalContent(safeName, elementType);
107
+ if (!localPath) {
108
+ // UX IMPROVEMENT: Provide helpful suggestions for finding content
109
+ const portfolioManager = PortfolioManager.getInstance();
110
+ const elementDir = portfolioManager.getElementDir(elementType);
111
+ return {
112
+ success: false,
113
+ error: {
114
+ success: false,
115
+ message: `Could not find ${elementType} named "${originalName || safeName}" in local portfolio.\n\n` +
116
+ `**Searched in**: ${elementDir}\n\n` +
117
+ `**Troubleshooting Tips**:\n` +
118
+ `• Check if the file exists using your file explorer\n` +
119
+ `• Try using the exact filename (without extension)\n` +
120
+ `• Use \`list_portfolio\` to see all available ${elementType}\n` +
121
+ `• If unsure of the type, omit --type and let the system detect it\n\n` +
122
+ `**Common name formats that work**:\n` +
123
+ `• "my-element" (kebab-case)\n` +
124
+ `• "My Element" (with spaces)\n` +
125
+ `• "MyElement" (PascalCase)\n` +
126
+ `• Partial matches are supported`,
127
+ error: 'CONTENT_NOT_FOUND'
128
+ }
129
+ };
130
+ }
131
+ }
132
+ else {
133
+ // CRITICAL FIX: No type provided - implement smart detection across ALL element types
134
+ // This prevents the previous hardcoded default to PERSONA and enables proper type detection
135
+ const detectionResult = await this.detectElementType(safeName);
136
+ if (!detectionResult.found) {
137
+ // UX IMPROVEMENT: Enhanced guidance with specific suggestions
138
+ const availableTypes = Object.values(ElementType).join(', ');
139
+ // Get suggestions for similar names
140
+ const suggestions = await this.generateNameSuggestions(safeName);
141
+ let message = `Content "${originalName || safeName}" not found in portfolio.\n\n`;
142
+ message += `🔍 **Searched in all element types**: ${availableTypes}\n\n`;
143
+ if (suggestions.length > 0) {
144
+ message += `💡 **Did you mean one of these?**\n`;
145
+ for (const suggestion of suggestions.slice(0, SEARCH_CONFIG.MAX_SUGGESTIONS)) {
146
+ message += ` • "${suggestion.name}" (${suggestion.type})\n`;
147
+ }
148
+ message += `\n`;
149
+ }
150
+ message += `🛠️ **Troubleshooting Steps**:\n`;
151
+ message += `1. 📝 Use \`list_portfolio\` to see all available content\n`;
152
+ message += `2. 🔍 Check exact spelling and try variations:\n`;
153
+ message += ` • "${(originalName || safeName).toLowerCase()}" (lowercase)\n`;
154
+ message += ` • "${(originalName || safeName).replace(/[^a-z0-9]/gi, '-').toLowerCase()}" (normalized)\n`;
155
+ if ((originalName || safeName).includes('.')) {
156
+ message += ` • "${(originalName || safeName).replace(/\./g, '')}" (no dots)\n`;
157
+ }
158
+ message += `3. 🎯 Specify element type: \`submit_content "${originalName || safeName}" --type=personas\`\n`;
159
+ message += `4. 📁 Check if file exists in portfolio directories\n\n`;
160
+ message += `📝 **Tip**: The system searches filenames AND metadata names with fuzzy matching.`;
161
+ return {
162
+ success: false,
163
+ error: {
164
+ success: false,
165
+ message,
166
+ error: 'CONTENT_NOT_FOUND'
167
+ }
168
+ };
169
+ }
170
+ if (detectionResult.matches.length > 1) {
171
+ // Multiple matches found - ask user to specify type
172
+ const matchDetails = detectionResult.matches.map(m => `- ${m.type}: ${m.path}`).join('\n');
173
+ return {
174
+ success: false,
175
+ error: {
176
+ success: false,
177
+ message: `Content "${originalName || safeName}" found in multiple element types:\n\n${matchDetails}\n\n` +
178
+ `Please specify the element type using the --type parameter to avoid ambiguity.`,
179
+ error: 'MULTIPLE_MATCHES_FOUND'
180
+ }
181
+ };
182
+ }
183
+ // Single match found - use it
184
+ const match = detectionResult.matches[0];
185
+ elementType = match.type;
186
+ localPath = match.path;
187
+ logger.info(`Smart detection: Found "${safeName}" as ${elementType}`, {
188
+ name: safeName,
189
+ detectedType: elementType,
190
+ path: localPath
191
+ });
192
+ }
193
+ return {
194
+ success: true,
195
+ elementType,
196
+ localPath
197
+ };
198
+ }
199
+ /**
200
+ * Validates file size and content security before processing
201
+ * @param localPath Path to the local file to validate
202
+ * @returns Validation result with content or error response
203
+ */
204
+ async validateFileAndContent(localPath) {
205
+ // SECURITY ENHANCEMENT (Task #7): Validate file path before processing
206
+ const pathValidation = await this.validatePortfolioPath(localPath);
207
+ if (!pathValidation.isValid) {
208
+ return {
209
+ success: false,
210
+ error: pathValidation.error
211
+ };
212
+ }
213
+ // Use the validated safe path for all subsequent operations
214
+ const safePath = pathValidation.safePath;
215
+ // Validate file size before reading
216
+ const stats = await fs.stat(safePath);
217
+ if (stats.size > FILE_SIZE_LIMITS.MAX_FILE_SIZE) {
218
+ SecurityMonitor.logSecurityEvent({
219
+ type: 'RATE_LIMIT_EXCEEDED',
220
+ severity: 'MEDIUM',
221
+ source: 'SubmitToPortfolioTool.execute',
222
+ details: `File size ${stats.size} exceeds limit of ${FILE_SIZE_LIMITS.MAX_FILE_SIZE}`
223
+ });
224
+ return {
225
+ success: false,
226
+ error: {
227
+ success: false,
228
+ message: `File size exceeds ${FILE_SIZE_LIMITS.MAX_FILE_SIZE_MB}MB limit`,
229
+ error: 'FILE_TOO_LARGE'
230
+ }
231
+ };
232
+ }
233
+ // Validate content security
234
+ const content = await fs.readFile(safePath, 'utf-8');
235
+ const validationResult = ContentValidator.validateAndSanitize(content);
236
+ if (!validationResult.isValid && validationResult.severity === 'critical') {
237
+ SecurityMonitor.logSecurityEvent({
238
+ type: 'CONTENT_INJECTION_ATTEMPT',
239
+ severity: 'HIGH',
240
+ source: 'SubmitToPortfolioTool.execute',
241
+ details: `Critical security issues detected: ${validationResult.detectedPatterns?.join(', ')}`
242
+ });
243
+ return {
244
+ success: false,
245
+ error: {
246
+ success: false,
247
+ message: `Content validation failed: ${validationResult.detectedPatterns?.join(', ')}`,
248
+ error: 'VALIDATION_FAILED'
249
+ }
250
+ };
251
+ }
252
+ return {
253
+ success: true,
254
+ content
255
+ };
256
+ }
257
+ /**
258
+ * Prepares metadata for the portfolio element
259
+ * @param safeName The normalized name of the element
260
+ * @param elementType The type of the element
261
+ * @param authStatus Authentication status containing username
262
+ * @returns Metadata object for the element
263
+ */
264
+ prepareElementMetadata(safeName, elementType, authStatus) {
265
+ return {
266
+ name: safeName,
267
+ description: `${elementType} submitted from local portfolio`,
268
+ author: authStatus.username || 'unknown',
269
+ created: new Date().toISOString(),
270
+ updated: new Date().toISOString(),
271
+ version: '1.0.0'
272
+ };
273
+ }
274
+ /**
275
+ * Validates GitHub token and checks for expiration before usage
276
+ * SECURITY ENHANCEMENT (Task #5): Token expiration validation to prevent stale token usage
277
+ * @param token The GitHub token to validate
278
+ * @returns Validation result with status and expiration info
279
+ */
280
+ async validateTokenBeforeUsage(token) {
281
+ try {
282
+ // Check token format first (basic validation)
283
+ if (!TokenManager.validateTokenFormat(token)) {
284
+ SecurityMonitor.logSecurityEvent({
285
+ type: 'TOKEN_VALIDATION_FAILURE',
286
+ severity: 'MEDIUM',
287
+ source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',
288
+ details: 'Token has invalid format'
289
+ });
290
+ return {
291
+ isValid: false,
292
+ error: {
293
+ success: false,
294
+ message: 'Invalid token format. Please re-authenticate.',
295
+ error: 'INVALID_TOKEN_FORMAT'
296
+ }
297
+ };
298
+ }
299
+ // Validate token with GitHub API to check expiration and permissions
300
+ const validationResult = await TokenManager.validateTokenScopes(token, {
301
+ required: ['repo'],
302
+ optional: ['user:email']
303
+ });
304
+ if (!validationResult.isValid) {
305
+ SecurityMonitor.logSecurityEvent({
306
+ type: 'TOKEN_VALIDATION_FAILURE',
307
+ severity: 'MEDIUM',
308
+ source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',
309
+ details: `Token validation failed: ${validationResult.error}`
310
+ });
311
+ return {
312
+ isValid: false,
313
+ error: {
314
+ success: false,
315
+ message: 'GitHub token is invalid or expired. Please re-authenticate.',
316
+ error: 'TOKEN_VALIDATION_FAILED'
317
+ }
318
+ };
319
+ }
320
+ // Check if token is near expiration (rate limit reset time can indicate token freshness)
321
+ let isNearExpiry = false;
322
+ if (validationResult.rateLimit?.resetTime) {
323
+ const now = new Date();
324
+ const timeUntilReset = validationResult.rateLimit.resetTime.getTime() - now.getTime();
325
+ const oneHour = 60 * 60 * 1000;
326
+ // Consider token "near expiry" if rate limit reset is more than 23 hours away
327
+ // (GitHub rate limits reset every hour, so this suggests token age)
328
+ if (timeUntilReset > 23 * oneHour) {
329
+ isNearExpiry = true;
330
+ logger.warn('GitHub token may be near expiration', {
331
+ tokenPrefix: TokenManager.getTokenPrefix(token),
332
+ rateLimitResetTime: validationResult.rateLimit.resetTime,
333
+ recommendation: 'Consider re-authenticating for long operations'
334
+ });
335
+ }
336
+ }
337
+ // Log successful validation
338
+ SecurityMonitor.logSecurityEvent({
339
+ type: 'TOKEN_VALIDATION_SUCCESS',
340
+ severity: 'LOW',
341
+ source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',
342
+ details: 'GitHub token validated successfully before usage',
343
+ metadata: {
344
+ tokenType: TokenManager.getTokenType(token),
345
+ scopes: validationResult.scopes,
346
+ rateLimitRemaining: validationResult.rateLimit?.remaining,
347
+ isNearExpiry
348
+ }
349
+ });
350
+ return {
351
+ isValid: true,
352
+ isNearExpiry
353
+ };
354
+ }
355
+ catch (error) {
356
+ // Handle rate limit exceeded specifically
357
+ if (error?.code === 'RATE_LIMIT_EXCEEDED') {
358
+ logger.warn('Token validation rate limited, allowing operation to proceed with cached status');
359
+ return { isValid: true }; // Allow to proceed if rate limited, as basic format check passed
360
+ }
361
+ SecurityMonitor.logSecurityEvent({
362
+ type: 'TOKEN_VALIDATION_FAILURE',
363
+ severity: 'HIGH',
364
+ source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',
365
+ details: `Token validation error: ${error.message || 'unknown error'}`
366
+ });
367
+ return {
368
+ isValid: false,
369
+ error: {
370
+ success: false,
371
+ message: 'Unable to validate GitHub token. Please check your connection and try again.',
372
+ error: 'TOKEN_VALIDATION_ERROR'
373
+ }
374
+ };
375
+ }
376
+ }
377
+ /**
378
+ * Enhanced path validation for portfolio operations with comprehensive security checks
379
+ * SECURITY ENHANCEMENT (Task #7): Additional validation for special characters and malicious patterns
380
+ * @param filePath The file path to validate
381
+ * @returns Validation result with secure path or error response
382
+ */
383
+ async validatePortfolioPath(filePath) {
384
+ try {
385
+ // Basic null/undefined check
386
+ if (!filePath || typeof filePath !== 'string') {
387
+ SecurityMonitor.logSecurityEvent({
388
+ type: 'PATH_TRAVERSAL_ATTEMPT',
389
+ severity: 'MEDIUM',
390
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
391
+ details: 'Invalid path provided - null, undefined, or non-string'
392
+ });
393
+ return {
394
+ isValid: false,
395
+ error: {
396
+ success: false,
397
+ message: 'Invalid file path provided',
398
+ error: 'INVALID_PATH'
399
+ }
400
+ };
401
+ }
402
+ // Check for suspicious patterns that could indicate path traversal or injection
403
+ const suspiciousPatterns = [
404
+ /\.\./, // Path traversal
405
+ /\/\.\./, // Unix path traversal
406
+ /\\\.\./, // Windows path traversal
407
+ /\x00/, // Null bytes
408
+ /[\x01-\x1f\x7f-\x9f]/, // Control characters
409
+ /[<>:"|?*]/, // Invalid filename characters on Windows
410
+ /^(CON|PRN|AUX|NUL|COM[1-9]|LPT[1-9])$/i, // Reserved Windows names
411
+ /^\./, // Hidden files (starting with dot)
412
+ /\s+$/, // Trailing whitespace
413
+ /^[\s]*$/, // Only whitespace
414
+ /%[0-9a-fA-F]{2}/, // URL encoding (potential bypass attempt)
415
+ /\\x[0-9a-fA-F]{2}/, // Hex encoding
416
+ /\$\{.*\}/, // Template literal injection
417
+ /`.*`/, // Backtick injection
418
+ /[\\\/]{2,}/ // Multiple consecutive slashes
419
+ ];
420
+ for (const pattern of suspiciousPatterns) {
421
+ if (pattern.test(filePath)) {
422
+ SecurityMonitor.logSecurityEvent({
423
+ type: 'PATH_TRAVERSAL_ATTEMPT',
424
+ severity: 'HIGH',
425
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
426
+ details: `Suspicious pattern detected in file path: ${pattern.source}`,
427
+ metadata: {
428
+ pathLength: filePath.length,
429
+ pattern: pattern.source
430
+ }
431
+ });
432
+ return {
433
+ isValid: false,
434
+ error: {
435
+ success: false,
436
+ message: 'File path contains invalid or suspicious characters',
437
+ error: 'SUSPICIOUS_PATH_PATTERN'
438
+ }
439
+ };
440
+ }
441
+ }
442
+ // Check path length (prevent buffer overflow attempts)
443
+ const MAX_PATH_LENGTH = process.platform === 'win32' ? 260 : 4096;
444
+ if (filePath.length > MAX_PATH_LENGTH) {
445
+ SecurityMonitor.logSecurityEvent({
446
+ type: 'PATH_TRAVERSAL_ATTEMPT',
447
+ severity: 'MEDIUM',
448
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
449
+ details: `File path exceeds maximum length: ${filePath.length} > ${MAX_PATH_LENGTH}`
450
+ });
451
+ return {
452
+ isValid: false,
453
+ error: {
454
+ success: false,
455
+ message: 'File path is too long',
456
+ error: 'PATH_TOO_LONG'
457
+ }
458
+ };
459
+ }
460
+ // Normalize path to resolve any relative components safely
461
+ let normalizedPath;
462
+ try {
463
+ // Remove null bytes and normalize
464
+ const cleanPath = filePath.replace(/\x00/g, '');
465
+ normalizedPath = path.normalize(cleanPath);
466
+ // Ensure the normalized path doesn't escape the intended directory
467
+ if (normalizedPath.includes('..') || normalizedPath.startsWith('/') ||
468
+ (process.platform === 'win32' && /^[a-zA-Z]:/.test(normalizedPath))) {
469
+ throw new Error('Path normalization resulted in directory traversal');
470
+ }
471
+ }
472
+ catch (error) {
473
+ SecurityMonitor.logSecurityEvent({
474
+ type: 'PATH_TRAVERSAL_ATTEMPT',
475
+ severity: 'HIGH',
476
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
477
+ details: `Path normalization failed: ${error instanceof Error ? error.message : 'unknown error'}`
478
+ });
479
+ return {
480
+ isValid: false,
481
+ error: {
482
+ success: false,
483
+ message: 'File path could not be safely processed',
484
+ error: 'PATH_NORMALIZATION_FAILED'
485
+ }
486
+ };
487
+ }
488
+ // Validate file extension (only allow safe extensions for portfolio content)
489
+ const allowedExtensions = ['.md', '.markdown', '.txt', '.yml', '.yaml', '.json'];
490
+ const fileExtension = path.extname(normalizedPath).toLowerCase();
491
+ if (fileExtension && !allowedExtensions.includes(fileExtension)) {
492
+ SecurityMonitor.logSecurityEvent({
493
+ type: 'CONTENT_INJECTION_ATTEMPT',
494
+ severity: 'MEDIUM',
495
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
496
+ details: `Disallowed file extension: ${fileExtension}`,
497
+ metadata: {
498
+ allowedExtensions: allowedExtensions.join(', ')
499
+ }
500
+ });
501
+ return {
502
+ isValid: false,
503
+ error: {
504
+ success: false,
505
+ message: `File extension '${fileExtension}' is not allowed. Allowed extensions: ${allowedExtensions.join(', ')}`,
506
+ error: 'INVALID_FILE_EXTENSION'
507
+ }
508
+ };
509
+ }
510
+ // Validate filename characters (only allow safe characters)
511
+ const basename = path.basename(normalizedPath);
512
+ const safeFilenamePattern = /^[a-zA-Z0-9\-_.\s()[\]{}]+$/;
513
+ if (basename && !safeFilenamePattern.test(basename)) {
514
+ SecurityMonitor.logSecurityEvent({
515
+ type: 'CONTENT_INJECTION_ATTEMPT',
516
+ severity: 'MEDIUM',
517
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
518
+ details: 'Filename contains potentially dangerous characters',
519
+ metadata: {
520
+ filename: basename,
521
+ allowedPattern: safeFilenamePattern.source
522
+ }
523
+ });
524
+ return {
525
+ isValid: false,
526
+ error: {
527
+ success: false,
528
+ message: 'Filename contains invalid characters. Only letters, numbers, spaces, hyphens, underscores, dots, and common brackets are allowed.',
529
+ error: 'INVALID_FILENAME_CHARACTERS'
530
+ }
531
+ };
532
+ }
533
+ // Log successful validation
534
+ SecurityMonitor.logSecurityEvent({
535
+ type: 'CONTENT_INJECTION_ATTEMPT',
536
+ severity: 'LOW',
537
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
538
+ details: 'File path validation successful',
539
+ metadata: {
540
+ originalPathLength: filePath.length,
541
+ normalizedPathLength: normalizedPath.length,
542
+ fileExtension: fileExtension || 'none'
543
+ }
544
+ });
545
+ return {
546
+ isValid: true,
547
+ safePath: normalizedPath
548
+ };
549
+ }
550
+ catch (error) {
551
+ SecurityMonitor.logSecurityEvent({
552
+ type: 'PATH_TRAVERSAL_ATTEMPT',
553
+ severity: 'HIGH',
554
+ source: 'SubmitToPortfolioTool.validatePortfolioPath',
555
+ details: `Path validation error: ${error instanceof Error ? error.message : 'unknown error'}`
556
+ });
557
+ return {
558
+ isValid: false,
559
+ error: {
560
+ success: false,
561
+ message: 'Unable to validate file path. Please check the file path and try again.',
562
+ error: 'PATH_VALIDATION_ERROR'
563
+ }
564
+ };
565
+ }
566
+ }
567
+ /**
568
+ * Smart token management for long operations with refresh-like capabilities
569
+ * SECURITY ENHANCEMENT (Task #14): Token refresh logic for long operations
570
+ *
571
+ * Note: GitHub OAuth device flow tokens don't have traditional refresh tokens,
572
+ * but we can implement smart validation and guidance for long operations
573
+ *
574
+ * @param operationType Type of operation being performed
575
+ * @returns Token management result with recommendations
576
+ */
577
+ async manageTokenForLongOperation(operationType) {
578
+ try {
579
+ // Get current token
580
+ const token = await TokenManager.getGitHubTokenAsync();
581
+ if (!token) {
582
+ return {
583
+ canProceed: false,
584
+ error: {
585
+ success: false,
586
+ message: 'No GitHub token available. Please authenticate first.',
587
+ error: 'NO_TOKEN'
588
+ }
589
+ };
590
+ }
591
+ // Validate token for the specific operation
592
+ const validation = await this.validateTokenBeforeUsage(token);
593
+ if (!validation.isValid) {
594
+ return {
595
+ canProceed: false,
596
+ error: validation.error
597
+ };
598
+ }
599
+ // Check if this is a long operation that might benefit from fresh authentication
600
+ const longOperations = ['portfolio_creation', 'collection_submission'];
601
+ const isLongOperation = longOperations.includes(operationType);
602
+ // Get token type to determine refresh capabilities
603
+ const tokenType = TokenManager.getTokenType(token);
604
+ let refreshRecommended = false;
605
+ // For long operations, check token age and recommend refresh if needed
606
+ if (isLongOperation && validation.isNearExpiry) {
607
+ refreshRecommended = true;
608
+ SecurityMonitor.logSecurityEvent({
609
+ type: 'TOKEN_VALIDATION_SUCCESS',
610
+ severity: 'LOW',
611
+ source: 'SubmitToPortfolioTool.manageTokenForLongOperation',
612
+ details: 'Long operation detected with aging token - refresh recommended',
613
+ metadata: {
614
+ operationType,
615
+ tokenType,
616
+ refreshRecommended: true
617
+ }
618
+ });
619
+ logger.warn('Long operation with potentially aging token detected', {
620
+ operationType,
621
+ tokenType,
622
+ recommendation: 'Consider re-authenticating if operation fails'
623
+ });
624
+ }
625
+ // For OAuth tokens in long operations, we can provide guidance
626
+ if (tokenType === 'OAuth Access Token' && isLongOperation) {
627
+ logger.info('OAuth token detected for long operation', {
628
+ operationType,
629
+ tokenType,
630
+ guidance: 'OAuth tokens are time-limited. If operation fails, re-authenticate using setup_github_auth'
631
+ });
632
+ }
633
+ // Log successful token management
634
+ SecurityMonitor.logSecurityEvent({
635
+ type: 'TOKEN_VALIDATION_SUCCESS',
636
+ severity: 'LOW',
637
+ source: 'SubmitToPortfolioTool.manageTokenForLongOperation',
638
+ details: 'Token management successful for long operation',
639
+ metadata: {
640
+ operationType,
641
+ tokenType,
642
+ isLongOperation,
643
+ refreshRecommended
644
+ }
645
+ });
646
+ return {
647
+ canProceed: true,
648
+ token,
649
+ refreshRecommended
650
+ };
651
+ }
652
+ catch (error) {
653
+ SecurityMonitor.logSecurityEvent({
654
+ type: 'TOKEN_VALIDATION_FAILURE',
655
+ severity: 'MEDIUM',
656
+ source: 'SubmitToPortfolioTool.manageTokenForLongOperation',
657
+ details: `Token management error: ${error.message || 'unknown error'}`
658
+ });
659
+ return {
660
+ canProceed: false,
661
+ error: {
662
+ success: false,
663
+ message: 'Unable to manage token for operation. Please check your authentication and try again.',
664
+ error: 'TOKEN_MANAGEMENT_ERROR'
665
+ }
666
+ };
667
+ }
668
+ }
669
+ /**
670
+ * Provides user guidance for token refresh when operations fail due to token issues
671
+ * SECURITY ENHANCEMENT (Task #14): User guidance for authentication refresh
672
+ */
673
+ formatTokenRefreshGuidance(operationType, tokenType) {
674
+ let guidance = '\n\n🔄 **Token Refresh Guidance**:\n';
675
+ if (tokenType === 'OAuth Access Token') {
676
+ guidance += '• Your OAuth token may have expired\n';
677
+ guidance += '• Run `setup_github_auth` to authenticate again\n';
678
+ guidance += '• This will generate a fresh token for continued access\n';
679
+ }
680
+ else if (tokenType === 'Personal Access Token') {
681
+ guidance += '• Your Personal Access Token may have expired\n';
682
+ guidance += '• Check your GitHub settings: https://github.com/settings/tokens\n';
683
+ guidance += '• Generate a new token if needed and update GITHUB_TOKEN environment variable\n';
684
+ }
685
+ else {
686
+ guidance += '• Your GitHub token may have expired or been revoked\n';
687
+ guidance += '• Re-authenticate using `setup_github_auth`\n';
688
+ guidance += '• Ensure your token has the required permissions\n';
689
+ }
690
+ guidance += `\n**Operation**: ${operationType}\n`;
691
+ guidance += '**Required scopes**: repo, user:email\n\n';
692
+ guidance += '💡 **Tip**: Fresh tokens work better for complex operations like portfolio creation.';
693
+ return guidance;
694
+ }
695
+ /**
696
+ * Sets up GitHub repository access and ensures portfolio repository exists
697
+ * @param authStatus Authentication status containing username
698
+ * @returns Setup result or error response
699
+ */
700
+ async setupGitHubRepository(authStatus) {
701
+ // SECURITY ENHANCEMENT (Task #14): Smart token management for long operations
702
+ const tokenManagement = await this.manageTokenForLongOperation('portfolio_creation');
703
+ if (!tokenManagement.canProceed) {
704
+ return {
705
+ success: false,
706
+ error: tokenManagement.error
707
+ };
708
+ }
709
+ const token = tokenManagement.token;
710
+ // Provide user guidance if refresh is recommended for this long operation
711
+ if (tokenManagement.refreshRecommended) {
712
+ const tokenType = TokenManager.getTokenType(token);
713
+ const guidance = this.formatTokenRefreshGuidance('portfolio creation', tokenType);
714
+ logger.warn(`Token refresh recommended for portfolio creation:${guidance}`);
715
+ }
716
+ this.portfolioManager.setToken(token);
717
+ // Check if portfolio exists and create if needed
718
+ const username = authStatus.username || 'unknown';
719
+ const portfolioExists = await this.portfolioManager.checkPortfolioExists(username);
720
+ if (!portfolioExists) {
721
+ logger.info('Creating portfolio repository...');
722
+ // Request consent for portfolio creation
723
+ const repoUrl = await this.portfolioManager.createPortfolio(username, true);
724
+ if (!repoUrl) {
725
+ return {
726
+ success: false,
727
+ error: {
728
+ success: false,
729
+ message: 'Failed to create portfolio repository',
730
+ error: 'CREATE_FAILED'
731
+ }
732
+ };
733
+ }
734
+ }
735
+ return { success: true };
736
+ }
737
+ /**
738
+ * Submits element to portfolio and handles the complete response workflow
739
+ * @param safeName The normalized name of the element
740
+ * @param elementType The type of the element
741
+ * @param metadata The metadata for the element
742
+ * @param content The content of the element
743
+ * @param authStatus Authentication status containing username and token
744
+ * @returns Complete submission result with success message or error
745
+ */
746
+ async submitElementAndHandleResponse(safeName, elementType, metadata, content, authStatus) {
747
+ // Create element structure to save
748
+ const element = {
749
+ type: elementType,
750
+ metadata,
751
+ content
752
+ };
753
+ // TYPE SAFETY FIX #2: Use adapter pattern instead of complex type casting
754
+ // Previously: element as unknown as Parameters<typeof this.portfolioManager.saveElement>[0]
755
+ // Now: Clean adapter pattern that implements IElement interface properly
756
+ const adapter = new PortfolioElementAdapter(element);
757
+ // UX IMPROVEMENT: Add retry logic for transient failures
758
+ const fileUrl = await this.saveElementWithRetry(adapter, safeName, elementType);
759
+ if (!fileUrl) {
760
+ return {
761
+ success: false,
762
+ message: 'Failed to save element to GitHub portfolio after multiple attempts.\n\n' +
763
+ '💡 **Troubleshooting Tips**:\n' +
764
+ '• Check your GitHub authentication: `gh auth status`\n' +
765
+ '• Verify repository permissions\n' +
766
+ '• Try again in a few minutes (GitHub API rate limits)\n' +
767
+ '• Check GitHub status: https://status.github.com',
768
+ error: 'SAVE_FAILED'
769
+ };
770
+ }
771
+ // Log successful submission (DMCP-SEC-006)
772
+ logger.info(`Successfully submitted ${safeName} to GitHub portfolio`, {
773
+ elementType,
774
+ username: authStatus.username,
775
+ fileUrl
776
+ });
777
+ // SECURITY ENHANCEMENT (Task #14): Smart token management for collection submission
778
+ const collectionTokenManagement = await this.manageTokenForLongOperation('collection_submission');
779
+ if (!collectionTokenManagement.canProceed) {
780
+ // Token management failed for collection submission, but main submission succeeded
781
+ const errorMessage = collectionTokenManagement.error?.message || 'Token management failed';
782
+ return {
783
+ success: true,
784
+ message: `✅ Successfully uploaded ${safeName} to your GitHub portfolio!\n📁 Portfolio URL: ${fileUrl}\n\n⚠️ Collection submission skipped: ${errorMessage}`,
785
+ url: fileUrl
786
+ };
787
+ }
788
+ const token = collectionTokenManagement.token;
789
+ // Provide refresh guidance if recommended for collection submission
790
+ if (collectionTokenManagement.refreshRecommended) {
791
+ const tokenType = TokenManager.getTokenType(token);
792
+ logger.info('Collection submission proceeding with aging token', {
793
+ tokenType,
794
+ recommendation: 'If collection submission fails, try re-authenticating with setup_github_auth'
795
+ });
796
+ }
797
+ // ENHANCEMENT (Issue #549): Ask user if they want to submit to collection
798
+ // This completes the community contribution workflow
799
+ const collectionSubmissionResult = await this.promptForCollectionSubmission({
800
+ elementName: safeName,
801
+ elementType,
802
+ portfolioUrl: fileUrl,
803
+ username: authStatus.username || 'unknown',
804
+ metadata,
805
+ token
806
+ });
807
+ // Build the response message based on what happened
808
+ let message = `✅ Successfully uploaded ${safeName} to your GitHub portfolio!\n`;
809
+ message += `📁 Portfolio URL: ${fileUrl}\n\n`;
810
+ if (collectionSubmissionResult.submitted) {
811
+ message += `🎉 Also submitted to DollhouseMCP collection for community review!\n`;
812
+ message += `📋 Issue: ${collectionSubmissionResult.issueUrl}`;
813
+ }
814
+ else if (collectionSubmissionResult.declined) {
815
+ message += `💡 You can submit to the collection later using the same command.`;
816
+ }
817
+ else if (collectionSubmissionResult.error) {
818
+ message += `⚠️ Collection submission failed: ${collectionSubmissionResult.error}\n`;
819
+ message += `💡 You can manually submit at: https://github.com/DollhouseMCP/collection/issues/new`;
820
+ }
821
+ return {
822
+ success: true,
823
+ message,
824
+ url: fileUrl
825
+ };
826
+ }
827
+ async execute(params) {
828
+ try {
829
+ // Validate and normalize input parameters
830
+ const validationResult = await this.validateAndNormalizeParams(params);
831
+ if (!validationResult.success) {
832
+ return validationResult.error;
833
+ }
834
+ const safeName = validationResult.safeName;
835
+ // Check authentication status
836
+ const authResult = await this.checkAuthentication();
837
+ if (!authResult.success) {
838
+ return authResult.error;
839
+ }
840
+ const authStatus = authResult.authStatus;
841
+ // Find content locally with smart type detection
842
+ const contentResult = await this.discoverContentWithTypeDetection(safeName, params.type, params.name);
843
+ if (!contentResult.success) {
844
+ return contentResult.error;
845
+ }
846
+ const elementType = contentResult.elementType;
847
+ const localPath = contentResult.localPath;
848
+ // Validate file and content security
849
+ const securityResult = await this.validateFileAndContent(localPath);
850
+ if (!securityResult.success) {
851
+ return securityResult.error;
852
+ }
853
+ const content = securityResult.content;
854
+ // Get user consent (placeholder for now - could add interactive prompt later)
855
+ logger.info(`Preparing to submit ${safeName} to GitHub portfolio`);
856
+ // Prepare metadata for element
857
+ const metadata = this.prepareElementMetadata(safeName, elementType, authStatus);
858
+ // Set up GitHub repository access
859
+ const repoResult = await this.setupGitHubRepository(authStatus);
860
+ if (!repoResult.success) {
861
+ return repoResult.error;
862
+ }
863
+ // Submit element to portfolio and handle collection submission
864
+ return await this.submitElementAndHandleResponse(safeName, elementType, metadata, content, authStatus);
865
+ }
866
+ catch (error) {
867
+ // SECURITY ENHANCEMENT (Task #14): Enhanced error handling with token refresh guidance
868
+ ErrorHandler.logError('submitToPortfolio', error, {
869
+ elementName: params.name,
870
+ elementType: params.type
871
+ });
872
+ // Check if error is token-related and provide refresh guidance
873
+ const errorMessage = error instanceof Error ? error.message : String(error);
874
+ const isTokenError = errorMessage.toLowerCase().includes('token') ||
875
+ errorMessage.toLowerCase().includes('auth') ||
876
+ errorMessage.toLowerCase().includes('401') ||
877
+ errorMessage.toLowerCase().includes('403');
878
+ let formattedError = ErrorHandler.formatForResponse(error);
879
+ if (isTokenError) {
880
+ try {
881
+ // Get current token to determine type for guidance
882
+ const currentToken = await TokenManager.getGitHubTokenAsync();
883
+ if (currentToken) {
884
+ const tokenType = TokenManager.getTokenType(currentToken);
885
+ const refreshGuidance = this.formatTokenRefreshGuidance('portfolio submission', tokenType);
886
+ // Append refresh guidance to error message
887
+ if (formattedError.message) {
888
+ formattedError.message += refreshGuidance;
889
+ }
890
+ }
891
+ }
892
+ catch (tokenError) {
893
+ // If we can't get token info, provide generic guidance
894
+ formattedError.message += '\n\n🔄 **Authentication Issue**: Try running `setup_github_auth` to refresh your authentication.';
895
+ }
896
+ }
897
+ return formattedError;
898
+ }
899
+ }
900
+ /**
901
+ * Prompts user to submit content to the DollhouseMCP collection
902
+ * ENHANCEMENT (Issue #549): Complete the community contribution workflow
903
+ */
904
+ async promptForCollectionSubmission(params) {
905
+ try {
906
+ // Create a simple prompt message for the user
907
+ // Note: In MCP context, we can't do interactive prompts, so we'll need to
908
+ // either make this automatic or require a parameter
909
+ // For now, let's check if the user has set an environment variable
910
+ // to auto-submit to collection (opt-in behavior)
911
+ const autoSubmit = process.env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION === 'true';
912
+ if (!autoSubmit) {
913
+ // User hasn't opted in to auto-submission
914
+ logger.info('Collection submission skipped (set DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION=true to enable)');
915
+ return { submitted: false, declined: true };
916
+ }
917
+ logger.info('Auto-submitting to DollhouseMCP collection...');
918
+ // Create the issue in the collection repository
919
+ const issueUrl = await this.createCollectionIssue({
920
+ ...params,
921
+ token: params.token
922
+ });
923
+ if (issueUrl) {
924
+ logger.info('Successfully created collection submission issue', { issueUrl });
925
+ return { submitted: true, declined: false, issueUrl };
926
+ }
927
+ else {
928
+ return { submitted: false, declined: false, error: 'Failed to create issue' };
929
+ }
930
+ }
931
+ catch (error) {
932
+ logger.error('Error in collection submission prompt', { error });
933
+ return {
934
+ submitted: false,
935
+ declined: false,
936
+ error: error instanceof Error ? error.message : 'Unknown error'
937
+ };
938
+ }
939
+ }
940
+ /**
941
+ * Creates an issue in the DollhouseMCP/collection repository
942
+ * ENHANCEMENT (Issue #549): GitHub API integration for collection submission
943
+ */
944
+ async createCollectionIssue(params) {
945
+ try {
946
+ // Format the issue title
947
+ const title = `[${params.elementType}] Add ${params.elementName} by @${params.username}`;
948
+ // Format the issue body with all relevant information
949
+ const body = `## New ${params.elementType} Submission
950
+
951
+ ` +
952
+ `**Name**: ${params.elementName}\n` +
953
+ `**Author**: @${params.username}\n` +
954
+ `**Type**: ${params.elementType}\n` +
955
+ `**Description**: ${params.metadata.description || 'No description provided'}\n\n` +
956
+ `### Portfolio Link\n` +
957
+ `${params.portfolioUrl}\n\n` +
958
+ `### Metadata\n` +
959
+ `\`\`\`json\n${JSON.stringify(params.metadata, null, 2)}\n\`\`\`\n\n` +
960
+ `### Review Checklist\n` +
961
+ `- [ ] Content is appropriate and follows community guidelines\n` +
962
+ `- [ ] No security vulnerabilities or malicious patterns\n` +
963
+ `- [ ] Metadata is complete and accurate\n` +
964
+ `- [ ] Element works as described\n` +
965
+ `- [ ] No duplicate of existing collection content\n\n` +
966
+ `---\n` +
967
+ `*This submission was created automatically via the DollhouseMCP submit_content tool.*`;
968
+ // Determine labels based on element type
969
+ const labels = [
970
+ 'contribution', // All submissions get this
971
+ 'pending-review', // Needs review
972
+ params.elementType.toLowerCase() // Element type label
973
+ ];
974
+ // PERFORMANCE OPTIMIZATION (Task #6): Use GitHub rate limiter for API calls
975
+ // This prevents hitting GitHub rate limits and provides better error handling
976
+ const issueUrl = await githubRateLimiter.queueRequest('create-collection-issue', async () => {
977
+ const url = 'https://api.github.com/repos/DollhouseMCP/collection/issues';
978
+ // Create AbortController for timeout
979
+ const controller = new AbortController();
980
+ const timeoutId = setTimeout(() => controller.abort(), getValidatedTimeout());
981
+ try {
982
+ const response = await fetch(url, {
983
+ method: 'POST',
984
+ headers: {
985
+ 'Accept': 'application/vnd.github.v3+json',
986
+ 'Authorization': `Bearer ${params.token}`,
987
+ 'Content-Type': 'application/json',
988
+ 'User-Agent': 'DollhouseMCP/1.0'
989
+ },
990
+ body: JSON.stringify({
991
+ title,
992
+ body,
993
+ labels
994
+ }),
995
+ signal: controller.signal
996
+ });
997
+ clearTimeout(timeoutId);
998
+ // PERFORMANCE OPTIMIZATION (Task #15): Enhanced rate limit logging
999
+ // Log rate limit headers for diagnostics
1000
+ const rateLimitRemaining = response.headers.get('X-RateLimit-Remaining');
1001
+ const rateLimitReset = response.headers.get('X-RateLimit-Reset');
1002
+ const rateLimitLimit = response.headers.get('X-RateLimit-Limit');
1003
+ logger.debug('GitHub API rate limit status', {
1004
+ operation: 'create-collection-issue',
1005
+ remaining: rateLimitRemaining,
1006
+ limit: rateLimitLimit,
1007
+ resetTime: rateLimitReset ? new Date(parseInt(rateLimitReset) * 1000) : undefined,
1008
+ responseStatus: response.status
1009
+ });
1010
+ // Log warning if approaching rate limit
1011
+ if (rateLimitRemaining && parseInt(rateLimitRemaining) < 100) {
1012
+ logger.warn('Approaching GitHub API rate limit', {
1013
+ operation: 'create-collection-issue',
1014
+ remaining: rateLimitRemaining,
1015
+ resetTime: rateLimitReset ? new Date(parseInt(rateLimitReset) * 1000) : undefined,
1016
+ recommendation: 'Consider reducing API usage frequency or authenticating for higher limits'
1017
+ });
1018
+ }
1019
+ if (!response.ok) {
1020
+ const errorText = await response.text();
1021
+ logger.error('GitHub API error creating issue', {
1022
+ status: response.status,
1023
+ statusText: response.statusText,
1024
+ error: errorText,
1025
+ rateLimitRemaining,
1026
+ rateLimitReset
1027
+ });
1028
+ if (response.status === 404) {
1029
+ logger.error('Collection repository not found or no access');
1030
+ }
1031
+ else if (response.status === 403) {
1032
+ logger.error('Permission denied to create issue in collection repo');
1033
+ }
1034
+ else if (response.status === 401) {
1035
+ logger.error('Authentication failed for collection submission');
1036
+ }
1037
+ throw new Error(`GitHub API error: ${response.status} ${response.statusText}`);
1038
+ }
1039
+ const data = await response.json();
1040
+ return data.html_url;
1041
+ }
1042
+ catch (fetchError) {
1043
+ // Re-throw to outer catch block
1044
+ throw fetchError;
1045
+ }
1046
+ finally {
1047
+ clearTimeout(timeoutId);
1048
+ }
1049
+ }, 'high' // High priority for collection submission
1050
+ );
1051
+ return issueUrl;
1052
+ }
1053
+ catch (error) {
1054
+ // Handle timeout specifically
1055
+ if (error.name === 'AbortError') {
1056
+ logger.error(`GitHub API request timeout after ${getValidatedTimeout()}ms`);
1057
+ }
1058
+ else {
1059
+ logger.error('Failed to create collection issue', {
1060
+ error: error.message || error
1061
+ });
1062
+ }
1063
+ return null;
1064
+ }
1065
+ }
1066
+ async findLocalContent(name, type) {
1067
+ try {
1068
+ // METADATA INDEX FIX: Use portfolio index for fast metadata-based lookups
1069
+ // This solves the critical issue where "Safe Roundtrip Tester" couldn't be found
1070
+ // because findLocalContent only searched filenames, not metadata names
1071
+ const indexManager = PortfolioIndexManager.getInstance();
1072
+ // UX IMPROVEMENT: Enhanced search with fuzzy matching
1073
+ const indexEntry = await indexManager.findByName(name, {
1074
+ elementType: type,
1075
+ fuzzyMatch: true
1076
+ });
1077
+ if (indexEntry) {
1078
+ logger.debug('Found content via metadata index', {
1079
+ searchName: name,
1080
+ metadataName: indexEntry.metadata.name,
1081
+ filename: indexEntry.filename,
1082
+ filePath: indexEntry.filePath,
1083
+ type
1084
+ });
1085
+ return indexEntry.filePath;
1086
+ }
1087
+ // FALLBACK: Use original file discovery if index lookup fails
1088
+ // This maintains backward compatibility and handles edge cases
1089
+ logger.debug('Index lookup failed, falling back to file discovery', { name, type });
1090
+ const portfolioManager = PortfolioManager.getInstance();
1091
+ const portfolioDir = portfolioManager.getElementDir(type);
1092
+ // UX IMPROVEMENT: Try multiple search strategies for better user experience
1093
+ let file = await FileDiscoveryUtil.findFile(portfolioDir, name, {
1094
+ extensions: ['.md', '.json', '.yaml', '.yml'],
1095
+ partialMatch: true,
1096
+ cacheResults: true
1097
+ });
1098
+ // If not found, try normalizing the name (e.g., "J.A.R.V.I.S." -> "j-a-r-v-i-s")
1099
+ if (!file) {
1100
+ const normalizedName = name.toLowerCase()
1101
+ .replace(/[^a-z0-9]/gi, '-') // Replace non-alphanumeric with dashes
1102
+ .replace(/-+/g, '-') // Replace multiple dashes with single dash
1103
+ .replace(/^-|-$/g, ''); // Remove leading/trailing dashes
1104
+ if (normalizedName !== name.toLowerCase()) {
1105
+ logger.debug('Trying normalized name search', {
1106
+ original: name,
1107
+ normalized: normalizedName,
1108
+ type
1109
+ });
1110
+ file = await FileDiscoveryUtil.findFile(portfolioDir, normalizedName, {
1111
+ extensions: ['.md', '.json', '.yaml', '.yml'],
1112
+ partialMatch: true,
1113
+ cacheResults: true
1114
+ });
1115
+ }
1116
+ }
1117
+ // If still not found, try searching by display name patterns
1118
+ if (!file) {
1119
+ // Try common variations like removing dots, spaces, etc.
1120
+ const variations = [
1121
+ name.replace(/\./g, ''), // Remove dots: "J.A.R.V.I.S." -> "JARVIS"
1122
+ name.replace(/\s+/g, '-'), // Replace spaces with dashes
1123
+ name.replace(/[\s\.]/g, ''), // Remove spaces and dots
1124
+ name.replace(/[\s\.]/g, '-'), // Replace spaces and dots with dashes
1125
+ ].filter(v => v !== name && v.length > 0);
1126
+ for (const variation of variations) {
1127
+ file = await FileDiscoveryUtil.findFile(portfolioDir, variation, {
1128
+ extensions: ['.md', '.json', '.yaml', '.yml'],
1129
+ partialMatch: true,
1130
+ cacheResults: true
1131
+ });
1132
+ if (file) {
1133
+ logger.debug('Found content using name variation', {
1134
+ original: name,
1135
+ variation,
1136
+ file,
1137
+ type
1138
+ });
1139
+ break;
1140
+ }
1141
+ }
1142
+ }
1143
+ if (file) {
1144
+ logger.debug('Found local content file via fallback', { name, type, file });
1145
+ return file;
1146
+ }
1147
+ logger.debug('No content found', { name, type });
1148
+ return null;
1149
+ }
1150
+ catch (error) {
1151
+ logger.error('Error finding local content', {
1152
+ name,
1153
+ type,
1154
+ error: error instanceof Error ? error.message : String(error)
1155
+ });
1156
+ return null;
1157
+ }
1158
+ }
1159
+ /**
1160
+ * Smart element type detection - searches across ALL element types for content
1161
+ * PERFORMANCE OPTIMIZATION (Task #9): Uses early termination for exact matches
1162
+ * This replaces the previous hardcoded default to PERSONA and enables proper type detection
1163
+ *
1164
+ * @param name The content name to search for
1165
+ * @returns Detection result with found matches across all element types
1166
+ */
1167
+ async detectElementType(name) {
1168
+ try {
1169
+ // PERFORMANCE OPTIMIZATION (Task #9): Use early termination search utility
1170
+ // Create search functions for each element type
1171
+ const elementTypes = Object.values(ElementType);
1172
+ const searchFunctions = elementTypes.map((type) => async () => {
1173
+ try {
1174
+ const filePath = await this.findLocalContent(name, type);
1175
+ if (filePath) {
1176
+ return { type: type, path: filePath };
1177
+ }
1178
+ return null;
1179
+ }
1180
+ catch (error) {
1181
+ // Log unexpected errors but don't fail the search
1182
+ if (error?.code !== 'ENOENT' && error?.code !== 'ENOTDIR') {
1183
+ logger.debug(`Error searching ${type} directory for content detection`, {
1184
+ name,
1185
+ type,
1186
+ error: error?.message || String(error),
1187
+ code: error?.code
1188
+ });
1189
+ }
1190
+ // Return null instead of throwing to let other searches continue
1191
+ return null;
1192
+ }
1193
+ });
1194
+ // PERFORMANCE OPTIMIZATION (Task #9): Define exact match criteria
1195
+ const isExactMatch = (match) => {
1196
+ const filename = path.basename(match.path, path.extname(match.path));
1197
+ return filename.toLowerCase() === name.toLowerCase();
1198
+ };
1199
+ // Execute searches with early termination optimization
1200
+ const searchResults = await EarlyTerminationSearch.executeWithEarlyTermination(searchFunctions, isExactMatch, {
1201
+ operationName: 'element-type-detection',
1202
+ timeoutAfterExactMatch: 1000, // Wait 1 second for other searches after exact match
1203
+ maxParallelSearches: 8 // Limit concurrent searches to avoid overwhelming the system
1204
+ });
1205
+ // PERFORMANCE OPTIMIZATION (Task #8): Enhanced batch operation reporting
1206
+ const batchResults = {
1207
+ name,
1208
+ totalSearches: searchResults.totalSearches,
1209
+ completedSearches: searchResults.completedSearches,
1210
+ matches: searchResults.matches.length,
1211
+ failures: searchResults.failures.length,
1212
+ exactMatchFound: !!searchResults.exactMatch,
1213
+ exactMatchType: searchResults.exactMatch?.type,
1214
+ earlyTerminationTriggered: searchResults.earlyTerminationTriggered,
1215
+ performanceGain: searchResults.performanceGain,
1216
+ matchedTypes: searchResults.matches.map(m => m.type),
1217
+ failedTypes: searchResults.failures.map(f => elementTypes[f.index]).filter(Boolean)
1218
+ };
1219
+ logger.debug('Element type detection completed with early termination optimization', batchResults);
1220
+ // PERFORMANCE OPTIMIZATION (Task #8): Clear reporting of partial failures
1221
+ if (searchResults.failures.length > 0) {
1222
+ logger.warn('Some element type searches failed during batch operation', {
1223
+ name,
1224
+ failures: searchResults.failures.map(f => ({
1225
+ type: elementTypes[f.index] || 'unknown',
1226
+ error: f.error.substring(0, 100) // Truncate long error messages
1227
+ })),
1228
+ successRate: `${searchResults.completedSearches}/${searchResults.totalSearches}`,
1229
+ impactOnResults: searchResults.matches.length > 0
1230
+ ? 'No impact - matches found in successful searches'
1231
+ : 'Potential impact - no matches found'
1232
+ });
1233
+ // If we have failures and no matches, provide actionable guidance
1234
+ if (searchResults.matches.length === 0 && searchResults.failures.length > 0) {
1235
+ logger.warn('Batch operation had failures and no matches found', {
1236
+ name,
1237
+ recommendation: 'Consider checking file permissions or portfolio structure',
1238
+ failureCount: searchResults.failures.length,
1239
+ totalSearches: searchResults.totalSearches
1240
+ });
1241
+ }
1242
+ }
1243
+ // Log performance gains from early termination
1244
+ if (searchResults.earlyTerminationTriggered) {
1245
+ logger.info('Early termination optimization applied successfully', {
1246
+ name,
1247
+ exactMatchType: searchResults.exactMatch?.type,
1248
+ performanceGain: searchResults.performanceGain,
1249
+ searchesCompleted: searchResults.completedSearches,
1250
+ searchesTotal: searchResults.totalSearches
1251
+ });
1252
+ }
1253
+ return {
1254
+ found: searchResults.matches.length > 0,
1255
+ matches: searchResults.matches
1256
+ };
1257
+ }
1258
+ catch (error) {
1259
+ logger.error('Error in element type detection', {
1260
+ name,
1261
+ error: error instanceof Error ? error.message : String(error)
1262
+ });
1263
+ // Return empty result on detection failure
1264
+ return {
1265
+ found: false,
1266
+ matches: []
1267
+ };
1268
+ }
1269
+ }
1270
+ /**
1271
+ * UX IMPROVEMENT: Generate name suggestions for similar content
1272
+ * PERFORMANCE OPTIMIZATION (Task #8): Enhanced batch operation handling with clear partial failure reporting
1273
+ * Helps users find content when exact matches fail
1274
+ */
1275
+ async generateNameSuggestions(searchName) {
1276
+ try {
1277
+ const suggestions = [];
1278
+ const searchLower = searchName.toLowerCase();
1279
+ const elementTypes = Object.values(ElementType);
1280
+ // Track batch operation results for better diagnostics
1281
+ const batchResults = {
1282
+ searchName,
1283
+ totalTypes: elementTypes.length,
1284
+ successfulScans: 0,
1285
+ failedScans: 0,
1286
+ failureDetails: [],
1287
+ totalSuggestions: 0,
1288
+ suggestionsByType: {}
1289
+ };
1290
+ // Process all element types for suggestions
1291
+ for (const elementType of elementTypes) {
1292
+ try {
1293
+ const portfolioManager = PortfolioManager.getInstance();
1294
+ const elementDir = portfolioManager.getElementDir(elementType);
1295
+ // Get files in this directory
1296
+ const files = await FileDiscoveryUtil.findFile(elementDir, '*', {
1297
+ extensions: ['.md', '.json', '.yaml', '.yml'],
1298
+ partialMatch: false,
1299
+ cacheResults: true
1300
+ });
1301
+ let typeSuggestions = 0;
1302
+ if (Array.isArray(files)) {
1303
+ for (const filePath of files) {
1304
+ const basename = path.basename(filePath, path.extname(filePath));
1305
+ // Calculate similarity using simple metrics
1306
+ if (this.calculateSimilarity(searchLower, basename.toLowerCase()) > SEARCH_CONFIG.MIN_SIMILARITY_SCORE) {
1307
+ suggestions.push({
1308
+ name: basename,
1309
+ type: elementType
1310
+ });
1311
+ typeSuggestions++;
1312
+ }
1313
+ }
1314
+ }
1315
+ else if (files) {
1316
+ const basename = path.basename(files, path.extname(files));
1317
+ if (this.calculateSimilarity(searchLower, basename.toLowerCase()) > SEARCH_CONFIG.MIN_SIMILARITY_SCORE) {
1318
+ suggestions.push({
1319
+ name: basename,
1320
+ type: elementType
1321
+ });
1322
+ typeSuggestions++;
1323
+ }
1324
+ }
1325
+ batchResults.successfulScans++;
1326
+ batchResults.suggestionsByType[elementType] = typeSuggestions;
1327
+ }
1328
+ catch (error) {
1329
+ // PERFORMANCE OPTIMIZATION (Task #8): Track and report partial failures
1330
+ batchResults.failedScans++;
1331
+ batchResults.failureDetails.push({
1332
+ type: elementType,
1333
+ error: error instanceof Error ? error.message : String(error)
1334
+ });
1335
+ // Log individual failures for diagnostics
1336
+ logger.debug('Failed to scan element type for suggestions', {
1337
+ elementType,
1338
+ searchName,
1339
+ error: error instanceof Error ? error.message : String(error)
1340
+ });
1341
+ }
1342
+ }
1343
+ batchResults.totalSuggestions = suggestions.length;
1344
+ // PERFORMANCE OPTIMIZATION (Task #8): Comprehensive batch operation reporting
1345
+ logger.debug('Name suggestion batch operation completed', {
1346
+ ...batchResults,
1347
+ successRate: `${batchResults.successfulScans}/${batchResults.totalTypes}`,
1348
+ // Don't log full failure details at debug level to avoid spam
1349
+ hasFailures: batchResults.failedScans > 0
1350
+ });
1351
+ // Report failures clearly if they occurred
1352
+ if (batchResults.failedScans > 0) {
1353
+ logger.warn('Some element type scans failed during name suggestion generation', {
1354
+ searchName,
1355
+ failedTypes: batchResults.failureDetails.map(f => f.type),
1356
+ successfulTypes: batchResults.successfulScans,
1357
+ impactOnResults: batchResults.totalSuggestions > 0
1358
+ ? 'Partial impact - suggestions found from successful scans'
1359
+ : 'Potential impact - no suggestions generated',
1360
+ recommendation: batchResults.totalSuggestions === 0 && batchResults.failedScans > 0
1361
+ ? 'Check portfolio directory structure and file permissions'
1362
+ : 'Suggestion generation partially successful despite some failures'
1363
+ });
1364
+ }
1365
+ // Sort by similarity (higher is better) and return top suggestions
1366
+ const sortedSuggestions = suggestions.sort((a, b) => {
1367
+ const simA = this.calculateSimilarity(searchLower, a.name.toLowerCase());
1368
+ const simB = this.calculateSimilarity(searchLower, b.name.toLowerCase());
1369
+ return simB - simA;
1370
+ });
1371
+ logger.debug('Name suggestions generated successfully', {
1372
+ searchName,
1373
+ totalSuggestions: sortedSuggestions.length,
1374
+ topSuggestions: sortedSuggestions.slice(0, 3).map(s => s.name)
1375
+ });
1376
+ return sortedSuggestions;
1377
+ }
1378
+ catch (error) {
1379
+ logger.warn('Failed to generate name suggestions - batch operation failed completely', {
1380
+ searchName,
1381
+ error: error instanceof Error ? error.message : String(error),
1382
+ recommendation: 'Check portfolio structure and permissions'
1383
+ });
1384
+ return [];
1385
+ }
1386
+ }
1387
+ /**
1388
+ * Simple similarity calculation using Levenshtein-like approach
1389
+ * Returns value between 0 and 1, where 1 is identical
1390
+ */
1391
+ calculateSimilarity(str1, str2) {
1392
+ // Handle exact matches
1393
+ if (str1 === str2)
1394
+ return 1;
1395
+ // Handle substring matches
1396
+ if (str1.includes(str2) || str2.includes(str1))
1397
+ return 0.8;
1398
+ // Handle partial matches
1399
+ const longer = str1.length > str2.length ? str1 : str2;
1400
+ const shorter = str1.length > str2.length ? str2 : str1;
1401
+ if (longer.length === 0)
1402
+ return 0;
1403
+ // Count common characters
1404
+ let common = 0;
1405
+ for (let i = 0; i < shorter.length; i++) {
1406
+ if (longer.includes(shorter[i])) {
1407
+ common++;
1408
+ }
1409
+ }
1410
+ return common / longer.length;
1411
+ }
1412
+ /**
1413
+ * UX IMPROVEMENT: Save element with automatic retry logic for transient failures
1414
+ * Handles common GitHub API issues like rate limits and temporary network problems
1415
+ */
1416
+ async saveElementWithRetry(adapter, elementName, elementType, maxRetries = RETRY_CONFIG.MAX_ATTEMPTS) {
1417
+ let lastError = null;
1418
+ for (let attempt = 1; attempt <= maxRetries; attempt++) {
1419
+ try {
1420
+ logger.debug(`Attempting to save element (attempt ${attempt}/${maxRetries})`, {
1421
+ elementName,
1422
+ elementType,
1423
+ attempt
1424
+ });
1425
+ const fileUrl = await this.portfolioManager.saveElement(adapter, true);
1426
+ if (fileUrl) {
1427
+ if (attempt > 1) {
1428
+ logger.info(`Element saved successfully after ${attempt} attempts`, {
1429
+ elementName,
1430
+ elementType,
1431
+ fileUrl
1432
+ });
1433
+ }
1434
+ return fileUrl;
1435
+ }
1436
+ // If saveElement returns null, treat as a failure but don't retry immediately
1437
+ lastError = new Error(`saveElement returned null on attempt ${attempt}`);
1438
+ }
1439
+ catch (error) {
1440
+ lastError = error;
1441
+ const isRetryable = this.isRetryableError(error);
1442
+ logger.warn(`Save attempt ${attempt} failed`, {
1443
+ elementName,
1444
+ elementType,
1445
+ attempt,
1446
+ error: error.message,
1447
+ isRetryable,
1448
+ willRetry: isRetryable && attempt < maxRetries
1449
+ });
1450
+ // If this is not a retryable error, fail immediately
1451
+ if (!isRetryable) {
1452
+ logger.error('Non-retryable error encountered, aborting retries', {
1453
+ elementName,
1454
+ error: error.message
1455
+ });
1456
+ break;
1457
+ }
1458
+ // If we have more attempts, wait before retrying
1459
+ if (attempt < maxRetries) {
1460
+ const delay = calculateRetryDelay(attempt);
1461
+ logger.debug(`Waiting ${delay}ms before retry`, { attempt, delay });
1462
+ await new Promise(resolve => setTimeout(resolve, delay));
1463
+ }
1464
+ }
1465
+ }
1466
+ // All attempts failed
1467
+ logger.error(`All ${maxRetries} save attempts failed`, {
1468
+ elementName,
1469
+ elementType,
1470
+ lastError: lastError?.message
1471
+ });
1472
+ return null;
1473
+ }
1474
+ /**
1475
+ * Determine if an error is worth retrying
1476
+ * Retryable: network issues, rate limits, temporary GitHub API problems
1477
+ * Non-retryable: authentication issues, validation errors, permanent failures
1478
+ */
1479
+ isRetryableError(error) {
1480
+ const errorMessage = error?.message?.toLowerCase() || '';
1481
+ const errorCode = error?.code;
1482
+ const statusCode = error?.status || error?.statusCode;
1483
+ // Network and timeout errors
1484
+ if (errorCode === 'ENOTFOUND' || errorCode === 'ECONNRESET' || errorCode === 'ETIMEDOUT') {
1485
+ return true;
1486
+ }
1487
+ // GitHub API rate limits
1488
+ if (statusCode === 429 || errorMessage.includes('rate limit')) {
1489
+ return true;
1490
+ }
1491
+ // Temporary GitHub API issues
1492
+ if (statusCode >= 500 && statusCode < 600) {
1493
+ return true;
1494
+ }
1495
+ // Temporary GitHub API problems
1496
+ if (errorMessage.includes('temporarily unavailable') ||
1497
+ errorMessage.includes('service unavailable') ||
1498
+ errorMessage.includes('internal server error')) {
1499
+ return true;
1500
+ }
1501
+ // Connection issues
1502
+ if (errorMessage.includes('connection') &&
1503
+ (errorMessage.includes('timeout') || errorMessage.includes('reset'))) {
1504
+ return true;
1505
+ }
1506
+ // Don't retry authentication or permission issues
1507
+ if (statusCode === 401 || statusCode === 403 ||
1508
+ errorMessage.includes('unauthorized') ||
1509
+ errorMessage.includes('forbidden') ||
1510
+ errorMessage.includes('authentication')) {
1511
+ return false;
1512
+ }
1513
+ // Don't retry validation errors
1514
+ if (statusCode === 400 || statusCode === 422 ||
1515
+ errorMessage.includes('invalid') ||
1516
+ errorMessage.includes('validation')) {
1517
+ return false;
1518
+ }
1519
+ // Default to not retrying for unknown errors to avoid infinite loops
1520
+ return false;
1521
+ }
1522
+ }
1523
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"submitToPortfolioTool.js","sourceRoot":"","sources":["../../../src/tools/portfolio/submitToPortfolioTool.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Tool for submitting content to GitHub portfolio repositories\n * Replaces the broken issue-based submission with direct repository saves\n * \n * FIXES IMPLEMENTED (PR #503):\n * 1. TYPE SAFETY FIX #1 (Issue #497): Changed apiCache from 'any' to proper APICache type\n * 2. TYPE SAFETY FIX #2 (Issue #497): Replaced complex type casting with PortfolioElementAdapter\n * 3. PERFORMANCE (PR #496 recommendation): Using FileDiscoveryUtil for optimized file search\n */\n\nimport { GitHubAuthManager } from '../../auth/GitHubAuthManager.js';\nimport { PortfolioRepoManager } from '../../portfolio/PortfolioRepoManager.js';\nimport { TokenManager } from '../../security/tokenManager.js';\nimport { ContentValidator } from '../../security/contentValidator.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { PortfolioIndexManager } from '../../portfolio/PortfolioIndexManager.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { logger } from '../../utils/logger.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { PathValidator } from '../../security/pathValidator.js';\nimport { APICache } from '../../cache/APICache.js';\nimport { PortfolioElementAdapter } from './PortfolioElementAdapter.js';\nimport { FileDiscoveryUtil } from '../../utils/FileDiscoveryUtil.js';\nimport { ErrorHandler, ErrorCategory } from '../../utils/ErrorHandler.js';\nimport { \n  GITHUB_API_TIMEOUT, \n  FILE_SIZE_LIMITS, \n  RETRY_CONFIG, \n  SEARCH_CONFIG,\n  PortfolioElementMetadata,\n  getValidatedTimeout,\n  calculateRetryDelay\n} from '../../config/portfolio-constants.js';\nimport { githubRateLimiter } from '../../utils/GitHubRateLimiter.js';\nimport { EarlyTerminationSearch } from '../../utils/EarlyTerminationSearch.js';\nimport * as path from 'path';\nimport * as fs from 'fs/promises';\n\nexport interface SubmitToPortfolioParams {\n  name: string;\n  type?: ElementType;\n}\n\nexport interface PortfolioElement {\n  type: ElementType;\n  metadata: PortfolioElementMetadata;\n  content: string;\n}\n\nexport interface SubmitToPortfolioResult {\n  success: boolean;\n  message: string;\n  url?: string;\n  error?: string;\n}\n\nexport interface ElementDetectionMatch {\n  type: ElementType;\n  path: string;\n}\n\nexport interface ElementDetectionResult {\n  found: boolean;\n  matches: ElementDetectionMatch[];\n}\n\nexport class SubmitToPortfolioTool {\n  private authManager: GitHubAuthManager;\n  private portfolioManager: PortfolioRepoManager;\n  private contentValidator: ContentValidator;\n\n  constructor(apiCache: APICache) {\n    // TYPE SAFETY FIX #1: Proper typing for apiCache parameter\n    // Previously: constructor(apiCache: any)\n    // Now: constructor(apiCache: APICache) with proper import\n    this.authManager = new GitHubAuthManager(apiCache);\n    this.portfolioManager = new PortfolioRepoManager();\n    this.contentValidator = new ContentValidator();\n  }\n\n  /**\n   * Validates and normalizes input parameters to prevent Unicode attacks and ensure data safety\n   * @param params The input parameters from the user\n   * @returns Validation result with normalized name or error response\n   */\n  private async validateAndNormalizeParams(params: SubmitToPortfolioParams): Promise<{\n    success: boolean;\n    safeName?: string;\n    error?: SubmitToPortfolioResult;\n  }> {\n    // Normalize user input to prevent Unicode attacks (DMCP-SEC-004)\n    const normalizedName = UnicodeValidator.normalize(params.name);\n    if (!normalizedName.isValid) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'MEDIUM',\n        source: 'SubmitToPortfolioTool.execute',\n        details: `Invalid Unicode in element name: ${normalizedName.detectedIssues?.[0] || 'unknown error'}`\n      });\n      return {\n        success: false,\n        error: {\n          success: false,\n          message: `Invalid characters in element name: ${normalizedName.detectedIssues?.[0] || 'unknown error'}`,\n          error: 'INVALID_INPUT'\n        }\n      };\n    }\n    \n    return {\n      success: true,\n      safeName: normalizedName.normalizedContent\n    };\n  }\n\n  /**\n   * Checks if the user is authenticated with GitHub\n   * @returns Authentication check result with status or error response\n   */\n  private async checkAuthentication(): Promise<{\n    success: boolean;\n    authStatus?: any;\n    error?: SubmitToPortfolioResult;\n  }> {\n    const authStatus = await this.authManager.getAuthStatus();\n    if (!authStatus.isAuthenticated) {\n      // Log authentication required (using existing event type)\n      logger.warn('User attempted portfolio submission without authentication');\n      return {\n        success: false,\n        error: {\n          success: false,\n          message: 'Not authenticated. Please authenticate first using the GitHub OAuth flow.\\n\\n' +\n                   'Visit: https://docs.anthropic.com/en/docs/claude-code/oauth-setup\\n' +\n                   'Or run: gh auth login --web',\n          error: 'NOT_AUTHENTICATED'\n        }\n      };\n    }\n\n    return {\n      success: true,\n      authStatus\n    };\n  }\n\n  /**\n   * Discovers content locally with smart type detection\n   * @param safeName The normalized name to search for\n   * @param explicitType Optional explicit element type provided by user\n   * @param originalName Original user-provided name for error messages\n   * @returns Content discovery result with element type and path or error response\n   */\n  private async discoverContentWithTypeDetection(\n    safeName: string, \n    explicitType?: ElementType, \n    originalName?: string\n  ): Promise<{\n    success: boolean;\n    elementType?: ElementType;\n    localPath?: string;\n    error?: SubmitToPortfolioResult;\n  }> {\n    let elementType = explicitType;\n    let localPath: string | null = null;\n    \n    if (elementType) {\n      // Type explicitly provided - search in that specific directory only\n      localPath = await this.findLocalContent(safeName, elementType);\n      if (!localPath) {\n        // UX IMPROVEMENT: Provide helpful suggestions for finding content\n        const portfolioManager = PortfolioManager.getInstance();\n        const elementDir = portfolioManager.getElementDir(elementType);\n        \n        return {\n          success: false,\n          error: {\n            success: false,\n            message: `Could not find ${elementType} named \"${originalName || safeName}\" in local portfolio.\\n\\n` +\n                    `**Searched in**: ${elementDir}\\n\\n` +\n                    `**Troubleshooting Tips**:\\n` +\n                    `• Check if the file exists using your file explorer\\n` +\n                    `• Try using the exact filename (without extension)\\n` +\n                    `• Use \\`list_portfolio\\` to see all available ${elementType}\\n` +\n                    `• If unsure of the type, omit --type and let the system detect it\\n\\n` +\n                    `**Common name formats that work**:\\n` +\n                    `• \"my-element\" (kebab-case)\\n` +\n                    `• \"My Element\" (with spaces)\\n` +\n                    `• \"MyElement\" (PascalCase)\\n` +\n                    `• Partial matches are supported`,\n            error: 'CONTENT_NOT_FOUND'\n          }\n        };\n      }\n    } else {\n      // CRITICAL FIX: No type provided - implement smart detection across ALL element types\n      // This prevents the previous hardcoded default to PERSONA and enables proper type detection\n      const detectionResult = await this.detectElementType(safeName);\n      \n      if (!detectionResult.found) {\n        // UX IMPROVEMENT: Enhanced guidance with specific suggestions\n        const availableTypes = Object.values(ElementType).join(', ');\n        \n        // Get suggestions for similar names\n        const suggestions = await this.generateNameSuggestions(safeName);\n        \n        let message = `Content \"${originalName || safeName}\" not found in portfolio.\\n\\n`;\n        message += `🔍 **Searched in all element types**: ${availableTypes}\\n\\n`;\n        \n        if (suggestions.length > 0) {\n          message += `💡 **Did you mean one of these?**\\n`;\n          for (const suggestion of suggestions.slice(0, SEARCH_CONFIG.MAX_SUGGESTIONS)) {\n            message += `  • \"${suggestion.name}\" (${suggestion.type})\\n`;\n          }\n          message += `\\n`;\n        }\n        \n        message += `🛠️ **Troubleshooting Steps**:\\n`;\n        message += `1. 📝 Use \\`list_portfolio\\` to see all available content\\n`;\n        message += `2. 🔍 Check exact spelling and try variations:\\n`;\n        message += `   • \"${(originalName || safeName).toLowerCase()}\" (lowercase)\\n`;\n        message += `   • \"${(originalName || safeName).replace(/[^a-z0-9]/gi, '-').toLowerCase()}\" (normalized)\\n`;\n        if ((originalName || safeName).includes('.')) {\n          message += `   • \"${(originalName || safeName).replace(/\\./g, '')}\" (no dots)\\n`;\n        }\n        message += `3. 🎯 Specify element type: \\`submit_content \"${originalName || safeName}\" --type=personas\\`\\n`;\n        message += `4. 📁 Check if file exists in portfolio directories\\n\\n`;\n        message += `📝 **Tip**: The system searches filenames AND metadata names with fuzzy matching.`;\n        \n        return {\n          success: false,\n          error: {\n            success: false,\n            message,\n            error: 'CONTENT_NOT_FOUND'\n          }\n        };\n      }\n      \n      if (detectionResult.matches.length > 1) {\n        // Multiple matches found - ask user to specify type\n        const matchDetails = detectionResult.matches.map(m => `- ${m.type}: ${m.path}`).join('\\n');\n        return {\n          success: false,\n          error: {\n            success: false,\n            message: `Content \"${originalName || safeName}\" found in multiple element types:\\n\\n${matchDetails}\\n\\n` +\n                    `Please specify the element type using the --type parameter to avoid ambiguity.`,\n            error: 'MULTIPLE_MATCHES_FOUND'\n          }\n        };\n      }\n      \n      // Single match found - use it\n      const match = detectionResult.matches[0];\n      elementType = match.type;\n      localPath = match.path;\n      \n      logger.info(`Smart detection: Found \"${safeName}\" as ${elementType}`, { \n        name: safeName,\n        detectedType: elementType,\n        path: localPath\n      });\n    }\n\n    return {\n      success: true,\n      elementType,\n      localPath\n    };\n  }\n\n  /**\n   * Validates file size and content security before processing\n   * @param localPath Path to the local file to validate\n   * @returns Validation result with content or error response\n   */\n  private async validateFileAndContent(localPath: string): Promise<{\n    success: boolean;\n    content?: string;\n    error?: SubmitToPortfolioResult;\n  }> {\n    // SECURITY ENHANCEMENT (Task #7): Validate file path before processing\n    const pathValidation = await this.validatePortfolioPath(localPath);\n    if (!pathValidation.isValid) {\n      return {\n        success: false,\n        error: pathValidation.error\n      };\n    }\n\n    // Use the validated safe path for all subsequent operations\n    const safePath = pathValidation.safePath!;\n\n    // Validate file size before reading\n    const stats = await fs.stat(safePath);\n    if (stats.size > FILE_SIZE_LIMITS.MAX_FILE_SIZE) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'RATE_LIMIT_EXCEEDED',\n        severity: 'MEDIUM',\n        source: 'SubmitToPortfolioTool.execute',\n        details: `File size ${stats.size} exceeds limit of ${FILE_SIZE_LIMITS.MAX_FILE_SIZE}`\n      });\n      return {\n        success: false,\n        error: {\n          success: false,\n          message: `File size exceeds ${FILE_SIZE_LIMITS.MAX_FILE_SIZE_MB}MB limit`,\n          error: 'FILE_TOO_LARGE'\n        }\n      };\n    }\n\n    // Validate content security\n    const content = await fs.readFile(safePath, 'utf-8');\n    const validationResult = ContentValidator.validateAndSanitize(content);\n\n    if (!validationResult.isValid && validationResult.severity === 'critical') {\n      SecurityMonitor.logSecurityEvent({\n        type: 'CONTENT_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'SubmitToPortfolioTool.execute',\n        details: `Critical security issues detected: ${validationResult.detectedPatterns?.join(', ')}`\n      });\n      return {\n        success: false,\n        error: {\n          success: false,\n          message: `Content validation failed: ${validationResult.detectedPatterns?.join(', ')}`,\n          error: 'VALIDATION_FAILED'\n        }\n      };\n    }\n\n    return {\n      success: true,\n      content\n    };\n  }\n\n  /**\n   * Prepares metadata for the portfolio element\n   * @param safeName The normalized name of the element\n   * @param elementType The type of the element\n   * @param authStatus Authentication status containing username\n   * @returns Metadata object for the element\n   */\n  private prepareElementMetadata(\n    safeName: string, \n    elementType: ElementType, \n    authStatus: any\n  ): PortfolioElementMetadata {\n    return {\n      name: safeName,\n      description: `${elementType} submitted from local portfolio`,\n      author: authStatus.username || 'unknown',\n      created: new Date().toISOString(),\n      updated: new Date().toISOString(),\n      version: '1.0.0'\n    };\n  }\n\n  /**\n   * Validates GitHub token and checks for expiration before usage\n   * SECURITY ENHANCEMENT (Task #5): Token expiration validation to prevent stale token usage\n   * @param token The GitHub token to validate\n   * @returns Validation result with status and expiration info\n   */\n  private async validateTokenBeforeUsage(token: string): Promise<{\n    isValid: boolean;\n    isNearExpiry?: boolean;\n    error?: SubmitToPortfolioResult;\n  }> {\n    try {\n      // Check token format first (basic validation)\n      if (!TokenManager.validateTokenFormat(token)) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'TOKEN_VALIDATION_FAILURE',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',\n          details: 'Token has invalid format'\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'Invalid token format. Please re-authenticate.',\n            error: 'INVALID_TOKEN_FORMAT'\n          }\n        };\n      }\n\n      // Validate token with GitHub API to check expiration and permissions\n      const validationResult = await TokenManager.validateTokenScopes(token, {\n        required: ['repo'],\n        optional: ['user:email']\n      });\n\n      if (!validationResult.isValid) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'TOKEN_VALIDATION_FAILURE',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',\n          details: `Token validation failed: ${validationResult.error}`\n        });\n\n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'GitHub token is invalid or expired. Please re-authenticate.',\n            error: 'TOKEN_VALIDATION_FAILED'\n          }\n        };\n      }\n\n      // Check if token is near expiration (rate limit reset time can indicate token freshness)\n      let isNearExpiry = false;\n      if (validationResult.rateLimit?.resetTime) {\n        const now = new Date();\n        const timeUntilReset = validationResult.rateLimit.resetTime.getTime() - now.getTime();\n        const oneHour = 60 * 60 * 1000;\n        \n        // Consider token \"near expiry\" if rate limit reset is more than 23 hours away\n        // (GitHub rate limits reset every hour, so this suggests token age)\n        if (timeUntilReset > 23 * oneHour) {\n          isNearExpiry = true;\n          logger.warn('GitHub token may be near expiration', {\n            tokenPrefix: TokenManager.getTokenPrefix(token),\n            rateLimitResetTime: validationResult.rateLimit.resetTime,\n            recommendation: 'Consider re-authenticating for long operations'\n          });\n        }\n      }\n\n      // Log successful validation\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_SUCCESS',\n        severity: 'LOW',\n        source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',\n        details: 'GitHub token validated successfully before usage',\n        metadata: {\n          tokenType: TokenManager.getTokenType(token),\n          scopes: validationResult.scopes,\n          rateLimitRemaining: validationResult.rateLimit?.remaining,\n          isNearExpiry\n        }\n      });\n\n      return {\n        isValid: true,\n        isNearExpiry\n      };\n\n    } catch (error: any) {\n      // Handle rate limit exceeded specifically\n      if (error?.code === 'RATE_LIMIT_EXCEEDED') {\n        logger.warn('Token validation rate limited, allowing operation to proceed with cached status');\n        return { isValid: true }; // Allow to proceed if rate limited, as basic format check passed\n      }\n\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_FAILURE',\n        severity: 'HIGH',\n        source: 'SubmitToPortfolioTool.validateTokenBeforeUsage',\n        details: `Token validation error: ${error.message || 'unknown error'}`\n      });\n\n      return {\n        isValid: false,\n        error: {\n          success: false,\n          message: 'Unable to validate GitHub token. Please check your connection and try again.',\n          error: 'TOKEN_VALIDATION_ERROR'\n        }\n      };\n    }\n  }\n\n  /**\n   * Enhanced path validation for portfolio operations with comprehensive security checks\n   * SECURITY ENHANCEMENT (Task #7): Additional validation for special characters and malicious patterns\n   * @param filePath The file path to validate\n   * @returns Validation result with secure path or error response\n   */\n  private async validatePortfolioPath(filePath: string): Promise<{\n    isValid: boolean;\n    safePath?: string;\n    error?: SubmitToPortfolioResult;\n  }> {\n    try {\n      // Basic null/undefined check\n      if (!filePath || typeof filePath !== 'string') {\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validatePortfolioPath',\n          details: 'Invalid path provided - null, undefined, or non-string'\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'Invalid file path provided',\n            error: 'INVALID_PATH'\n          }\n        };\n      }\n\n      // Check for suspicious patterns that could indicate path traversal or injection\n      const suspiciousPatterns = [\n        /\\.\\./,                    // Path traversal\n        /\\/\\.\\./,                  // Unix path traversal\n        /\\\\\\.\\./,                  // Windows path traversal\n        /\\x00/,                    // Null bytes\n        /[\\x01-\\x1f\\x7f-\\x9f]/,    // Control characters\n        /[<>:\"|?*]/,               // Invalid filename characters on Windows\n        /^(CON|PRN|AUX|NUL|COM[1-9]|LPT[1-9])$/i, // Reserved Windows names\n        /^\\./,                     // Hidden files (starting with dot)\n        /\\s+$/,                    // Trailing whitespace\n        /^[\\s]*$/,                 // Only whitespace\n        /%[0-9a-fA-F]{2}/,         // URL encoding (potential bypass attempt)\n        /\\\\x[0-9a-fA-F]{2}/,       // Hex encoding\n        /\\$\\{.*\\}/,                // Template literal injection\n        /`.*`/,                    // Backtick injection\n        /[\\\\\\/]{2,}/               // Multiple consecutive slashes\n      ];\n\n      for (const pattern of suspiciousPatterns) {\n        if (pattern.test(filePath)) {\n          SecurityMonitor.logSecurityEvent({\n            type: 'PATH_TRAVERSAL_ATTEMPT',\n            severity: 'HIGH',\n            source: 'SubmitToPortfolioTool.validatePortfolioPath',\n            details: `Suspicious pattern detected in file path: ${pattern.source}`,\n            metadata: {\n              pathLength: filePath.length,\n              pattern: pattern.source\n            }\n          });\n          \n          return {\n            isValid: false,\n            error: {\n              success: false,\n              message: 'File path contains invalid or suspicious characters',\n              error: 'SUSPICIOUS_PATH_PATTERN'\n            }\n          };\n        }\n      }\n\n      // Check path length (prevent buffer overflow attempts)\n      const MAX_PATH_LENGTH = process.platform === 'win32' ? 260 : 4096;\n      if (filePath.length > MAX_PATH_LENGTH) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validatePortfolioPath',\n          details: `File path exceeds maximum length: ${filePath.length} > ${MAX_PATH_LENGTH}`\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'File path is too long',\n            error: 'PATH_TOO_LONG'\n          }\n        };\n      }\n\n      // Normalize path to resolve any relative components safely\n      let normalizedPath: string;\n      try {\n        // Remove null bytes and normalize\n        const cleanPath = filePath.replace(/\\x00/g, '');\n        normalizedPath = path.normalize(cleanPath);\n        \n        // Ensure the normalized path doesn't escape the intended directory\n        if (normalizedPath.includes('..') || normalizedPath.startsWith('/') || \n            (process.platform === 'win32' && /^[a-zA-Z]:/.test(normalizedPath))) {\n          throw new Error('Path normalization resulted in directory traversal');\n        }\n      } catch (error) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'HIGH',\n          source: 'SubmitToPortfolioTool.validatePortfolioPath',\n          details: `Path normalization failed: ${error instanceof Error ? error.message : 'unknown error'}`\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'File path could not be safely processed',\n            error: 'PATH_NORMALIZATION_FAILED'\n          }\n        };\n      }\n\n      // Validate file extension (only allow safe extensions for portfolio content)\n      const allowedExtensions = ['.md', '.markdown', '.txt', '.yml', '.yaml', '.json'];\n      const fileExtension = path.extname(normalizedPath).toLowerCase();\n      \n      if (fileExtension && !allowedExtensions.includes(fileExtension)) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validatePortfolioPath',\n          details: `Disallowed file extension: ${fileExtension}`,\n          metadata: {\n            allowedExtensions: allowedExtensions.join(', ')\n          }\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: `File extension '${fileExtension}' is not allowed. Allowed extensions: ${allowedExtensions.join(', ')}`,\n            error: 'INVALID_FILE_EXTENSION'\n          }\n        };\n      }\n\n      // Validate filename characters (only allow safe characters)\n      const basename = path.basename(normalizedPath);\n      const safeFilenamePattern = /^[a-zA-Z0-9\\-_.\\s()[\\]{}]+$/;\n      \n      if (basename && !safeFilenamePattern.test(basename)) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: 'MEDIUM',\n          source: 'SubmitToPortfolioTool.validatePortfolioPath',\n          details: 'Filename contains potentially dangerous characters',\n          metadata: {\n            filename: basename,\n            allowedPattern: safeFilenamePattern.source\n          }\n        });\n        \n        return {\n          isValid: false,\n          error: {\n            success: false,\n            message: 'Filename contains invalid characters. Only letters, numbers, spaces, hyphens, underscores, dots, and common brackets are allowed.',\n            error: 'INVALID_FILENAME_CHARACTERS'\n          }\n        };\n      }\n\n      // Log successful validation\n      SecurityMonitor.logSecurityEvent({\n        type: 'CONTENT_INJECTION_ATTEMPT',\n        severity: 'LOW',\n        source: 'SubmitToPortfolioTool.validatePortfolioPath',\n        details: 'File path validation successful',\n        metadata: {\n          originalPathLength: filePath.length,\n          normalizedPathLength: normalizedPath.length,\n          fileExtension: fileExtension || 'none'\n        }\n      });\n\n      return {\n        isValid: true,\n        safePath: normalizedPath\n      };\n\n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'HIGH',\n        source: 'SubmitToPortfolioTool.validatePortfolioPath',\n        details: `Path validation error: ${error instanceof Error ? error.message : 'unknown error'}`\n      });\n      \n      return {\n        isValid: false,\n        error: {\n          success: false,\n          message: 'Unable to validate file path. Please check the file path and try again.',\n          error: 'PATH_VALIDATION_ERROR'\n        }\n      };\n    }\n  }\n\n  /**\n   * Smart token management for long operations with refresh-like capabilities\n   * SECURITY ENHANCEMENT (Task #14): Token refresh logic for long operations\n   * \n   * Note: GitHub OAuth device flow tokens don't have traditional refresh tokens,\n   * but we can implement smart validation and guidance for long operations\n   * \n   * @param operationType Type of operation being performed\n   * @returns Token management result with recommendations\n   */\n  private async manageTokenForLongOperation(operationType: 'portfolio_creation' | 'collection_submission' | 'file_upload'): Promise<{\n    canProceed: boolean;\n    token?: string;\n    refreshRecommended?: boolean;\n    error?: SubmitToPortfolioResult;\n  }> {\n    try {\n      // Get current token\n      const token = await TokenManager.getGitHubTokenAsync();\n      if (!token) {\n        return {\n          canProceed: false,\n          error: {\n            success: false,\n            message: 'No GitHub token available. Please authenticate first.',\n            error: 'NO_TOKEN'\n          }\n        };\n      }\n\n      // Validate token for the specific operation\n      const validation = await this.validateTokenBeforeUsage(token);\n      if (!validation.isValid) {\n        return {\n          canProceed: false,\n          error: validation.error\n        };\n      }\n\n      // Check if this is a long operation that might benefit from fresh authentication\n      const longOperations = ['portfolio_creation', 'collection_submission'];\n      const isLongOperation = longOperations.includes(operationType);\n\n      // Get token type to determine refresh capabilities\n      const tokenType = TokenManager.getTokenType(token);\n      let refreshRecommended = false;\n\n      // For long operations, check token age and recommend refresh if needed\n      if (isLongOperation && validation.isNearExpiry) {\n        refreshRecommended = true;\n        \n        SecurityMonitor.logSecurityEvent({\n          type: 'TOKEN_VALIDATION_SUCCESS',\n          severity: 'LOW',\n          source: 'SubmitToPortfolioTool.manageTokenForLongOperation',\n          details: 'Long operation detected with aging token - refresh recommended',\n          metadata: {\n            operationType,\n            tokenType,\n            refreshRecommended: true\n          }\n        });\n\n        logger.warn('Long operation with potentially aging token detected', {\n          operationType,\n          tokenType,\n          recommendation: 'Consider re-authenticating if operation fails'\n        });\n      }\n\n      // For OAuth tokens in long operations, we can provide guidance\n      if (tokenType === 'OAuth Access Token' && isLongOperation) {\n        logger.info('OAuth token detected for long operation', {\n          operationType,\n          tokenType,\n          guidance: 'OAuth tokens are time-limited. If operation fails, re-authenticate using setup_github_auth'\n        });\n      }\n\n      // Log successful token management\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_SUCCESS',\n        severity: 'LOW',\n        source: 'SubmitToPortfolioTool.manageTokenForLongOperation',\n        details: 'Token management successful for long operation',\n        metadata: {\n          operationType,\n          tokenType,\n          isLongOperation,\n          refreshRecommended\n        }\n      });\n\n      return {\n        canProceed: true,\n        token,\n        refreshRecommended\n      };\n\n    } catch (error: any) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_FAILURE',\n        severity: 'MEDIUM',\n        source: 'SubmitToPortfolioTool.manageTokenForLongOperation',\n        details: `Token management error: ${error.message || 'unknown error'}`\n      });\n\n      return {\n        canProceed: false,\n        error: {\n          success: false,\n          message: 'Unable to manage token for operation. Please check your authentication and try again.',\n          error: 'TOKEN_MANAGEMENT_ERROR'\n        }\n      };\n    }\n  }\n\n  /**\n   * Provides user guidance for token refresh when operations fail due to token issues\n   * SECURITY ENHANCEMENT (Task #14): User guidance for authentication refresh\n   */\n  private formatTokenRefreshGuidance(operationType: string, tokenType: string): string {\n    let guidance = '\\n\\n🔄 **Token Refresh Guidance**:\\n';\n    \n    if (tokenType === 'OAuth Access Token') {\n      guidance += '• Your OAuth token may have expired\\n';\n      guidance += '• Run `setup_github_auth` to authenticate again\\n';\n      guidance += '• This will generate a fresh token for continued access\\n';\n    } else if (tokenType === 'Personal Access Token') {\n      guidance += '• Your Personal Access Token may have expired\\n';\n      guidance += '• Check your GitHub settings: https://github.com/settings/tokens\\n';\n      guidance += '• Generate a new token if needed and update GITHUB_TOKEN environment variable\\n';\n    } else {\n      guidance += '• Your GitHub token may have expired or been revoked\\n';\n      guidance += '• Re-authenticate using `setup_github_auth`\\n';\n      guidance += '• Ensure your token has the required permissions\\n';\n    }\n\n    guidance += `\\n**Operation**: ${operationType}\\n`;\n    guidance += '**Required scopes**: repo, user:email\\n\\n';\n    guidance += '💡 **Tip**: Fresh tokens work better for complex operations like portfolio creation.';\n\n    return guidance;\n  }\n\n  /**\n   * Sets up GitHub repository access and ensures portfolio repository exists\n   * @param authStatus Authentication status containing username\n   * @returns Setup result or error response\n   */\n  private async setupGitHubRepository(authStatus: any): Promise<{\n    success: boolean;\n    error?: SubmitToPortfolioResult;\n  }> {\n    // SECURITY ENHANCEMENT (Task #14): Smart token management for long operations\n    const tokenManagement = await this.manageTokenForLongOperation('portfolio_creation');\n    if (!tokenManagement.canProceed) {\n      return {\n        success: false,\n        error: tokenManagement.error\n      };\n    }\n\n    const token = tokenManagement.token!;\n\n    // Provide user guidance if refresh is recommended for this long operation\n    if (tokenManagement.refreshRecommended) {\n      const tokenType = TokenManager.getTokenType(token);\n      const guidance = this.formatTokenRefreshGuidance('portfolio creation', tokenType);\n      logger.warn(`Token refresh recommended for portfolio creation:${guidance}`);\n    }\n\n    this.portfolioManager.setToken(token);\n\n    // Check if portfolio exists and create if needed\n    const username = authStatus.username || 'unknown';\n    const portfolioExists = await this.portfolioManager.checkPortfolioExists(username);\n    \n    if (!portfolioExists) {\n      logger.info('Creating portfolio repository...');\n      // Request consent for portfolio creation\n      const repoUrl = await this.portfolioManager.createPortfolio(username, true);\n      if (!repoUrl) {\n        return {\n          success: false,\n          error: {\n            success: false,\n            message: 'Failed to create portfolio repository',\n            error: 'CREATE_FAILED'\n          }\n        };\n      }\n    }\n\n    return { success: true };\n  }\n\n  /**\n   * Submits element to portfolio and handles the complete response workflow\n   * @param safeName The normalized name of the element\n   * @param elementType The type of the element\n   * @param metadata The metadata for the element\n   * @param content The content of the element\n   * @param authStatus Authentication status containing username and token\n   * @returns Complete submission result with success message or error\n   */\n  private async submitElementAndHandleResponse(\n    safeName: string,\n    elementType: ElementType,\n    metadata: PortfolioElementMetadata,\n    content: string,\n    authStatus: any\n  ): Promise<SubmitToPortfolioResult> {\n    // Create element structure to save\n    const element: PortfolioElement = {\n      type: elementType,\n      metadata,\n      content\n    };\n    \n    // TYPE SAFETY FIX #2: Use adapter pattern instead of complex type casting\n    // Previously: element as unknown as Parameters<typeof this.portfolioManager.saveElement>[0]\n    // Now: Clean adapter pattern that implements IElement interface properly\n    const adapter = new PortfolioElementAdapter(element);\n    \n    // UX IMPROVEMENT: Add retry logic for transient failures\n    const fileUrl = await this.saveElementWithRetry(adapter, safeName, elementType);\n    \n    if (!fileUrl) {\n      return {\n        success: false,\n        message: 'Failed to save element to GitHub portfolio after multiple attempts.\\n\\n' +\n                '💡 **Troubleshooting Tips**:\\n' +\n                '• Check your GitHub authentication: `gh auth status`\\n' +\n                '• Verify repository permissions\\n' +\n                '• Try again in a few minutes (GitHub API rate limits)\\n' +\n                '• Check GitHub status: https://status.github.com',\n        error: 'SAVE_FAILED'\n      };\n    }\n\n    // Log successful submission (DMCP-SEC-006)\n    logger.info(`Successfully submitted ${safeName} to GitHub portfolio`, {\n      elementType,\n      username: authStatus.username,\n      fileUrl\n    });\n\n    // SECURITY ENHANCEMENT (Task #14): Smart token management for collection submission\n    const collectionTokenManagement = await this.manageTokenForLongOperation('collection_submission');\n    if (!collectionTokenManagement.canProceed) {\n      // Token management failed for collection submission, but main submission succeeded\n      const errorMessage = collectionTokenManagement.error?.message || 'Token management failed';\n      return {\n        success: true,\n        message: `✅ Successfully uploaded ${safeName} to your GitHub portfolio!\\n📁 Portfolio URL: ${fileUrl}\\n\\n⚠️ Collection submission skipped: ${errorMessage}`,\n        url: fileUrl\n      };\n    }\n\n    const token = collectionTokenManagement.token!;\n\n    // Provide refresh guidance if recommended for collection submission\n    if (collectionTokenManagement.refreshRecommended) {\n      const tokenType = TokenManager.getTokenType(token);\n      logger.info('Collection submission proceeding with aging token', {\n        tokenType,\n        recommendation: 'If collection submission fails, try re-authenticating with setup_github_auth'\n      });\n    }\n\n    // ENHANCEMENT (Issue #549): Ask user if they want to submit to collection\n    // This completes the community contribution workflow\n    const collectionSubmissionResult = await this.promptForCollectionSubmission({\n      elementName: safeName,\n      elementType,\n      portfolioUrl: fileUrl,\n      username: authStatus.username || 'unknown',\n      metadata,\n      token\n    });\n\n    // Build the response message based on what happened\n    let message = `✅ Successfully uploaded ${safeName} to your GitHub portfolio!\\n`;\n    message += `📁 Portfolio URL: ${fileUrl}\\n\\n`;\n    \n    if (collectionSubmissionResult.submitted) {\n      message += `🎉 Also submitted to DollhouseMCP collection for community review!\\n`;\n      message += `📋 Issue: ${collectionSubmissionResult.issueUrl}`;\n    } else if (collectionSubmissionResult.declined) {\n      message += `💡 You can submit to the collection later using the same command.`;\n    } else if (collectionSubmissionResult.error) {\n      message += `⚠️ Collection submission failed: ${collectionSubmissionResult.error}\\n`;\n      message += `💡 You can manually submit at: https://github.com/DollhouseMCP/collection/issues/new`;\n    }\n\n    return {\n      success: true,\n      message,\n      url: fileUrl\n    };\n  }\n\n  async execute(params: SubmitToPortfolioParams): Promise<SubmitToPortfolioResult> {\n    try {\n      // Validate and normalize input parameters\n      const validationResult = await this.validateAndNormalizeParams(params);\n      if (!validationResult.success) {\n        return validationResult.error!;\n      }\n      const safeName = validationResult.safeName!;\n\n      // Check authentication status\n      const authResult = await this.checkAuthentication();\n      if (!authResult.success) {\n        return authResult.error!;\n      }\n      const authStatus = authResult.authStatus!;\n\n      // Find content locally with smart type detection\n      const contentResult = await this.discoverContentWithTypeDetection(safeName!, params.type, params.name);\n      if (!contentResult.success) {\n        return contentResult.error!;\n      }\n      const elementType = contentResult.elementType!;\n      const localPath = contentResult.localPath!;\n\n      // Validate file and content security\n      const securityResult = await this.validateFileAndContent(localPath);\n      if (!securityResult.success) {\n        return securityResult.error!;\n      }\n      const content = securityResult.content!;\n\n      // Get user consent (placeholder for now - could add interactive prompt later)\n      logger.info(`Preparing to submit ${safeName} to GitHub portfolio`);\n\n      // Prepare metadata for element\n      const metadata = this.prepareElementMetadata(safeName!, elementType, authStatus);\n\n      // Set up GitHub repository access\n      const repoResult = await this.setupGitHubRepository(authStatus);\n      if (!repoResult.success) {\n        return repoResult.error!;\n      }\n\n      // Submit element to portfolio and handle collection submission\n      return await this.submitElementAndHandleResponse(\n        safeName!, \n        elementType, \n        metadata, \n        content, \n        authStatus\n      );\n\n    } catch (error) {\n      // SECURITY ENHANCEMENT (Task #14): Enhanced error handling with token refresh guidance\n      ErrorHandler.logError('submitToPortfolio', error, {\n        elementName: params.name,\n        elementType: params.type\n      });\n\n      // Check if error is token-related and provide refresh guidance\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      const isTokenError = errorMessage.toLowerCase().includes('token') || \n                          errorMessage.toLowerCase().includes('auth') ||\n                          errorMessage.toLowerCase().includes('401') ||\n                          errorMessage.toLowerCase().includes('403');\n\n      let formattedError = ErrorHandler.formatForResponse(error);\n\n      if (isTokenError) {\n        try {\n          // Get current token to determine type for guidance\n          const currentToken = await TokenManager.getGitHubTokenAsync();\n          if (currentToken) {\n            const tokenType = TokenManager.getTokenType(currentToken);\n            const refreshGuidance = this.formatTokenRefreshGuidance('portfolio submission', tokenType);\n            \n            // Append refresh guidance to error message\n            if (formattedError.message) {\n              formattedError.message += refreshGuidance;\n            }\n          }\n        } catch (tokenError) {\n          // If we can't get token info, provide generic guidance\n          formattedError.message += '\\n\\n🔄 **Authentication Issue**: Try running `setup_github_auth` to refresh your authentication.';\n        }\n      }\n\n      return formattedError;\n    }\n  }\n\n  /**\n   * Prompts user to submit content to the DollhouseMCP collection\n   * ENHANCEMENT (Issue #549): Complete the community contribution workflow\n   */\n  private async promptForCollectionSubmission(params: {\n    elementName: string;\n    elementType: ElementType;\n    portfolioUrl: string;\n    username: string;\n    metadata: PortfolioElementMetadata;\n    token: string;\n  }): Promise<{ submitted: boolean; declined: boolean; error?: string; issueUrl?: string }> {\n    try {\n      // Create a simple prompt message for the user\n      // Note: In MCP context, we can't do interactive prompts, so we'll need to\n      // either make this automatic or require a parameter\n      \n      // For now, let's check if the user has set an environment variable\n      // to auto-submit to collection (opt-in behavior)\n      const autoSubmit = process.env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION === 'true';\n      \n      if (!autoSubmit) {\n        // User hasn't opted in to auto-submission\n        logger.info('Collection submission skipped (set DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION=true to enable)');\n        return { submitted: false, declined: true };\n      }\n\n      logger.info('Auto-submitting to DollhouseMCP collection...');\n\n      // Create the issue in the collection repository\n      const issueUrl = await this.createCollectionIssue({\n        ...params,\n        token: params.token\n      });\n\n      if (issueUrl) {\n        logger.info('Successfully created collection submission issue', { issueUrl });\n        return { submitted: true, declined: false, issueUrl };\n      } else {\n        return { submitted: false, declined: false, error: 'Failed to create issue' };\n      }\n\n    } catch (error) {\n      logger.error('Error in collection submission prompt', { error });\n      return {\n        submitted: false,\n        declined: false,\n        error: error instanceof Error ? error.message : 'Unknown error'\n      };\n    }\n  }\n\n  /**\n   * Creates an issue in the DollhouseMCP/collection repository\n   * ENHANCEMENT (Issue #549): GitHub API integration for collection submission\n   */\n  private async createCollectionIssue(params: {\n    elementName: string;\n    elementType: ElementType;\n    portfolioUrl: string;\n    username: string;\n    metadata: PortfolioElementMetadata;\n    token: string;\n  }): Promise<string | null> {\n    try {\n\n      // Format the issue title\n      const title = `[${params.elementType}] Add ${params.elementName} by @${params.username}`;\n\n      // Format the issue body with all relevant information\n      const body = `## New ${params.elementType} Submission\n\n` +\n        `**Name**: ${params.elementName}\\n` +\n        `**Author**: @${params.username}\\n` +\n        `**Type**: ${params.elementType}\\n` +\n        `**Description**: ${params.metadata.description || 'No description provided'}\\n\\n` +\n        `### Portfolio Link\\n` +\n        `${params.portfolioUrl}\\n\\n` +\n        `### Metadata\\n` +\n        `\\`\\`\\`json\\n${JSON.stringify(params.metadata, null, 2)}\\n\\`\\`\\`\\n\\n` +\n        `### Review Checklist\\n` +\n        `- [ ] Content is appropriate and follows community guidelines\\n` +\n        `- [ ] No security vulnerabilities or malicious patterns\\n` +\n        `- [ ] Metadata is complete and accurate\\n` +\n        `- [ ] Element works as described\\n` +\n        `- [ ] No duplicate of existing collection content\\n\\n` +\n        `---\\n` +\n        `*This submission was created automatically via the DollhouseMCP submit_content tool.*`;\n\n      // Determine labels based on element type\n      const labels = [\n        'contribution',  // All submissions get this\n        'pending-review', // Needs review\n        params.elementType.toLowerCase() // Element type label\n      ];\n\n      // PERFORMANCE OPTIMIZATION (Task #6): Use GitHub rate limiter for API calls\n      // This prevents hitting GitHub rate limits and provides better error handling\n      const issueUrl = await githubRateLimiter.queueRequest(\n        'create-collection-issue',\n        async () => {\n          const url = 'https://api.github.com/repos/DollhouseMCP/collection/issues';\n          \n          // Create AbortController for timeout\n          const controller = new AbortController();\n          const timeoutId = setTimeout(() => controller.abort(), getValidatedTimeout());\n          \n          try {\n            const response = await fetch(url, {\n              method: 'POST',\n              headers: {\n                'Accept': 'application/vnd.github.v3+json',\n                'Authorization': `Bearer ${params.token}`,\n                'Content-Type': 'application/json',\n                'User-Agent': 'DollhouseMCP/1.0'\n              },\n              body: JSON.stringify({\n                title,\n                body,\n                labels\n              }),\n              signal: controller.signal\n            });\n            \n            clearTimeout(timeoutId);\n\n            // PERFORMANCE OPTIMIZATION (Task #15): Enhanced rate limit logging\n            // Log rate limit headers for diagnostics\n            const rateLimitRemaining = response.headers.get('X-RateLimit-Remaining');\n            const rateLimitReset = response.headers.get('X-RateLimit-Reset');\n            const rateLimitLimit = response.headers.get('X-RateLimit-Limit');\n            \n            logger.debug('GitHub API rate limit status', {\n              operation: 'create-collection-issue',\n              remaining: rateLimitRemaining,\n              limit: rateLimitLimit,\n              resetTime: rateLimitReset ? new Date(parseInt(rateLimitReset) * 1000) : undefined,\n              responseStatus: response.status\n            });\n            \n            // Log warning if approaching rate limit\n            if (rateLimitRemaining && parseInt(rateLimitRemaining) < 100) {\n              logger.warn('Approaching GitHub API rate limit', {\n                operation: 'create-collection-issue',\n                remaining: rateLimitRemaining,\n                resetTime: rateLimitReset ? new Date(parseInt(rateLimitReset) * 1000) : undefined,\n                recommendation: 'Consider reducing API usage frequency or authenticating for higher limits'\n              });\n            }\n\n            if (!response.ok) {\n              const errorText = await response.text();\n              logger.error('GitHub API error creating issue', { \n                status: response.status, \n                statusText: response.statusText,\n                error: errorText,\n                rateLimitRemaining,\n                rateLimitReset\n              });\n              \n              if (response.status === 404) {\n                logger.error('Collection repository not found or no access');\n              } else if (response.status === 403) {\n                logger.error('Permission denied to create issue in collection repo');\n              } else if (response.status === 401) {\n                logger.error('Authentication failed for collection submission');\n              }\n              throw new Error(`GitHub API error: ${response.status} ${response.statusText}`);\n            }\n\n            const data = await response.json();\n            return data.html_url;\n            \n          } catch (fetchError: any) {\n            // Re-throw to outer catch block\n            throw fetchError;\n          } finally {\n            clearTimeout(timeoutId);\n          }\n        },\n        'high' // High priority for collection submission\n      );\n      \n      return issueUrl;\n\n    } catch (error: any) {\n      // Handle timeout specifically\n      if (error.name === 'AbortError') {\n        logger.error(`GitHub API request timeout after ${getValidatedTimeout()}ms`);\n      } else {\n        logger.error('Failed to create collection issue', { \n          error: error.message || error\n        });\n      }\n      return null;\n    }\n  }\n\n  private async findLocalContent(name: string, type: ElementType): Promise<string | null> {\n    try {\n      // METADATA INDEX FIX: Use portfolio index for fast metadata-based lookups\n      // This solves the critical issue where \"Safe Roundtrip Tester\" couldn't be found\n      // because findLocalContent only searched filenames, not metadata names\n      const indexManager = PortfolioIndexManager.getInstance();\n      \n      // UX IMPROVEMENT: Enhanced search with fuzzy matching\n      const indexEntry = await indexManager.findByName(name, { \n        elementType: type,\n        fuzzyMatch: true\n      });\n      \n      if (indexEntry) {\n        logger.debug('Found content via metadata index', { \n          searchName: name, \n          metadataName: indexEntry.metadata.name,\n          filename: indexEntry.filename,\n          filePath: indexEntry.filePath,\n          type \n        });\n        return indexEntry.filePath;\n      }\n      \n      // FALLBACK: Use original file discovery if index lookup fails\n      // This maintains backward compatibility and handles edge cases\n      logger.debug('Index lookup failed, falling back to file discovery', { name, type });\n      \n      const portfolioManager = PortfolioManager.getInstance();\n      const portfolioDir = portfolioManager.getElementDir(type);\n      \n      // UX IMPROVEMENT: Try multiple search strategies for better user experience\n      let file = await FileDiscoveryUtil.findFile(portfolioDir, name, {\n        extensions: ['.md', '.json', '.yaml', '.yml'],\n        partialMatch: true,\n        cacheResults: true\n      });\n      \n      // If not found, try normalizing the name (e.g., \"J.A.R.V.I.S.\" -> \"j-a-r-v-i-s\")\n      if (!file) {\n        const normalizedName = name.toLowerCase()\n          .replace(/[^a-z0-9]/gi, '-')  // Replace non-alphanumeric with dashes\n          .replace(/-+/g, '-')         // Replace multiple dashes with single dash\n          .replace(/^-|-$/g, '');      // Remove leading/trailing dashes\n          \n        if (normalizedName !== name.toLowerCase()) {\n          logger.debug('Trying normalized name search', { \n            original: name, \n            normalized: normalizedName,\n            type \n          });\n          \n          file = await FileDiscoveryUtil.findFile(portfolioDir, normalizedName, {\n            extensions: ['.md', '.json', '.yaml', '.yml'],\n            partialMatch: true,\n            cacheResults: true\n          });\n        }\n      }\n      \n      // If still not found, try searching by display name patterns\n      if (!file) {\n        // Try common variations like removing dots, spaces, etc.\n        const variations = [\n          name.replace(/\\./g, ''),        // Remove dots: \"J.A.R.V.I.S.\" -> \"JARVIS\"\n          name.replace(/\\s+/g, '-'),      // Replace spaces with dashes\n          name.replace(/[\\s\\.]/g, ''),    // Remove spaces and dots\n          name.replace(/[\\s\\.]/g, '-'),   // Replace spaces and dots with dashes\n        ].filter(v => v !== name && v.length > 0);\n        \n        for (const variation of variations) {\n          file = await FileDiscoveryUtil.findFile(portfolioDir, variation, {\n            extensions: ['.md', '.json', '.yaml', '.yml'],\n            partialMatch: true,\n            cacheResults: true\n          });\n          \n          if (file) {\n            logger.debug('Found content using name variation', {\n              original: name,\n              variation,\n              file,\n              type\n            });\n            break;\n          }\n        }\n      }\n      \n      if (file) {\n        logger.debug('Found local content file via fallback', { name, type, file });\n        return file;\n      }\n      \n      logger.debug('No content found', { name, type });\n      return null;\n      \n    } catch (error) {\n      logger.error('Error finding local content', {\n        name,\n        type,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Smart element type detection - searches across ALL element types for content\n   * PERFORMANCE OPTIMIZATION (Task #9): Uses early termination for exact matches\n   * This replaces the previous hardcoded default to PERSONA and enables proper type detection\n   * \n   * @param name The content name to search for\n   * @returns Detection result with found matches across all element types\n   */\n  private async detectElementType(name: string): Promise<ElementDetectionResult> {\n    try {\n      // PERFORMANCE OPTIMIZATION (Task #9): Use early termination search utility\n      // Create search functions for each element type\n      const elementTypes = Object.values(ElementType);\n      const searchFunctions = elementTypes.map((type) => async () => {\n        try {\n          const filePath = await this.findLocalContent(name, type);\n          if (filePath) {\n            return { type: type as ElementType, path: filePath };\n          }\n          return null;\n        } catch (error: any) {\n          // Log unexpected errors but don't fail the search\n          if (error?.code !== 'ENOENT' && error?.code !== 'ENOTDIR') {\n            logger.debug(`Error searching ${type} directory for content detection`, { \n              name,\n              type,\n              error: error?.message || String(error),\n              code: error?.code \n            });\n          }\n          // Return null instead of throwing to let other searches continue\n          return null;\n        }\n      });\n\n      // PERFORMANCE OPTIMIZATION (Task #9): Define exact match criteria\n      const isExactMatch = (match: ElementDetectionMatch): boolean => {\n        const filename = path.basename(match.path, path.extname(match.path));\n        return filename.toLowerCase() === name.toLowerCase();\n      };\n\n      // Execute searches with early termination optimization\n      const searchResults = await EarlyTerminationSearch.executeWithEarlyTermination(\n        searchFunctions,\n        isExactMatch,\n        {\n          operationName: 'element-type-detection',\n          timeoutAfterExactMatch: 1000, // Wait 1 second for other searches after exact match\n          maxParallelSearches: 8 // Limit concurrent searches to avoid overwhelming the system\n        }\n      );\n\n      // PERFORMANCE OPTIMIZATION (Task #8): Enhanced batch operation reporting\n      const batchResults = {\n        name,\n        totalSearches: searchResults.totalSearches,\n        completedSearches: searchResults.completedSearches,\n        matches: searchResults.matches.length,\n        failures: searchResults.failures.length,\n        exactMatchFound: !!searchResults.exactMatch,\n        exactMatchType: searchResults.exactMatch?.type,\n        earlyTerminationTriggered: searchResults.earlyTerminationTriggered,\n        performanceGain: searchResults.performanceGain,\n        matchedTypes: searchResults.matches.map(m => m.type),\n        failedTypes: searchResults.failures.map(f => elementTypes[f.index]).filter(Boolean)\n      };\n\n      logger.debug('Element type detection completed with early termination optimization', batchResults);\n\n      // PERFORMANCE OPTIMIZATION (Task #8): Clear reporting of partial failures\n      if (searchResults.failures.length > 0) {\n        logger.warn('Some element type searches failed during batch operation', {\n          name,\n          failures: searchResults.failures.map(f => ({\n            type: elementTypes[f.index] || 'unknown',\n            error: f.error.substring(0, 100) // Truncate long error messages\n          })),\n          successRate: `${searchResults.completedSearches}/${searchResults.totalSearches}`,\n          impactOnResults: searchResults.matches.length > 0 \n            ? 'No impact - matches found in successful searches' \n            : 'Potential impact - no matches found'\n        });\n\n        // If we have failures and no matches, provide actionable guidance\n        if (searchResults.matches.length === 0 && searchResults.failures.length > 0) {\n          logger.warn('Batch operation had failures and no matches found', {\n            name,\n            recommendation: 'Consider checking file permissions or portfolio structure',\n            failureCount: searchResults.failures.length,\n            totalSearches: searchResults.totalSearches\n          });\n        }\n      }\n\n      // Log performance gains from early termination\n      if (searchResults.earlyTerminationTriggered) {\n        logger.info('Early termination optimization applied successfully', {\n          name,\n          exactMatchType: searchResults.exactMatch?.type,\n          performanceGain: searchResults.performanceGain,\n          searchesCompleted: searchResults.completedSearches,\n          searchesTotal: searchResults.totalSearches\n        });\n      }\n\n      return {\n        found: searchResults.matches.length > 0,\n        matches: searchResults.matches\n      };\n\n    } catch (error) {\n      logger.error('Error in element type detection', {\n        name,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      \n      // Return empty result on detection failure\n      return {\n        found: false,\n        matches: []\n      };\n    }\n  }\n\n  /**\n   * UX IMPROVEMENT: Generate name suggestions for similar content\n   * PERFORMANCE OPTIMIZATION (Task #8): Enhanced batch operation handling with clear partial failure reporting\n   * Helps users find content when exact matches fail\n   */\n  private async generateNameSuggestions(searchName: string): Promise<Array<{name: string, type: string}>> {\n    try {\n      const suggestions: Array<{name: string, type: string}> = [];\n      const searchLower = searchName.toLowerCase();\n      const elementTypes = Object.values(ElementType);\n      \n      // Track batch operation results for better diagnostics\n      const batchResults = {\n        searchName,\n        totalTypes: elementTypes.length,\n        successfulScans: 0,\n        failedScans: 0,\n        failureDetails: [] as Array<{ type: ElementType; error: string }>,\n        totalSuggestions: 0,\n        suggestionsByType: {} as Record<string, number>\n      };\n      \n      // Process all element types for suggestions\n      for (const elementType of elementTypes) {\n        try {\n          const portfolioManager = PortfolioManager.getInstance();\n          const elementDir = portfolioManager.getElementDir(elementType);\n          \n          // Get files in this directory\n          const files = await FileDiscoveryUtil.findFile(elementDir, '*', {\n            extensions: ['.md', '.json', '.yaml', '.yml'],\n            partialMatch: false,\n            cacheResults: true\n          });\n          \n          let typeSuggestions = 0;\n          \n          if (Array.isArray(files)) {\n            for (const filePath of files) {\n              const basename = path.basename(filePath, path.extname(filePath));\n              \n              // Calculate similarity using simple metrics\n              if (this.calculateSimilarity(searchLower, basename.toLowerCase()) > SEARCH_CONFIG.MIN_SIMILARITY_SCORE) {\n                suggestions.push({\n                  name: basename,\n                  type: elementType\n                });\n                typeSuggestions++;\n              }\n            }\n          } else if (files) {\n            const basename = path.basename(files, path.extname(files));\n            if (this.calculateSimilarity(searchLower, basename.toLowerCase()) > SEARCH_CONFIG.MIN_SIMILARITY_SCORE) {\n              suggestions.push({\n                name: basename,\n                type: elementType\n              });\n              typeSuggestions++;\n            }\n          }\n          \n          batchResults.successfulScans++;\n          batchResults.suggestionsByType[elementType] = typeSuggestions;\n          \n        } catch (error) {\n          // PERFORMANCE OPTIMIZATION (Task #8): Track and report partial failures\n          batchResults.failedScans++;\n          batchResults.failureDetails.push({\n            type: elementType,\n            error: error instanceof Error ? error.message : String(error)\n          });\n          \n          // Log individual failures for diagnostics\n          logger.debug('Failed to scan element type for suggestions', {\n            elementType,\n            searchName,\n            error: error instanceof Error ? error.message : String(error)\n          });\n        }\n      }\n      \n      batchResults.totalSuggestions = suggestions.length;\n      \n      // PERFORMANCE OPTIMIZATION (Task #8): Comprehensive batch operation reporting\n      logger.debug('Name suggestion batch operation completed', {\n        ...batchResults,\n        successRate: `${batchResults.successfulScans}/${batchResults.totalTypes}`,\n        // Don't log full failure details at debug level to avoid spam\n        hasFailures: batchResults.failedScans > 0\n      });\n      \n      // Report failures clearly if they occurred\n      if (batchResults.failedScans > 0) {\n        logger.warn('Some element type scans failed during name suggestion generation', {\n          searchName,\n          failedTypes: batchResults.failureDetails.map(f => f.type),\n          successfulTypes: batchResults.successfulScans,\n          impactOnResults: batchResults.totalSuggestions > 0 \n            ? 'Partial impact - suggestions found from successful scans' \n            : 'Potential impact - no suggestions generated',\n          recommendation: batchResults.totalSuggestions === 0 && batchResults.failedScans > 0\n            ? 'Check portfolio directory structure and file permissions'\n            : 'Suggestion generation partially successful despite some failures'\n        });\n      }\n      \n      // Sort by similarity (higher is better) and return top suggestions\n      const sortedSuggestions = suggestions.sort((a, b) => {\n        const simA = this.calculateSimilarity(searchLower, a.name.toLowerCase());\n        const simB = this.calculateSimilarity(searchLower, b.name.toLowerCase());\n        return simB - simA;\n      });\n      \n      logger.debug('Name suggestions generated successfully', {\n        searchName,\n        totalSuggestions: sortedSuggestions.length,\n        topSuggestions: sortedSuggestions.slice(0, 3).map(s => s.name)\n      });\n      \n      return sortedSuggestions;\n      \n    } catch (error) {\n      logger.warn('Failed to generate name suggestions - batch operation failed completely', { \n        searchName, \n        error: error instanceof Error ? error.message : String(error),\n        recommendation: 'Check portfolio structure and permissions'\n      });\n      return [];\n    }\n  }\n  \n  /**\n   * Simple similarity calculation using Levenshtein-like approach\n   * Returns value between 0 and 1, where 1 is identical\n   */\n  private calculateSimilarity(str1: string, str2: string): number {\n    // Handle exact matches\n    if (str1 === str2) return 1;\n    \n    // Handle substring matches\n    if (str1.includes(str2) || str2.includes(str1)) return 0.8;\n    \n    // Handle partial matches\n    const longer = str1.length > str2.length ? str1 : str2;\n    const shorter = str1.length > str2.length ? str2 : str1;\n    \n    if (longer.length === 0) return 0;\n    \n    // Count common characters\n    let common = 0;\n    for (let i = 0; i < shorter.length; i++) {\n      if (longer.includes(shorter[i])) {\n        common++;\n      }\n    }\n    \n    return common / longer.length;\n  }\n  \n  /**\n   * UX IMPROVEMENT: Save element with automatic retry logic for transient failures\n   * Handles common GitHub API issues like rate limits and temporary network problems\n   */\n  private async saveElementWithRetry(\n    adapter: PortfolioElementAdapter, \n    elementName: string, \n    elementType: ElementType,\n    maxRetries: number = RETRY_CONFIG.MAX_ATTEMPTS\n  ): Promise<string | null> {\n    let lastError: any = null;\n    \n    for (let attempt = 1; attempt <= maxRetries; attempt++) {\n      try {\n        logger.debug(`Attempting to save element (attempt ${attempt}/${maxRetries})`, {\n          elementName,\n          elementType,\n          attempt\n        });\n        \n        const fileUrl = await this.portfolioManager.saveElement(adapter, true);\n        \n        if (fileUrl) {\n          if (attempt > 1) {\n            logger.info(`Element saved successfully after ${attempt} attempts`, {\n              elementName,\n              elementType,\n              fileUrl\n            });\n          }\n          return fileUrl;\n        }\n        \n        // If saveElement returns null, treat as a failure but don't retry immediately\n        lastError = new Error(`saveElement returned null on attempt ${attempt}`);\n        \n      } catch (error: any) {\n        lastError = error;\n        const isRetryable = this.isRetryableError(error);\n        \n        logger.warn(`Save attempt ${attempt} failed`, {\n          elementName,\n          elementType,\n          attempt,\n          error: error.message,\n          isRetryable,\n          willRetry: isRetryable && attempt < maxRetries\n        });\n        \n        // If this is not a retryable error, fail immediately\n        if (!isRetryable) {\n          logger.error('Non-retryable error encountered, aborting retries', {\n            elementName,\n            error: error.message\n          });\n          break;\n        }\n        \n        // If we have more attempts, wait before retrying\n        if (attempt < maxRetries) {\n          const delay = calculateRetryDelay(attempt);\n          logger.debug(`Waiting ${delay}ms before retry`, { attempt, delay });\n          await new Promise(resolve => setTimeout(resolve, delay));\n        }\n      }\n    }\n    \n    // All attempts failed\n    logger.error(`All ${maxRetries} save attempts failed`, {\n      elementName,\n      elementType,\n      lastError: lastError?.message\n    });\n    \n    return null;\n  }\n  \n  /**\n   * Determine if an error is worth retrying\n   * Retryable: network issues, rate limits, temporary GitHub API problems\n   * Non-retryable: authentication issues, validation errors, permanent failures\n   */\n  private isRetryableError(error: any): boolean {\n    const errorMessage = error?.message?.toLowerCase() || '';\n    const errorCode = error?.code;\n    const statusCode = error?.status || error?.statusCode;\n    \n    // Network and timeout errors\n    if (errorCode === 'ENOTFOUND' || errorCode === 'ECONNRESET' || errorCode === 'ETIMEDOUT') {\n      return true;\n    }\n    \n    // GitHub API rate limits\n    if (statusCode === 429 || errorMessage.includes('rate limit')) {\n      return true;\n    }\n    \n    // Temporary GitHub API issues\n    if (statusCode >= 500 && statusCode < 600) {\n      return true;\n    }\n    \n    // Temporary GitHub API problems\n    if (errorMessage.includes('temporarily unavailable') || \n        errorMessage.includes('service unavailable') ||\n        errorMessage.includes('internal server error')) {\n      return true;\n    }\n    \n    // Connection issues\n    if (errorMessage.includes('connection') && \n        (errorMessage.includes('timeout') || errorMessage.includes('reset'))) {\n      return true;\n    }\n    \n    // Don't retry authentication or permission issues\n    if (statusCode === 401 || statusCode === 403 || \n        errorMessage.includes('unauthorized') || \n        errorMessage.includes('forbidden') ||\n        errorMessage.includes('authentication')) {\n      return false;\n    }\n    \n    // Don't retry validation errors\n    if (statusCode === 400 || statusCode === 422 ||\n        errorMessage.includes('invalid') ||\n        errorMessage.includes('validation')) {\n      return false;\n    }\n    \n    // Default to not retrying for unknown errors to avoid infinite loops\n    return false;\n  }\n}"]}