@dollhousemcp/mcp-server 2.0.27 → 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 +13 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/mcp-aql/OperationSchema.js +2 -2
- package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/evaluatePermission.js +9 -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/console/IngestRoutes.d.ts +7 -1
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +28 -6
- package/dist/web/console/LeaderForwardingSink.d.ts +6 -1
- package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
- package/dist/web/console/LeaderForwardingSink.js +8 -2
- package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
- package/dist/web/console/UnifiedConsole.js +6 -3
- package/dist/web/console/sessionClientPlatform.d.ts +11 -0
- package/dist/web/console/sessionClientPlatform.d.ts.map +1 -0
- package/dist/web/console/sessionClientPlatform.js +83 -0
- package/dist/web/public/permissions.js +10 -0
- package/dist/web/public/sessions.css +89 -9
- package/dist/web/public/sessions.js +160 -4
- package/dist/web/public/setup.js +40 -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/scripts/pretooluse-dollhouse.sh +36 -2
- package/scripts/pretooluse-vscode.sh +5 -2
- package/scripts/pretooluse-windsurf.sh +5 -2
- package/server.json +2 -2
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACrE;
|
|
1
|
+
{"version":3,"file":"evaluatePermission.d.ts","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,4BAA4B,CAAC;AAC9D,OAAO,KAAK,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AACtG,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAEnE,wEAAwE;AACxE,qBAAa,yBAA0B,SAAQ,KAAK;IAClD,SAAgB,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,CAAC;IACpF,SAAgB,QAAQ,EAAE,MAAM,CAAC;gBAG/B,OAAO,EAAE,MAAM,EACf,KAAK,EAAE,YAAY,GAAG,gBAAgB,GAAG,QAAQ,GAAG,eAAe,EACnE,QAAQ,EAAE,MAAM,EAChB,KAAK,CAAC,EAAE,OAAO;CAOlB;AAED,+CAA+C;AAC/C,MAAM,WAAW,sBAAsB;IACrC,uBAAuB,EAAE,WAAW,CAAC;IACrC,YAAY,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,wBAAwB,CAAC;IACjG,qBAAqB,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,KAAK,mBAAmB,CAAC;IAChI,iBAAiB,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC;CACrE;AAqDD,iCAAiC;AACjC,eAAO,MAAM,mBAAmB,UAAkC,CAAC;AAiBnE;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,QAAQ,EAAE,OAAO,GAAG,MAAM,GAAG,KAAK,EAClC,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,MAAM,CAAC,EAAE,MAAM,GACd,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAYzB;AAED;;;;;;GAMG;AACH,wBAAsB,kBAAkB,CACtC,MAAM,EAAE;IAAE,SAAS,CAAC,EAAE,OAAO,CAAC;IAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAAC,QAAQ,CAAC,EAAE,OAAO,CAAC;IAAC,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,EAC1F,IAAI,EAAE,sBAAsB,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAiDlC"}
|
|
@@ -40,9 +40,15 @@ function formatCursor(decision, reason) {
|
|
|
40
40
|
function formatWindsurf(decision, reason) {
|
|
41
41
|
return withReason({ allowed: decision === 'allow' }, reason);
|
|
42
42
|
}
|
|
43
|
-
/** Codex:
|
|
43
|
+
/** Codex: explicit PreToolUse payload, maps 'ask' to 'deny' */
|
|
44
44
|
function formatCodex(decision, reason) {
|
|
45
|
-
return {
|
|
45
|
+
return {
|
|
46
|
+
hookSpecificOutput: {
|
|
47
|
+
hookEventName: 'PreToolUse',
|
|
48
|
+
permissionDecision: decision === 'ask' ? 'deny' : decision,
|
|
49
|
+
permissionDecisionReason: reason ?? '',
|
|
50
|
+
},
|
|
51
|
+
};
|
|
46
52
|
}
|
|
47
53
|
/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */
|
|
48
54
|
function formatClaudeCode(decision, reason) {
|
|
@@ -141,4 +147,4 @@ export async function evaluatePermission(params, deps) {
|
|
|
141
147
|
// Default: allow
|
|
142
148
|
return formatPermissionResponse('allow', platform, input);
|
|
143
149
|
}
|
|
144
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex: wraps in hookSpecificOutput, maps 'ask' to 'deny' */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  return { hookSpecificOutput: withReason({ permissionDecision: decision === 'ask' ? 'deny' : decision }, reason) };\n}\n\n/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: withReason({\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision,\n    }, reason, 'permissionDecisionReason'),\n  };\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  vscode: formatClaudeCode,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\nfunction warnOnUnknownPlatform(platform: string): void {\n  void import('../../utils/logger.js')\n    .then(({ logger }) => {\n      logger.warn(\n        `[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`,\n      );\n    })\n    .catch((error) => {\n      console.warn(\n        `[evaluatePermission] Failed to load logger while handling unknown platform \"${platform}\".`,\n        error,\n      );\n    });\n}\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      warnOnUnknownPlatform(platform);\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown; session_id?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n  const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''\n    ? params.session_id\n    : undefined;\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements(sessionId);\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"evaluatePermission.js","sourceRoot":"","sources":["../../../src/handlers/mcp-aql/evaluatePermission.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation for PreToolUse hooks across all AI platforms.\n *\n * Provides the `evaluate_permission` MCP-AQL READ operation, enabling\n * Claude Code, Cursor, Gemini CLI, Windsurf, and Codex to use\n * DollhouseMCP as their permission evaluation backend via hooks.\n *\n * Three-stage evaluation pipeline:\n * 1. Rate limiting — prevents abuse\n * 2. Static tool classification — built-in allow/deny rules\n * 3. Element policy evaluation — active element gatekeeper policies\n *\n * Returns platform-specific response formats so each platform's hook\n * script receives the JSON shape it expects.\n */\n\nimport type { RateLimiter } from '../../utils/RateLimiter.js';\nimport type { ToolClassificationResult, CliToolPolicyResult } from './policies/ToolClassification.js';\nimport type { ActiveElement } from './policies/ElementPolicies.js';\n\n/** Error thrown when permission evaluation fails at a specific stage */\nexport class PermissionEvaluationError extends Error {\n  public readonly stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch';\n  public readonly toolName: string;\n\n  constructor(\n    message: string,\n    stage: 'rate_limit' | 'classification' | 'policy' | 'element_fetch',\n    toolName: string,\n    cause?: unknown,\n  ) {\n    super(message, cause ? { cause } : undefined);\n    this.name = 'PermissionEvaluationError';\n    this.stage = stage;\n    this.toolName = toolName;\n  }\n}\n\n/** Dependencies injected from MCPAQLHandler */\nexport interface EvaluatePermissionDeps {\n  permissionPromptLimiter: RateLimiter;\n  classifyTool: (toolName: string, toolInput: Record<string, unknown>) => ToolClassificationResult;\n  evaluateCliToolPolicy: (toolName: string, toolInput: Record<string, unknown>, elements: ActiveElement[]) => CliToolPolicyResult;\n  getActiveElements: (sessionId?: string) => Promise<ActiveElement[]>;\n}\n\n/** Optional reason field, only included when reason is provided */\nfunction withReason(obj: Record<string, unknown>, reason?: string, key = 'reason'): Record<string, unknown> {\n  return reason ? { ...obj, [key]: reason } : obj;\n}\n\n/** Gemini: maps 'ask' to 'deny' (no interactive support) */\nfunction formatGemini(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ decision: decision === 'ask' ? 'deny' : decision }, reason);\n}\n\n/** Cursor: uses 'permission' field instead of 'decision' */\nfunction formatCursor(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ permission: decision }, reason);\n}\n\n/** Windsurf: uses boolean 'allowed' field */\nfunction formatWindsurf(decision: string, reason?: string): Record<string, unknown> {\n  return withReason({ allowed: decision === 'allow' }, reason);\n}\n\n/** Codex: explicit PreToolUse payload, maps 'ask' to 'deny' */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: {\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision === 'ask' ? 'deny' : decision,\n      permissionDecisionReason: reason ?? '',\n    },\n  };\n}\n\n/** Claude Code (default): uses hookSpecificOutput.permissionDecision for PreToolUse */\nfunction formatClaudeCode(decision: string, reason?: string): Record<string, unknown> {\n  return {\n    hookSpecificOutput: withReason({\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision,\n    }, reason, 'permissionDecisionReason'),\n  };\n}\n\n/** Platform formatter lookup */\nconst platformFormatters: Record<string, (decision: string, reason?: string) => Record<string, unknown>> = {\n  gemini: formatGemini,\n  cursor: formatCursor,\n  windsurf: formatWindsurf,\n  codex: formatCodex,\n  vscode: formatClaudeCode,\n  claude_code: formatClaudeCode,\n};\n\n/** Known platform identifiers */\nexport const SUPPORTED_PLATFORMS = Object.keys(platformFormatters);\n\nfunction warnOnUnknownPlatform(platform: string): void {\n  void import('../../utils/logger.js')\n    .then(({ logger }) => {\n      logger.warn(\n        `[evaluatePermission] Unknown platform \"${platform}\", defaulting to claude_code format. Supported: ${SUPPORTED_PLATFORMS.join(', ')}`,\n      );\n    })\n    .catch((error) => {\n      console.warn(\n        `[evaluatePermission] Failed to load logger while handling unknown platform \"${platform}\".`,\n        error,\n      );\n    });\n}\n\n/**\n * Format permission evaluation response for platform-specific hook scripts.\n * Each platform expects a different JSON shape from its hook response.\n *\n * Unknown platforms default to claude_code format with a warning log.\n */\nexport function formatPermissionResponse(\n  decision: 'allow' | 'deny' | 'ask',\n  platform: string,\n  _input: Record<string, unknown>,\n  reason?: string,\n): Record<string, unknown> {\n  switch (platform) {\n    case 'gemini': return formatGemini(decision, reason);\n    case 'cursor': return formatCursor(decision, reason);\n    case 'windsurf': return formatWindsurf(decision, reason);\n    case 'codex': return formatCodex(decision, reason);\n    case 'claude_code': return formatClaudeCode(decision, reason);\n    default:\n      // Import lazily to avoid circular dependency at module load time\n      warnOnUnknownPlatform(platform);\n      return formatClaudeCode(decision, reason);\n  }\n}\n\n/**\n * Evaluate a CLI permission request for PreToolUse hooks.\n *\n * @param params - Tool name, input, and target platform\n * @param deps - Injected dependencies from MCPAQLHandler\n * @returns Platform-formatted permission decision\n */\nexport async function evaluatePermission(\n  params: { tool_name?: unknown; input?: unknown; platform?: unknown; session_id?: unknown },\n  deps: EvaluatePermissionDeps,\n): Promise<Record<string, unknown>> {\n  const toolName = typeof params.tool_name === 'string' ? params.tool_name : '';\n  const inputRaw = params.input;\n  const input = (inputRaw && typeof inputRaw === 'object')\n    ? inputRaw as Record<string, unknown>\n    : {};\n  const platform = typeof params.platform === 'string' ? params.platform : 'claude_code';\n  const sessionId = typeof params.session_id === 'string' && params.session_id.trim() !== ''\n    ? params.session_id\n    : undefined;\n\n  // Rate limit\n  const rateStatus = deps.permissionPromptLimiter.checkLimit();\n  if (!rateStatus.allowed) {\n    return formatPermissionResponse('deny', platform, input, 'Rate limit exceeded');\n  }\n  deps.permissionPromptLimiter.consumeToken();\n\n  // Stage 1: Static classification\n  const classification = deps.classifyTool(toolName, input);\n  if (classification.behavior === 'allow') {\n    return formatPermissionResponse('allow', platform, input);\n  }\n  if (classification.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, classification.reason);\n  }\n\n  // Stage 2: Element policy evaluation\n  let elements: ActiveElement[];\n  try {\n    elements = await deps.getActiveElements(sessionId);\n  } catch (err) {\n    throw new PermissionEvaluationError(\n      `Failed to fetch active elements for policy evaluation: ${err instanceof Error ? err.message : String(err)}`,\n      'element_fetch', toolName, err,\n    );\n  }\n  const decision = deps.evaluateCliToolPolicy(toolName, input, elements);\n\n  if (decision.behavior === 'deny') {\n    return formatPermissionResponse('deny', platform, input, decision.message);\n  }\n  if (decision.behavior === 'confirm') {\n    return formatPermissionResponse('ask', platform, input,\n      decision.message || 'Requires confirmation per element policy');\n  }\n\n  // Default: allow\n  return formatPermissionResponse('allow', platform, input);\n}\n"]}
|
|
@@ -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"}
|