@dollhousemcp/mcp-server 1.7.4 → 1.8.1
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 +118 -0
- package/README.github.md +90 -1
- package/README.md +1 -1
- package/README.md.backup +90 -1
- package/README.npm.md +1 -1
- package/dist/collection/CollectionBrowser.js +2 -2
- package/dist/collection/CollectionSearch.js +3 -3
- package/dist/collection/PersonaDetails.js +2 -2
- package/dist/config/ConfigManager.d.ts +20 -0
- package/dist/config/ConfigManager.d.ts.map +1 -1
- package/dist/config/ConfigManager.js +48 -4
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ConfigHandler.d.ts +30 -0
- package/dist/handlers/ConfigHandler.d.ts.map +1 -1
- package/dist/handlers/ConfigHandler.js +149 -21
- package/dist/index.d.ts +4 -6
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +73 -52
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -1
- package/dist/portfolio/GitHubPortfolioIndexer.js +8 -5
- package/dist/portfolio/PortfolioIndexManager.d.ts +1 -1
- package/dist/portfolio/PortfolioIndexManager.js +2 -2
- package/dist/portfolio/PortfolioRepoManager.d.ts +16 -2
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioRepoManager.js +95 -18
- package/dist/portfolio/PortfolioSyncManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioSyncManager.js +23 -15
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +11 -1
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +11 -6
- package/dist/server/tools/CollectionTools.js +6 -6
- package/dist/server/tools/ConfigTools.js +2 -2
- package/dist/server/tools/ConfigToolsV2.js +5 -5
- package/dist/server/tools/PortfolioTools.d.ts.map +1 -1
- package/dist/server/tools/PortfolioTools.js +16 -5
- package/dist/server/types.d.ts +2 -0
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
- package/dist/tools/portfolio/submitToPortfolioTool.js +6 -5
- package/package.json +7 -6
|
@@ -14,12 +14,23 @@ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
|
14
14
|
import { SecurityMonitor } from '../security/securityMonitor.js';
|
|
15
15
|
import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
|
|
16
16
|
export class PortfolioRepoManager {
|
|
17
|
-
static
|
|
17
|
+
static DEFAULT_PORTFOLIO_REPO_NAME = 'dollhouse-portfolio';
|
|
18
18
|
static DEFAULT_DESCRIPTION = 'My DollhouseMCP element portfolio';
|
|
19
19
|
static GITHUB_API_BASE = 'https://api.github.com';
|
|
20
20
|
token = null;
|
|
21
|
-
|
|
21
|
+
repositoryName;
|
|
22
|
+
constructor(repositoryName) {
|
|
22
23
|
// Token will be retrieved when needed
|
|
24
|
+
// Support custom repository names or use default
|
|
25
|
+
this.repositoryName = repositoryName ||
|
|
26
|
+
process.env.TEST_GITHUB_REPO ||
|
|
27
|
+
PortfolioRepoManager.DEFAULT_PORTFOLIO_REPO_NAME;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Get the configured repository name
|
|
31
|
+
*/
|
|
32
|
+
getRepositoryName() {
|
|
33
|
+
return this.repositoryName;
|
|
23
34
|
}
|
|
24
35
|
/**
|
|
25
36
|
* Set the GitHub token for API calls
|
|
@@ -88,8 +99,24 @@ export class PortfolioRepoManager {
|
|
|
88
99
|
if (response.status === 404) {
|
|
89
100
|
return null; // Not found is often expected
|
|
90
101
|
}
|
|
91
|
-
|
|
102
|
+
// Check if response is ok BEFORE trying to parse JSON
|
|
92
103
|
if (!response.ok) {
|
|
104
|
+
// Try to parse error details if response is JSON
|
|
105
|
+
let data = {};
|
|
106
|
+
// HTTP headers are case-insensitive, check both cases for robustness
|
|
107
|
+
const contentType = response.headers.get('content-type') || response.headers.get('Content-Type');
|
|
108
|
+
if (contentType && contentType.toLowerCase().includes('application/json')) {
|
|
109
|
+
try {
|
|
110
|
+
data = await response.json();
|
|
111
|
+
}
|
|
112
|
+
catch (jsonError) {
|
|
113
|
+
// JSON parsing failed for error response - continue with empty data
|
|
114
|
+
// This can happen if GitHub returns malformed JSON or content-type mismatch
|
|
115
|
+
if (process.env.DEBUG) {
|
|
116
|
+
console.debug('Failed to parse JSON error response:', jsonError);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
93
120
|
// Create error with status code attached for better classification
|
|
94
121
|
let errorMessage = data.message || `GitHub API error: ${response.status}`;
|
|
95
122
|
let errorCode = 'PORTFOLIO_SYNC_005'; // Default
|
|
@@ -125,6 +152,8 @@ export class PortfolioRepoManager {
|
|
|
125
152
|
error.code = errorCode;
|
|
126
153
|
throw error;
|
|
127
154
|
}
|
|
155
|
+
// Parse JSON only after we know response is ok
|
|
156
|
+
const data = await response.json();
|
|
128
157
|
return data;
|
|
129
158
|
}
|
|
130
159
|
/**
|
|
@@ -136,7 +165,7 @@ export class PortfolioRepoManager {
|
|
|
136
165
|
// MEDIUM FIX: Normalize username to prevent Unicode attacks
|
|
137
166
|
const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;
|
|
138
167
|
try {
|
|
139
|
-
const repo = await this.githubRequest(`/repos/${normalizedUsername}/${
|
|
168
|
+
const repo = await this.githubRequest(`/repos/${normalizedUsername}/${this.repositoryName}`);
|
|
140
169
|
return repo !== null;
|
|
141
170
|
}
|
|
142
171
|
catch (error) {
|
|
@@ -171,7 +200,7 @@ export class PortfolioRepoManager {
|
|
|
171
200
|
metadata: { username: normalizedUsername }
|
|
172
201
|
});
|
|
173
202
|
// Check if portfolio already exists
|
|
174
|
-
const existingRepo = await this.githubRequest(`/repos/${normalizedUsername}/${
|
|
203
|
+
const existingRepo = await this.githubRequest(`/repos/${normalizedUsername}/${this.repositoryName}`);
|
|
175
204
|
if (existingRepo && existingRepo.html_url) {
|
|
176
205
|
logger.info(`Portfolio already exists for ${normalizedUsername}`);
|
|
177
206
|
return existingRepo.html_url;
|
|
@@ -179,7 +208,7 @@ export class PortfolioRepoManager {
|
|
|
179
208
|
// Create the portfolio repository
|
|
180
209
|
try {
|
|
181
210
|
const repo = await this.githubRequest('/user/repos', 'POST', {
|
|
182
|
-
name:
|
|
211
|
+
name: this.repositoryName,
|
|
183
212
|
description: PortfolioRepoManager.DEFAULT_DESCRIPTION,
|
|
184
213
|
private: false,
|
|
185
214
|
auto_init: true
|
|
@@ -194,7 +223,7 @@ export class PortfolioRepoManager {
|
|
|
194
223
|
logger.info(`Portfolio repository already exists for ${normalizedUsername} (race condition handled)`);
|
|
195
224
|
// Re-check for the existing repository and return its URL
|
|
196
225
|
try {
|
|
197
|
-
const existingRepo = await this.githubRequest(`/repos/${normalizedUsername}/${
|
|
226
|
+
const existingRepo = await this.githubRequest(`/repos/${normalizedUsername}/${this.repositoryName}`);
|
|
198
227
|
if (existingRepo && existingRepo.html_url) {
|
|
199
228
|
return existingRepo.html_url;
|
|
200
229
|
}
|
|
@@ -224,9 +253,9 @@ export class PortfolioRepoManager {
|
|
|
224
253
|
}
|
|
225
254
|
// Validate element before saving
|
|
226
255
|
this.validateElement(element);
|
|
227
|
-
//
|
|
228
|
-
|
|
229
|
-
const username =
|
|
256
|
+
// CRITICAL FIX: Use authenticated user's username, NOT element author (Issue #913)
|
|
257
|
+
// The portfolio belongs to the authenticated user, not the element's author
|
|
258
|
+
const username = await this.getUsername();
|
|
230
259
|
logger.info(`User consented to save element ${element.id} to portfolio`);
|
|
231
260
|
// LOW FIX: Add security audit logging for element save (DMCP-SEC-006)
|
|
232
261
|
SecurityMonitor.logSecurityEvent({
|
|
@@ -255,7 +284,7 @@ export class PortfolioRepoManager {
|
|
|
255
284
|
// First, check if file exists to determine if this is create or update
|
|
256
285
|
let existingFile = null;
|
|
257
286
|
try {
|
|
258
|
-
existingFile = await this.githubRequest(`/repos/${username}/${
|
|
287
|
+
existingFile = await this.githubRequest(`/repos/${username}/${this.repositoryName}/contents/${filePath}`);
|
|
259
288
|
}
|
|
260
289
|
catch (checkError) {
|
|
261
290
|
// IMPORTANT: Authentication and rate limit errors must be re-thrown!
|
|
@@ -288,7 +317,7 @@ export class PortfolioRepoManager {
|
|
|
288
317
|
});
|
|
289
318
|
// Return the existing file URL instead of creating duplicate commit
|
|
290
319
|
const existingUrl = existingFile.html_url ||
|
|
291
|
-
`https://github.com/${username}/${
|
|
320
|
+
`https://github.com/${username}/${this.repositoryName}/blob/main/${filePath}`;
|
|
292
321
|
return existingUrl;
|
|
293
322
|
}
|
|
294
323
|
}
|
|
@@ -305,7 +334,7 @@ export class PortfolioRepoManager {
|
|
|
305
334
|
if (existingFile && existingFile.sha) {
|
|
306
335
|
requestBody.sha = existingFile.sha;
|
|
307
336
|
}
|
|
308
|
-
const result = await this.githubRequest(`/repos/${username}/${
|
|
337
|
+
const result = await this.githubRequest(`/repos/${username}/${this.repositoryName}/contents/${filePath}`, 'PUT', requestBody);
|
|
309
338
|
// FIX: GitHub API response structure varies - handle all cases
|
|
310
339
|
// The response may have commit data at different levels or not at all
|
|
311
340
|
if (!result) {
|
|
@@ -328,7 +357,7 @@ export class PortfolioRepoManager {
|
|
|
328
357
|
}
|
|
329
358
|
// Path 3: Generate URL from response data
|
|
330
359
|
else if (result.content?.path) {
|
|
331
|
-
commitUrl = `https://github.com/${username}/${
|
|
360
|
+
commitUrl = `https://github.com/${username}/${this.repositoryName}/blob/main/${result.content.path}`;
|
|
332
361
|
}
|
|
333
362
|
// Path 4: Fallback to repository URL (guaranteed to be set)
|
|
334
363
|
else {
|
|
@@ -338,7 +367,7 @@ export class PortfolioRepoManager {
|
|
|
338
367
|
hasCommit: !!result.commit,
|
|
339
368
|
hasContent: !!result.content
|
|
340
369
|
});
|
|
341
|
-
commitUrl = `https://github.com/${username}/${
|
|
370
|
+
commitUrl = `https://github.com/${username}/${this.repositoryName}/tree/main/${element.type}`;
|
|
342
371
|
}
|
|
343
372
|
logger.debug('Successfully saved element to GitHub portfolio', {
|
|
344
373
|
element: element.id,
|
|
@@ -449,14 +478,14 @@ These elements can be imported into your DollhouseMCP installation.
|
|
|
449
478
|
---
|
|
450
479
|
*Generated by DollhouseMCP*
|
|
451
480
|
`;
|
|
452
|
-
await this.githubRequest(`/repos/${username}/${
|
|
481
|
+
await this.githubRequest(`/repos/${username}/${this.repositoryName}/contents/README.md`, 'PUT', {
|
|
453
482
|
message: 'Initialize portfolio structure',
|
|
454
483
|
content: Buffer.from(readmeContent).toString('base64')
|
|
455
484
|
});
|
|
456
485
|
// Create directory placeholders
|
|
457
486
|
const directories = ['personas', 'skills', 'templates', 'agents', 'memories', 'ensembles'];
|
|
458
487
|
for (const dir of directories) {
|
|
459
|
-
await this.githubRequest(`/repos/${username}/${
|
|
488
|
+
await this.githubRequest(`/repos/${username}/${this.repositoryName}/contents/${dir}/.gitkeep`, 'PUT', {
|
|
460
489
|
message: `Create ${dir} directory`,
|
|
461
490
|
content: Buffer.from('').toString('base64')
|
|
462
491
|
});
|
|
@@ -512,5 +541,53 @@ These elements can be imported into your DollhouseMCP installation.
|
|
|
512
541
|
// Fallback to basic markdown format
|
|
513
542
|
return `# ${element.metadata.name}\n\n${element.metadata.description || ''}`;
|
|
514
543
|
}
|
|
544
|
+
/**
|
|
545
|
+
* Get the authenticated user's username
|
|
546
|
+
*/
|
|
547
|
+
async getUsername() {
|
|
548
|
+
const response = await this.githubRequest('/user');
|
|
549
|
+
if (!response || !response.login) {
|
|
550
|
+
throw new Error('Failed to get GitHub username');
|
|
551
|
+
}
|
|
552
|
+
return response.login;
|
|
553
|
+
}
|
|
554
|
+
/**
|
|
555
|
+
* Get file content from GitHub repository
|
|
556
|
+
* Used for pull operations to download elements
|
|
557
|
+
*/
|
|
558
|
+
async getFileContent(path, username, repository) {
|
|
559
|
+
try {
|
|
560
|
+
// Use provided username/repository or defaults
|
|
561
|
+
const repoUser = username || await this.getUsername();
|
|
562
|
+
const repoName = repository || this.repositoryName;
|
|
563
|
+
logger.info('Fetching file content from GitHub', {
|
|
564
|
+
path,
|
|
565
|
+
username: repoUser,
|
|
566
|
+
repository: repoName
|
|
567
|
+
});
|
|
568
|
+
const response = await this.githubRequest(`/repos/${repoUser}/${repoName}/contents/${path}`);
|
|
569
|
+
if (!response || !response.content) {
|
|
570
|
+
throw new Error(`No content found at path: ${path}`);
|
|
571
|
+
}
|
|
572
|
+
// Decode base64 content
|
|
573
|
+
const decodedContent = Buffer.from(response.content, 'base64').toString('utf-8');
|
|
574
|
+
return decodedContent;
|
|
575
|
+
}
|
|
576
|
+
catch (error) {
|
|
577
|
+
logger.error('Failed to get file content from GitHub', {
|
|
578
|
+
error,
|
|
579
|
+
path
|
|
580
|
+
});
|
|
581
|
+
if (error instanceof Error) {
|
|
582
|
+
if (error.message.includes('404')) {
|
|
583
|
+
throw new Error(`File not found at path: ${path}`);
|
|
584
|
+
}
|
|
585
|
+
if (error.message.includes('401') || error.message.includes('403')) {
|
|
586
|
+
throw new Error(`Authentication failed. Please check your GitHub token.`);
|
|
587
|
+
}
|
|
588
|
+
}
|
|
589
|
+
throw error;
|
|
590
|
+
}
|
|
591
|
+
}
|
|
515
592
|
}
|
|
516
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioRepoManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PortfolioRepoManager - Manages GitHub portfolio repositories for element storage\n * \n * Key Features:\n * - EXPLICIT CONSENT required for all operations\n * - Creates portfolio repositories in user's GitHub account\n * - Saves elements to appropriate directories\n * - Handles API failures gracefully\n * - Provides audit logging for consent decisions\n */\n\nimport { IElement } from '../types/elements/IElement.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { logger } from '../utils/logger.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\n\nexport interface PortfolioRepoOptions {\n  description?: string;\n  private?: boolean;\n  auto_init?: boolean;\n}\n\nexport class PortfolioRepoManager {\n  private static readonly PORTFOLIO_REPO_NAME = 'dollhouse-portfolio';\n  private static readonly DEFAULT_DESCRIPTION = 'My DollhouseMCP element portfolio';\n  private static readonly GITHUB_API_BASE = 'https://api.github.com';\n  \n  private token: string | null = null;\n\n  constructor() {\n    // Token will be retrieved when needed\n  }\n\n  /**\n   * Set the GitHub token for API calls\n   * Used when token is already available from TokenManager\n   */\n  public setToken(token: string): void {\n    this.token = token;\n  }\n\n  /**\n   * Get GitHub token for API calls with validation\n   * SECURITY FIX: Added token validation to prevent token validation bypass (DMCP-SEC-002)\n   * Method name includes 'validate' to satisfy security scanner pattern\n   */\n  private async getTokenAndValidate(): Promise<string> {\n    if (!this.token) {\n      this.token = await TokenManager.getGitHubTokenAsync();\n      if (!this.token) {\n        throw new Error('GitHub authentication required. Please use setup_github_auth first.');\n      }\n      \n      // CRITICAL FIX: Validate token before use to prevent bypass attacks\n      // Using validateTokenScopes with minimal required scopes for portfolio operations\n      const validationResult = await TokenManager.validateTokenScopes(this.token, {\n        required: ['public_repo'] // Minimum scope needed for portfolio operations\n      });\n      \n      if (!validationResult.isValid) {\n        this.token = null;\n        throw new Error(`Invalid or expired GitHub token: ${validationResult.error || 'Please re-authenticate.'}`);\n      }\n      \n      // LOW FIX: Add audit logging for security operations (DMCP-SEC-006)\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_SUCCESS',\n        severity: 'LOW',\n        source: 'PortfolioRepoManager.getToken',\n        details: 'GitHub token validated successfully for portfolio operations'\n      });\n    }\n    return this.token;\n  }\n\n  /**\n   * Make authenticated GitHub API request\n   * Made public to support GitHubPortfolioIndexer operations\n   */\n  public async githubRequest(\n    path: string,\n    method: string = 'GET',\n    body?: any\n  ): Promise<any> {\n    const token = await this.getTokenAndValidate();\n    const url = `${PortfolioRepoManager.GITHUB_API_BASE}${path}`;\n    \n    const options: RequestInit = {\n      method,\n      headers: {\n        'Authorization': `Bearer ${token}`,\n        'Accept': 'application/vnd.github.v3+json',\n        'Content-Type': 'application/json',\n        'User-Agent': 'DollhouseMCP/1.0'\n      }\n    };\n\n    if (body) {\n      options.body = JSON.stringify(body);\n    }\n\n    const response = await fetch(url, options);\n    \n    // Check if response exists before accessing properties\n    if (!response) {\n      const error: any = new Error('No response received from GitHub API');\n      error.status = 0;\n      error.code = 'PORTFOLIO_SYNC_005';\n      throw error;\n    }\n    \n    if (response.status === 404) {\n      return null; // Not found is often expected\n    }\n\n    const data = await response.json();\n\n    if (!response.ok) {\n      // Create error with status code attached for better classification\n      let errorMessage = data.message || `GitHub API error: ${response.status}`;\n      let errorCode = 'PORTFOLIO_SYNC_005'; // Default\n      \n      switch (response.status) {\n        case 401:\n          errorMessage = 'GitHub authentication failed. Please check your token.';\n          errorCode = 'PORTFOLIO_SYNC_001';\n          break;\n        case 403:\n          if (data.message?.includes('rate limit')) {\n            errorMessage = `GitHub API rate limit exceeded: ${data.message}`;\n            errorCode = 'PORTFOLIO_SYNC_006';\n          } else {\n            errorMessage = `GitHub API access forbidden: ${data.message || 'insufficient permissions'}`;\n            errorCode = 'PORTFOLIO_SYNC_001'; // Treat as auth issue\n          }\n          break;\n        case 422:\n          // Validation failed - often means repository already exists\n          errorMessage = `Repository validation failed: ${data.message || 'name already exists on this account'}`;\n          errorCode = 'PORTFOLIO_SYNC_003';\n          break;\n        case 500:\n          errorMessage = 'GitHub API server error. Please try again later.';\n          errorCode = 'PORTFOLIO_SYNC_005';\n          break;\n        default:\n          errorMessage = `GitHub API error (${response.status}): ${data.message || 'Unknown error'}`;\n      }\n      \n      const error: any = new Error(errorMessage);\n      error.status = response.status;\n      error.code = errorCode;\n      throw error;\n    }\n\n    return data;\n  }\n\n  /**\n   * Check if portfolio repository exists for a user\n   * No consent required - this is a read-only operation\n   * SECURITY FIX: Added Unicode normalization for user input (DMCP-SEC-004)\n   */\n  async checkPortfolioExists(username: string): Promise<boolean> {\n    // MEDIUM FIX: Normalize username to prevent Unicode attacks\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    try {\n      const repo = await this.githubRequest(\n        `/repos/${normalizedUsername}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}`\n      );\n      return repo !== null;\n    } catch (error) {\n      // Repository doesn't exist or API error - both return false\n      ErrorHandler.logError('PortfolioRepoManager.checkIfRepoExists', error, { username });\n      return false;\n    }\n  }\n\n  /**\n   * Create portfolio repository with EXPLICIT user consent\n   * @throws Error if user declines consent or if consent is not provided\n   */\n  async createPortfolio(username: string, consent: boolean | undefined): Promise<string> {\n    // MEDIUM FIX: Normalize username to prevent Unicode attacks (DMCP-SEC-004)\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    \n    // CRITICAL: Validate consent is explicitly provided\n    if (consent === undefined) {\n      throw new Error('Consent is required for portfolio creation');\n    }\n\n    if (!consent) {\n      logger.info(`User declined portfolio creation for ${username}`);\n      throw new Error('User declined portfolio creation');\n    }\n\n    // Log consent for audit trail\n    logger.info(`User consented to portfolio creation for ${normalizedUsername}`);\n    \n    // LOW FIX: Add security audit logging (DMCP-SEC-006)\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'PortfolioRepoManager.createPortfolio',\n      details: `User ${normalizedUsername} consented to portfolio creation`,\n      metadata: { username: normalizedUsername }\n    });\n\n    // Check if portfolio already exists\n    const existingRepo = await this.githubRequest(\n      `/repos/${normalizedUsername}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}`\n    );\n    \n    if (existingRepo && existingRepo.html_url) {\n      logger.info(`Portfolio already exists for ${normalizedUsername}`);\n      return existingRepo.html_url;\n    }\n\n    // Create the portfolio repository\n    try {\n      const repo = await this.githubRequest(\n        '/user/repos',\n        'POST',\n        {\n          name: PortfolioRepoManager.PORTFOLIO_REPO_NAME,\n          description: PortfolioRepoManager.DEFAULT_DESCRIPTION,\n          private: false,\n          auto_init: true\n        }\n      );\n\n      // Initialize portfolio structure\n      await this.generatePortfolioStructure(normalizedUsername);\n\n      return repo.html_url;\n    } catch (error: any) {\n      // Handle race condition: if repository was created between our check and creation attempt\n      if (error.message && error.message.includes('name already exists')) {\n        logger.info(`Portfolio repository already exists for ${normalizedUsername} (race condition handled)`);\n        \n        // Re-check for the existing repository and return its URL\n        try {\n          const existingRepo = await this.githubRequest(\n            `/repos/${normalizedUsername}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}`\n          );\n          if (existingRepo && existingRepo.html_url) {\n            return existingRepo.html_url;\n          }\n        } catch (recheckError) {\n          ErrorHandler.logError('PortfolioRepoManager.recheckExistingRepo', recheckError, { username: normalizedUsername });\n        }\n        \n        // If we can't get the existing repo, throw a more specific error\n        throw new Error(`Portfolio repository already exists for ${normalizedUsername}. Please check your GitHub account.`);\n      }\n      \n      ErrorHandler.logError('PortfolioRepoManager.createPortfolioRepo', error, { username: normalizedUsername });\n      throw ErrorHandler.wrapError(error, `Failed to create portfolio repository for ${normalizedUsername}. ${error.message || 'Unknown error occurred.'}`, ErrorCategory.NETWORK_ERROR);\n    }\n  }\n\n  /**\n   * Save element to portfolio with EXPLICIT user consent\n   * @throws Error if user declines consent or element is invalid\n   */\n  async saveElement(element: IElement, consent: boolean | undefined): Promise<string> {\n    // CRITICAL: Validate consent is explicitly provided\n    if (consent === undefined) {\n      throw new Error('Consent is required to save element');\n    }\n\n    if (!consent) {\n      logger.info(`User declined to save element ${element.id} to portfolio`);\n      throw new Error('User declined to save element to portfolio');\n    }\n\n    // Validate element before saving\n    this.validateElement(element);\n\n    // MEDIUM FIX: Normalize username from element metadata (DMCP-SEC-004)\n    const rawUsername = element.metadata.author || 'anonymous';\n    const username = UnicodeValidator.normalize(rawUsername).normalizedContent;\n    logger.info(`User consented to save element ${element.id} to portfolio`);\n    \n    // LOW FIX: Add security audit logging for element save (DMCP-SEC-006)\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'PortfolioRepoManager.saveElement',\n      details: `User consented to save element ${element.id} to portfolio`,\n      metadata: { \n        elementId: element.id,\n        elementType: element.type,\n        username \n      }\n    });\n\n    // Generate file path based on element type\n    // FIX: Don't add 's' - element.type is already plural (e.g., 'personas', 'skills')\n    const fileName = PortfolioRepoManager.generateFileName(element.metadata.name);\n    const filePath = `${element.type}/${fileName}.md`;\n\n    // Prepare content (could be markdown with frontmatter)\n    const content = this.formatElementContent(element);\n    \n    // DIAGNOSTIC: Log content size before sending to GitHub\n    logger.debug(`[CONTENT-TRACE] Saving element ${element.id} to GitHub - content size: ${content.length} chars`);\n    logger.debug(`[CONTENT-TRACE] First 200 chars: ${content.substring(0, 200)}`);\n    logger.debug(`[CONTENT-TRACE] Last 200 chars: ${content.substring(Math.max(0, content.length - 200))}`);\n\n    // Save to GitHub\n    try {\n      // First, check if file exists to determine if this is create or update\n      let existingFile = null;\n      try {\n        existingFile = await this.githubRequest(\n          `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`\n        );\n      } catch (checkError: any) {\n        // IMPORTANT: Authentication and rate limit errors must be re-thrown!\n        // These are NOT \"file doesn't exist\" scenarios - they indicate we can't\n        // access the API at all. Only 404 (and similar) should be treated as \n        // \"file doesn't exist\". This ensures auth errors are properly reported\n        // with correct error codes (e.g., PORTFOLIO_SYNC_001 for auth failures).\n        // See PR #846 and test: portfolio-single-upload.qa.test.ts\n        if (checkError.status === 401 || checkError.code === 'PORTFOLIO_SYNC_001') {\n          throw checkError; // Authentication error - don't continue\n        }\n        if (checkError.status === 403 || checkError.code === 'PORTFOLIO_SYNC_006') {\n          throw checkError; // Rate limit or permission error - don't continue\n        }\n        // For other errors (like 404), assume file doesn't exist and continue\n        // with file creation. This is the expected flow for new files.\n        logger.debug(`File check returned error (likely doesn't exist): ${filePath}`);\n        existingFile = null;\n      }\n\n      // DUPLICATE DETECTION (Issue #792): Check if content is identical\n      if (existingFile && existingFile.content) {\n        // Decode existing content from base64\n        const existingContent = Buffer.from(existingFile.content, 'base64').toString('utf-8');\n        \n        // Compare with new content\n        if (existingContent === content) {\n          logger.info('Skipping duplicate portfolio upload - content identical', {\n            elementId: element.id,\n            elementName: element.metadata.name,\n            filePath\n          });\n          \n          // Return the existing file URL instead of creating duplicate commit\n          const existingUrl = existingFile.html_url || \n            `https://github.com/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/blob/main/${filePath}`;\n          \n          return existingUrl;\n        }\n      }\n\n      // Create or update the file (only if content is different)\n      // DEBUG: Log what we're about to send\n      logger.debug(`[DEBUG] Creating/updating file. existingFile: ${!!existingFile}, sha: ${existingFile?.sha}`);\n      \n      const requestBody: any = {\n        message: existingFile ? \n          `Update ${element.metadata.name} in portfolio` : \n          `Add ${element.metadata.name} to portfolio`,\n        content: Buffer.from(content).toString('base64')\n      };\n      \n      // Only include sha if we have an existing file with a sha\n      if (existingFile && existingFile.sha) {\n        requestBody.sha = existingFile.sha;\n      }\n      \n      const result = await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`,\n        'PUT',\n        requestBody\n      );\n\n      // FIX: GitHub API response structure varies - handle all cases\n      // The response may have commit data at different levels or not at all\n      if (!result) {\n        logger.error('[PORTFOLIO_SYNC_004] GitHub API returned null response', {\n          element: element.id,\n          username,\n          filePath\n        });\n        throw new Error(`[PORTFOLIO_SYNC_004] GitHub API returned null response for ${element.metadata.name}`);\n      }\n\n      // Try multiple paths to get the commit URL\n      let commitUrl: string;\n      \n      // Path 1: result.commit.html_url (standard for content API)\n      if (result.commit?.html_url) {\n        commitUrl = result.commit.html_url;\n      }\n      // Path 2: result.content.html_url (some API responses)\n      else if (result.content?.html_url) {\n        commitUrl = result.content.html_url;\n      }\n      // Path 3: Generate URL from response data\n      else if (result.content?.path) {\n        commitUrl = `https://github.com/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/blob/main/${result.content.path}`;\n      }\n      // Path 4: Fallback to repository URL (guaranteed to be set)\n      else {\n        logger.warn('[PORTFOLIO_SYNC_004] Could not extract commit URL from GitHub response, using fallback', {\n          element: element.id,\n          responseKeys: Object.keys(result),\n          hasCommit: !!result.commit,\n          hasContent: !!result.content\n        });\n        commitUrl = `https://github.com/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/tree/main/${element.type}`;\n      }\n\n      logger.debug('Successfully saved element to GitHub portfolio', {\n        element: element.id,\n        username,\n        filePath,\n        commitUrl\n      });\n\n      return commitUrl;\n    } catch (error: any) {\n      // Use error code if already set by githubRequest\n      let errorCode = error.code || 'PORTFOLIO_SYNC_005'; // Default network error\n      let enhancedMessage = 'Failed to save element to portfolio';\n      \n      // Check error status first (more reliable than message parsing)\n      if (error.status) {\n        switch (error.status) {\n          case 401:\n            errorCode = 'PORTFOLIO_SYNC_001';\n            enhancedMessage = 'GitHub authentication failed. Please re-authenticate.';\n            break;\n          case 403:\n            if (error.message?.includes('rate limit')) {\n              errorCode = 'PORTFOLIO_SYNC_006';\n              enhancedMessage = 'GitHub API rate limit exceeded. Please try again later.';\n            } else {\n              errorCode = 'PORTFOLIO_SYNC_001';\n              enhancedMessage = 'GitHub API access forbidden. Check token permissions.';\n            }\n            break;\n          case 404:\n            errorCode = 'PORTFOLIO_SYNC_002';\n            enhancedMessage = 'GitHub portfolio repository not found. Please run init_portfolio first.';\n            break;\n          case 422:\n            errorCode = 'PORTFOLIO_SYNC_003';\n            enhancedMessage = 'Repository validation failed.';\n            break;\n          default:\n            // Keep the error code from githubRequest if set\n            if (!error.code) {\n              errorCode = 'PORTFOLIO_SYNC_005';\n            }\n        }\n      } else if (!error.code) {\n        // Fall back to message parsing only if no status code available\n        if (error.message?.includes('401') || error.message?.includes('authentication')) {\n          errorCode = 'PORTFOLIO_SYNC_001';\n          enhancedMessage = 'GitHub authentication failed. Please re-authenticate.';\n        } else if (error.message?.includes('404') || error.message?.includes('not found')) {\n          errorCode = 'PORTFOLIO_SYNC_002';\n          enhancedMessage = 'GitHub portfolio repository not found. Please run init_portfolio first.';\n        } else if (error.message?.includes('403') || error.message?.includes('rate limit')) {\n          errorCode = 'PORTFOLIO_SYNC_006';\n          enhancedMessage = 'GitHub API rate limit exceeded. Please try again later.';\n        } else if (error.message?.includes('Cannot read properties')) {\n          errorCode = 'PORTFOLIO_SYNC_004';\n          enhancedMessage = `GitHub API response parsing error: ${error.message}`;\n        }\n      }\n      \n      logger.error(`[${errorCode}] ${enhancedMessage}`, { \n        elementId: element.id,\n        username,\n        originalError: error.message,\n        errorStatus: error.status,\n        stack: error.stack\n      });\n      \n      ErrorHandler.logError('PortfolioRepoManager.saveElementToRepo', error, { \n        elementId: element.id,\n        username,\n        errorCode,\n        errorStatus: error.status\n      });\n      \n      // Throw error with code for better handling upstream\n      const wrappedError = ErrorHandler.wrapError(error, `[${errorCode}] ${enhancedMessage}`, ErrorCategory.NETWORK_ERROR);\n      (wrappedError as any).code = errorCode;\n      (wrappedError as any).status = error.status;\n      throw wrappedError;\n    }\n  }\n\n  /**\n   * Generate initial portfolio structure with README and directories\n   * SECURITY: Username already normalized by calling methods\n   */\n  async generatePortfolioStructure(username: string): Promise<void> {\n    // Create README.md\n    const readmeContent = `# DollhouseMCP Portfolio\n\nThis is my personal collection of DollhouseMCP elements.\n\n## Structure\n\n- **personas/** - Behavioral profiles\n- **skills/** - Discrete capabilities  \n- **templates/** - Reusable content structures\n- **agents/** - Autonomous actors\n- **memories/** - Persistent context\n- **ensembles/** - Element groups\n\n## Usage\n\nThese elements can be imported into your DollhouseMCP installation.\n\n---\n*Generated by DollhouseMCP*\n`;\n\n    await this.githubRequest(\n      `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/README.md`,\n      'PUT',\n      {\n        message: 'Initialize portfolio structure',\n        content: Buffer.from(readmeContent).toString('base64')\n      }\n    );\n\n    // Create directory placeholders\n    const directories = ['personas', 'skills', 'templates', 'agents', 'memories', 'ensembles'];\n    \n    for (const dir of directories) {\n      await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${dir}/.gitkeep`,\n        'PUT',\n        {\n          message: `Create ${dir} directory`,\n          content: Buffer.from('').toString('base64')\n        }\n      );\n    }\n  }\n\n  /**\n   * Validate element before saving\n   * @throws Error if element is invalid\n   */\n  private validateElement(element: IElement): void {\n    if (!element.metadata.name) {\n      throw new Error('Invalid element: name is required');\n    }\n\n    if (!element.id) {\n      throw new Error('Invalid element: id is required');\n    }\n\n    if (!element.type) {\n      throw new Error('Invalid element: type is required');\n    }\n  }\n\n  /**\n   * Generate safe filename from element name\n   * SECURITY: Additional Unicode normalization for filenames\n   * SECURITY FIX: Fixed ReDoS vulnerability with input length limit and optimized regex\n   */\n  public static generateFileName(name: string): string {\n    // SECURITY FIX: Limit input length to prevent ReDoS attacks\n    // Even with optimized regex, very long inputs could cause performance issues\n    const MAX_FILENAME_LENGTH = 255; // Common filesystem limit\n    \n    // Normalize to prevent Unicode attacks in filenames\n    const normalizedName = UnicodeValidator.normalize(name).normalizedContent;\n    \n    // Truncate to safe length BEFORE regex operations\n    const truncatedName = normalizedName.slice(0, MAX_FILENAME_LENGTH);\n    \n    // SECURITY FIX: Optimized regex operations to prevent ReDoS\n    // 1. Convert non-alphanumeric sequences to single dash\n    // 2. Remove leading/trailing dashes in a single pass using trim\n    const safeName = truncatedName\n      .toLowerCase()\n      .replace(/[^a-z0-9]+/g, '-')\n      .replace(/^-+/, '')  // Remove leading dashes\n      .replace(/-+$/, ''); // Remove trailing dashes\n    \n    // Ensure we have a valid filename (not empty after cleaning)\n    return safeName || 'unnamed';\n  }\n\n  /**\n   * Format element content for storage\n   */\n  private formatElementContent(element: IElement): string {\n    // Serialize the element or create basic markdown\n    if (element.serialize) {\n      return element.serialize();\n    }\n    // Fallback to basic markdown format\n    return `# ${element.metadata.name}\\n\\n${element.metadata.description || ''}`;\n  }\n}"]}
|
|
593
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioRepoManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PortfolioRepoManager - Manages GitHub portfolio repositories for element storage\n * \n * Key Features:\n * - EXPLICIT CONSENT required for all operations\n * - Creates portfolio repositories in user's GitHub account\n * - Saves elements to appropriate directories\n * - Handles API failures gracefully\n * - Provides audit logging for consent decisions\n */\n\nimport { IElement } from '../types/elements/IElement.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { logger } from '../utils/logger.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\n\nexport interface PortfolioRepoOptions {\n  description?: string;\n  private?: boolean;\n  auto_init?: boolean;\n}\n\nexport class PortfolioRepoManager {\n  private static readonly DEFAULT_PORTFOLIO_REPO_NAME = 'dollhouse-portfolio';\n  private static readonly DEFAULT_DESCRIPTION = 'My DollhouseMCP element portfolio';\n  private static readonly GITHUB_API_BASE = 'https://api.github.com';\n  \n  private token: string | null = null;\n  private repositoryName: string;\n\n  constructor(repositoryName?: string) {\n    // Token will be retrieved when needed\n    // Support custom repository names or use default\n    this.repositoryName = repositoryName || \n                         process.env.TEST_GITHUB_REPO || \n                         PortfolioRepoManager.DEFAULT_PORTFOLIO_REPO_NAME;\n  }\n\n  /**\n   * Get the configured repository name\n   */\n  public getRepositoryName(): string {\n    return this.repositoryName;\n  }\n\n  /**\n   * Set the GitHub token for API calls\n   * Used when token is already available from TokenManager\n   */\n  public setToken(token: string): void {\n    this.token = token;\n  }\n\n  /**\n   * Get GitHub token for API calls with validation\n   * SECURITY FIX: Added token validation to prevent token validation bypass (DMCP-SEC-002)\n   * Method name includes 'validate' to satisfy security scanner pattern\n   */\n  private async getTokenAndValidate(): Promise<string> {\n    if (!this.token) {\n      this.token = await TokenManager.getGitHubTokenAsync();\n      if (!this.token) {\n        throw new Error('GitHub authentication required. Please use setup_github_auth first.');\n      }\n      \n      // CRITICAL FIX: Validate token before use to prevent bypass attacks\n      // Using validateTokenScopes with minimal required scopes for portfolio operations\n      const validationResult = await TokenManager.validateTokenScopes(this.token, {\n        required: ['public_repo'] // Minimum scope needed for portfolio operations\n      });\n      \n      if (!validationResult.isValid) {\n        this.token = null;\n        throw new Error(`Invalid or expired GitHub token: ${validationResult.error || 'Please re-authenticate.'}`);\n      }\n      \n      // LOW FIX: Add audit logging for security operations (DMCP-SEC-006)\n      SecurityMonitor.logSecurityEvent({\n        type: 'TOKEN_VALIDATION_SUCCESS',\n        severity: 'LOW',\n        source: 'PortfolioRepoManager.getToken',\n        details: 'GitHub token validated successfully for portfolio operations'\n      });\n    }\n    return this.token;\n  }\n\n  /**\n   * Make authenticated GitHub API request\n   * Made public to support GitHubPortfolioIndexer operations\n   */\n  public async githubRequest(\n    path: string,\n    method: string = 'GET',\n    body?: any\n  ): Promise<any> {\n    const token = await this.getTokenAndValidate();\n    const url = `${PortfolioRepoManager.GITHUB_API_BASE}${path}`;\n    \n    const options: RequestInit = {\n      method,\n      headers: {\n        'Authorization': `Bearer ${token}`,\n        'Accept': 'application/vnd.github.v3+json',\n        'Content-Type': 'application/json',\n        'User-Agent': 'DollhouseMCP/1.0'\n      }\n    };\n\n    if (body) {\n      options.body = JSON.stringify(body);\n    }\n\n    const response = await fetch(url, options);\n    \n    // Check if response exists before accessing properties\n    if (!response) {\n      const error: any = new Error('No response received from GitHub API');\n      error.status = 0;\n      error.code = 'PORTFOLIO_SYNC_005';\n      throw error;\n    }\n    \n    if (response.status === 404) {\n      return null; // Not found is often expected\n    }\n\n    // Check if response is ok BEFORE trying to parse JSON\n    if (!response.ok) {\n      // Try to parse error details if response is JSON\n      let data: any = {};\n      // HTTP headers are case-insensitive, check both cases for robustness\n      const contentType = response.headers.get('content-type') || response.headers.get('Content-Type');\n      if (contentType && contentType.toLowerCase().includes('application/json')) {\n        try {\n          data = await response.json();\n        } catch (jsonError) {\n          // JSON parsing failed for error response - continue with empty data\n          // This can happen if GitHub returns malformed JSON or content-type mismatch\n          if (process.env.DEBUG) {\n            console.debug('Failed to parse JSON error response:', jsonError);\n          }\n        }\n      }\n\n      // Create error with status code attached for better classification\n      let errorMessage = data.message || `GitHub API error: ${response.status}`;\n      let errorCode = 'PORTFOLIO_SYNC_005'; // Default\n\n      switch (response.status) {\n        case 401:\n          errorMessage = 'GitHub authentication failed. Please check your token.';\n          errorCode = 'PORTFOLIO_SYNC_001';\n          break;\n        case 403:\n          if (data.message?.includes('rate limit')) {\n            errorMessage = `GitHub API rate limit exceeded: ${data.message}`;\n            errorCode = 'PORTFOLIO_SYNC_006';\n          } else {\n            errorMessage = `GitHub API access forbidden: ${data.message || 'insufficient permissions'}`;\n            errorCode = 'PORTFOLIO_SYNC_001'; // Treat as auth issue\n          }\n          break;\n        case 422:\n          // Validation failed - often means repository already exists\n          errorMessage = `Repository validation failed: ${data.message || 'name already exists on this account'}`;\n          errorCode = 'PORTFOLIO_SYNC_003';\n          break;\n        case 500:\n          errorMessage = 'GitHub API server error. Please try again later.';\n          errorCode = 'PORTFOLIO_SYNC_005';\n          break;\n        default:\n          errorMessage = `GitHub API error (${response.status}): ${data.message || 'Unknown error'}`;\n      }\n\n      const error: any = new Error(errorMessage);\n      error.status = response.status;\n      error.code = errorCode;\n      throw error;\n    }\n\n    // Parse JSON only after we know response is ok\n    const data = await response.json();\n\n    return data;\n  }\n\n  /**\n   * Check if portfolio repository exists for a user\n   * No consent required - this is a read-only operation\n   * SECURITY FIX: Added Unicode normalization for user input (DMCP-SEC-004)\n   */\n  async checkPortfolioExists(username: string): Promise<boolean> {\n    // MEDIUM FIX: Normalize username to prevent Unicode attacks\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    try {\n      const repo = await this.githubRequest(\n        `/repos/${normalizedUsername}/${this.repositoryName}`\n      );\n      return repo !== null;\n    } catch (error) {\n      // Repository doesn't exist or API error - both return false\n      ErrorHandler.logError('PortfolioRepoManager.checkIfRepoExists', error, { username });\n      return false;\n    }\n  }\n\n  /**\n   * Create portfolio repository with EXPLICIT user consent\n   * @throws Error if user declines consent or if consent is not provided\n   */\n  async createPortfolio(username: string, consent: boolean | undefined): Promise<string> {\n    // MEDIUM FIX: Normalize username to prevent Unicode attacks (DMCP-SEC-004)\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    \n    // CRITICAL: Validate consent is explicitly provided\n    if (consent === undefined) {\n      throw new Error('Consent is required for portfolio creation');\n    }\n\n    if (!consent) {\n      logger.info(`User declined portfolio creation for ${username}`);\n      throw new Error('User declined portfolio creation');\n    }\n\n    // Log consent for audit trail\n    logger.info(`User consented to portfolio creation for ${normalizedUsername}`);\n    \n    // LOW FIX: Add security audit logging (DMCP-SEC-006)\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'PortfolioRepoManager.createPortfolio',\n      details: `User ${normalizedUsername} consented to portfolio creation`,\n      metadata: { username: normalizedUsername }\n    });\n\n    // Check if portfolio already exists\n    const existingRepo = await this.githubRequest(\n      `/repos/${normalizedUsername}/${this.repositoryName}`\n    );\n    \n    if (existingRepo && existingRepo.html_url) {\n      logger.info(`Portfolio already exists for ${normalizedUsername}`);\n      return existingRepo.html_url;\n    }\n\n    // Create the portfolio repository\n    try {\n      const repo = await this.githubRequest(\n        '/user/repos',\n        'POST',\n        {\n          name: this.repositoryName,\n          description: PortfolioRepoManager.DEFAULT_DESCRIPTION,\n          private: false,\n          auto_init: true\n        }\n      );\n\n      // Initialize portfolio structure\n      await this.generatePortfolioStructure(normalizedUsername);\n\n      return repo.html_url;\n    } catch (error: any) {\n      // Handle race condition: if repository was created between our check and creation attempt\n      if (error.message && error.message.includes('name already exists')) {\n        logger.info(`Portfolio repository already exists for ${normalizedUsername} (race condition handled)`);\n        \n        // Re-check for the existing repository and return its URL\n        try {\n          const existingRepo = await this.githubRequest(\n            `/repos/${normalizedUsername}/${this.repositoryName}`\n          );\n          if (existingRepo && existingRepo.html_url) {\n            return existingRepo.html_url;\n          }\n        } catch (recheckError) {\n          ErrorHandler.logError('PortfolioRepoManager.recheckExistingRepo', recheckError, { username: normalizedUsername });\n        }\n        \n        // If we can't get the existing repo, throw a more specific error\n        throw new Error(`Portfolio repository already exists for ${normalizedUsername}. Please check your GitHub account.`);\n      }\n      \n      ErrorHandler.logError('PortfolioRepoManager.createPortfolioRepo', error, { username: normalizedUsername });\n      throw ErrorHandler.wrapError(error, `Failed to create portfolio repository for ${normalizedUsername}. ${error.message || 'Unknown error occurred.'}`, ErrorCategory.NETWORK_ERROR);\n    }\n  }\n\n  /**\n   * Save element to portfolio with EXPLICIT user consent\n   * @throws Error if user declines consent or element is invalid\n   */\n  async saveElement(element: IElement, consent: boolean | undefined): Promise<string> {\n    // CRITICAL: Validate consent is explicitly provided\n    if (consent === undefined) {\n      throw new Error('Consent is required to save element');\n    }\n\n    if (!consent) {\n      logger.info(`User declined to save element ${element.id} to portfolio`);\n      throw new Error('User declined to save element to portfolio');\n    }\n\n    // Validate element before saving\n    this.validateElement(element);\n\n    // CRITICAL FIX: Use authenticated user's username, NOT element author (Issue #913)\n    // The portfolio belongs to the authenticated user, not the element's author\n    const username = await this.getUsername();\n    logger.info(`User consented to save element ${element.id} to portfolio`);\n    \n    // LOW FIX: Add security audit logging for element save (DMCP-SEC-006)\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'PortfolioRepoManager.saveElement',\n      details: `User consented to save element ${element.id} to portfolio`,\n      metadata: { \n        elementId: element.id,\n        elementType: element.type,\n        username \n      }\n    });\n\n    // Generate file path based on element type\n    // FIX: Don't add 's' - element.type is already plural (e.g., 'personas', 'skills')\n    const fileName = PortfolioRepoManager.generateFileName(element.metadata.name);\n    const filePath = `${element.type}/${fileName}.md`;\n\n    // Prepare content (could be markdown with frontmatter)\n    const content = this.formatElementContent(element);\n    \n    // DIAGNOSTIC: Log content size before sending to GitHub\n    logger.debug(`[CONTENT-TRACE] Saving element ${element.id} to GitHub - content size: ${content.length} chars`);\n    logger.debug(`[CONTENT-TRACE] First 200 chars: ${content.substring(0, 200)}`);\n    logger.debug(`[CONTENT-TRACE] Last 200 chars: ${content.substring(Math.max(0, content.length - 200))}`);\n\n    // Save to GitHub\n    try {\n      // First, check if file exists to determine if this is create or update\n      let existingFile = null;\n      try {\n        existingFile = await this.githubRequest(\n          `/repos/${username}/${this.repositoryName}/contents/${filePath}`\n        );\n      } catch (checkError: any) {\n        // IMPORTANT: Authentication and rate limit errors must be re-thrown!\n        // These are NOT \"file doesn't exist\" scenarios - they indicate we can't\n        // access the API at all. Only 404 (and similar) should be treated as \n        // \"file doesn't exist\". This ensures auth errors are properly reported\n        // with correct error codes (e.g., PORTFOLIO_SYNC_001 for auth failures).\n        // See PR #846 and test: portfolio-single-upload.qa.test.ts\n        if (checkError.status === 401 || checkError.code === 'PORTFOLIO_SYNC_001') {\n          throw checkError; // Authentication error - don't continue\n        }\n        if (checkError.status === 403 || checkError.code === 'PORTFOLIO_SYNC_006') {\n          throw checkError; // Rate limit or permission error - don't continue\n        }\n        // For other errors (like 404), assume file doesn't exist and continue\n        // with file creation. This is the expected flow for new files.\n        logger.debug(`File check returned error (likely doesn't exist): ${filePath}`);\n        existingFile = null;\n      }\n\n      // DUPLICATE DETECTION (Issue #792): Check if content is identical\n      if (existingFile && existingFile.content) {\n        // Decode existing content from base64\n        const existingContent = Buffer.from(existingFile.content, 'base64').toString('utf-8');\n        \n        // Compare with new content\n        if (existingContent === content) {\n          logger.info('Skipping duplicate portfolio upload - content identical', {\n            elementId: element.id,\n            elementName: element.metadata.name,\n            filePath\n          });\n          \n          // Return the existing file URL instead of creating duplicate commit\n          const existingUrl = existingFile.html_url || \n            `https://github.com/${username}/${this.repositoryName}/blob/main/${filePath}`;\n          \n          return existingUrl;\n        }\n      }\n\n      // Create or update the file (only if content is different)\n      // DEBUG: Log what we're about to send\n      logger.debug(`[DEBUG] Creating/updating file. existingFile: ${!!existingFile}, sha: ${existingFile?.sha}`);\n      \n      const requestBody: any = {\n        message: existingFile ? \n          `Update ${element.metadata.name} in portfolio` : \n          `Add ${element.metadata.name} to portfolio`,\n        content: Buffer.from(content).toString('base64')\n      };\n      \n      // Only include sha if we have an existing file with a sha\n      if (existingFile && existingFile.sha) {\n        requestBody.sha = existingFile.sha;\n      }\n      \n      const result = await this.githubRequest(\n        `/repos/${username}/${this.repositoryName}/contents/${filePath}`,\n        'PUT',\n        requestBody\n      );\n\n      // FIX: GitHub API response structure varies - handle all cases\n      // The response may have commit data at different levels or not at all\n      if (!result) {\n        logger.error('[PORTFOLIO_SYNC_004] GitHub API returned null response', {\n          element: element.id,\n          username,\n          filePath\n        });\n        throw new Error(`[PORTFOLIO_SYNC_004] GitHub API returned null response for ${element.metadata.name}`);\n      }\n\n      // Try multiple paths to get the commit URL\n      let commitUrl: string;\n      \n      // Path 1: result.commit.html_url (standard for content API)\n      if (result.commit?.html_url) {\n        commitUrl = result.commit.html_url;\n      }\n      // Path 2: result.content.html_url (some API responses)\n      else if (result.content?.html_url) {\n        commitUrl = result.content.html_url;\n      }\n      // Path 3: Generate URL from response data\n      else if (result.content?.path) {\n        commitUrl = `https://github.com/${username}/${this.repositoryName}/blob/main/${result.content.path}`;\n      }\n      // Path 4: Fallback to repository URL (guaranteed to be set)\n      else {\n        logger.warn('[PORTFOLIO_SYNC_004] Could not extract commit URL from GitHub response, using fallback', {\n          element: element.id,\n          responseKeys: Object.keys(result),\n          hasCommit: !!result.commit,\n          hasContent: !!result.content\n        });\n        commitUrl = `https://github.com/${username}/${this.repositoryName}/tree/main/${element.type}`;\n      }\n\n      logger.debug('Successfully saved element to GitHub portfolio', {\n        element: element.id,\n        username,\n        filePath,\n        commitUrl\n      });\n\n      return commitUrl;\n    } catch (error: any) {\n      // Use error code if already set by githubRequest\n      let errorCode = error.code || 'PORTFOLIO_SYNC_005'; // Default network error\n      let enhancedMessage = 'Failed to save element to portfolio';\n      \n      // Check error status first (more reliable than message parsing)\n      if (error.status) {\n        switch (error.status) {\n          case 401:\n            errorCode = 'PORTFOLIO_SYNC_001';\n            enhancedMessage = 'GitHub authentication failed. Please re-authenticate.';\n            break;\n          case 403:\n            if (error.message?.includes('rate limit')) {\n              errorCode = 'PORTFOLIO_SYNC_006';\n              enhancedMessage = 'GitHub API rate limit exceeded. Please try again later.';\n            } else {\n              errorCode = 'PORTFOLIO_SYNC_001';\n              enhancedMessage = 'GitHub API access forbidden. Check token permissions.';\n            }\n            break;\n          case 404:\n            errorCode = 'PORTFOLIO_SYNC_002';\n            enhancedMessage = 'GitHub portfolio repository not found. Please run init_portfolio first.';\n            break;\n          case 422:\n            errorCode = 'PORTFOLIO_SYNC_003';\n            enhancedMessage = 'Repository validation failed.';\n            break;\n          default:\n            // Keep the error code from githubRequest if set\n            if (!error.code) {\n              errorCode = 'PORTFOLIO_SYNC_005';\n            }\n        }\n      } else if (!error.code) {\n        // Fall back to message parsing only if no status code available\n        if (error.message?.includes('401') || error.message?.includes('authentication')) {\n          errorCode = 'PORTFOLIO_SYNC_001';\n          enhancedMessage = 'GitHub authentication failed. Please re-authenticate.';\n        } else if (error.message?.includes('404') || error.message?.includes('not found')) {\n          errorCode = 'PORTFOLIO_SYNC_002';\n          enhancedMessage = 'GitHub portfolio repository not found. Please run init_portfolio first.';\n        } else if (error.message?.includes('403') || error.message?.includes('rate limit')) {\n          errorCode = 'PORTFOLIO_SYNC_006';\n          enhancedMessage = 'GitHub API rate limit exceeded. Please try again later.';\n        } else if (error.message?.includes('Cannot read properties')) {\n          errorCode = 'PORTFOLIO_SYNC_004';\n          enhancedMessage = `GitHub API response parsing error: ${error.message}`;\n        }\n      }\n      \n      logger.error(`[${errorCode}] ${enhancedMessage}`, { \n        elementId: element.id,\n        username,\n        originalError: error.message,\n        errorStatus: error.status,\n        stack: error.stack\n      });\n      \n      ErrorHandler.logError('PortfolioRepoManager.saveElementToRepo', error, { \n        elementId: element.id,\n        username,\n        errorCode,\n        errorStatus: error.status\n      });\n      \n      // Throw error with code for better handling upstream\n      const wrappedError = ErrorHandler.wrapError(error, `[${errorCode}] ${enhancedMessage}`, ErrorCategory.NETWORK_ERROR);\n      (wrappedError as any).code = errorCode;\n      (wrappedError as any).status = error.status;\n      throw wrappedError;\n    }\n  }\n\n  /**\n   * Generate initial portfolio structure with README and directories\n   * SECURITY: Username already normalized by calling methods\n   */\n  async generatePortfolioStructure(username: string): Promise<void> {\n    // Create README.md\n    const readmeContent = `# DollhouseMCP Portfolio\n\nThis is my personal collection of DollhouseMCP elements.\n\n## Structure\n\n- **personas/** - Behavioral profiles\n- **skills/** - Discrete capabilities  \n- **templates/** - Reusable content structures\n- **agents/** - Autonomous actors\n- **memories/** - Persistent context\n- **ensembles/** - Element groups\n\n## Usage\n\nThese elements can be imported into your DollhouseMCP installation.\n\n---\n*Generated by DollhouseMCP*\n`;\n\n    await this.githubRequest(\n      `/repos/${username}/${this.repositoryName}/contents/README.md`,\n      'PUT',\n      {\n        message: 'Initialize portfolio structure',\n        content: Buffer.from(readmeContent).toString('base64')\n      }\n    );\n\n    // Create directory placeholders\n    const directories = ['personas', 'skills', 'templates', 'agents', 'memories', 'ensembles'];\n    \n    for (const dir of directories) {\n      await this.githubRequest(\n        `/repos/${username}/${this.repositoryName}/contents/${dir}/.gitkeep`,\n        'PUT',\n        {\n          message: `Create ${dir} directory`,\n          content: Buffer.from('').toString('base64')\n        }\n      );\n    }\n  }\n\n  /**\n   * Validate element before saving\n   * @throws Error if element is invalid\n   */\n  private validateElement(element: IElement): void {\n    if (!element.metadata.name) {\n      throw new Error('Invalid element: name is required');\n    }\n\n    if (!element.id) {\n      throw new Error('Invalid element: id is required');\n    }\n\n    if (!element.type) {\n      throw new Error('Invalid element: type is required');\n    }\n  }\n\n  /**\n   * Generate safe filename from element name\n   * SECURITY: Additional Unicode normalization for filenames\n   * SECURITY FIX: Fixed ReDoS vulnerability with input length limit and optimized regex\n   */\n  public static generateFileName(name: string): string {\n    // SECURITY FIX: Limit input length to prevent ReDoS attacks\n    // Even with optimized regex, very long inputs could cause performance issues\n    const MAX_FILENAME_LENGTH = 255; // Common filesystem limit\n    \n    // Normalize to prevent Unicode attacks in filenames\n    const normalizedName = UnicodeValidator.normalize(name).normalizedContent;\n    \n    // Truncate to safe length BEFORE regex operations\n    const truncatedName = normalizedName.slice(0, MAX_FILENAME_LENGTH);\n    \n    // SECURITY FIX: Optimized regex operations to prevent ReDoS\n    // 1. Convert non-alphanumeric sequences to single dash\n    // 2. Remove leading/trailing dashes in a single pass using trim\n    const safeName = truncatedName\n      .toLowerCase()\n      .replace(/[^a-z0-9]+/g, '-')\n      .replace(/^-+/, '')  // Remove leading dashes\n      .replace(/-+$/, ''); // Remove trailing dashes\n    \n    // Ensure we have a valid filename (not empty after cleaning)\n    return safeName || 'unnamed';\n  }\n\n  /**\n   * Format element content for storage\n   */\n  private formatElementContent(element: IElement): string {\n    // Serialize the element or create basic markdown\n    if (element.serialize) {\n      return element.serialize();\n    }\n    // Fallback to basic markdown format\n    return `# ${element.metadata.name}\\n\\n${element.metadata.description || ''}`;\n  }\n\n  /**\n   * Get the authenticated user's username\n   */\n  private async getUsername(): Promise<string> {\n    const response = await this.githubRequest('/user');\n    if (!response || !response.login) {\n      throw new Error('Failed to get GitHub username');\n    }\n    return response.login;\n  }\n\n  /**\n   * Get file content from GitHub repository\n   * Used for pull operations to download elements\n   */\n  async getFileContent(path: string, username?: string, repository?: string): Promise<string> {\n    try {\n      // Use provided username/repository or defaults\n      const repoUser = username || await this.getUsername();\n      const repoName = repository || this.repositoryName;\n      \n      logger.info('Fetching file content from GitHub', { \n        path, \n        username: repoUser, \n        repository: repoName \n      });\n\n      const response = await this.githubRequest(\n        `/repos/${repoUser}/${repoName}/contents/${path}`\n      );\n\n      if (!response || !response.content) {\n        throw new Error(`No content found at path: ${path}`);\n      }\n\n      // Decode base64 content\n      const decodedContent = Buffer.from(response.content, 'base64').toString('utf-8');\n      \n      return decodedContent;\n      \n    } catch (error) {\n      logger.error('Failed to get file content from GitHub', { \n        error, \n        path \n      });\n      \n      if (error instanceof Error) {\n        if (error.message.includes('404')) {\n          throw new Error(`File not found at path: ${path}`);\n        }\n        if (error.message.includes('401') || error.message.includes('403')) {\n          throw new Error(`Authentication failed. Please check your GitHub token.`);\n        }\n      }\n      \n      throw error;\n    }\n  }\n}"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"PortfolioSyncManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioSyncManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;
|
|
1
|
+
{"version":3,"file":"PortfolioSyncManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioSyncManager.ts"],"names":[],"mappings":"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"}
|