@dollhousemcp/mcp-server 1.6.9 → 1.6.10
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 +8 -0
- package/README.md +3 -3
- package/dist/config/error-codes.d.ts +29 -0
- package/dist/config/error-codes.d.ts.map +1 -0
- package/dist/config/error-codes.js +70 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.d.ts.map +1 -1
- package/dist/generated/version.js +3 -3
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioRepoManager.js +22 -3
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +6 -1
- 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.js +5 -5
- package/dist/services/BuildInfoService.d.ts +1 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -1
- package/dist/services/BuildInfoService.js +3 -2
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts +30 -1
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
- package/dist/tools/portfolio/submitToPortfolioTool.js +600 -76
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,13 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## [1.6.10] - 2025-08-28
|
|
4
|
+
|
|
5
|
+
### Fixed
|
|
6
|
+
- Collection submission pipeline now includes full markdown content with frontmatter (#818)
|
|
7
|
+
- Added missing `localPath` parameter to `submitElementAndHandleResponse()` call
|
|
8
|
+
- Fixes "No frontmatter found" errors in collection workflow
|
|
9
|
+
- Enables proper processing of submitted elements
|
|
10
|
+
|
|
3
11
|
## [1.6.9] - 2025-08-27
|
|
4
12
|
|
|
5
13
|
Fix content truncation in create_persona tool - personas were being truncated at 1000 chars
|
package/README.md
CHANGED
|
@@ -30,7 +30,7 @@ A comprehensive Model Context Protocol (MCP) server that enables dynamic AI pers
|
|
|
30
30
|
**🏪 Collection**: https://github.com/DollhouseMCP/collection
|
|
31
31
|
**📦 NPM Package**: https://www.npmjs.com/package/@dollhousemcp/mcp-server
|
|
32
32
|
**🌍 Website**: https://dollhousemcp.com (planned)
|
|
33
|
-
**📦 Version**: v1.6.
|
|
33
|
+
**📦 Version**: v1.6.10
|
|
34
34
|
|
|
35
35
|
> **🎉 New in v1.6.9**: Fixed version update script to prevent wrong file creation and package-lock.json corruption. The release workflow now works reliably with proper version management.
|
|
36
36
|
|
|
@@ -1147,7 +1147,7 @@ DollhouseMCP/
|
|
|
1147
1147
|
├── scripts/ # Management and utility scripts
|
|
1148
1148
|
├── Dockerfile # Multi-stage Docker build
|
|
1149
1149
|
├── docker-compose.yml # Production and development configs
|
|
1150
|
-
├── package.json # Project config (dollhousemcp v1.
|
|
1150
|
+
├── package.json # Project config (dollhousemcp v1.6.10)
|
|
1151
1151
|
├── tsconfig.json # TypeScript configuration
|
|
1152
1152
|
├── jest.config.cjs # Jest test configuration
|
|
1153
1153
|
├── setup.sh # Automated installation script
|
|
@@ -1547,7 +1547,7 @@ This project is licensed under the **AGPL-3.0** License with Platform Stability
|
|
|
1547
1547
|
|
|
1548
1548
|
## 🏷️ Version History
|
|
1549
1549
|
|
|
1550
|
-
### v1.6.9 - August 26, 2025
|
|
1550
|
+
### v1.6.9 - August 26, 2025
|
|
1551
1551
|
|
|
1552
1552
|
**Critical Fixes**: Fixed OAuth helper NPM packaging and performance testing workflow
|
|
1553
1553
|
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Error codes for collection submission process
|
|
3
|
+
* Implements detailed error tracking as specified in Issue #785
|
|
4
|
+
*/
|
|
5
|
+
export declare enum CollectionErrorCode {
|
|
6
|
+
COLL_AUTH_001 = "COLL_AUTH_001",// Token validation failed
|
|
7
|
+
COLL_AUTH_002 = "COLL_AUTH_002",// Token missing 'public_repo' scope
|
|
8
|
+
COLL_AUTH_003 = "COLL_AUTH_003",// OAuth helper not running
|
|
9
|
+
COLL_AUTH_004 = "COLL_AUTH_004",// Token expired during operation
|
|
10
|
+
COLL_API_001 = "COLL_API_001",// Rate limit exceeded
|
|
11
|
+
COLL_API_002 = "COLL_API_002",// Repository not found
|
|
12
|
+
COLL_API_003 = "COLL_API_003",// Issue creation failed
|
|
13
|
+
COLL_API_004 = "COLL_API_004",// Network timeout
|
|
14
|
+
COLL_CFG_001 = "COLL_CFG_001",// Auto-submit disabled
|
|
15
|
+
COLL_CFG_002 = "COLL_CFG_002",// Collection repo not configured
|
|
16
|
+
COLL_VAL_001 = "COLL_VAL_001",// Invalid element format
|
|
17
|
+
COLL_VAL_002 = "COLL_VAL_002"
|
|
18
|
+
}
|
|
19
|
+
export declare const CollectionErrorMessages: Record<CollectionErrorCode, string>;
|
|
20
|
+
export declare const CollectionErrorSolutions: Record<CollectionErrorCode, string>;
|
|
21
|
+
/**
|
|
22
|
+
* Format a collection error for user display
|
|
23
|
+
* @param code The error code
|
|
24
|
+
* @param step Current step number
|
|
25
|
+
* @param totalSteps Total number of steps
|
|
26
|
+
* @param details Additional error details
|
|
27
|
+
*/
|
|
28
|
+
export declare function formatCollectionError(code: CollectionErrorCode, step: number, totalSteps: number, details?: string): string;
|
|
29
|
+
//# sourceMappingURL=error-codes.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"error-codes.d.ts","sourceRoot":"","sources":["../../src/config/error-codes.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,oBAAY,mBAAmB;IAE7B,aAAa,kBAAkB,CAAE,0BAA0B;IAC3D,aAAa,kBAAkB,CAAE,oCAAoC;IACrE,aAAa,kBAAkB,CAAE,2BAA2B;IAC5D,aAAa,kBAAkB,CAAE,iCAAiC;IAGlE,YAAY,iBAAiB,CAAE,sBAAsB;IACrD,YAAY,iBAAiB,CAAE,uBAAuB;IACtD,YAAY,iBAAiB,CAAE,wBAAwB;IACvD,YAAY,iBAAiB,CAAE,kBAAkB;IAGjD,YAAY,iBAAiB,CAAE,uBAAuB;IACtD,YAAY,iBAAiB,CAAE,iCAAiC;IAGhE,YAAY,iBAAiB,CAAE,yBAAyB;IACxD,YAAY,iBAAiB;CAC9B;AAED,eAAO,MAAM,uBAAuB,EAAE,MAAM,CAAC,mBAAmB,EAAE,MAAM,CAgBvE,CAAC;AAEF,eAAO,MAAM,wBAAwB,EAAE,MAAM,CAAC,mBAAmB,EAAE,MAAM,CAgBxE,CAAC;AAEF;;;;;;GAMG;AACH,wBAAgB,qBAAqB,CACnC,IAAI,EAAE,mBAAmB,EACzB,IAAI,EAAE,MAAM,EACZ,UAAU,EAAE,MAAM,EAClB,OAAO,CAAC,EAAE,MAAM,GACf,MAAM,CAcR"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Error codes for collection submission process
|
|
3
|
+
* Implements detailed error tracking as specified in Issue #785
|
|
4
|
+
*/
|
|
5
|
+
export var CollectionErrorCode;
|
|
6
|
+
(function (CollectionErrorCode) {
|
|
7
|
+
// Authentication & Authorization (COLL_AUTH_xxx)
|
|
8
|
+
CollectionErrorCode["COLL_AUTH_001"] = "COLL_AUTH_001";
|
|
9
|
+
CollectionErrorCode["COLL_AUTH_002"] = "COLL_AUTH_002";
|
|
10
|
+
CollectionErrorCode["COLL_AUTH_003"] = "COLL_AUTH_003";
|
|
11
|
+
CollectionErrorCode["COLL_AUTH_004"] = "COLL_AUTH_004";
|
|
12
|
+
// GitHub API Operations (COLL_API_xxx)
|
|
13
|
+
CollectionErrorCode["COLL_API_001"] = "COLL_API_001";
|
|
14
|
+
CollectionErrorCode["COLL_API_002"] = "COLL_API_002";
|
|
15
|
+
CollectionErrorCode["COLL_API_003"] = "COLL_API_003";
|
|
16
|
+
CollectionErrorCode["COLL_API_004"] = "COLL_API_004";
|
|
17
|
+
// Configuration (COLL_CFG_xxx)
|
|
18
|
+
CollectionErrorCode["COLL_CFG_001"] = "COLL_CFG_001";
|
|
19
|
+
CollectionErrorCode["COLL_CFG_002"] = "COLL_CFG_002";
|
|
20
|
+
// Content Validation (COLL_VAL_xxx)
|
|
21
|
+
CollectionErrorCode["COLL_VAL_001"] = "COLL_VAL_001";
|
|
22
|
+
CollectionErrorCode["COLL_VAL_002"] = "COLL_VAL_002"; // Content exceeds size limit
|
|
23
|
+
})(CollectionErrorCode || (CollectionErrorCode = {}));
|
|
24
|
+
export const CollectionErrorMessages = {
|
|
25
|
+
[CollectionErrorCode.COLL_AUTH_001]: "Token validation failed",
|
|
26
|
+
[CollectionErrorCode.COLL_AUTH_002]: "Token missing 'public_repo' scope",
|
|
27
|
+
[CollectionErrorCode.COLL_AUTH_003]: "OAuth helper not running",
|
|
28
|
+
[CollectionErrorCode.COLL_AUTH_004]: "Token expired during operation",
|
|
29
|
+
[CollectionErrorCode.COLL_API_001]: "Rate limit exceeded",
|
|
30
|
+
[CollectionErrorCode.COLL_API_002]: "Repository not found",
|
|
31
|
+
[CollectionErrorCode.COLL_API_003]: "Issue creation failed",
|
|
32
|
+
[CollectionErrorCode.COLL_API_004]: "Network timeout",
|
|
33
|
+
[CollectionErrorCode.COLL_CFG_001]: "Auto-submit disabled",
|
|
34
|
+
[CollectionErrorCode.COLL_CFG_002]: "Collection repo not configured",
|
|
35
|
+
[CollectionErrorCode.COLL_VAL_001]: "Invalid element format",
|
|
36
|
+
[CollectionErrorCode.COLL_VAL_002]: "Content exceeds size limit"
|
|
37
|
+
};
|
|
38
|
+
export const CollectionErrorSolutions = {
|
|
39
|
+
[CollectionErrorCode.COLL_AUTH_001]: "Run 'setup_github_auth' to re-authenticate",
|
|
40
|
+
[CollectionErrorCode.COLL_AUTH_002]: "Re-authenticate with 'setup_github_auth' to get proper scopes",
|
|
41
|
+
[CollectionErrorCode.COLL_AUTH_003]: "Run 'setup_github_auth' to restart authentication",
|
|
42
|
+
[CollectionErrorCode.COLL_AUTH_004]: "Token expired. Run 'setup_github_auth' to refresh",
|
|
43
|
+
[CollectionErrorCode.COLL_API_001]: "Wait for rate limit to reset or use a different token",
|
|
44
|
+
[CollectionErrorCode.COLL_API_002]: "Verify repository exists and you have access",
|
|
45
|
+
[CollectionErrorCode.COLL_API_003]: "Check GitHub status and try again",
|
|
46
|
+
[CollectionErrorCode.COLL_API_004]: "Check network connection and retry",
|
|
47
|
+
[CollectionErrorCode.COLL_CFG_001]: "Run 'configure_collection_submission autoSubmit: true'",
|
|
48
|
+
[CollectionErrorCode.COLL_CFG_002]: "Check collection repository configuration",
|
|
49
|
+
[CollectionErrorCode.COLL_VAL_001]: "Verify element has proper metadata and content",
|
|
50
|
+
[CollectionErrorCode.COLL_VAL_002]: "Reduce content size to under 500KB"
|
|
51
|
+
};
|
|
52
|
+
/**
|
|
53
|
+
* Format a collection error for user display
|
|
54
|
+
* @param code The error code
|
|
55
|
+
* @param step Current step number
|
|
56
|
+
* @param totalSteps Total number of steps
|
|
57
|
+
* @param details Additional error details
|
|
58
|
+
*/
|
|
59
|
+
export function formatCollectionError(code, step, totalSteps, details) {
|
|
60
|
+
const message = CollectionErrorMessages[code];
|
|
61
|
+
const solution = CollectionErrorSolutions[code];
|
|
62
|
+
let output = `Collection Submission Failed at Step ${step}/${totalSteps}:\n`;
|
|
63
|
+
output += `Error ${code}: ${message}\n`;
|
|
64
|
+
if (details) {
|
|
65
|
+
output += `Details: ${details}\n`;
|
|
66
|
+
}
|
|
67
|
+
output += `Solution: ${solution}`;
|
|
68
|
+
return output;
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"error-codes.js","sourceRoot":"","sources":["../../src/config/error-codes.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,MAAM,CAAN,IAAY,mBAoBX;AApBD,WAAY,mBAAmB;IAC7B,iDAAiD;IACjD,sDAA+B,CAAA;IAC/B,sDAA+B,CAAA;IAC/B,sDAA+B,CAAA;IAC/B,sDAA+B,CAAA;IAE/B,uCAAuC;IACvC,oDAA6B,CAAA;IAC7B,oDAA6B,CAAA;IAC7B,oDAA6B,CAAA;IAC7B,oDAA6B,CAAA;IAE7B,+BAA+B;IAC/B,oDAA6B,CAAA;IAC7B,oDAA6B,CAAA;IAE7B,oCAAoC;IACpC,oDAA6B,CAAA;IAC7B,oDAA6B,CAAA,CAAE,6BAA6B;AAC9D,CAAC,EApBW,mBAAmB,KAAnB,mBAAmB,QAoB9B;AAED,MAAM,CAAC,MAAM,uBAAuB,GAAwC;IAC1E,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,yBAAyB;IAC9D,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,mCAAmC;IACxE,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,0BAA0B;IAC/D,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,gCAAgC;IAErE,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,qBAAqB;IACzD,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,sBAAsB;IAC1D,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,uBAAuB;IAC3D,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,iBAAiB;IAErD,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,sBAAsB;IAC1D,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,gCAAgC;IAEpE,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,wBAAwB;IAC5D,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,4BAA4B;CACjE,CAAC;AAEF,MAAM,CAAC,MAAM,wBAAwB,GAAwC;IAC3E,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,4CAA4C;IACjF,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,+DAA+D;IACpG,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,mDAAmD;IACxF,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAE,mDAAmD;IAExF,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,uDAAuD;IAC3F,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,8CAA8C;IAClF,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,mCAAmC;IACvE,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,oCAAoC;IAExE,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,wDAAwD;IAC5F,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,2CAA2C;IAE/E,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,gDAAgD;IACpF,CAAC,mBAAmB,CAAC,YAAY,CAAC,EAAE,oCAAoC;CACzE,CAAC;AAEF;;;;;;GAMG;AACH,MAAM,UAAU,qBAAqB,CACnC,IAAyB,EACzB,IAAY,EACZ,UAAkB,EAClB,OAAgB;IAEhB,MAAM,OAAO,GAAG,uBAAuB,CAAC,IAAI,CAAC,CAAC;IAC9C,MAAM,QAAQ,GAAG,wBAAwB,CAAC,IAAI,CAAC,CAAC;IAEhD,IAAI,MAAM,GAAG,wCAAwC,IAAI,IAAI,UAAU,KAAK,CAAC;IAC7E,MAAM,IAAI,SAAS,IAAI,KAAK,OAAO,IAAI,CAAC;IAExC,IAAI,OAAO,EAAE,CAAC;QACZ,MAAM,IAAI,YAAY,OAAO,IAAI,CAAC;IACpC,CAAC;IAED,MAAM,IAAI,aAAa,QAAQ,EAAE,CAAC;IAElC,OAAO,MAAM,CAAC;AAChB,CAAC","sourcesContent":["/**\n * Error codes for collection submission process\n * Implements detailed error tracking as specified in Issue #785\n */\n\nexport enum CollectionErrorCode {\n  // Authentication & Authorization (COLL_AUTH_xxx)\n  COLL_AUTH_001 = \"COLL_AUTH_001\", // Token validation failed\n  COLL_AUTH_002 = \"COLL_AUTH_002\", // Token missing 'public_repo' scope\n  COLL_AUTH_003 = \"COLL_AUTH_003\", // OAuth helper not running\n  COLL_AUTH_004 = \"COLL_AUTH_004\", // Token expired during operation\n  \n  // GitHub API Operations (COLL_API_xxx)\n  COLL_API_001 = \"COLL_API_001\", // Rate limit exceeded\n  COLL_API_002 = \"COLL_API_002\", // Repository not found\n  COLL_API_003 = \"COLL_API_003\", // Issue creation failed\n  COLL_API_004 = \"COLL_API_004\", // Network timeout\n  \n  // Configuration (COLL_CFG_xxx)\n  COLL_CFG_001 = \"COLL_CFG_001\", // Auto-submit disabled\n  COLL_CFG_002 = \"COLL_CFG_002\", // Collection repo not configured\n  \n  // Content Validation (COLL_VAL_xxx)\n  COLL_VAL_001 = \"COLL_VAL_001\", // Invalid element format\n  COLL_VAL_002 = \"COLL_VAL_002\"  // Content exceeds size limit\n}\n\nexport const CollectionErrorMessages: Record<CollectionErrorCode, string> = {\n  [CollectionErrorCode.COLL_AUTH_001]: \"Token validation failed\",\n  [CollectionErrorCode.COLL_AUTH_002]: \"Token missing 'public_repo' scope\",\n  [CollectionErrorCode.COLL_AUTH_003]: \"OAuth helper not running\",\n  [CollectionErrorCode.COLL_AUTH_004]: \"Token expired during operation\",\n  \n  [CollectionErrorCode.COLL_API_001]: \"Rate limit exceeded\",\n  [CollectionErrorCode.COLL_API_002]: \"Repository not found\",\n  [CollectionErrorCode.COLL_API_003]: \"Issue creation failed\",\n  [CollectionErrorCode.COLL_API_004]: \"Network timeout\",\n  \n  [CollectionErrorCode.COLL_CFG_001]: \"Auto-submit disabled\",\n  [CollectionErrorCode.COLL_CFG_002]: \"Collection repo not configured\",\n  \n  [CollectionErrorCode.COLL_VAL_001]: \"Invalid element format\",\n  [CollectionErrorCode.COLL_VAL_002]: \"Content exceeds size limit\"\n};\n\nexport const CollectionErrorSolutions: Record<CollectionErrorCode, string> = {\n  [CollectionErrorCode.COLL_AUTH_001]: \"Run 'setup_github_auth' to re-authenticate\",\n  [CollectionErrorCode.COLL_AUTH_002]: \"Re-authenticate with 'setup_github_auth' to get proper scopes\",\n  [CollectionErrorCode.COLL_AUTH_003]: \"Run 'setup_github_auth' to restart authentication\",\n  [CollectionErrorCode.COLL_AUTH_004]: \"Token expired. Run 'setup_github_auth' to refresh\",\n  \n  [CollectionErrorCode.COLL_API_001]: \"Wait for rate limit to reset or use a different token\",\n  [CollectionErrorCode.COLL_API_002]: \"Verify repository exists and you have access\",\n  [CollectionErrorCode.COLL_API_003]: \"Check GitHub status and try again\",\n  [CollectionErrorCode.COLL_API_004]: \"Check network connection and retry\",\n  \n  [CollectionErrorCode.COLL_CFG_001]: \"Run 'configure_collection_submission autoSubmit: true'\",\n  [CollectionErrorCode.COLL_CFG_002]: \"Check collection repository configuration\",\n  \n  [CollectionErrorCode.COLL_VAL_001]: \"Verify element has proper metadata and content\",\n  [CollectionErrorCode.COLL_VAL_002]: \"Reduce content size to under 500KB\"\n};\n\n/**\n * Format a collection error for user display\n * @param code The error code\n * @param step Current step number\n * @param totalSteps Total number of steps\n * @param details Additional error details\n */\nexport function formatCollectionError(\n  code: CollectionErrorCode,\n  step: number,\n  totalSteps: number,\n  details?: string\n): string {\n  const message = CollectionErrorMessages[code];\n  const solution = CollectionErrorSolutions[code];\n  \n  let output = `Collection Submission Failed at Step ${step}/${totalSteps}:\\n`;\n  output += `Error ${code}: ${message}\\n`;\n  \n  if (details) {\n    output += `Details: ${details}\\n`;\n  }\n  \n  output += `Solution: ${solution}`;\n  \n  return output;\n}"]}
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export declare const PACKAGE_VERSION = "1.6.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2025-08-
|
|
5
|
+
export declare const PACKAGE_VERSION = "1.6.10";
|
|
6
|
+
export declare const BUILD_TIMESTAMP = "2025-08-28T19:40:14.869Z";
|
|
7
7
|
export declare const BUILD_TYPE: 'npm' | 'git';
|
|
8
8
|
export declare const PACKAGE_NAME = "@dollhousemcp/mcp-server";
|
|
9
9
|
//# sourceMappingURL=version.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"version.d.ts","sourceRoot":"","sources":["../../src/generated/version.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,eAAO,MAAM,eAAe,
|
|
1
|
+
{"version":3,"file":"version.d.ts","sourceRoot":"","sources":["../../src/generated/version.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,eAAO,MAAM,eAAe,WAAW,CAAC;AACxC,eAAO,MAAM,eAAe,6BAA6B,CAAC;AAC1D,eAAO,MAAM,UAAU,EAAE,KAAK,GAAG,KAAa,CAAC;AAC/C,eAAO,MAAM,YAAY,6BAA6B,CAAC"}
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export const PACKAGE_VERSION = '1.6.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2025-08-
|
|
5
|
+
export const PACKAGE_VERSION = '1.6.10';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2025-08-28T19:40:14.869Z';
|
|
7
7
|
export const BUILD_TYPE = 'npm';
|
|
8
8
|
export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsUUFBUSxDQUFDO0FBQ3hDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjYuMTAnO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RJTUVTVEFNUCA9ICcyMDI1LTA4LTI4VDE5OjQwOjE0Ljg2OVonO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RZUEU6ICducG0nIHwgJ2dpdCcgPSAnbnBtJztcbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSAnQGRvbGxob3VzZW1jcC9tY3Atc2VydmVyJztcbiJdfQ==
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"PortfolioRepoManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,+BAA+B,CAAC;AAOzD,MAAM,WAAW,oBAAoB;IACnC,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAyB;IACpE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAuC;IAClF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAA4B;IAEnE,OAAO,CAAC,KAAK,CAAuB;;IAMpC;;;OAGG;IACI,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAIpC;;;;OAIG;YACW,mBAAmB;IA6BjC;;OAEG;YACW,aAAa;IA0D3B;;;;OAIG;IACG,oBAAoB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAe9D;;;OAGG;IACG,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IA+EtF;;;OAGG;IACG,WAAW,CAAC,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"PortfolioRepoManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,+BAA+B,CAAC;AAOzD,MAAM,WAAW,oBAAoB;IACnC,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAyB;IACpE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAuC;IAClF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAA4B;IAEnE,OAAO,CAAC,KAAK,CAAuB;;IAMpC;;;OAGG;IACI,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAIpC;;;;OAIG;YACW,mBAAmB;IA6BjC;;OAEG;YACW,aAAa;IA0D3B;;;;OAIG;IACG,oBAAoB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAe9D;;;OAGG;IACG,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IA+EtF;;;OAGG;IACG,WAAW,CAAC,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IA2KnF;;;OAGG;IACG,0BAA0B,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA+CjE;;;OAGG;IACH,OAAO,CAAC,eAAe;IAcvB;;;;OAIG;IACH,OAAO,CAAC,gBAAgB;IAwBxB;;OAEG;IACH,OAAO,CAAC,oBAAoB;CAQ7B"}
|
|
@@ -232,9 +232,28 @@ export class PortfolioRepoManager {
|
|
|
232
232
|
try {
|
|
233
233
|
// First, check if file exists to determine if this is create or update
|
|
234
234
|
const existingFile = await this.githubRequest(`/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`);
|
|
235
|
-
//
|
|
235
|
+
// DUPLICATE DETECTION (Issue #792): Check if content is identical
|
|
236
|
+
if (existingFile && existingFile.content) {
|
|
237
|
+
// Decode existing content from base64
|
|
238
|
+
const existingContent = Buffer.from(existingFile.content, 'base64').toString('utf-8');
|
|
239
|
+
// Compare with new content
|
|
240
|
+
if (existingContent === content) {
|
|
241
|
+
logger.info('Skipping duplicate portfolio upload - content identical', {
|
|
242
|
+
elementId: element.id,
|
|
243
|
+
elementName: element.metadata.name,
|
|
244
|
+
filePath
|
|
245
|
+
});
|
|
246
|
+
// Return the existing file URL instead of creating duplicate commit
|
|
247
|
+
const existingUrl = existingFile.html_url ||
|
|
248
|
+
`https://github.com/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/blob/main/${filePath}`;
|
|
249
|
+
return existingUrl;
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
// Create or update the file (only if content is different)
|
|
236
253
|
const result = await this.githubRequest(`/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`, 'PUT', {
|
|
237
|
-
message:
|
|
254
|
+
message: existingFile ?
|
|
255
|
+
`Update ${element.metadata.name} in portfolio` :
|
|
256
|
+
`Add ${element.metadata.name} to portfolio`,
|
|
238
257
|
content: Buffer.from(content).toString('base64'),
|
|
239
258
|
sha: existingFile?.sha // Include SHA if updating existing file
|
|
240
259
|
});
|
|
@@ -408,4 +427,4 @@ These elements can be imported into your DollhouseMCP installation.
|
|
|
408
427
|
return `# ${element.metadata.name}\n\n${element.metadata.description || ''}`;
|
|
409
428
|
}
|
|
410
429
|
}
|
|
411
|
-
//# 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   */\n  private 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    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      // Provide more specific error messages for common status codes\n      let errorMessage = data.message || `GitHub API error: ${response.status}`;\n      \n      switch (response.status) {\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          break;\n        case 401:\n          errorMessage = 'GitHub authentication failed. Please check your token.';\n          break;\n        case 403:\n          errorMessage = `GitHub API access forbidden: ${data.message || 'insufficient permissions or rate limit exceeded'}`;\n          break;\n        case 500:\n          errorMessage = 'GitHub API server error. Please try again later.';\n          break;\n        default:\n          errorMessage = `GitHub API error (${response.status}): ${data.message || 'Unknown error'}`;\n      }\n      \n      throw new Error(errorMessage);\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 = this.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      const existingFile = await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`\n      );\n\n      // Create or update the file\n      const result = await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`,\n        'PUT',\n        {\n          message: `Add ${element.metadata.name} to portfolio`,\n          content: Buffer.from(content).toString('base64'),\n          sha: existingFile?.sha // Include SHA if updating existing file\n        }\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      // Enhanced error reporting with specific error codes\n      let errorCode = 'PORTFOLIO_SYNC_005'; // Default network error\n      let enhancedMessage = 'Failed to save element to portfolio';\n      \n      // Check for specific error conditions\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 of null')) {\n        errorCode = 'PORTFOLIO_SYNC_004';\n        enhancedMessage = `GitHub API response parsing error: ${error.message}`;\n      }\n      \n      logger.error(`[${errorCode}] ${enhancedMessage}`, { \n        elementId: element.id,\n        username,\n        originalError: error.message,\n        stack: error.stack\n      });\n      \n      ErrorHandler.logError('PortfolioRepoManager.saveElementToRepo', error, { \n        elementId: element.id,\n        username,\n        errorCode\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      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  private 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}"]}
|
|
430
|
+
//# 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   */\n  private 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    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      // Provide more specific error messages for common status codes\n      let errorMessage = data.message || `GitHub API error: ${response.status}`;\n      \n      switch (response.status) {\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          break;\n        case 401:\n          errorMessage = 'GitHub authentication failed. Please check your token.';\n          break;\n        case 403:\n          errorMessage = `GitHub API access forbidden: ${data.message || 'insufficient permissions or rate limit exceeded'}`;\n          break;\n        case 500:\n          errorMessage = 'GitHub API server error. Please try again later.';\n          break;\n        default:\n          errorMessage = `GitHub API error (${response.status}): ${data.message || 'Unknown error'}`;\n      }\n      \n      throw new Error(errorMessage);\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 = this.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      const existingFile = await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`\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      const result = await this.githubRequest(\n        `/repos/${username}/${PortfolioRepoManager.PORTFOLIO_REPO_NAME}/contents/${filePath}`,\n        'PUT',\n        {\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          sha: existingFile?.sha // Include SHA if updating existing file\n        }\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      // Enhanced error reporting with specific error codes\n      let errorCode = 'PORTFOLIO_SYNC_005'; // Default network error\n      let enhancedMessage = 'Failed to save element to portfolio';\n      \n      // Check for specific error conditions\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 of null')) {\n        errorCode = 'PORTFOLIO_SYNC_004';\n        enhancedMessage = `GitHub API response parsing error: ${error.message}`;\n      }\n      \n      logger.error(`[${errorCode}] ${enhancedMessage}`, { \n        elementId: element.id,\n        username,\n        originalError: error.message,\n        stack: error.stack\n      });\n      \n      ErrorHandler.logError('PortfolioRepoManager.saveElementToRepo', error, { \n        elementId: element.id,\n        username,\n        errorCode\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      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  private 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}"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"suppressions.d.ts","sourceRoot":"","sources":["../../../../src/security/audit/config/suppressions.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAKH,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;CAChB;AAqCD,eAAO,MAAM,YAAY,EAAE,WAAW,
|
|
1
|
+
{"version":3,"file":"suppressions.d.ts","sourceRoot":"","sources":["../../../../src/security/audit/config/suppressions.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAKH,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;CAChB;AAqCD,eAAO,MAAM,YAAY,EAAE,WAAW,EA8WrC,CAAC;AAEF;;GAEG;AACH,wBAAgB,oBAAoB,IAAI,MAAM,EAAE,CA8B/C;AAuJD;;;GAGG;AACH,wBAAgB,cAAc,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO,CAgDzE;AAED;;GAEG;AACH,wBAAgB,qBAAqB,IAAI,IAAI,CAE5C;AAED;;GAEG;AACH,wBAAgB,mBAAmB,IAAI;IACrC,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC/B,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACpC,CAiBA"}
|