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