@dollhousemcp/mcp-server 2.0.28 → 2.0.29
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 +7 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/services/BuildInfoService.d.ts +6 -1
- package/dist/services/BuildInfoService.d.ts.map +1 -1
- package/dist/services/BuildInfoService.js +23 -3
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
- package/dist/tools/portfolio/submitToPortfolioTool.js +4 -3
- package/dist/utils/permissionHooks.d.ts +18 -0
- package/dist/utils/permissionHooks.d.ts.map +1 -1
- package/dist/utils/permissionHooks.js +182 -15
- package/dist/web/public/permissions.js +10 -0
- package/dist/web/public/setup.js +36 -0
- package/dist/web/routes/permissionRoutes.d.ts +1 -0
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +17 -7
- package/dist/web/routes/setupRoutes.d.ts +5 -1
- package/dist/web/routes/setupRoutes.d.ts.map +1 -1
- package/dist/web/routes/setupRoutes.js +28 -14
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +5 -1
- package/package.json +1 -1
- package/server.json +2 -2
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,12 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## [2.0.29] - 2026-04-21
|
|
4
|
+
|
|
5
|
+
- Added explicit session platform metadata and update-available status in the web console session dropdown
|
|
6
|
+
- Improved dropdown alignment for session status, platform, and uptime columns
|
|
7
|
+
- Fixed Codex pre-tool-use hook JSON output and expanded hook contract coverage across supported clients
|
|
8
|
+
- Automatically verify and refresh installed local permission hook assets when they go stale
|
|
9
|
+
|
|
3
10
|
## [2.0.28] - 2026-04-21
|
|
4
11
|
|
|
5
12
|
- Add explicit session client metadata and update-available status in the web console dropdown, with improved layout alignment.
|
|
@@ -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 = "2.0.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2026-04-
|
|
5
|
+
export declare const PACKAGE_VERSION = "2.0.29";
|
|
6
|
+
export declare const BUILD_TIMESTAMP = "2026-04-21T19:29:55.786Z";
|
|
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
|
|
@@ -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 = '2.0.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2026-04-
|
|
5
|
+
export const PACKAGE_VERSION = '2.0.29';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2026-04-21T19:29:55.786Z';
|
|
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsUUFBUSxDQUFDO0FBQ3hDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjknO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RJTUVTVEFNUCA9ICcyMDI2LTA0LTIxVDE5OjI5OjU1Ljc4NlonO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RZUEU6ICducG0nIHwgJ2dpdCcgPSAnbnBtJztcbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSAnQGRvbGxob3VzZW1jcC9tY3Atc2VydmVyJztcbiJdfQ==
|
|
@@ -23,7 +23,6 @@ export interface BuildInfo {
|
|
|
23
23
|
type: 'git' | 'npm' | 'unknown';
|
|
24
24
|
gitCommit?: string;
|
|
25
25
|
gitBranch?: string;
|
|
26
|
-
collectionFix?: string;
|
|
27
26
|
};
|
|
28
27
|
runtime: {
|
|
29
28
|
nodeVersion: string;
|
|
@@ -45,6 +44,12 @@ export interface BuildInfo {
|
|
|
45
44
|
uptime: number;
|
|
46
45
|
mcpConnection: boolean;
|
|
47
46
|
};
|
|
47
|
+
permissionHooks?: {
|
|
48
|
+
installedHosts: string[];
|
|
49
|
+
currentHosts: string[];
|
|
50
|
+
repairedHosts: string[];
|
|
51
|
+
needsRepairHosts: string[];
|
|
52
|
+
};
|
|
48
53
|
/** Issue #706: Startup timing and readiness status. */
|
|
49
54
|
startup?: {
|
|
50
55
|
status: 'ready' | 'initializing';
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"BuildInfoService.d.ts","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AAEpE,OAAO,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;
|
|
1
|
+
{"version":3,"file":"BuildInfoService.d.ts","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AAEpE,OAAO,KAAK,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAIhF,MAAM,WAAW,SAAS;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,gBAAgB,EAAE,MAAM,CAAC;IACzB,aAAa,EAAE,KAAK,GAAG,SAAS,CAAC;IACjC,OAAO,EAAE;QACP,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,MAAM,CAAC;KACjB,CAAC;IACF,KAAK,EAAE;QACL,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,KAAK,GAAG,KAAK,GAAG,SAAS,CAAC;QAChC,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,SAAS,CAAC,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,OAAO,EAAE;QACP,WAAW,EAAE,MAAM,CAAC;QACpB,QAAQ,EAAE,MAAM,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,MAAM,EAAE,MAAM,CAAC;QACf,WAAW,EAAE,MAAM,CAAC,WAAW,CAAC;KACjC,CAAC;IACF,WAAW,EAAE;QACX,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,YAAY,EAAE,OAAO,CAAC;QACtB,aAAa,EAAE,OAAO,CAAC;QACvB,OAAO,EAAE,OAAO,CAAC;QACjB,QAAQ,EAAE,OAAO,CAAC;QAClB,UAAU,CAAC,EAAE,MAAM,CAAC;KACrB,CAAC;IACF,MAAM,EAAE;QACN,SAAS,EAAE,IAAI,CAAC;QAChB,MAAM,EAAE,MAAM,CAAC;QACf,aAAa,EAAE,OAAO,CAAC;KACxB,CAAC;IACF,eAAe,CAAC,EAAE;QAChB,cAAc,EAAE,MAAM,EAAE,CAAC;QACzB,YAAY,EAAE,MAAM,EAAE,CAAC;QACvB,aAAa,EAAE,MAAM,EAAE,CAAC;QACxB,gBAAgB,EAAE,MAAM,EAAE,CAAC;KAC5B,CAAC;IACF,uDAAuD;IACvD,OAAO,CAAC,EAAE;QACR,MAAM,EAAE,OAAO,GAAG,cAAc,CAAC;QACjC,qBAAqB,EAAE,OAAO,CAAC;QAC/B,QAAQ,EAAE,MAAM,CAAC;QACjB,eAAe,CAAC,EAAE,aAAa,CAAC;KACjC,CAAC;CACH;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAO;IACjC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAyB;IACxD,qEAAqE;IACrE,OAAO,CAAC,YAAY,CAA6B;IACjD,2DAA2D;IAC3D,OAAO,CAAC,oBAAoB,CAAgC;gBAEhD,cAAc,EAAE,sBAAsB;IAKlD;;;OAGG;IACH,eAAe,CAAC,KAAK,EAAE,YAAY,GAAG,IAAI;IAI1C;;OAEG;IACH,uBAAuB,CAAC,OAAO,EAAE,MAAM,OAAO,GAAG,IAAI;IAIrD;;;;;;;OAOG;IACU,YAAY,IAAI,OAAO,CAAC,SAAS,CAAC;IAuF/C;;;;OAIG;IACI,eAAe,CAAC,IAAI,EAAE,SAAS,GAAG,MAAM;IA+G/C;;;OAGG;YACW,UAAU;IAcxB;;;OAGG;YACW,aAAa;IA+B3B,OAAO,CAAC,YAAY;IAepB,OAAO,CAAC,YAAY;CAIrB"}
|
|
@@ -12,6 +12,7 @@ import * as child_process from 'child_process';
|
|
|
12
12
|
import { logger } from '../utils/logger.js';
|
|
13
13
|
import { PACKAGE_NAME, PACKAGE_VERSION, BUILD_TIMESTAMP, BUILD_TYPE } from '../generated/version.js';
|
|
14
14
|
import { resolveSessionIdentity } from './sessionIdentity.js';
|
|
15
|
+
import { getPermissionHookAuditSummary } from '../utils/permissionHooks.js';
|
|
15
16
|
export class BuildInfoService {
|
|
16
17
|
startTime;
|
|
17
18
|
fileOperations;
|
|
@@ -48,7 +49,8 @@ export class BuildInfoService {
|
|
|
48
49
|
// Use Promise.allSettled to collect all available info, even if some sources fail
|
|
49
50
|
const results = await Promise.allSettled([
|
|
50
51
|
this.getGitInfo(),
|
|
51
|
-
this.getDockerInfo()
|
|
52
|
+
this.getDockerInfo(),
|
|
53
|
+
getPermissionHookAuditSummary(),
|
|
52
54
|
]);
|
|
53
55
|
// Package info comes from build-time generated constants
|
|
54
56
|
const packageInfo = { name: PACKAGE_NAME, version: PACKAGE_VERSION };
|
|
@@ -58,6 +60,9 @@ export class BuildInfoService {
|
|
|
58
60
|
const dockerInfo = results[1].status === 'fulfilled'
|
|
59
61
|
? results[1].value
|
|
60
62
|
: { isDocker: false, info: undefined };
|
|
63
|
+
const permissionHookInfo = results[2].status === 'fulfilled'
|
|
64
|
+
? results[2].value
|
|
65
|
+
: { installedHosts: [], currentHosts: [], repairedHosts: [], needsRepairHosts: [] };
|
|
61
66
|
// Log any failures for diagnostics
|
|
62
67
|
const failures = [];
|
|
63
68
|
if (results[0].status === 'rejected') {
|
|
@@ -66,6 +71,9 @@ export class BuildInfoService {
|
|
|
66
71
|
if (results[1].status === 'rejected') {
|
|
67
72
|
failures.push(`docker info: ${results[1].reason}`);
|
|
68
73
|
}
|
|
74
|
+
if (results[2].status === 'rejected') {
|
|
75
|
+
failures.push(`permission hook audit: ${results[2].reason}`);
|
|
76
|
+
}
|
|
69
77
|
if (failures.length > 0) {
|
|
70
78
|
logger.debug(`Build info collection had ${failures.length} failure(s): ${failures.join('; ')}`);
|
|
71
79
|
}
|
|
@@ -90,7 +98,6 @@ export class BuildInfoService {
|
|
|
90
98
|
type: BUILD_TYPE,
|
|
91
99
|
gitCommit: gitInfo.commit,
|
|
92
100
|
gitBranch: gitInfo.branch,
|
|
93
|
-
collectionFix: 'v1.6.9-beta1-collection-fix' // Version identifier for verification
|
|
94
101
|
},
|
|
95
102
|
runtime: {
|
|
96
103
|
nodeVersion: process.version,
|
|
@@ -112,6 +119,7 @@ export class BuildInfoService {
|
|
|
112
119
|
uptime: Date.now() - this.startTime.getTime(),
|
|
113
120
|
mcpConnection: true // We're connected if this method is being called via MCP
|
|
114
121
|
},
|
|
122
|
+
permissionHooks: permissionHookInfo,
|
|
115
123
|
startup: startupInfo,
|
|
116
124
|
};
|
|
117
125
|
}
|
|
@@ -176,6 +184,18 @@ export class BuildInfoService {
|
|
|
176
184
|
lines.push(`- **Started**: ${info.server.startTime.toISOString()}`);
|
|
177
185
|
lines.push(`- **Uptime**: ${this.formatUptime(info.server.uptime / 1000)}`);
|
|
178
186
|
lines.push(`- **MCP Connection**: ${info.server.mcpConnection ? '✅ Connected' : '❌ Disconnected'}`);
|
|
187
|
+
if (info.permissionHooks) {
|
|
188
|
+
const installedHosts = info.permissionHooks.installedHosts.length > 0
|
|
189
|
+
? info.permissionHooks.installedHosts.join(', ')
|
|
190
|
+
: 'None';
|
|
191
|
+
const currentHosts = info.permissionHooks.currentHosts.length > 0
|
|
192
|
+
? info.permissionHooks.currentHosts.join(', ')
|
|
193
|
+
: 'None';
|
|
194
|
+
const needsRepairHosts = info.permissionHooks.needsRepairHosts.length > 0
|
|
195
|
+
? info.permissionHooks.needsRepairHosts.join(', ')
|
|
196
|
+
: 'None';
|
|
197
|
+
lines.push('', '## 🔐 Permission Hooks', `- **Installed Hosts**: ${installedHosts}`, `- **Current Assets**: ${currentHosts}`, `- **Needs Repair**: ${needsRepairHosts}`);
|
|
198
|
+
}
|
|
179
199
|
// Issue #706: Startup timing
|
|
180
200
|
if (info.startup) {
|
|
181
201
|
lines.push('');
|
|
@@ -271,4 +291,4 @@ export class BuildInfoService {
|
|
|
271
291
|
return `${mb.toFixed(1)} MB`;
|
|
272
292
|
}
|
|
273
293
|
}
|
|
274
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BuildInfoService.js","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BuildInfoService - Provides build and runtime information\n * Separated from main index.ts to avoid making that file larger\n * \n * SECURITY FIX (PR #614):\n * 1. DMCP-SEC-004: FALSE POSITIVE SUPPRESSION - No user input Unicode normalization needed\n *    This service only processes system information (git, package.json, environment variables)\n *    The MCP tool 'get_build_info' takes NO parameters (empty inputSchema)\n *    No user-provided data flows through this service that requires Unicode normalization\n */\n\nimport * as child_process from 'child_process';\nimport { logger } from '../utils/logger.js';\nimport { IFileOperationsService } from './FileOperationsService.js';\nimport { PACKAGE_NAME, PACKAGE_VERSION, BUILD_TIMESTAMP, BUILD_TYPE } from '../generated/version.js';\nimport type { StartupTimer, StartupReport } from '../telemetry/StartupTimer.js';\nimport { resolveSessionIdentity } from './sessionIdentity.js';\n\nexport interface BuildInfo {\n  sessionId: string;\n  runtimeSessionId: string;\n  sessionSource: 'env' | 'derived';\n  package: {\n    name: string;\n    version: string;\n  };\n  build: {\n    timestamp?: string;\n    type: 'git' | 'npm' | 'unknown';\n    gitCommit?: string;\n    gitBranch?: string;\n    collectionFix?: string;  // Version identifier for verification\n  };\n  runtime: {\n    nodeVersion: string;\n    platform: string;\n    arch: string;\n    uptime: number;\n    memoryUsage: NodeJS.MemoryUsage;\n  };\n  environment: {\n    nodeEnv?: string;\n    isProduction: boolean;\n    isDevelopment: boolean;\n    isDebug: boolean;\n    isDocker: boolean;\n    dockerInfo?: string;\n  };\n  server: {\n    startTime: Date;\n    uptime: number;\n    mcpConnection: boolean;\n  };\n  /** Issue #706: Startup timing and readiness status. */\n  startup?: {\n    status: 'ready' | 'initializing';\n    deferredSetupComplete: boolean;\n    uptimeMs: number;\n    startupTimingMs?: StartupReport;\n  };\n}\n\nexport class BuildInfoService {\n  private readonly startTime: Date;\n  private readonly fileOperations: IFileOperationsService;\n  /** Issue #706: Optional startup timer for timing instrumentation. */\n  private startupTimer: StartupTimer | null = null;\n  /** Issue #706: Callback to check deferred setup status. */\n  private deferredSetupChecker: (() => boolean) | null = null;\n\n  constructor(fileOperations: IFileOperationsService) {\n    this.startTime = new Date();\n    this.fileOperations = fileOperations;\n  }\n\n  /**\n   * Issue #706: Wire in startup instrumentation after DI is ready.\n   * Called from Container to avoid circular dependency at registration time.\n   */\n  setStartupTimer(timer: StartupTimer): void {\n    this.startupTimer = timer;\n  }\n\n  /**\n   * Issue #706: Wire in deferred setup status checker.\n   */\n  setDeferredSetupChecker(checker: () => boolean): void {\n    this.deferredSetupChecker = checker;\n  }\n\n  /**\n   * Get comprehensive build information\n   * SECURITY NOTE: This method processes only system-generated data\n   * No user input is involved - all data comes from filesystem, git, and Node.js process\n   *\n   * Uses Promise.allSettled to collect partial results even if some sources fail,\n   * providing maximum information availability with graceful degradation.\n   */\n  public async getBuildInfo(): Promise<BuildInfo> {\n    // Use Promise.allSettled to collect all available info, even if some sources fail\n    const results = await Promise.allSettled([\n      this.getGitInfo(),\n      this.getDockerInfo()\n    ]);\n\n    // Package info comes from build-time generated constants\n    const packageInfo = { name: PACKAGE_NAME, version: PACKAGE_VERSION };\n\n    const gitInfo = results[0].status === 'fulfilled'\n      ? results[0].value\n      : { commit: undefined, branch: undefined };\n\n    const dockerInfo = results[1].status === 'fulfilled'\n      ? results[1].value\n      : { isDocker: false, info: undefined };\n\n    // Log any failures for diagnostics\n    const failures: string[] = [];\n    if (results[0].status === 'rejected') {\n      failures.push(`git info: ${results[0].reason}`);\n    }\n    if (results[1].status === 'rejected') {\n      failures.push(`docker info: ${results[1].reason}`);\n    }\n\n    if (failures.length > 0) {\n      logger.debug(`Build info collection had ${failures.length} failure(s): ${failures.join('; ')}`);\n    }\n\n    // Build timestamp comes from build-time generated constants\n    const buildTimestamp = BUILD_TIMESTAMP;\n\n    // Issue #706: Startup timing and readiness\n    const deferredComplete = this.deferredSetupChecker ? this.deferredSetupChecker() : true;\n    const startupInfo: BuildInfo['startup'] = {\n      status: deferredComplete ? 'ready' : 'initializing',\n      deferredSetupComplete: deferredComplete,\n      uptimeMs: Date.now() - this.startTime.getTime(),\n      startupTimingMs: this.startupTimer?.getReport(),\n    };\n    const sessionIdentity = resolveSessionIdentity();\n\n    return {\n      sessionId: sessionIdentity.sessionId,\n      runtimeSessionId: sessionIdentity.runtimeSessionId,\n      sessionSource: sessionIdentity.source,\n      package: packageInfo,\n      build: {\n        timestamp: buildTimestamp,\n        type: BUILD_TYPE,\n        gitCommit: gitInfo.commit,\n        gitBranch: gitInfo.branch,\n        collectionFix: 'v1.6.9-beta1-collection-fix'  // Version identifier for verification\n      },\n      runtime: {\n        nodeVersion: process.version,\n        platform: process.platform,\n        arch: process.arch,\n        uptime: process.uptime(),\n        memoryUsage: process.memoryUsage()\n      },\n      environment: {\n        nodeEnv: process.env.NODE_ENV,\n        isProduction: process.env.NODE_ENV === 'production',\n        isDevelopment: process.env.NODE_ENV === 'development',\n        isDebug: process.env.DEBUG === 'true' || process.env.DEBUG === '1',\n        isDocker: dockerInfo.isDocker,\n        dockerInfo: dockerInfo.info\n      },\n      server: {\n        startTime: this.startTime,\n        uptime: Date.now() - this.startTime.getTime(),\n        mcpConnection: true // We're connected if this method is being called via MCP\n      },\n      startup: startupInfo,\n    };\n  }\n\n  /**\n   * Format build info as user-friendly markdown\n   * SECURITY NOTE: Only formats system-generated data - no user input processing\n   * All input data comes from getBuildInfo() which only reads system information\n   */\n  public formatBuildInfo(info: BuildInfo): string {\n    const lines: string[] = [];\n    \n    lines.push('# 🔧 Build Information\\n');\n    \n    // Package info\n    lines.push('## 📦 Package');\n    lines.push(`- **Name**: ${info.package.name}`);\n    lines.push(`- **Version**: ${info.package.version}`);\n    lines.push('');\n\n    const identitySourceLabel = info.sessionSource === 'env'\n      ? 'Explicit environment'\n      : 'Derived from workspace context';\n    const sessionLines = [\n      '## 🪪 Session',\n      `- **Session ID**: ${info.sessionId}`,\n      `- **Identity Source**: ${identitySourceLabel}`,\n    ];\n    if (info.runtimeSessionId !== info.sessionId) {\n      sessionLines.splice(2, 0, `- **Runtime Session ID**: ${info.runtimeSessionId}`);\n    }\n    lines.push(...sessionLines, '');\n    \n    // Build info\n    lines.push('## 🏗️ Build');\n    lines.push(`- **Type**: ${info.build.type}`);\n    if (info.build.timestamp) {\n      lines.push(`- **Timestamp**: ${info.build.timestamp}`);\n    }\n    if (info.build.gitCommit) {\n      lines.push(`- **Git Commit**: \\`${info.build.gitCommit}\\``);\n    }\n    if (info.build.gitBranch) {\n      lines.push(`- **Git Branch**: ${info.build.gitBranch}`);\n    }\n    lines.push('');\n    \n    // Runtime info\n    lines.push('## 💻 Runtime');\n    lines.push(`- **Node.js**: ${info.runtime.nodeVersion}`);\n    lines.push(`- **Platform**: ${info.runtime.platform}`);\n    lines.push(`- **Architecture**: ${info.runtime.arch}`);\n    lines.push(`- **Process Uptime**: ${this.formatUptime(info.runtime.uptime)}`);\n    lines.push(`- **Memory Usage**: ${this.formatMemory(info.runtime.memoryUsage.heapUsed)} / ${this.formatMemory(info.runtime.memoryUsage.heapTotal)}`);\n    lines.push('');\n    \n    // Environment info\n    lines.push('## ⚙️ Environment');\n    lines.push(`- **NODE_ENV**: ${info.environment.nodeEnv || 'not set'}`);\n    lines.push(`- **Mode**: ${info.environment.isProduction ? 'Production' : info.environment.isDevelopment ? 'Development' : 'Unknown'}`);\n    lines.push(`- **Debug**: ${info.environment.isDebug ? 'Enabled' : 'Disabled'}`);\n    lines.push(`- **Docker**: ${info.environment.isDocker ? 'Yes' : 'No'}`);\n    if (info.environment.dockerInfo) {\n      lines.push(`- **Container**: ${info.environment.dockerInfo}`);\n    }\n    lines.push('');\n    \n    // Server info\n    lines.push('## 🚀 Server');\n    lines.push(`- **Started**: ${info.server.startTime.toISOString()}`);\n    lines.push(`- **Uptime**: ${this.formatUptime(info.server.uptime / 1000)}`);\n    lines.push(`- **MCP Connection**: ${info.server.mcpConnection ? '✅ Connected' : '❌ Disconnected'}`);\n\n    // Issue #706: Startup timing\n    if (info.startup) {\n      lines.push('');\n      lines.push('## ⏱️ Startup');\n      lines.push(`- **Status**: ${info.startup.status === 'ready' ? '✅ Ready' : '⏳ Initializing'}`);\n      lines.push(`- **Deferred Setup**: ${info.startup.deferredSetupComplete ? 'Complete' : 'In Progress'}`);\n      if (info.startup.startupTimingMs) {\n        const timing = info.startup.startupTimingMs;\n        lines.push(`- **Critical Path**: ${timing.criticalPathMs}ms`);\n        lines.push(`- **Deferred Work**: ${timing.deferredMs}ms`);\n        lines.push(`- **Total Startup**: ${timing.totalMs}ms`);\n        if (timing.connectAtMs !== null) {\n          lines.push(`- **Time to Connect**: ${timing.connectAtMs}ms`);\n        }\n        if (timing.phases.length > 0) {\n          lines.push('- **Phases**:');\n          for (const phase of timing.phases) {\n            const tag = phase.critical ? 'critical' : 'deferred';\n            lines.push(`  - ${phase.name}: ${phase.durationMs}ms (${tag})`);\n          }\n        }\n      }\n    }\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * SECURITY NOTE: No Unicode normalization needed - executes system git commands\n   * Data source: Git CLI output (system-controlled), no user input\n   */\n  private async getGitInfo(): Promise<{ commit?: string; branch?: string }> {\n    try {\n      // SECURITY NOTE: Git commands return system-controlled data - not user input\n      // Git commit hashes and branch names are controlled by git, no Unicode normalization needed\n      const commit = child_process.execSync('git rev-parse --short HEAD', { encoding: 'utf-8' }).trim();\n      const branch = child_process.execSync('git rev-parse --abbrev-ref HEAD', { encoding: 'utf-8' }).trim();\n      \n      return { commit, branch };\n    } catch {\n      // Not in a git repository or git not available\n      return {};\n    }\n  }\n\n  /**\n   * SECURITY NOTE: No Unicode normalization needed - reads container runtime files\n   * Data source: System cgroup files (container-controlled), no user input\n   */\n  private async getDockerInfo(): Promise<{ isDocker: boolean; info?: string }> {\n    try {\n      // SECURITY NOTE: Reading system cgroup file - controlled by container runtime, not user input\n      // Container runtime generates this file content, no Unicode normalization needed\n      const cgroupContent = await this.fileOperations.readFile('/proc/1/cgroup', {\n        source: 'BuildInfoService.getDockerInfo'\n      });\n      const isDocker = cgroupContent.includes('docker') || cgroupContent.includes('containerd');\n      \n      if (isDocker) {\n        // Try to get container ID\n        const containerId = cgroupContent\n          .split('\\n')\n          .find(line => line.includes('docker'))\n          ?.split('/')\n          .pop()\n          ?.substring(0, 12);\n        \n        return {\n          isDocker: true,\n          info: containerId ? `Container ID: ${containerId}` : 'Running in Docker'\n        };\n      }\n      \n      return { isDocker: false };\n    } catch {\n      // Not in Docker or /proc not available\n      return { isDocker: false };\n    }\n  }\n\n  private formatUptime(seconds: number): string {\n    const days = Math.floor(seconds / 86400);\n    const hours = Math.floor((seconds % 86400) / 3600);\n    const minutes = Math.floor((seconds % 3600) / 60);\n    const secs = Math.floor(seconds % 60);\n    \n    const parts: string[] = [];\n    if (days > 0) parts.push(`${days}d`);\n    if (hours > 0) parts.push(`${hours}h`);\n    if (minutes > 0) parts.push(`${minutes}m`);\n    if (secs > 0 || parts.length === 0) parts.push(`${secs}s`);\n    \n    return parts.join(' ');\n  }\n\n  private formatMemory(bytes: number): string {\n    const mb = bytes / 1024 / 1024;\n    return `${mb.toFixed(1)} MB`;\n  }\n}\n"]}
|
|
294
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BuildInfoService.js","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BuildInfoService - Provides build and runtime information\n * Separated from main index.ts to avoid making that file larger\n * \n * SECURITY FIX (PR #614):\n * 1. DMCP-SEC-004: FALSE POSITIVE SUPPRESSION - No user input Unicode normalization needed\n *    This service only processes system information (git, package.json, environment variables)\n *    The MCP tool 'get_build_info' takes NO parameters (empty inputSchema)\n *    No user-provided data flows through this service that requires Unicode normalization\n */\n\nimport * as child_process from 'child_process';\nimport { logger } from '../utils/logger.js';\nimport { IFileOperationsService } from './FileOperationsService.js';\nimport { PACKAGE_NAME, PACKAGE_VERSION, BUILD_TIMESTAMP, BUILD_TYPE } from '../generated/version.js';\nimport type { StartupTimer, StartupReport } from '../telemetry/StartupTimer.js';\nimport { resolveSessionIdentity } from './sessionIdentity.js';\nimport { getPermissionHookAuditSummary } from '../utils/permissionHooks.js';\n\nexport interface BuildInfo {\n  sessionId: string;\n  runtimeSessionId: string;\n  sessionSource: 'env' | 'derived';\n  package: {\n    name: string;\n    version: string;\n  };\n  build: {\n    timestamp?: string;\n    type: 'git' | 'npm' | 'unknown';\n    gitCommit?: string;\n    gitBranch?: string;\n  };\n  runtime: {\n    nodeVersion: string;\n    platform: string;\n    arch: string;\n    uptime: number;\n    memoryUsage: NodeJS.MemoryUsage;\n  };\n  environment: {\n    nodeEnv?: string;\n    isProduction: boolean;\n    isDevelopment: boolean;\n    isDebug: boolean;\n    isDocker: boolean;\n    dockerInfo?: string;\n  };\n  server: {\n    startTime: Date;\n    uptime: number;\n    mcpConnection: boolean;\n  };\n  permissionHooks?: {\n    installedHosts: string[];\n    currentHosts: string[];\n    repairedHosts: string[];\n    needsRepairHosts: string[];\n  };\n  /** Issue #706: Startup timing and readiness status. */\n  startup?: {\n    status: 'ready' | 'initializing';\n    deferredSetupComplete: boolean;\n    uptimeMs: number;\n    startupTimingMs?: StartupReport;\n  };\n}\n\nexport class BuildInfoService {\n  private readonly startTime: Date;\n  private readonly fileOperations: IFileOperationsService;\n  /** Issue #706: Optional startup timer for timing instrumentation. */\n  private startupTimer: StartupTimer | null = null;\n  /** Issue #706: Callback to check deferred setup status. */\n  private deferredSetupChecker: (() => boolean) | null = null;\n\n  constructor(fileOperations: IFileOperationsService) {\n    this.startTime = new Date();\n    this.fileOperations = fileOperations;\n  }\n\n  /**\n   * Issue #706: Wire in startup instrumentation after DI is ready.\n   * Called from Container to avoid circular dependency at registration time.\n   */\n  setStartupTimer(timer: StartupTimer): void {\n    this.startupTimer = timer;\n  }\n\n  /**\n   * Issue #706: Wire in deferred setup status checker.\n   */\n  setDeferredSetupChecker(checker: () => boolean): void {\n    this.deferredSetupChecker = checker;\n  }\n\n  /**\n   * Get comprehensive build information\n   * SECURITY NOTE: This method processes only system-generated data\n   * No user input is involved - all data comes from filesystem, git, and Node.js process\n   *\n   * Uses Promise.allSettled to collect partial results even if some sources fail,\n   * providing maximum information availability with graceful degradation.\n   */\n  public async getBuildInfo(): Promise<BuildInfo> {\n    // Use Promise.allSettled to collect all available info, even if some sources fail\n    const results = await Promise.allSettled([\n      this.getGitInfo(),\n      this.getDockerInfo(),\n      getPermissionHookAuditSummary(),\n    ]);\n\n    // Package info comes from build-time generated constants\n    const packageInfo = { name: PACKAGE_NAME, version: PACKAGE_VERSION };\n\n    const gitInfo = results[0].status === 'fulfilled'\n      ? results[0].value\n      : { commit: undefined, branch: undefined };\n\n    const dockerInfo = results[1].status === 'fulfilled'\n      ? results[1].value\n      : { isDocker: false, info: undefined };\n    const permissionHookInfo = results[2].status === 'fulfilled'\n      ? results[2].value\n      : { installedHosts: [], currentHosts: [], repairedHosts: [], needsRepairHosts: [] };\n\n    // Log any failures for diagnostics\n    const failures: string[] = [];\n    if (results[0].status === 'rejected') {\n      failures.push(`git info: ${results[0].reason}`);\n    }\n    if (results[1].status === 'rejected') {\n      failures.push(`docker info: ${results[1].reason}`);\n    }\n    if (results[2].status === 'rejected') {\n      failures.push(`permission hook audit: ${results[2].reason}`);\n    }\n\n    if (failures.length > 0) {\n      logger.debug(`Build info collection had ${failures.length} failure(s): ${failures.join('; ')}`);\n    }\n\n    // Build timestamp comes from build-time generated constants\n    const buildTimestamp = BUILD_TIMESTAMP;\n\n    // Issue #706: Startup timing and readiness\n    const deferredComplete = this.deferredSetupChecker ? this.deferredSetupChecker() : true;\n    const startupInfo: BuildInfo['startup'] = {\n      status: deferredComplete ? 'ready' : 'initializing',\n      deferredSetupComplete: deferredComplete,\n      uptimeMs: Date.now() - this.startTime.getTime(),\n      startupTimingMs: this.startupTimer?.getReport(),\n    };\n    const sessionIdentity = resolveSessionIdentity();\n\n    return {\n      sessionId: sessionIdentity.sessionId,\n      runtimeSessionId: sessionIdentity.runtimeSessionId,\n      sessionSource: sessionIdentity.source,\n      package: packageInfo,\n      build: {\n        timestamp: buildTimestamp,\n        type: BUILD_TYPE,\n        gitCommit: gitInfo.commit,\n        gitBranch: gitInfo.branch,\n      },\n      runtime: {\n        nodeVersion: process.version,\n        platform: process.platform,\n        arch: process.arch,\n        uptime: process.uptime(),\n        memoryUsage: process.memoryUsage()\n      },\n      environment: {\n        nodeEnv: process.env.NODE_ENV,\n        isProduction: process.env.NODE_ENV === 'production',\n        isDevelopment: process.env.NODE_ENV === 'development',\n        isDebug: process.env.DEBUG === 'true' || process.env.DEBUG === '1',\n        isDocker: dockerInfo.isDocker,\n        dockerInfo: dockerInfo.info\n      },\n      server: {\n        startTime: this.startTime,\n        uptime: Date.now() - this.startTime.getTime(),\n        mcpConnection: true // We're connected if this method is being called via MCP\n      },\n      permissionHooks: permissionHookInfo,\n      startup: startupInfo,\n    };\n  }\n\n  /**\n   * Format build info as user-friendly markdown\n   * SECURITY NOTE: Only formats system-generated data - no user input processing\n   * All input data comes from getBuildInfo() which only reads system information\n   */\n  public formatBuildInfo(info: BuildInfo): string {\n    const lines: string[] = [];\n    \n    lines.push('# 🔧 Build Information\\n');\n    \n    // Package info\n    lines.push('## 📦 Package');\n    lines.push(`- **Name**: ${info.package.name}`);\n    lines.push(`- **Version**: ${info.package.version}`);\n    lines.push('');\n\n    const identitySourceLabel = info.sessionSource === 'env'\n      ? 'Explicit environment'\n      : 'Derived from workspace context';\n    const sessionLines = [\n      '## 🪪 Session',\n      `- **Session ID**: ${info.sessionId}`,\n      `- **Identity Source**: ${identitySourceLabel}`,\n    ];\n    if (info.runtimeSessionId !== info.sessionId) {\n      sessionLines.splice(2, 0, `- **Runtime Session ID**: ${info.runtimeSessionId}`);\n    }\n    lines.push(...sessionLines, '');\n    \n    // Build info\n    lines.push('## 🏗️ Build');\n    lines.push(`- **Type**: ${info.build.type}`);\n    if (info.build.timestamp) {\n      lines.push(`- **Timestamp**: ${info.build.timestamp}`);\n    }\n    if (info.build.gitCommit) {\n      lines.push(`- **Git Commit**: \\`${info.build.gitCommit}\\``);\n    }\n    if (info.build.gitBranch) {\n      lines.push(`- **Git Branch**: ${info.build.gitBranch}`);\n    }\n    lines.push('');\n    \n    // Runtime info\n    lines.push('## 💻 Runtime');\n    lines.push(`- **Node.js**: ${info.runtime.nodeVersion}`);\n    lines.push(`- **Platform**: ${info.runtime.platform}`);\n    lines.push(`- **Architecture**: ${info.runtime.arch}`);\n    lines.push(`- **Process Uptime**: ${this.formatUptime(info.runtime.uptime)}`);\n    lines.push(`- **Memory Usage**: ${this.formatMemory(info.runtime.memoryUsage.heapUsed)} / ${this.formatMemory(info.runtime.memoryUsage.heapTotal)}`);\n    lines.push('');\n    \n    // Environment info\n    lines.push('## ⚙️ Environment');\n    lines.push(`- **NODE_ENV**: ${info.environment.nodeEnv || 'not set'}`);\n    lines.push(`- **Mode**: ${info.environment.isProduction ? 'Production' : info.environment.isDevelopment ? 'Development' : 'Unknown'}`);\n    lines.push(`- **Debug**: ${info.environment.isDebug ? 'Enabled' : 'Disabled'}`);\n    lines.push(`- **Docker**: ${info.environment.isDocker ? 'Yes' : 'No'}`);\n    if (info.environment.dockerInfo) {\n      lines.push(`- **Container**: ${info.environment.dockerInfo}`);\n    }\n    lines.push('');\n    \n    // Server info\n    lines.push('## 🚀 Server');\n    lines.push(`- **Started**: ${info.server.startTime.toISOString()}`);\n    lines.push(`- **Uptime**: ${this.formatUptime(info.server.uptime / 1000)}`);\n    lines.push(`- **MCP Connection**: ${info.server.mcpConnection ? '✅ Connected' : '❌ Disconnected'}`);\n\n    if (info.permissionHooks) {\n      const installedHosts = info.permissionHooks.installedHosts.length > 0\n        ? info.permissionHooks.installedHosts.join(', ')\n        : 'None';\n      const currentHosts = info.permissionHooks.currentHosts.length > 0\n        ? info.permissionHooks.currentHosts.join(', ')\n        : 'None';\n      const needsRepairHosts = info.permissionHooks.needsRepairHosts.length > 0\n        ? info.permissionHooks.needsRepairHosts.join(', ')\n        : 'None';\n\n      lines.push(\n        '',\n        '## 🔐 Permission Hooks',\n        `- **Installed Hosts**: ${installedHosts}`,\n        `- **Current Assets**: ${currentHosts}`,\n        `- **Needs Repair**: ${needsRepairHosts}`,\n      );\n    }\n\n    // Issue #706: Startup timing\n    if (info.startup) {\n      lines.push('');\n      lines.push('## ⏱️ Startup');\n      lines.push(`- **Status**: ${info.startup.status === 'ready' ? '✅ Ready' : '⏳ Initializing'}`);\n      lines.push(`- **Deferred Setup**: ${info.startup.deferredSetupComplete ? 'Complete' : 'In Progress'}`);\n      if (info.startup.startupTimingMs) {\n        const timing = info.startup.startupTimingMs;\n        lines.push(`- **Critical Path**: ${timing.criticalPathMs}ms`);\n        lines.push(`- **Deferred Work**: ${timing.deferredMs}ms`);\n        lines.push(`- **Total Startup**: ${timing.totalMs}ms`);\n        if (timing.connectAtMs !== null) {\n          lines.push(`- **Time to Connect**: ${timing.connectAtMs}ms`);\n        }\n        if (timing.phases.length > 0) {\n          lines.push('- **Phases**:');\n          for (const phase of timing.phases) {\n            const tag = phase.critical ? 'critical' : 'deferred';\n            lines.push(`  - ${phase.name}: ${phase.durationMs}ms (${tag})`);\n          }\n        }\n      }\n    }\n\n    return lines.join('\\n');\n  }\n\n  /**\n   * SECURITY NOTE: No Unicode normalization needed - executes system git commands\n   * Data source: Git CLI output (system-controlled), no user input\n   */\n  private async getGitInfo(): Promise<{ commit?: string; branch?: string }> {\n    try {\n      // SECURITY NOTE: Git commands return system-controlled data - not user input\n      // Git commit hashes and branch names are controlled by git, no Unicode normalization needed\n      const commit = child_process.execSync('git rev-parse --short HEAD', { encoding: 'utf-8' }).trim();\n      const branch = child_process.execSync('git rev-parse --abbrev-ref HEAD', { encoding: 'utf-8' }).trim();\n      \n      return { commit, branch };\n    } catch {\n      // Not in a git repository or git not available\n      return {};\n    }\n  }\n\n  /**\n   * SECURITY NOTE: No Unicode normalization needed - reads container runtime files\n   * Data source: System cgroup files (container-controlled), no user input\n   */\n  private async getDockerInfo(): Promise<{ isDocker: boolean; info?: string }> {\n    try {\n      // SECURITY NOTE: Reading system cgroup file - controlled by container runtime, not user input\n      // Container runtime generates this file content, no Unicode normalization needed\n      const cgroupContent = await this.fileOperations.readFile('/proc/1/cgroup', {\n        source: 'BuildInfoService.getDockerInfo'\n      });\n      const isDocker = cgroupContent.includes('docker') || cgroupContent.includes('containerd');\n      \n      if (isDocker) {\n        // Try to get container ID\n        const containerId = cgroupContent\n          .split('\\n')\n          .find(line => line.includes('docker'))\n          ?.split('/')\n          .pop()\n          ?.substring(0, 12);\n        \n        return {\n          isDocker: true,\n          info: containerId ? `Container ID: ${containerId}` : 'Running in Docker'\n        };\n      }\n      \n      return { isDocker: false };\n    } catch {\n      // Not in Docker or /proc not available\n      return { isDocker: false };\n    }\n  }\n\n  private formatUptime(seconds: number): string {\n    const days = Math.floor(seconds / 86400);\n    const hours = Math.floor((seconds % 86400) / 3600);\n    const minutes = Math.floor((seconds % 3600) / 60);\n    const secs = Math.floor(seconds % 60);\n    \n    const parts: string[] = [];\n    if (days > 0) parts.push(`${days}d`);\n    if (hours > 0) parts.push(`${hours}h`);\n    if (minutes > 0) parts.push(`${minutes}m`);\n    if (secs > 0 || parts.length === 0) parts.push(`${secs}s`);\n    \n    return parts.join(' ');\n  }\n\n  private formatMemory(bytes: number): string {\n    const mb = bytes / 1024 / 1024;\n    return `${mb.toFixed(1)} MB`;\n  }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"submitToPortfolioTool.d.ts","sourceRoot":"","sources":["../../../src/tools/portfolio/submitToPortfolioTool.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,yCAAyC,CAAC;AAC/E,OAAO,EAAE,YAAY,EAAE,MAAM,gCAAgC,CAAC;AAE9D,OAAO,EAAE,gBAAgB,EAA2B,MAAM,qCAAqC,CAAC;AAChG,OAAO,EAAE,qBAAqB,EAAE,MAAM,0CAA0C,CAAC;AACjF,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAIvD,OAAO,EAAE,QAAQ,EAAE,MAAM,yBAAyB,CAAC;AAYnD,OAAO,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,uBAAuB,EAAE,MAAM,YAAY,CAAC;AAGhG,YAAY,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,uBAAuB,EAAE,CAAC;AACnF,OAAO,EAAE,YAAY,EAAE,MAAM,kCAAkC,CAAC;AAKhE,OAAO,EAAE,sBAAsB,EAAE,MAAM,yCAAyC,CAAC;
|
|
1
|
+
{"version":3,"file":"submitToPortfolioTool.d.ts","sourceRoot":"","sources":["../../../src/tools/portfolio/submitToPortfolioTool.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,EAAE,iBAAiB,EAAE,MAAM,iCAAiC,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,yCAAyC,CAAC;AAC/E,OAAO,EAAE,YAAY,EAAE,MAAM,gCAAgC,CAAC;AAE9D,OAAO,EAAE,gBAAgB,EAA2B,MAAM,qCAAqC,CAAC;AAChG,OAAO,EAAE,qBAAqB,EAAE,MAAM,0CAA0C,CAAC;AACjF,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAIvD,OAAO,EAAE,QAAQ,EAAE,MAAM,yBAAyB,CAAC;AAYnD,OAAO,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,uBAAuB,EAAE,MAAM,YAAY,CAAC;AAGhG,YAAY,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,uBAAuB,EAAE,CAAC;AACnF,OAAO,EAAE,YAAY,EAAE,MAAM,kCAAkC,CAAC;AAKhE,OAAO,EAAE,sBAAsB,EAAE,MAAM,yCAAyC,CAAC;AAMjF,MAAM,WAAW,qBAAqB;IACpC,IAAI,EAAE,WAAW,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,sBAAsB;IACrC,KAAK,EAAE,OAAO,CAAC;IACf,OAAO,EAAE,qBAAqB,EAAE,CAAC;CAClC;AAED,MAAM,WAAW,iCAAiC;IAChD,WAAW,EAAE,iBAAiB,CAAC;IAC/B,oBAAoB,EAAE,oBAAoB,CAAC;IAC3C,gBAAgB,EAAE,gBAAgB,CAAC;IACnC,qBAAqB,EAAE,qBAAqB,CAAC;IAC7C,WAAW,EAAE,YAAY,CAAC;IAC1B,cAAc,EAAE,sBAAsB,CAAC;IACvC,YAAY,EAAE,YAAY,CAAC;CAC5B;AAED,qBAAa,qBAAqB;IAChC,OAAO,CAAC,WAAW,CAAoB;IACvC,OAAO,CAAC,oBAAoB,CAAuB;IACnD,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,qBAAqB,CAAwB;IACrD,OAAO,CAAC,WAAW,CAAe;IAClC,OAAO,CAAC,cAAc,CAAyB;IAC/C,OAAO,CAAC,YAAY,CAAe;gBAEvB,QAAQ,EAAE,QAAQ,EAAE,YAAY,EAAE,iCAAiC;IA0B/E;;;;OAIG;YACW,0BAA0B;IA8BxC;;;OAGG;YACW,mBAAmB;IA2BjC;;;;;;OAMG;YACW,gCAAgC;IAsH9C;;;;OAIG;YACW,sBAAsB;IA+DpC;;;;;;OAMG;YACW,sBAAsB;IA2EpC;;;OAGG;IACH,OAAO,CAAC,oBAAoB;IAoC5B;;;;;;OAMG;YACW,sBAAsB;IAiEpC;;;;;OAKG;YACW,wBAAwB;IAsItC;;;;;OAKG;YACW,qBAAqB;IA6NnC;;;;;;;;;OASG;YACW,2BAA2B;IA6GzC;;;OAGG;IACH,OAAO,CAAC,0BAA0B;IAwBlC;;;;OAIG;YACW,qBAAqB;IA+CnC;;;;;;;OAOG;YACW,oBAAoB;IA8DlC;;;;;;OAMG;YACW,kBAAkB;IAkDhC;;;;;;;;OAQG;YACW,8BAA8B;IAyJtC,OAAO,CAAC,MAAM,EAAE,uBAAuB,GAAG,OAAO,CAAC,uBAAuB,CAAC;IAoJhF;;;OAGG;YACW,6BAA6B;IAuE3C;;;OAGG;YACW,qBAAqB;YAwRrB,gBAAgB;IA6I9B;;;;;;;OAOG;YACW,iBAAiB;IAoH/B;;;;OAIG;YACW,uBAAuB;IA6HrC;;;OAGG;IACH,OAAO,CAAC,mBAAmB;IAwB3B;;;OAGG;YACW,oBAAoB;IAyElC;;;;OAIG;IACH,OAAO,CAAC,gBAAgB;CAmDzB"}
|