@dollhousemcp/mcp-server 2.0.29 → 2.0.31
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 +10 -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 +6 -3
- package/dist/services/BuildInfoService.d.ts +5 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -1
- package/dist/services/BuildInfoService.js +44 -8
- package/dist/utils/permissionHookInstallers.d.ts +27 -0
- package/dist/utils/permissionHookInstallers.d.ts.map +1 -0
- package/dist/utils/permissionHookInstallers.js +465 -0
- package/dist/utils/permissionHookShared.d.ts +165 -0
- package/dist/utils/permissionHookShared.d.ts.map +1 -0
- package/dist/utils/permissionHookShared.js +425 -0
- package/dist/utils/permissionHookStatus.d.ts +10 -0
- package/dist/utils/permissionHookStatus.d.ts.map +1 -0
- package/dist/utils/permissionHookStatus.js +260 -0
- package/dist/utils/permissionHooks.d.ts +3 -91
- package/dist/utils/permissionHooks.d.ts.map +1 -1
- package/dist/utils/permissionHooks.js +4 -947
- package/dist/web/routes/healthRoutes.d.ts +3 -0
- package/dist/web/routes/healthRoutes.d.ts.map +1 -1
- package/dist/web/routes/healthRoutes.js +24 -2
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +21 -2
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +9 -2
- package/package.json +3 -1
- package/scripts/permission-hook-config.sh +67 -0
- package/scripts/pretooluse-dollhouse.sh +185 -38
- package/scripts/pretooluse-vscode.sh +23 -10
- package/scripts/pretooluse-windsurf.sh +6 -6
- 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;AAyDD,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,13 +40,16 @@ function formatCursor(decision, reason) {
|
|
|
40
40
|
function formatWindsurf(decision, reason) {
|
|
41
41
|
return withReason({ allowed: decision === 'allow' }, reason);
|
|
42
42
|
}
|
|
43
|
-
/** Codex
|
|
43
|
+
/** Codex PreToolUse currently supports block/deny only; allow is empty stdout. */
|
|
44
44
|
function formatCodex(decision, reason) {
|
|
45
|
+
if (decision === 'allow') {
|
|
46
|
+
return {};
|
|
47
|
+
}
|
|
45
48
|
return {
|
|
46
49
|
hookSpecificOutput: {
|
|
47
50
|
hookEventName: 'PreToolUse',
|
|
48
51
|
permissionDecision: decision === 'ask' ? 'deny' : decision,
|
|
49
|
-
permissionDecisionReason: reason
|
|
52
|
+
...(reason ? { permissionDecisionReason: reason } : {}),
|
|
50
53
|
},
|
|
51
54
|
};
|
|
52
55
|
}
|
|
@@ -147,4 +150,4 @@ export async function evaluatePermission(params, deps) {
|
|
|
147
150
|
// Default: allow
|
|
148
151
|
return formatPermissionResponse('allow', platform, input);
|
|
149
152
|
}
|
|
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"]}
|
|
153
|
+
//# 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 PreToolUse currently supports block/deny only; allow is empty stdout. */\nfunction formatCodex(decision: string, reason?: string): Record<string, unknown> {\n  if (decision === 'allow') {\n    return {};\n  }\n\n  return {\n    hookSpecificOutput: {\n      hookEventName: 'PreToolUse',\n      permissionDecision: decision === 'ask' ? 'deny' : decision,\n      ...(reason ? { 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"]}
|
|
@@ -10,6 +10,7 @@
|
|
|
10
10
|
*/
|
|
11
11
|
import { IFileOperationsService } from './FileOperationsService.js';
|
|
12
12
|
import type { StartupTimer, StartupReport } from '../telemetry/StartupTimer.js';
|
|
13
|
+
import { type PermissionHookDiagnosticRecord, type PermissionHookHealthSummary, type PermissionHookStartupRepairSummary } from '../utils/permissionHooks.js';
|
|
13
14
|
export interface BuildInfo {
|
|
14
15
|
sessionId: string;
|
|
15
16
|
runtimeSessionId: string;
|
|
@@ -45,10 +46,14 @@ export interface BuildInfo {
|
|
|
45
46
|
mcpConnection: boolean;
|
|
46
47
|
};
|
|
47
48
|
permissionHooks?: {
|
|
49
|
+
health: PermissionHookHealthSummary;
|
|
48
50
|
installedHosts: string[];
|
|
49
51
|
currentHosts: string[];
|
|
50
52
|
repairedHosts: string[];
|
|
51
53
|
needsRepairHosts: string[];
|
|
54
|
+
diagnosticsPath: string;
|
|
55
|
+
lastDiagnostic: PermissionHookDiagnosticRecord | null;
|
|
56
|
+
lastStartupRepair: PermissionHookStartupRepairSummary | null;
|
|
52
57
|
};
|
|
53
58
|
/** Issue #706: Startup timing and readiness status. */
|
|
54
59
|
startup?: {
|
|
@@ -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;AAEhF,OAAO,EAEL,KAAK,8BAA8B,EAEnC,KAAK,2BAA2B,EAChC,KAAK,kCAAkC,EACxC,MAAM,6BAA6B,CAAC;AAErC,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,MAAM,EAAE,2BAA2B,CAAC;QACpC,cAAc,EAAE,MAAM,EAAE,CAAC;QACzB,YAAY,EAAE,MAAM,EAAE,CAAC;QACvB,aAAa,EAAE,MAAM,EAAE,CAAC;QACxB,gBAAgB,EAAE,MAAM,EAAE,CAAC;QAC3B,eAAe,EAAE,MAAM,CAAC;QACxB,cAAc,EAAE,8BAA8B,GAAG,IAAI,CAAC;QACtD,iBAAiB,EAAE,kCAAkC,GAAG,IAAI,CAAC;KAC9D,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;IAuG/C;;;;OAIG;IACI,eAAe,CAAC,IAAI,EAAE,SAAS,GAAG,MAAM;IAoJ/C;;;OAGG;YACW,UAAU;IAcxB;;;OAGG;YACW,aAAa;IA+B3B,OAAO,CAAC,YAAY;IAepB,OAAO,CAAC,YAAY;CAIrB"}
|
|
@@ -12,7 +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
|
+
import { getPermissionHookAuditSummary, summarizePermissionHookHealth, } from '../utils/permissionHooks.js';
|
|
16
16
|
export class BuildInfoService {
|
|
17
17
|
startTime;
|
|
18
18
|
fileOperations;
|
|
@@ -62,17 +62,27 @@ export class BuildInfoService {
|
|
|
62
62
|
: { isDocker: false, info: undefined };
|
|
63
63
|
const permissionHookInfo = results[2].status === 'fulfilled'
|
|
64
64
|
? results[2].value
|
|
65
|
-
: {
|
|
65
|
+
: {
|
|
66
|
+
installedHosts: [],
|
|
67
|
+
currentHosts: [],
|
|
68
|
+
repairedHosts: [],
|
|
69
|
+
needsRepairHosts: [],
|
|
70
|
+
diagnosticsPath: '',
|
|
71
|
+
lastDiagnostic: null,
|
|
72
|
+
lastStartupRepair: null,
|
|
73
|
+
};
|
|
74
|
+
const permissionHookHealth = summarizePermissionHookHealth(permissionHookInfo);
|
|
75
|
+
const formatSettledReason = (reason) => (reason instanceof Error ? `${reason.name}: ${reason.message}` : String(reason));
|
|
66
76
|
// Log any failures for diagnostics
|
|
67
77
|
const failures = [];
|
|
68
78
|
if (results[0].status === 'rejected') {
|
|
69
|
-
failures.push(`git info: ${results[0].reason}`);
|
|
79
|
+
failures.push(`git info: ${formatSettledReason(results[0].reason)}`);
|
|
70
80
|
}
|
|
71
81
|
if (results[1].status === 'rejected') {
|
|
72
|
-
failures.push(`docker info: ${results[1].reason}`);
|
|
82
|
+
failures.push(`docker info: ${formatSettledReason(results[1].reason)}`);
|
|
73
83
|
}
|
|
74
84
|
if (results[2].status === 'rejected') {
|
|
75
|
-
failures.push(`permission hook audit: ${results[2].reason}`);
|
|
85
|
+
failures.push(`permission hook audit: ${formatSettledReason(results[2].reason)}`);
|
|
76
86
|
}
|
|
77
87
|
if (failures.length > 0) {
|
|
78
88
|
logger.debug(`Build info collection had ${failures.length} failure(s): ${failures.join('; ')}`);
|
|
@@ -119,7 +129,10 @@ export class BuildInfoService {
|
|
|
119
129
|
uptime: Date.now() - this.startTime.getTime(),
|
|
120
130
|
mcpConnection: true // We're connected if this method is being called via MCP
|
|
121
131
|
},
|
|
122
|
-
permissionHooks:
|
|
132
|
+
permissionHooks: {
|
|
133
|
+
...permissionHookInfo,
|
|
134
|
+
health: permissionHookHealth,
|
|
135
|
+
},
|
|
123
136
|
startup: startupInfo,
|
|
124
137
|
};
|
|
125
138
|
}
|
|
@@ -194,7 +207,30 @@ export class BuildInfoService {
|
|
|
194
207
|
const needsRepairHosts = info.permissionHooks.needsRepairHosts.length > 0
|
|
195
208
|
? info.permissionHooks.needsRepairHosts.join(', ')
|
|
196
209
|
: 'None';
|
|
197
|
-
|
|
210
|
+
const lastStartupRepair = info.permissionHooks.lastStartupRepair;
|
|
211
|
+
const startupRepairIssues = lastStartupRepair
|
|
212
|
+
? lastStartupRepair.hostResults
|
|
213
|
+
.filter((result) => result.outcome === 'needs_repair' || result.outcome === 'error')
|
|
214
|
+
.map((result) => result.repairError ? `${result.host} (${result.repairError})` : result.host)
|
|
215
|
+
.join('; ')
|
|
216
|
+
: '';
|
|
217
|
+
lines.push('', '## 🔐 Permission Hooks', `- **Health**: ${info.permissionHooks.health.status.toUpperCase()} — ${info.permissionHooks.health.message}`, `- **Installed Hosts**: ${installedHosts}`, `- **Current Assets**: ${currentHosts}`, `- **Needs Repair**: ${needsRepairHosts}`);
|
|
218
|
+
if (info.permissionHooks.diagnosticsPath) {
|
|
219
|
+
lines.push(`- **Diagnostics Log**: ${info.permissionHooks.diagnosticsPath}`);
|
|
220
|
+
}
|
|
221
|
+
if (lastStartupRepair) {
|
|
222
|
+
lines.push(`- **Last Startup Audit**: ${lastStartupRepair.completedAt} (${lastStartupRepair.durationMs}ms)`, `- **Startup Repairs Applied**: ${lastStartupRepair.repairedCount}`, `- **Startup Repair Issues**: ${startupRepairIssues || 'None'}`);
|
|
223
|
+
}
|
|
224
|
+
if (info.permissionHooks.lastDiagnostic) {
|
|
225
|
+
const lastDiagnostic = info.permissionHooks.lastDiagnostic;
|
|
226
|
+
const diagnosticOutcome = lastDiagnostic.outcome
|
|
227
|
+
? `${lastDiagnostic.event} / ${lastDiagnostic.outcome}`
|
|
228
|
+
: lastDiagnostic.event;
|
|
229
|
+
lines.push(`- **Last Diagnostic Event**: ${lastDiagnostic.timestamp} (${diagnosticOutcome})`, `- **Last Diagnostic Stage**: ${lastDiagnostic.stage}`, `- **Last Diagnostic Input Bytes**: ${lastDiagnostic.rawInputLength ?? 0}`, `- **Last Diagnostic Normalized Bytes**: ${lastDiagnostic.normalizedResponseLength ?? 0}`, `- **Last Diagnostic Emitted Bytes**: ${lastDiagnostic.emittedResponseLength ?? 0}`);
|
|
230
|
+
if (lastDiagnostic.reason) {
|
|
231
|
+
lines.push(`- **Last Diagnostic Reason**: ${lastDiagnostic.reason}`);
|
|
232
|
+
}
|
|
233
|
+
}
|
|
198
234
|
}
|
|
199
235
|
// Issue #706: Startup timing
|
|
200
236
|
if (info.startup) {
|
|
@@ -291,4 +327,4 @@ export class BuildInfoService {
|
|
|
291
327
|
return `${mb.toFixed(1)} MB`;
|
|
292
328
|
}
|
|
293
329
|
}
|
|
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"]}
|
|
330
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BuildInfoService.js","sourceRoot":"","sources":["../../src/services/BuildInfoService.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,KAAK,aAAa,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAE5C,OAAO,EAAE,YAAY,EAAE,eAAe,EAAE,eAAe,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AAErG,OAAO,EAAE,sBAAsB,EAAE,MAAM,sBAAsB,CAAC;AAC9D,OAAO,EACL,6BAA6B,EAE7B,6BAA6B,GAG9B,MAAM,6BAA6B,CAAC;AAuDrC,MAAM,OAAO,gBAAgB;IACV,SAAS,CAAO;IAChB,cAAc,CAAyB;IACxD,qEAAqE;IAC7D,YAAY,GAAwB,IAAI,CAAC;IACjD,2DAA2D;IACnD,oBAAoB,GAA2B,IAAI,CAAC;IAE5D,YAAY,cAAsC;QAChD,IAAI,CAAC,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC;QAC5B,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC;IACvC,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,KAAmB;QACjC,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;IAC5B,CAAC;IAED;;OAEG;IACH,uBAAuB,CAAC,OAAsB;QAC5C,IAAI,CAAC,oBAAoB,GAAG,OAAO,CAAC;IACtC,CAAC;IAED;;;;;;;OAOG;IACI,KAAK,CAAC,YAAY;QACvB,kFAAkF;QAClF,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CAAC;YACvC,IAAI,CAAC,UAAU,EAAE;YACjB,IAAI,CAAC,aAAa,EAAE;YACpB,6BAA6B,EAAE;SAChC,CAAC,CAAC;QAEH,yDAAyD;QACzD,MAAM,WAAW,GAAG,EAAE,IAAI,EAAE,YAAY,EAAE,OAAO,EAAE,eAAe,EAAE,CAAC;QAErE,MAAM,OAAO,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,WAAW;YAC/C,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK;YAClB,CAAC,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC;QAE7C,MAAM,UAAU,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,WAAW;YAClD,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK;YAClB,CAAC,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,IAAI,EAAE,SAAS,EAAE,CAAC;QACzC,MAAM,kBAAkB,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,WAAW;YAC1D,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK;YAClB,CAAC,CAAC;gBACA,cAAc,EAAE,EAAE;gBAClB,YAAY,EAAE,EAAE;gBAChB,aAAa,EAAE,EAAE;gBACjB,gBAAgB,EAAE,EAAE;gBACpB,eAAe,EAAE,EAAE;gBACnB,cAAc,EAAE,IAAI;gBACpB,iBAAiB,EAAE,IAAI;aACxB,CAAC;QACJ,MAAM,oBAAoB,GAAG,6BAA6B,CAAC,kBAAkB,CAAC,CAAC;QAE/E,MAAM,mBAAmB,GAAG,CAAC,MAAe,EAAU,EAAE,CAAC,CACvD,MAAM,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,IAAI,KAAK,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAC/E,CAAC;QAEF,mCAAmC;QACnC,MAAM,QAAQ,GAAa,EAAE,CAAC;QAC9B,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;YACrC,QAAQ,CAAC,IAAI,CAAC,aAAa,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACvE,CAAC;QACD,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;YACrC,QAAQ,CAAC,IAAI,CAAC,gBAAgB,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC1E,CAAC;QACD,IAAI,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;YACrC,QAAQ,CAAC,IAAI,CAAC,0BAA0B,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACpF,CAAC;QAED,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxB,MAAM,CAAC,KAAK,CAAC,6BAA6B,QAAQ,CAAC,MAAM,gBAAgB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAClG,CAAC;QAED,4DAA4D;QAC5D,MAAM,cAAc,GAAG,eAAe,CAAC;QAEvC,2CAA2C;QAC3C,MAAM,gBAAgB,GAAG,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;QACxF,MAAM,WAAW,GAAyB;YACxC,MAAM,EAAE,gBAAgB,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,cAAc;YACnD,qBAAqB,EAAE,gBAAgB;YACvC,QAAQ,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE;YAC/C,eAAe,EAAE,IAAI,CAAC,YAAY,EAAE,SAAS,EAAE;SAChD,CAAC;QACF,MAAM,eAAe,GAAG,sBAAsB,EAAE,CAAC;QAEjD,OAAO;YACL,SAAS,EAAE,eAAe,CAAC,SAAS;YACpC,gBAAgB,EAAE,eAAe,CAAC,gBAAgB;YAClD,aAAa,EAAE,eAAe,CAAC,MAAM;YACrC,OAAO,EAAE,WAAW;YACpB,KAAK,EAAE;gBACL,SAAS,EAAE,cAAc;gBACzB,IAAI,EAAE,UAAU;gBAChB,SAAS,EAAE,OAAO,CAAC,MAAM;gBACzB,SAAS,EAAE,OAAO,CAAC,MAAM;aAC1B;YACD,OAAO,EAAE;gBACP,WAAW,EAAE,OAAO,CAAC,OAAO;gBAC5B,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,IAAI,EAAE,OAAO,CAAC,IAAI;gBAClB,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE;gBACxB,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE;aACnC;YACD,WAAW,EAAE;gBACX,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ;gBAC7B,YAAY,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY;gBACnD,aAAa,EAAE,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,aAAa;gBACrD,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,GAAG;gBAClE,QAAQ,EAAE,UAAU,CAAC,QAAQ;gBAC7B,UAAU,EAAE,UAAU,CAAC,IAAI;aAC5B;YACD,MAAM,EAAE;gBACN,SAAS,EAAE,IAAI,CAAC,SAAS;gBACzB,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE;gBAC7C,aAAa,EAAE,IAAI,CAAC,yDAAyD;aAC9E;YACD,eAAe,EAAE;gBACf,GAAG,kBAAkB;gBACrB,MAAM,EAAE,oBAAoB;aAC7B;YACD,OAAO,EAAE,WAAW;SACrB,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACI,eAAe,CAAC,IAAe;QACpC,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,KAAK,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;QAEvC,eAAe;QACf,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAC5B,KAAK,CAAC,IAAI,CAAC,eAAe,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;QAC/C,KAAK,CAAC,IAAI,CAAC,kBAAkB,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC;QACrD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,MAAM,mBAAmB,GAAG,IAAI,CAAC,aAAa,KAAK,KAAK;YACtD,CAAC,CAAC,sBAAsB;YACxB,CAAC,CAAC,gCAAgC,CAAC;QACrC,MAAM,YAAY,GAAG;YACnB,eAAe;YACf,qBAAqB,IAAI,CAAC,SAAS,EAAE;YACrC,0BAA0B,mBAAmB,EAAE;SAChD,CAAC;QACF,IAAI,IAAI,CAAC,gBAAgB,KAAK,IAAI,CAAC,SAAS,EAAE,CAAC;YAC7C,YAAY,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,6BAA6B,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC;QAClF,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,GAAG,YAAY,EAAE,EAAE,CAAC,CAAC;QAEhC,aAAa;QACb,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,eAAe,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC;QAC7C,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,oBAAoB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;QACzD,CAAC;QACD,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,uBAAuB,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,CAAC;QAC9D,CAAC;QACD,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,qBAAqB,IAAI,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;QAC1D,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,eAAe;QACf,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAC5B,KAAK,CAAC,IAAI,CAAC,kBAAkB,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;QACzD,KAAK,CAAC,IAAI,CAAC,mBAAmB,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QACvD,KAAK,CAAC,IAAI,CAAC,uBAAuB,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;QACvD,KAAK,CAAC,IAAI,CAAC,yBAAyB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC9E,KAAK,CAAC,IAAI,CAAC,uBAAuB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;QACrJ,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,mBAAmB;QACnB,KAAK,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;QAChC,KAAK,CAAC,IAAI,CAAC,mBAAmB,IAAI,CAAC,WAAW,CAAC,OAAO,IAAI,SAAS,EAAE,CAAC,CAAC;QACvE,KAAK,CAAC,IAAI,CAAC,eAAe,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC;QACvI,KAAK,CAAC,IAAI,CAAC,gBAAgB,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;QAChF,KAAK,CAAC,IAAI,CAAC,iBAAiB,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QACxE,IAAI,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;YAChC,KAAK,CAAC,IAAI,CAAC,oBAAoB,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC,CAAC;QAChE,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,cAAc;QACd,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,kBAAkB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QACpE,KAAK,CAAC,IAAI,CAAC,iBAAiB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC,CAAC;QAC5E,KAAK,CAAC,IAAI,CAAC,yBAAyB,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,gBAAgB,EAAE,CAAC,CAAC;QAEpG,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC;gBACnE,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC;gBAChD,CAAC,CAAC,MAAM,CAAC;YACX,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC;gBAC/D,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC;gBAC9C,CAAC,CAAC,MAAM,CAAC;YACX,MAAM,gBAAgB,GAAG,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC;gBACvE,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC;gBAClD,CAAC,CAAC,MAAM,CAAC;YACX,MAAM,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC;YACjE,MAAM,mBAAmB,GAAG,iBAAiB;gBAC3C,CAAC,CAAC,iBAAiB,CAAC,WAAW;qBAC5B,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,OAAO,KAAK,cAAc,IAAI,MAAM,CAAC,OAAO,KAAK,OAAO,CAAC;qBACnF,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,IAAI,KAAK,MAAM,CAAC,WAAW,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC;qBAC5F,IAAI,CAAC,IAAI,CAAC;gBACb,CAAC,CAAC,EAAE,CAAC;YAEP,KAAK,CAAC,IAAI,CACR,EAAE,EACF,wBAAwB,EACxB,iBAAiB,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,OAAO,EAAE,EAC5G,0BAA0B,cAAc,EAAE,EAC1C,yBAAyB,YAAY,EAAE,EACvC,uBAAuB,gBAAgB,EAAE,CAC1C,CAAC;YAEF,IAAI,IAAI,CAAC,eAAe,CAAC,eAAe,EAAE,CAAC;gBACzC,KAAK,CAAC,IAAI,CAAC,0BAA0B,IAAI,CAAC,eAAe,CAAC,eAAe,EAAE,CAAC,CAAC;YAC/E,CAAC;YAED,IAAI,iBAAiB,EAAE,CAAC;gBACtB,KAAK,CAAC,IAAI,CACR,6BAA6B,iBAAiB,CAAC,WAAW,KAAK,iBAAiB,CAAC,UAAU,KAAK,EAChG,kCAAkC,iBAAiB,CAAC,aAAa,EAAE,EACnE,gCAAgC,mBAAmB,IAAI,MAAM,EAAE,CAChE,CAAC;YACJ,CAAC;YAED,IAAI,IAAI,CAAC,eAAe,CAAC,cAAc,EAAE,CAAC;gBACxC,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC;gBAC3D,MAAM,iBAAiB,GAAG,cAAc,CAAC,OAAO;oBAC9C,CAAC,CAAC,GAAG,cAAc,CAAC,KAAK,MAAM,cAAc,CAAC,OAAO,EAAE;oBACvD,CAAC,CAAC,cAAc,CAAC,KAAK,CAAC;gBACzB,KAAK,CAAC,IAAI,CACR,gCAAgC,cAAc,CAAC,SAAS,KAAK,iBAAiB,GAAG,EACjF,gCAAgC,cAAc,CAAC,KAAK,EAAE,EACtD,sCAAsC,cAAc,CAAC,cAAc,IAAI,CAAC,EAAE,EAC1E,2CAA2C,cAAc,CAAC,wBAAwB,IAAI,CAAC,EAAE,EACzF,wCAAwC,cAAc,CAAC,qBAAqB,IAAI,CAAC,EAAE,CACpF,CAAC;gBACF,IAAI,cAAc,CAAC,MAAM,EAAE,CAAC;oBAC1B,KAAK,CAAC,IAAI,CAAC,iCAAiC,cAAc,CAAC,MAAM,EAAE,CAAC,CAAC;gBACvE,CAAC;YACH,CAAC;QACH,CAAC;QAED,6BAA6B;QAC7B,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YAC5B,KAAK,CAAC,IAAI,CAAC,iBAAiB,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,gBAAgB,EAAE,CAAC,CAAC;YAC9F,KAAK,CAAC,IAAI,CAAC,yBAAyB,IAAI,CAAC,OAAO,CAAC,qBAAqB,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,aAAa,EAAE,CAAC,CAAC;YACvG,IAAI,IAAI,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC;gBACjC,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC;gBAC5C,KAAK,CAAC,IAAI,CAAC,wBAAwB,MAAM,CAAC,cAAc,IAAI,CAAC,CAAC;gBAC9D,KAAK,CAAC,IAAI,CAAC,wBAAwB,MAAM,CAAC,UAAU,IAAI,CAAC,CAAC;gBAC1D,KAAK,CAAC,IAAI,CAAC,wBAAwB,MAAM,CAAC,OAAO,IAAI,CAAC,CAAC;gBACvD,IAAI,MAAM,CAAC,WAAW,KAAK,IAAI,EAAE,CAAC;oBAChC,KAAK,CAAC,IAAI,CAAC,0BAA0B,MAAM,CAAC,WAAW,IAAI,CAAC,CAAC;gBAC/D,CAAC;gBACD,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC7B,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;oBAC5B,KAAK,MAAM,KAAK,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;wBAClC,MAAM,GAAG,GAAG,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC;wBACrD,KAAK,CAAC,IAAI,CAAC,OAAO,KAAK,CAAC,IAAI,KAAK,KAAK,CAAC,UAAU,OAAO,GAAG,GAAG,CAAC,CAAC;oBAClE,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,UAAU;QACtB,IAAI,CAAC;YACH,6EAA6E;YAC7E,4FAA4F;YAC5F,MAAM,MAAM,GAAG,aAAa,CAAC,QAAQ,CAAC,4BAA4B,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;YAClG,MAAM,MAAM,GAAG,aAAa,CAAC,QAAQ,CAAC,iCAAiC,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;YAEvG,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC;QAC5B,CAAC;QAAC,MAAM,CAAC;YACP,+CAA+C;YAC/C,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,aAAa;QACzB,IAAI,CAAC;YACH,8FAA8F;YAC9F,iFAAiF;YACjF,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,gBAAgB,EAAE;gBACzE,MAAM,EAAE,gCAAgC;aACzC,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,aAAa,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,aAAa,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;YAE1F,IAAI,QAAQ,EAAE,CAAC;gBACb,0BAA0B;gBAC1B,MAAM,WAAW,GAAG,aAAa;qBAC9B,KAAK,CAAC,IAAI,CAAC;qBACX,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;oBACtC,EAAE,KAAK,CAAC,GAAG,CAAC;qBACX,GAAG,EAAE;oBACN,EAAE,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBAErB,OAAO;oBACL,QAAQ,EAAE,IAAI;oBACd,IAAI,EAAE,WAAW,CAAC,CAAC,CAAC,iBAAiB,WAAW,EAAE,CAAC,CAAC,CAAC,mBAAmB;iBACzE,CAAC;YACJ,CAAC;YAED,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;QAC7B,CAAC;QAAC,MAAM,CAAC;YACP,uCAAuC;YACvC,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAEO,YAAY,CAAC,OAAe;QAClC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC;QACzC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,GAAG,IAAI,CAAC,CAAC;QACnD,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;QAClD,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;QAEtC,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,IAAI,IAAI,GAAG,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;QACrC,IAAI,KAAK,GAAG,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC;QACvC,IAAI,OAAO,GAAG,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,GAAG,OAAO,GAAG,CAAC,CAAC;QAC3C,IAAI,IAAI,GAAG,CAAC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;QAE3D,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAEO,YAAY,CAAC,KAAa;QAChC,MAAM,EAAE,GAAG,KAAK,GAAG,IAAI,GAAG,IAAI,CAAC;QAC/B,OAAO,GAAG,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC;IAC/B,CAAC;CACF","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 {\n  getPermissionHookAuditSummary,\n  type PermissionHookDiagnosticRecord,\n  summarizePermissionHookHealth,\n  type PermissionHookHealthSummary,\n  type PermissionHookStartupRepairSummary,\n} 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    health: PermissionHookHealthSummary;\n    installedHosts: string[];\n    currentHosts: string[];\n    repairedHosts: string[];\n    needsRepairHosts: string[];\n    diagnosticsPath: string;\n    lastDiagnostic: PermissionHookDiagnosticRecord | null;\n    lastStartupRepair: PermissionHookStartupRepairSummary | null;\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      : {\n        installedHosts: [],\n        currentHosts: [],\n        repairedHosts: [],\n        needsRepairHosts: [],\n        diagnosticsPath: '',\n        lastDiagnostic: null,\n        lastStartupRepair: null,\n      };\n    const permissionHookHealth = summarizePermissionHookHealth(permissionHookInfo);\n\n    const formatSettledReason = (reason: unknown): string => (\n      reason instanceof Error ? `${reason.name}: ${reason.message}` : String(reason)\n    );\n\n    // Log any failures for diagnostics\n    const failures: string[] = [];\n    if (results[0].status === 'rejected') {\n      failures.push(`git info: ${formatSettledReason(results[0].reason)}`);\n    }\n    if (results[1].status === 'rejected') {\n      failures.push(`docker info: ${formatSettledReason(results[1].reason)}`);\n    }\n    if (results[2].status === 'rejected') {\n      failures.push(`permission hook audit: ${formatSettledReason(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: {\n        ...permissionHookInfo,\n        health: permissionHookHealth,\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    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      const lastStartupRepair = info.permissionHooks.lastStartupRepair;\n      const startupRepairIssues = lastStartupRepair\n        ? lastStartupRepair.hostResults\n          .filter((result) => result.outcome === 'needs_repair' || result.outcome === 'error')\n          .map((result) => result.repairError ? `${result.host} (${result.repairError})` : result.host)\n          .join('; ')\n        : '';\n\n      lines.push(\n        '',\n        '## 🔐 Permission Hooks',\n        `- **Health**: ${info.permissionHooks.health.status.toUpperCase()} — ${info.permissionHooks.health.message}`,\n        `- **Installed Hosts**: ${installedHosts}`,\n        `- **Current Assets**: ${currentHosts}`,\n        `- **Needs Repair**: ${needsRepairHosts}`,\n      );\n\n      if (info.permissionHooks.diagnosticsPath) {\n        lines.push(`- **Diagnostics Log**: ${info.permissionHooks.diagnosticsPath}`);\n      }\n\n      if (lastStartupRepair) {\n        lines.push(\n          `- **Last Startup Audit**: ${lastStartupRepair.completedAt} (${lastStartupRepair.durationMs}ms)`,\n          `- **Startup Repairs Applied**: ${lastStartupRepair.repairedCount}`,\n          `- **Startup Repair Issues**: ${startupRepairIssues || 'None'}`,\n        );\n      }\n\n      if (info.permissionHooks.lastDiagnostic) {\n        const lastDiagnostic = info.permissionHooks.lastDiagnostic;\n        const diagnosticOutcome = lastDiagnostic.outcome\n          ? `${lastDiagnostic.event} / ${lastDiagnostic.outcome}`\n          : lastDiagnostic.event;\n        lines.push(\n          `- **Last Diagnostic Event**: ${lastDiagnostic.timestamp} (${diagnosticOutcome})`,\n          `- **Last Diagnostic Stage**: ${lastDiagnostic.stage}`,\n          `- **Last Diagnostic Input Bytes**: ${lastDiagnostic.rawInputLength ?? 0}`,\n          `- **Last Diagnostic Normalized Bytes**: ${lastDiagnostic.normalizedResponseLength ?? 0}`,\n          `- **Last Diagnostic Emitted Bytes**: ${lastDiagnostic.emittedResponseLength ?? 0}`,\n        );\n        if (lastDiagnostic.reason) {\n          lines.push(`- **Last Diagnostic Reason**: ${lastDiagnostic.reason}`);\n        }\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"]}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type InstallPermissionHookOptions, type InstallPermissionHookResult } from './permissionHookShared.js';
|
|
2
|
+
export declare function ensureClaudePreToolUseHook(parsed: Record<string, unknown>, command: string): {
|
|
3
|
+
changed: boolean;
|
|
4
|
+
parsed: Record<string, unknown>;
|
|
5
|
+
};
|
|
6
|
+
export declare function ensureVsCodePreToolUseHook(parsed: Record<string, unknown>, command: string): {
|
|
7
|
+
changed: boolean;
|
|
8
|
+
parsed: Record<string, unknown>;
|
|
9
|
+
};
|
|
10
|
+
export declare function ensureGeminiBeforeToolHook(parsed: Record<string, unknown>, command: string): {
|
|
11
|
+
changed: boolean;
|
|
12
|
+
parsed: Record<string, unknown>;
|
|
13
|
+
};
|
|
14
|
+
export declare function ensureCodexPreToolUseHook(parsed: Record<string, unknown>, command: string): {
|
|
15
|
+
changed: boolean;
|
|
16
|
+
parsed: Record<string, unknown>;
|
|
17
|
+
};
|
|
18
|
+
export declare function ensureCursorPreToolUseHook(parsed: Record<string, unknown>, command: string): {
|
|
19
|
+
changed: boolean;
|
|
20
|
+
parsed: Record<string, unknown>;
|
|
21
|
+
};
|
|
22
|
+
export declare function ensureWindsurfHooks(parsed: Record<string, unknown>, command: string): {
|
|
23
|
+
changed: boolean;
|
|
24
|
+
parsed: Record<string, unknown>;
|
|
25
|
+
};
|
|
26
|
+
export declare function installPermissionHook(client: string, options?: InstallPermissionHookOptions): Promise<InstallPermissionHookResult>;
|
|
27
|
+
//# sourceMappingURL=permissionHookInstallers.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionHookInstallers.d.ts","sourceRoot":"","sources":["../../src/utils/permissionHookInstallers.ts"],"names":[],"mappings":"AAGA,OAAO,EACL,KAAK,4BAA4B,EACjC,KAAK,2BAA2B,EAkBjC,MAAM,2BAA2B,CAAC;AAQnC,wBAAgB,0BAA0B,CACxC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAEvD;AAED,wBAAgB,0BAA0B,CACxC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAEvD;AAED,wBAAgB,0BAA0B,CACxC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAEvD;AAED,wBAAgB,yBAAyB,CACvC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAIvD;AAED,wBAAgB,0BAA0B,CACxC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CAyBvD;AAED,wBAAgB,mBAAmB,CACjC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC/B,OAAO,EAAE,MAAM,GACd;IAAE,OAAO,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAAE,CA4BvD;AA0cD,wBAAsB,qBAAqB,CACzC,MAAM,EAAE,MAAM,EACd,OAAO,GAAE,4BAAiC,GACzC,OAAO,CAAC,2BAA2B,CAAC,CAwCtC"}
|