@dollhousemcp/mcp-server 2.0.24 → 2.0.26

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.
Files changed (62) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/dist/auto-dollhouse/portDiscovery.d.ts.map +1 -1
  3. package/dist/auto-dollhouse/portDiscovery.js +7 -4
  4. package/dist/di/Container.d.ts.map +1 -1
  5. package/dist/di/Container.js +4 -2
  6. package/dist/generated/version.d.ts +2 -2
  7. package/dist/generated/version.js +3 -3
  8. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  9. package/dist/handlers/mcp-aql/MCPAQLHandler.js +18 -1
  10. package/dist/handlers/mcp-aql/OperationRouter.d.ts.map +1 -1
  11. package/dist/handlers/mcp-aql/OperationRouter.js +6 -1
  12. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  13. package/dist/handlers/mcp-aql/OperationSchema.js +16 -1
  14. package/dist/handlers/mcp-aql/SchemaDispatcher.d.ts.map +1 -1
  15. package/dist/handlers/mcp-aql/SchemaDispatcher.js +2 -1
  16. package/dist/index.js +14 -2
  17. package/dist/server/tools/BuildInfoTools.d.ts +1 -0
  18. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -1
  19. package/dist/server/tools/BuildInfoTools.js +2 -1
  20. package/dist/server/tools/MCPAQLTools.js +3 -1
  21. package/dist/services/ActivationStore.d.ts +8 -0
  22. package/dist/services/ActivationStore.d.ts.map +1 -1
  23. package/dist/services/ActivationStore.js +28 -29
  24. package/dist/services/BuildInfoService.d.ts +3 -0
  25. package/dist/services/BuildInfoService.d.ts.map +1 -1
  26. package/dist/services/BuildInfoService.js +18 -1
  27. package/dist/services/sessionIdentity.d.ts +24 -0
  28. package/dist/services/sessionIdentity.d.ts.map +1 -0
  29. package/dist/services/sessionIdentity.js +42 -0
  30. package/dist/utils/permissionAuthority.d.ts +38 -0
  31. package/dist/utils/permissionAuthority.d.ts.map +1 -0
  32. package/dist/utils/permissionAuthority.js +341 -0
  33. package/dist/utils/permissionHooks.d.ts.map +1 -1
  34. package/dist/utils/permissionHooks.js +10 -1
  35. package/dist/web/console/UnifiedConsole.d.ts +2 -0
  36. package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
  37. package/dist/web/console/UnifiedConsole.js +3 -1
  38. package/dist/web/portDiscovery.d.ts +7 -0
  39. package/dist/web/portDiscovery.d.ts.map +1 -1
  40. package/dist/web/portDiscovery.js +35 -4
  41. package/dist/web/public/app.js +28 -4
  42. package/dist/web/public/index.html +2 -0
  43. package/dist/web/public/permissions.css +456 -0
  44. package/dist/web/public/permissions.js +629 -12
  45. package/dist/web/public/sessions.css +119 -0
  46. package/dist/web/public/sessions.js +95 -9
  47. package/dist/web/public/setup.js +67 -7
  48. package/dist/web/public/styles.css +21 -2
  49. package/dist/web/routes/permissionRoutes.d.ts +4 -1
  50. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  51. package/dist/web/routes/permissionRoutes.js +118 -6
  52. package/dist/web/routes/setupRoutes.d.ts +18 -0
  53. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  54. package/dist/web/routes/setupRoutes.js +129 -49
  55. package/dist/web/server.d.ts +4 -0
  56. package/dist/web/server.d.ts.map +1 -1
  57. package/dist/web/server.js +19 -1
  58. package/package.json +4 -3
  59. package/scripts/pretooluse-dollhouse.sh +78 -12
  60. package/scripts/pretooluse-vscode.sh +6 -9
  61. package/scripts/pretooluse-windsurf.sh +6 -9
  62. package/server.json +2 -2
@@ -7,9 +7,13 @@
7
7
  * - Decision tracking ring buffer for the live dashboard feed
8
8
  */
9
9
  import express from 'express';
10
+ import { homedir } from 'node:os';
10
11
  import { logger } from '../../utils/logger.js';
11
12
  import { formatPermissionResponse } from '../../handlers/mcp-aql/evaluatePermission.js';
13
+ import { ensureLatestPortFile } from '../portDiscovery.js';
14
+ import { getPermissionHookStatusAsync } from '../../utils/permissionHooks.js';
12
15
  import { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';
16
+ import { PERMISSION_AUTHORITY_HOSTS, PERMISSION_AUTHORITY_MODES, readPermissionAuthorityState, setPermissionAuthorityMode, } from '../../utils/permissionAuthority.js';
13
17
  const PERMISSION_ROUTE_RATE_LIMIT_REQUESTS = 120;
14
18
  const PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS = 60_000;
15
19
  const DECISION_BUFFER_SIZE = 200;
@@ -194,11 +198,33 @@ function extractKnownPolicySessions(elements) {
194
198
  }
195
199
  return knownSessions.sort((a, b) => a.sessionId.localeCompare(b.sessionId));
196
200
  }
197
- /**
198
- * Register permission-related routes on a gateway router.
199
- * Must be called with the MCP-AQL handler for policy evaluation.
200
- */
201
- export function registerPermissionRoutes(router, handler) {
201
+ async function selfHealLatestPermissionPortFile(port) {
202
+ if (typeof port !== 'number' || !Number.isInteger(port) || port <= 0) {
203
+ return;
204
+ }
205
+ try {
206
+ await ensureLatestPortFile(port);
207
+ }
208
+ catch (err) {
209
+ logger.debug('[WebUI/Gateway] Could not refresh permission-server.port', {
210
+ error: err instanceof Error ? err.message : String(err),
211
+ port,
212
+ });
213
+ }
214
+ }
215
+ async function resolveInstalledPermissionAuthorityHosts(homeDir, authorityState) {
216
+ const installedStatuses = await Promise.all(PERMISSION_AUTHORITY_HOSTS.map(async (host) => ({
217
+ host,
218
+ status: await getPermissionHookStatusAsync(homeDir, host),
219
+ })));
220
+ const installedHosts = installedStatuses
221
+ .filter(({ status }) => status.installed)
222
+ .map(({ host }) => host);
223
+ const persistedHosts = Object.keys(authorityState.hosts || {}).filter((host) => PERMISSION_AUTHORITY_HOSTS.includes(host));
224
+ return Array.from(new Set(installedHosts.concat(persistedHosts)));
225
+ }
226
+ export function registerPermissionRoutes(router, handler, options = {}) {
227
+ const authorityHomeDir = options.homeDir ?? homedir();
202
228
  const decisionTracker = createPermissionDecisionTracker();
203
229
  /**
204
230
  * POST /api/evaluate_permission
@@ -208,6 +234,7 @@ export function registerPermissionRoutes(router, handler) {
208
234
  */
209
235
  const permissionLimiter = new SlidingWindowRateLimiter(PERMISSION_ROUTE_RATE_LIMIT_REQUESTS, PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS);
210
236
  router.post('/evaluate_permission', express.json(), async (req, res) => {
237
+ await selfHealLatestPermissionPortFile(req.socket.localPort);
211
238
  const body = req.body;
212
239
  const platform = typeof body.platform === 'string' ? body.platform.normalize('NFC') : 'claude_code';
213
240
  if (!permissionLimiter.tryAcquire()) {
@@ -261,6 +288,7 @@ export function registerPermissionRoutes(router, handler) {
261
288
  */
262
289
  router.get('/permissions/status', async (req, res) => {
263
290
  try {
291
+ await selfHealLatestPermissionPortFile(req.socket.localPort);
264
292
  const sessionId = typeof req.query['sessionId'] === 'string' && req.query['sessionId']
265
293
  ? req.query['sessionId']
266
294
  : undefined;
@@ -276,6 +304,8 @@ export function registerPermissionRoutes(router, handler) {
276
304
  return;
277
305
  }
278
306
  const data = opResult.data;
307
+ const authorityState = await readPermissionAuthorityState(authorityHomeDir);
308
+ const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);
279
309
  const elements = normalizePolicyElements((data.elements || []));
280
310
  const denyPatterns = data.combinedDenyPatterns ?? [];
281
311
  const allowPatterns = data.combinedAllowPatterns ?? [];
@@ -301,6 +331,10 @@ export function registerPermissionRoutes(router, handler) {
301
331
  permissionPromptActive: data.permissionPromptActive,
302
332
  hookInstalled: data.hookInstalled,
303
333
  hookHost: data.hookHost,
334
+ authority: authorityState,
335
+ authoritySupportedHosts: installedAuthorityHosts,
336
+ authoritySupportedModes: [...PERMISSION_AUTHORITY_MODES],
337
+ authorityAiMutable: false,
304
338
  enforcementReady: data.enforcementReady,
305
339
  invalidPolicyElementCount: data.invalidPolicyElementCount ?? 0,
306
340
  advisory: data.advisory,
@@ -312,5 +346,83 @@ export function registerPermissionRoutes(router, handler) {
312
346
  res.status(500).json({ error: 'Failed to get permission status' });
313
347
  }
314
348
  });
349
+ router.get('/permissions/authority', async (_req, res) => {
350
+ try {
351
+ const authorityState = await readPermissionAuthorityState(authorityHomeDir);
352
+ const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);
353
+ res.json({
354
+ ...authorityState,
355
+ supportedHosts: installedAuthorityHosts,
356
+ supportedModes: [...PERMISSION_AUTHORITY_MODES],
357
+ aiMutable: false,
358
+ });
359
+ }
360
+ catch (err) {
361
+ logger.error('[WebUI/Gateway] permissions/authority error:', err);
362
+ res.status(500).json({ error: 'Failed to get permission authority' });
363
+ }
364
+ });
365
+ router.post('/permissions/authority', express.json(), async (req, res) => {
366
+ try {
367
+ const host = normalizeAuthorityHost(req.body?.host);
368
+ const mode = normalizeAuthorityMode(req.body?.mode);
369
+ const reason = typeof req.body?.reason === 'string' && req.body.reason.trim() !== ''
370
+ ? req.body.reason.trim()
371
+ : undefined;
372
+ if (!host || !mode) {
373
+ res.status(400).json({
374
+ error: 'host and mode are required',
375
+ supportedHosts: [...PERMISSION_AUTHORITY_HOSTS],
376
+ supportedModes: [...PERMISSION_AUTHORITY_MODES],
377
+ });
378
+ return;
379
+ }
380
+ let policies;
381
+ if (mode === 'authoritative') {
382
+ const policyResult = asSingleResult(await handler.handleRead({
383
+ operation: 'get_effective_cli_policies',
384
+ params: { reporting_scope: 'dashboard' },
385
+ }));
386
+ if (!policyResult.success) {
387
+ res.status(500).json({ error: policyResult.error || 'Failed to fetch effective policies' });
388
+ return;
389
+ }
390
+ policies = policyResult.data;
391
+ }
392
+ const authorityState = await setPermissionAuthorityMode({
393
+ host,
394
+ mode,
395
+ reason,
396
+ homeDir: authorityHomeDir,
397
+ policies: mode === 'authoritative'
398
+ ? {
399
+ combinedAllowPatterns: asStringArray(policies?.combinedAllowPatterns),
400
+ combinedConfirmPatterns: asStringArray(policies?.combinedConfirmPatterns),
401
+ combinedDenyPatterns: asStringArray(policies?.combinedDenyPatterns),
402
+ }
403
+ : undefined,
404
+ });
405
+ res.json({ success: true, authority: authorityState });
406
+ }
407
+ catch (err) {
408
+ logger.error(`[WebUI/Gateway] permissions/authority update error (host=${String(req.body?.host ?? '')}, mode=${String(req.body?.mode ?? '')}):`, err);
409
+ res.status(500).json({
410
+ error: err instanceof Error ? err.message : 'Failed to update permission authority',
411
+ });
412
+ }
413
+ });
414
+ }
415
+ function normalizeAuthorityHost(value) {
416
+ return typeof value === 'string' && PERMISSION_AUTHORITY_HOSTS.includes(value)
417
+ ? value
418
+ : null;
419
+ }
420
+ function normalizeAuthorityMode(value) {
421
+ return typeof value === 'string' && PERMISSION_AUTHORITY_MODES.includes(value)
422
+ ? value
423
+ : null;
424
+ }
425
+ function asStringArray(value) {
426
+ return Array.isArray(value) ? value.filter((entry) => typeof entry === 'string') : [];
315
427
  }
316
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation HTTP routes and decision tracking.\n *\n * Provides:\n * - POST /evaluate_permission — evaluates tool permissions via MCP-AQL\n * - GET /permissions/status — returns current policies and recent decisions\n * - Decision tracking ring buffer for the live dashboard feed\n */\n\nimport express, { Router } from 'express';\nimport { logger } from '../../utils/logger.js';\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\nimport { formatPermissionResponse } from '../../handlers/mcp-aql/evaluatePermission.js';\n\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\n\n// ── Permission Decision Tracking ─────────────────────────────────────────────\n// Ring buffer of recent permission decisions for the live dashboard feed.\n\ninterface PermissionDecision {\n  id: string;\n  timestamp: string;\n  session_id?: string;\n  tool_name: string;\n  command?: string;\n  decision: string;\n  reason?: string;\n  platform?: string;\n  target?: string;\n  targetLabel?: string;\n  details?: PermissionDecisionDetail[];\n}\n\ninterface PermissionDecisionDetail {\n  label: string;\n  value: string;\n  monospace?: boolean;\n}\n\ninterface KnownPolicySession {\n  sessionId: string;\n  displayName: string;\n  source: 'policy';\n}\n\nconst PERMISSION_ROUTE_RATE_LIMIT_REQUESTS = 120;\nconst PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS = 60_000;\nconst DECISION_BUFFER_SIZE = 200;\n\ninterface PermissionDecisionTracker {\n  trackDecision(\n    sessionId: string | undefined,\n    toolName: string,\n    input: Record<string, unknown>,\n    result: Record<string, unknown>,\n    platform: string,\n  ): void;\n  getRecentDecisions(): PermissionDecision[];\n}\n\nconst CLAUDE_COMPATIBLE_HOOK_PLATFORMS = new Set(['claude_code', 'vscode']);\nconst NORMALIZABLE_PERMISSION_DECISIONS = new Set(['allow', 'deny', 'ask']);\ntype NormalizablePermissionDecision = 'allow' | 'deny' | 'ask';\n\n/** Extract a string field from a record, trying multiple keys in order */\nfunction extractString(obj: Record<string, unknown>, keys: string[], fallback: string): string {\n  for (const key of keys) {\n    const val = obj?.[key];\n    if (typeof val === 'string') return val;\n  }\n  return fallback;\n}\n\nfunction extractDecision(result: Record<string, unknown>): string {\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedDecision = (nested as Record<string, unknown>).permissionDecision;\n    if (typeof nestedDecision === 'string') return nestedDecision;\n  }\n\n  if (typeof result.permission === 'string') return result.permission;\n  if (typeof result.decision === 'string') return result.decision;\n  if (typeof result.behavior === 'string') return result.behavior;\n  if (typeof result.allowed === 'boolean') return result.allowed ? 'allow' : 'deny';\n  return 'unknown';\n}\n\nfunction extractReason(result: Record<string, unknown>): string {\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedReason = (nested as Record<string, unknown>).permissionDecisionReason;\n    if (typeof nestedReason === 'string') return nestedReason;\n  }\n\n  return extractString(result, ['reason', 'message'], '');\n}\n\nfunction shouldNormalizeClaudeHook(platform: string | undefined): boolean {\n  return platform !== undefined && CLAUDE_COMPATIBLE_HOOK_PLATFORMS.has(platform);\n}\n\nfunction normalizePermissionResponseForPlatform(\n  platform: string,\n  input: Record<string, unknown>,\n  result: Record<string, unknown>,\n): Record<string, unknown> {\n  if (!shouldNormalizeClaudeHook(platform)) {\n    return result;\n  }\n\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedDecision = (nested as Record<string, unknown>).permissionDecision;\n    if (typeof nestedDecision === 'string') {\n      return result;\n    }\n  }\n\n  const decision = extractDecision(result);\n  if (NORMALIZABLE_PERMISSION_DECISIONS.has(decision)) {\n    return formatPermissionResponse(\n      decision as NormalizablePermissionDecision,\n      platform,\n      input,\n      extractReason(result),\n    );\n  }\n\n  return formatPermissionResponse('allow', platform, input);\n}\n\nfunction buildDecisionDetails(\n  toolName: string,\n  input: Record<string, unknown>,\n  result: Record<string, unknown>,\n  platform: string,\n): { target?: string; targetLabel?: string; details: PermissionDecisionDetail[] } {\n  const details: PermissionDecisionDetail[] = [];\n\n  if (platform) {\n    details.push({ label: 'Platform', value: platform, monospace: true });\n  }\n\n  if (toolName === 'Bash' && typeof input.command === 'string' && input.command !== '') {\n    details.push({ label: 'Command', value: input.command, monospace: true });\n  }\n\n  const targetDescriptors: Array<{ key: string; label: string; monospace?: boolean }> = [\n    { key: 'file_path', label: 'File', monospace: true },\n    { key: 'path', label: 'Path', monospace: true },\n    { key: 'url', label: 'URL' },\n    { key: 'pattern', label: 'Pattern', monospace: true },\n    { key: 'query', label: 'Query' },\n    { key: 'element_name', label: 'Element', monospace: true },\n    { key: 'request_id', label: 'Request', monospace: true },\n  ];\n\n  let target: string | undefined;\n  let targetLabel: string | undefined;\n\n  for (const descriptor of targetDescriptors) {\n    const value = input[descriptor.key];\n    if (typeof value !== 'string' || value === '') {\n      continue;\n    }\n\n    target = value;\n    targetLabel = descriptor.label;\n    details.push({ label: descriptor.label, value, monospace: descriptor.monospace });\n    break;\n  }\n\n  const matchedPattern = extractString(result, ['matched_pattern', 'matchedPattern'], '');\n  if (matchedPattern !== '') {\n    details.push({ label: 'Matched Pattern', value: matchedPattern, monospace: true });\n  }\n\n  const policySource = extractString(result, ['policy_source', 'policySource'], '');\n  if (policySource !== '') {\n    details.push({ label: 'Policy Source', value: policySource, monospace: true });\n  }\n\n  return { target, targetLabel, details };\n}\n\nfunction createPermissionDecisionTracker(bufferSize = DECISION_BUFFER_SIZE): PermissionDecisionTracker {\n  const recentDecisions: PermissionDecision[] = [];\n  let decisionCounter = 0;\n\n  return {\n    trackDecision(\n      sessionId: string | undefined,\n      toolName: string,\n      input: Record<string, unknown>,\n      result: Record<string, unknown>,\n      platform: string,\n    ): void {\n      const detailState = buildDecisionDetails(toolName, input, result, platform);\n      const entry: PermissionDecision = {\n        id: `d-${++decisionCounter}`,\n        timestamp: new Date().toISOString(),\n        ...(sessionId ? { session_id: sessionId } : {}),\n        tool_name: toolName,\n        command: toolName === 'Bash' && typeof input?.command === 'string' ? input.command : undefined,\n        decision: extractDecision(result),\n        reason: extractReason(result),\n        platform,\n        target: detailState.target,\n        targetLabel: detailState.targetLabel,\n        details: detailState.details,\n      };\n      recentDecisions.unshift(entry);\n      if (recentDecisions.length > bufferSize) {\n        recentDecisions.length = bufferSize;\n      }\n    },\n    getRecentDecisions(): PermissionDecision[] {\n      return recentDecisions;\n    },\n  };\n}\n\nfunction mergeRuleArrays(...sources: unknown[]): string[] {\n  const merged = new Set<string>();\n  for (const source of sources) {\n    if (!Array.isArray(source)) continue;\n    for (const entry of source) {\n      if (typeof entry === 'string' && entry !== '') {\n        merged.add(entry);\n      }\n    }\n  }\n  return Array.from(merged);\n}\n\nfunction normalizePolicyElements(elements: Array<Record<string, unknown>>): Array<Record<string, unknown>> {\n  return elements.map((element) => ({\n    ...element,\n    element_name: resolveElementName(element),\n    allowRules: mergeRuleArrays(element.allowPatterns, element.allowOperations),\n    confirmRules: mergeRuleArrays(element.confirmPatterns, element.confirmOperations),\n    denyRules: mergeRuleArrays(element.denyPatterns, element.denyOperations),\n    invalidGatekeeperPolicy: !!element.invalidGatekeeperPolicy,\n    invalidGatekeeperMessage: typeof element.invalidGatekeeperMessage === 'string' ? element.invalidGatekeeperMessage : undefined,\n  }));\n}\n\nfunction resolveElementName(element: Record<string, unknown>): string {\n  if (typeof element.element_name === 'string') return element.element_name;\n  if (typeof element.name === 'string') return element.name;\n  return '';\n}\n\n/** Helper to extract single result from MCP-AQL batch response */\nfunction asSingleResult(results: unknown): { success: boolean; data?: unknown; error?: string } {\n  if (Array.isArray(results)) return results[0] || { success: false, error: 'Empty result' };\n  return results as { success: boolean; data?: unknown; error?: string };\n}\n\nfunction extractKnownPolicySessions(elements: Array<Record<string, unknown>>): KnownPolicySession[] {\n  const seen = new Set<string>();\n  const knownSessions: KnownPolicySession[] = [];\n\n  for (const element of elements) {\n    const sessionIds = Array.isArray(element.sessionIds) ? element.sessionIds : [];\n    for (const sessionId of sessionIds) {\n      if (typeof sessionId !== 'string' || sessionId === '' || seen.has(sessionId)) {\n        continue;\n      }\n\n      seen.add(sessionId);\n      knownSessions.push({\n        sessionId,\n        displayName: sessionId,\n        source: 'policy',\n      });\n    }\n  }\n\n  return knownSessions.sort((a, b) => a.sessionId.localeCompare(b.sessionId));\n}\n\n/**\n * Register permission-related routes on a gateway router.\n * Must be called with the MCP-AQL handler for policy evaluation.\n */\nexport function registerPermissionRoutes(router: Router, handler: MCPAQLHandler): void {\n  const decisionTracker = createPermissionDecisionTracker();\n  /**\n   * POST /api/evaluate_permission\n   * Permission evaluation endpoint for PreToolUse hooks.\n   * Routes through evaluate_permission MCP-AQL READ operation.\n   * Fail-open: returns allow on any error to avoid blocking the user.\n   */\n  const permissionLimiter = new SlidingWindowRateLimiter(\n    PERMISSION_ROUTE_RATE_LIMIT_REQUESTS,\n    PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS,\n  );\n  router.post('/evaluate_permission', express.json(), async (req, res) => {\n    const body = req.body as {\n      tool_name?: string;\n      input?: Record<string, unknown>;\n      platform?: string;\n      session_id?: string;\n    };\n    const platform = typeof body.platform === 'string' ? body.platform.normalize('NFC') : 'claude_code';\n\n    if (!permissionLimiter.tryAcquire()) {\n      res.json(formatPermissionResponse('allow', platform, {})); // fail open on rate limit\n      return;\n    }\n\n    // Unicode normalization (NFC) on string inputs to prevent homograph attacks\n    const tool_name = typeof body.tool_name === 'string' ? body.tool_name.normalize('NFC') : undefined;\n    const session_id = typeof body.session_id === 'string' ? body.session_id.normalize('NFC') : undefined;\n    const input = body.input;\n\n    if (!tool_name) {\n      res.json(formatPermissionResponse('allow', platform, input || {})); // fail open on bad input\n      return;\n    }\n\n    const startMs = Date.now();\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'evaluate_permission',\n        params: {\n          tool_name,\n          input: input || {},\n          platform,\n          ...(session_id ? { session_id } : {}),\n        },\n      }));\n      const elapsedMs = Date.now() - startMs;\n\n      if (!opResult.success) {\n        logger.warn(`[WebUI/Gateway] evaluate_permission failed (${elapsedMs}ms): ${opResult.error}`);\n        res.json(formatPermissionResponse('allow', platform, input || {})); // fail open\n        return;\n      }\n\n      const rawResult = opResult.data as Record<string, unknown>;\n      const responseData = normalizePermissionResponseForPlatform(\n        platform,\n        input || {},\n        rawResult,\n      );\n      const trackedResult = { ...rawResult, ...responseData };\n      const decision = extractDecision(responseData);\n      logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);\n\n      // Track decision for live dashboard feed\n      decisionTracker.trackDecision(session_id, tool_name, input || {}, trackedResult, platform);\n\n      res.json(responseData);\n    } catch (err) {\n      const elapsedMs = Date.now() - startMs;\n      logger.error(`[WebUI/Gateway] evaluate_permission error (${elapsedMs}ms):`, err);\n      res.json(formatPermissionResponse('allow', platform, input || {})); // fail open\n    }\n  });\n\n  /**\n   * GET /api/permissions/status\n   * Returns current permission policies and recent decisions\n   * for the live permissions dashboard.\n   */\n  router.get('/permissions/status', async (req, res) => {\n    try {\n      const sessionId = typeof req.query['sessionId'] === 'string' && req.query['sessionId']\n        ? req.query['sessionId']\n        : undefined;\n\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'get_effective_cli_policies',\n        params: {\n          reporting_scope: 'dashboard',\n          ...(sessionId ? { session_id: sessionId } : {}),\n        },\n      }));\n\n      if (!opResult.success) {\n        res.status(500).json({ error: opResult.error || 'Failed to get policies' });\n        return;\n      }\n\n      const data = opResult.data as Record<string, unknown>;\n      const elements = normalizePolicyElements((data.elements || []) as Array<Record<string, unknown>>);\n\n      const denyPatterns = (data.combinedDenyPatterns as string[] | undefined) ?? [];\n      const allowPatterns = (data.combinedAllowPatterns as string[] | undefined) ?? [];\n      const confirmPatterns = (data.combinedConfirmPatterns as string[] | undefined) ?? [];\n      const denyOperations = (data.combinedDenyOperations as string[] | undefined) ?? [];\n      const allowOperations = (data.combinedAllowOperations as string[] | undefined) ?? [];\n      const confirmOperations = (data.combinedConfirmOperations as string[] | undefined) ?? [];\n\n      res.json({\n        ...(sessionId ? { sessionId } : {}),\n        activeElementCount: data.activeElementCount,\n        hasAllowlist: data.hasAllowlist,\n        denyPatterns,\n        allowPatterns,\n        confirmPatterns,\n        denyOperations,\n        allowOperations,\n        confirmOperations,\n        denyRules: mergeRuleArrays(denyPatterns, denyOperations),\n        allowRules: mergeRuleArrays(allowPatterns, allowOperations),\n        confirmRules: mergeRuleArrays(confirmPatterns, confirmOperations),\n        elements,\n        knownSessions: extractKnownPolicySessions(elements),\n        permissionPromptActive: data.permissionPromptActive,\n        hookInstalled: data.hookInstalled,\n        hookHost: data.hookHost,\n        enforcementReady: data.enforcementReady,\n        invalidPolicyElementCount: data.invalidPolicyElementCount ?? 0,\n        advisory: data.advisory,\n        recentDecisions: decisionTracker.getRecentDecisions(),\n      });\n    } catch (err) {\n      logger.error('[WebUI/Gateway] permissions/status error:', err);\n      res.status(500).json({ error: 'Failed to get permission status' });\n    }\n  });\n}\n"]}
428
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation HTTP routes and decision tracking.\n *\n * Provides:\n * - POST /evaluate_permission — evaluates tool permissions via MCP-AQL\n * - GET /permissions/status — returns current policies and recent decisions\n * - Decision tracking ring buffer for the live dashboard feed\n */\n\nimport express, { Router } from 'express';\nimport { homedir } from 'node:os';\nimport { logger } from '../../utils/logger.js';\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\nimport { formatPermissionResponse } from '../../handlers/mcp-aql/evaluatePermission.js';\nimport { ensureLatestPortFile } from '../portDiscovery.js';\nimport { getPermissionHookStatusAsync } from '../../utils/permissionHooks.js';\n\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport {\n  type PermissionAuthorityHost,\n  type PermissionAuthorityMode,\n  PERMISSION_AUTHORITY_HOSTS,\n  PERMISSION_AUTHORITY_MODES,\n  readPermissionAuthorityState,\n  setPermissionAuthorityMode,\n} from '../../utils/permissionAuthority.js';\n\n// ── Permission Decision Tracking ─────────────────────────────────────────────\n// Ring buffer of recent permission decisions for the live dashboard feed.\n\ninterface PermissionDecision {\n  id: string;\n  timestamp: string;\n  session_id?: string;\n  tool_name: string;\n  command?: string;\n  decision: string;\n  reason?: string;\n  platform?: string;\n  target?: string;\n  targetLabel?: string;\n  details?: PermissionDecisionDetail[];\n}\n\ninterface PermissionDecisionDetail {\n  label: string;\n  value: string;\n  monospace?: boolean;\n}\n\ninterface KnownPolicySession {\n  sessionId: string;\n  displayName: string;\n  source: 'policy';\n}\n\nconst PERMISSION_ROUTE_RATE_LIMIT_REQUESTS = 120;\nconst PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS = 60_000;\nconst DECISION_BUFFER_SIZE = 200;\n\ninterface PermissionDecisionTracker {\n  trackDecision(\n    sessionId: string | undefined,\n    toolName: string,\n    input: Record<string, unknown>,\n    result: Record<string, unknown>,\n    platform: string,\n  ): void;\n  getRecentDecisions(): PermissionDecision[];\n}\n\nconst CLAUDE_COMPATIBLE_HOOK_PLATFORMS = new Set(['claude_code', 'vscode']);\nconst NORMALIZABLE_PERMISSION_DECISIONS = new Set(['allow', 'deny', 'ask']);\ntype NormalizablePermissionDecision = 'allow' | 'deny' | 'ask';\n\n/** Extract a string field from a record, trying multiple keys in order */\nfunction extractString(obj: Record<string, unknown>, keys: string[], fallback: string): string {\n  for (const key of keys) {\n    const val = obj?.[key];\n    if (typeof val === 'string') return val;\n  }\n  return fallback;\n}\n\nfunction extractDecision(result: Record<string, unknown>): string {\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedDecision = (nested as Record<string, unknown>).permissionDecision;\n    if (typeof nestedDecision === 'string') return nestedDecision;\n  }\n\n  if (typeof result.permission === 'string') return result.permission;\n  if (typeof result.decision === 'string') return result.decision;\n  if (typeof result.behavior === 'string') return result.behavior;\n  if (typeof result.allowed === 'boolean') return result.allowed ? 'allow' : 'deny';\n  return 'unknown';\n}\n\nfunction extractReason(result: Record<string, unknown>): string {\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedReason = (nested as Record<string, unknown>).permissionDecisionReason;\n    if (typeof nestedReason === 'string') return nestedReason;\n  }\n\n  return extractString(result, ['reason', 'message'], '');\n}\n\nfunction shouldNormalizeClaudeHook(platform: string | undefined): boolean {\n  return platform !== undefined && CLAUDE_COMPATIBLE_HOOK_PLATFORMS.has(platform);\n}\n\nfunction normalizePermissionResponseForPlatform(\n  platform: string,\n  input: Record<string, unknown>,\n  result: Record<string, unknown>,\n): Record<string, unknown> {\n  if (!shouldNormalizeClaudeHook(platform)) {\n    return result;\n  }\n\n  const nested = result.hookSpecificOutput;\n  if (nested && typeof nested === 'object' && !Array.isArray(nested)) {\n    const nestedDecision = (nested as Record<string, unknown>).permissionDecision;\n    if (typeof nestedDecision === 'string') {\n      return result;\n    }\n  }\n\n  const decision = extractDecision(result);\n  if (NORMALIZABLE_PERMISSION_DECISIONS.has(decision)) {\n    return formatPermissionResponse(\n      decision as NormalizablePermissionDecision,\n      platform,\n      input,\n      extractReason(result),\n    );\n  }\n\n  return formatPermissionResponse('allow', platform, input);\n}\n\nfunction buildDecisionDetails(\n  toolName: string,\n  input: Record<string, unknown>,\n  result: Record<string, unknown>,\n  platform: string,\n): { target?: string; targetLabel?: string; details: PermissionDecisionDetail[] } {\n  const details: PermissionDecisionDetail[] = [];\n\n  if (platform) {\n    details.push({ label: 'Platform', value: platform, monospace: true });\n  }\n\n  if (toolName === 'Bash' && typeof input.command === 'string' && input.command !== '') {\n    details.push({ label: 'Command', value: input.command, monospace: true });\n  }\n\n  const targetDescriptors: Array<{ key: string; label: string; monospace?: boolean }> = [\n    { key: 'file_path', label: 'File', monospace: true },\n    { key: 'path', label: 'Path', monospace: true },\n    { key: 'url', label: 'URL' },\n    { key: 'pattern', label: 'Pattern', monospace: true },\n    { key: 'query', label: 'Query' },\n    { key: 'element_name', label: 'Element', monospace: true },\n    { key: 'request_id', label: 'Request', monospace: true },\n  ];\n\n  let target: string | undefined;\n  let targetLabel: string | undefined;\n\n  for (const descriptor of targetDescriptors) {\n    const value = input[descriptor.key];\n    if (typeof value !== 'string' || value === '') {\n      continue;\n    }\n\n    target = value;\n    targetLabel = descriptor.label;\n    details.push({ label: descriptor.label, value, monospace: descriptor.monospace });\n    break;\n  }\n\n  const matchedPattern = extractString(result, ['matched_pattern', 'matchedPattern'], '');\n  if (matchedPattern !== '') {\n    details.push({ label: 'Matched Pattern', value: matchedPattern, monospace: true });\n  }\n\n  const policySource = extractString(result, ['policy_source', 'policySource'], '');\n  if (policySource !== '') {\n    details.push({ label: 'Policy Source', value: policySource, monospace: true });\n  }\n\n  return { target, targetLabel, details };\n}\n\nfunction createPermissionDecisionTracker(bufferSize = DECISION_BUFFER_SIZE): PermissionDecisionTracker {\n  const recentDecisions: PermissionDecision[] = [];\n  let decisionCounter = 0;\n\n  return {\n    trackDecision(\n      sessionId: string | undefined,\n      toolName: string,\n      input: Record<string, unknown>,\n      result: Record<string, unknown>,\n      platform: string,\n    ): void {\n      const detailState = buildDecisionDetails(toolName, input, result, platform);\n      const entry: PermissionDecision = {\n        id: `d-${++decisionCounter}`,\n        timestamp: new Date().toISOString(),\n        ...(sessionId ? { session_id: sessionId } : {}),\n        tool_name: toolName,\n        command: toolName === 'Bash' && typeof input?.command === 'string' ? input.command : undefined,\n        decision: extractDecision(result),\n        reason: extractReason(result),\n        platform,\n        target: detailState.target,\n        targetLabel: detailState.targetLabel,\n        details: detailState.details,\n      };\n      recentDecisions.unshift(entry);\n      if (recentDecisions.length > bufferSize) {\n        recentDecisions.length = bufferSize;\n      }\n    },\n    getRecentDecisions(): PermissionDecision[] {\n      return recentDecisions;\n    },\n  };\n}\n\nfunction mergeRuleArrays(...sources: unknown[]): string[] {\n  const merged = new Set<string>();\n  for (const source of sources) {\n    if (!Array.isArray(source)) continue;\n    for (const entry of source) {\n      if (typeof entry === 'string' && entry !== '') {\n        merged.add(entry);\n      }\n    }\n  }\n  return Array.from(merged);\n}\n\nfunction normalizePolicyElements(elements: Array<Record<string, unknown>>): Array<Record<string, unknown>> {\n  return elements.map((element) => ({\n    ...element,\n    element_name: resolveElementName(element),\n    allowRules: mergeRuleArrays(element.allowPatterns, element.allowOperations),\n    confirmRules: mergeRuleArrays(element.confirmPatterns, element.confirmOperations),\n    denyRules: mergeRuleArrays(element.denyPatterns, element.denyOperations),\n    invalidGatekeeperPolicy: !!element.invalidGatekeeperPolicy,\n    invalidGatekeeperMessage: typeof element.invalidGatekeeperMessage === 'string' ? element.invalidGatekeeperMessage : undefined,\n  }));\n}\n\nfunction resolveElementName(element: Record<string, unknown>): string {\n  if (typeof element.element_name === 'string') return element.element_name;\n  if (typeof element.name === 'string') return element.name;\n  return '';\n}\n\n/** Helper to extract single result from MCP-AQL batch response */\nfunction asSingleResult(results: unknown): { success: boolean; data?: unknown; error?: string } {\n  if (Array.isArray(results)) return results[0] || { success: false, error: 'Empty result' };\n  return results as { success: boolean; data?: unknown; error?: string };\n}\n\nfunction extractKnownPolicySessions(elements: Array<Record<string, unknown>>): KnownPolicySession[] {\n  const seen = new Set<string>();\n  const knownSessions: KnownPolicySession[] = [];\n\n  for (const element of elements) {\n    const sessionIds = Array.isArray(element.sessionIds) ? element.sessionIds : [];\n    for (const sessionId of sessionIds) {\n      if (typeof sessionId !== 'string' || sessionId === '' || seen.has(sessionId)) {\n        continue;\n      }\n\n      seen.add(sessionId);\n      knownSessions.push({\n        sessionId,\n        displayName: sessionId,\n        source: 'policy',\n      });\n    }\n  }\n\n  return knownSessions.sort((a, b) => a.sessionId.localeCompare(b.sessionId));\n}\n\nasync function selfHealLatestPermissionPortFile(port: number | undefined): Promise<void> {\n  if (typeof port !== 'number' || !Number.isInteger(port) || port <= 0) {\n    return;\n  }\n\n  try {\n    await ensureLatestPortFile(port);\n  } catch (err) {\n    logger.debug('[WebUI/Gateway] Could not refresh permission-server.port', {\n      error: err instanceof Error ? err.message : String(err),\n      port,\n    });\n  }\n}\n\nasync function resolveInstalledPermissionAuthorityHosts(\n  homeDir: string,\n  authorityState: Awaited<ReturnType<typeof readPermissionAuthorityState>>,\n): Promise<PermissionAuthorityHost[]> {\n  const installedStatuses = await Promise.all(\n    PERMISSION_AUTHORITY_HOSTS.map(async (host) => ({\n      host,\n      status: await getPermissionHookStatusAsync(homeDir, host),\n    })),\n  );\n\n  const installedHosts = installedStatuses\n    .filter(({ status }) => status.installed)\n    .map(({ host }) => host);\n\n  const persistedHosts = Object.keys(authorityState.hosts || {}).filter((host): host is PermissionAuthorityHost =>\n    (PERMISSION_AUTHORITY_HOSTS as readonly string[]).includes(host),\n  );\n\n  return Array.from(new Set(installedHosts.concat(persistedHosts)));\n}\n\n/**\n * Register permission-related routes on a gateway router.\n * Must be called with the MCP-AQL handler for policy evaluation.\n */\nexport interface RegisterPermissionRoutesOptions {\n  homeDir?: string;\n}\n\nexport function registerPermissionRoutes(\n  router: Router,\n  handler: MCPAQLHandler,\n  options: RegisterPermissionRoutesOptions = {},\n): void {\n  const authorityHomeDir = options.homeDir ?? homedir();\n  const decisionTracker = createPermissionDecisionTracker();\n  /**\n   * POST /api/evaluate_permission\n   * Permission evaluation endpoint for PreToolUse hooks.\n   * Routes through evaluate_permission MCP-AQL READ operation.\n   * Fail-open: returns allow on any error to avoid blocking the user.\n   */\n  const permissionLimiter = new SlidingWindowRateLimiter(\n    PERMISSION_ROUTE_RATE_LIMIT_REQUESTS,\n    PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS,\n  );\n  router.post('/evaluate_permission', express.json(), async (req, res) => {\n    await selfHealLatestPermissionPortFile(req.socket.localPort);\n\n    const body = req.body as {\n      tool_name?: string;\n      input?: Record<string, unknown>;\n      platform?: string;\n      session_id?: string;\n    };\n    const platform = typeof body.platform === 'string' ? body.platform.normalize('NFC') : 'claude_code';\n\n    if (!permissionLimiter.tryAcquire()) {\n      res.json(formatPermissionResponse('allow', platform, {})); // fail open on rate limit\n      return;\n    }\n\n    // Unicode normalization (NFC) on string inputs to prevent homograph attacks\n    const tool_name = typeof body.tool_name === 'string' ? body.tool_name.normalize('NFC') : undefined;\n    const session_id = typeof body.session_id === 'string' ? body.session_id.normalize('NFC') : undefined;\n    const input = body.input;\n\n    if (!tool_name) {\n      res.json(formatPermissionResponse('allow', platform, input || {})); // fail open on bad input\n      return;\n    }\n\n    const startMs = Date.now();\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'evaluate_permission',\n        params: {\n          tool_name,\n          input: input || {},\n          platform,\n          ...(session_id ? { session_id } : {}),\n        },\n      }));\n      const elapsedMs = Date.now() - startMs;\n\n      if (!opResult.success) {\n        logger.warn(`[WebUI/Gateway] evaluate_permission failed (${elapsedMs}ms): ${opResult.error}`);\n        res.json(formatPermissionResponse('allow', platform, input || {})); // fail open\n        return;\n      }\n\n      const rawResult = opResult.data as Record<string, unknown>;\n      const responseData = normalizePermissionResponseForPlatform(\n        platform,\n        input || {},\n        rawResult,\n      );\n      const trackedResult = { ...rawResult, ...responseData };\n      const decision = extractDecision(responseData);\n      logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);\n\n      // Track decision for live dashboard feed\n      decisionTracker.trackDecision(session_id, tool_name, input || {}, trackedResult, platform);\n\n      res.json(responseData);\n    } catch (err) {\n      const elapsedMs = Date.now() - startMs;\n      logger.error(`[WebUI/Gateway] evaluate_permission error (${elapsedMs}ms):`, err);\n      res.json(formatPermissionResponse('allow', platform, input || {})); // fail open\n    }\n  });\n\n  /**\n   * GET /api/permissions/status\n   * Returns current permission policies and recent decisions\n   * for the live permissions dashboard.\n   */\n  router.get('/permissions/status', async (req, res) => {\n    try {\n      await selfHealLatestPermissionPortFile(req.socket.localPort);\n\n      const sessionId = typeof req.query['sessionId'] === 'string' && req.query['sessionId']\n        ? req.query['sessionId']\n        : undefined;\n\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'get_effective_cli_policies',\n        params: {\n          reporting_scope: 'dashboard',\n          ...(sessionId ? { session_id: sessionId } : {}),\n        },\n      }));\n\n      if (!opResult.success) {\n        res.status(500).json({ error: opResult.error || 'Failed to get policies' });\n        return;\n      }\n\n      const data = opResult.data as Record<string, unknown>;\n      const authorityState = await readPermissionAuthorityState(authorityHomeDir);\n      const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);\n      const elements = normalizePolicyElements((data.elements || []) as Array<Record<string, unknown>>);\n\n      const denyPatterns = (data.combinedDenyPatterns as string[] | undefined) ?? [];\n      const allowPatterns = (data.combinedAllowPatterns as string[] | undefined) ?? [];\n      const confirmPatterns = (data.combinedConfirmPatterns as string[] | undefined) ?? [];\n      const denyOperations = (data.combinedDenyOperations as string[] | undefined) ?? [];\n      const allowOperations = (data.combinedAllowOperations as string[] | undefined) ?? [];\n      const confirmOperations = (data.combinedConfirmOperations as string[] | undefined) ?? [];\n\n      res.json({\n        ...(sessionId ? { sessionId } : {}),\n        activeElementCount: data.activeElementCount,\n        hasAllowlist: data.hasAllowlist,\n        denyPatterns,\n        allowPatterns,\n        confirmPatterns,\n        denyOperations,\n        allowOperations,\n        confirmOperations,\n        denyRules: mergeRuleArrays(denyPatterns, denyOperations),\n        allowRules: mergeRuleArrays(allowPatterns, allowOperations),\n        confirmRules: mergeRuleArrays(confirmPatterns, confirmOperations),\n        elements,\n        knownSessions: extractKnownPolicySessions(elements),\n        permissionPromptActive: data.permissionPromptActive,\n        hookInstalled: data.hookInstalled,\n        hookHost: data.hookHost,\n        authority: authorityState,\n        authoritySupportedHosts: installedAuthorityHosts,\n        authoritySupportedModes: [...PERMISSION_AUTHORITY_MODES],\n        authorityAiMutable: false,\n        enforcementReady: data.enforcementReady,\n        invalidPolicyElementCount: data.invalidPolicyElementCount ?? 0,\n        advisory: data.advisory,\n        recentDecisions: decisionTracker.getRecentDecisions(),\n      });\n    } catch (err) {\n      logger.error('[WebUI/Gateway] permissions/status error:', err);\n      res.status(500).json({ error: 'Failed to get permission status' });\n    }\n  });\n\n  router.get('/permissions/authority', async (_req, res) => {\n    try {\n      const authorityState = await readPermissionAuthorityState(authorityHomeDir);\n      const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);\n      res.json({\n        ...authorityState,\n        supportedHosts: installedAuthorityHosts,\n        supportedModes: [...PERMISSION_AUTHORITY_MODES],\n        aiMutable: false,\n      });\n    } catch (err) {\n      logger.error('[WebUI/Gateway] permissions/authority error:', err);\n      res.status(500).json({ error: 'Failed to get permission authority' });\n    }\n  });\n\n  router.post('/permissions/authority', express.json(), async (req, res) => {\n    try {\n      const host = normalizeAuthorityHost(req.body?.host);\n      const mode = normalizeAuthorityMode(req.body?.mode);\n      const reason = typeof req.body?.reason === 'string' && req.body.reason.trim() !== ''\n        ? req.body.reason.trim()\n        : undefined;\n\n      if (!host || !mode) {\n        res.status(400).json({\n          error: 'host and mode are required',\n          supportedHosts: [...PERMISSION_AUTHORITY_HOSTS],\n          supportedModes: [...PERMISSION_AUTHORITY_MODES],\n        });\n        return;\n      }\n\n      let policies: Record<string, unknown> | undefined;\n      if (mode === 'authoritative') {\n        const policyResult = asSingleResult(await handler.handleRead({\n          operation: 'get_effective_cli_policies',\n          params: { reporting_scope: 'dashboard' },\n        }));\n\n        if (!policyResult.success) {\n          res.status(500).json({ error: policyResult.error || 'Failed to fetch effective policies' });\n          return;\n        }\n\n        policies = policyResult.data as Record<string, unknown>;\n      }\n\n      const authorityState = await setPermissionAuthorityMode({\n        host,\n        mode,\n        reason,\n        homeDir: authorityHomeDir,\n        policies: mode === 'authoritative'\n          ? {\n            combinedAllowPatterns: asStringArray(policies?.combinedAllowPatterns),\n            combinedConfirmPatterns: asStringArray(policies?.combinedConfirmPatterns),\n            combinedDenyPatterns: asStringArray(policies?.combinedDenyPatterns),\n          }\n          : undefined,\n      });\n\n      res.json({ success: true, authority: authorityState });\n    } catch (err) {\n      logger.error(\n        `[WebUI/Gateway] permissions/authority update error (host=${String(req.body?.host ?? '')}, mode=${String(req.body?.mode ?? '')}):`,\n        err,\n      );\n      res.status(500).json({\n        error: err instanceof Error ? err.message : 'Failed to update permission authority',\n      });\n    }\n  });\n}\n\nfunction normalizeAuthorityHost(value: unknown): PermissionAuthorityHost | null {\n  return typeof value === 'string' && PERMISSION_AUTHORITY_HOSTS.includes(value as PermissionAuthorityHost)\n    ? value as PermissionAuthorityHost\n    : null;\n}\n\nfunction normalizeAuthorityMode(value: unknown): PermissionAuthorityMode | null {\n  return typeof value === 'string' && PERMISSION_AUTHORITY_MODES.includes(value as PermissionAuthorityMode)\n    ? value as PermissionAuthorityMode\n    : null;\n}\n\nfunction asStringArray(value: unknown): string[] {\n  return Array.isArray(value) ? value.filter((entry): entry is string => typeof entry === 'string') : [];\n}\n"]}
@@ -9,6 +9,23 @@
9
9
  */
10
10
  import type { Request, Response } from 'express';
11
11
  import { type InstallPermissionHookResult } from '../../utils/permissionHooks.js';
12
+ /**
13
+ * Create setup handlers (Express 5 compatible — plain handler functions, not Router).
14
+ */
15
+ interface DetectResult {
16
+ installed: boolean;
17
+ configPath: string | null;
18
+ currentConfig?: Record<string, unknown>;
19
+ serverKey?: string;
20
+ }
21
+ /**
22
+ * Parse a TOML config file for a DollhouseMCP server entry.
23
+ *
24
+ * Detection prefers the canonical lowercase Codex section name first, then
25
+ * falls back to older Dollhouse-related section names so stale configs are
26
+ * still visible in the UI instead of being mistaken for a fresh install.
27
+ */
28
+ export declare function parseTomlConfig(raw: string): Omit<DetectResult, 'configPath'>;
12
29
  export declare function createSetupRoutes(opts?: {
13
30
  /** Override install-mcp runner. For testing only — prefix signals test-only use. */
14
31
  _runInstallMcp?: (client: string, version?: string) => Promise<string>;
@@ -89,4 +106,5 @@ export declare function ensureNvmLauncher(home?: string, nvmDirOverride?: string
89
106
  */
90
107
  export declare function patchConfigForNvmLauncher(client: string, wrapperPath: string, configPathOverride?: string): Promise<void>;
91
108
  export declare function installJsonMcpClientConfig(client: string, version?: string, configPathOverride?: string): Promise<string>;
109
+ export {};
92
110
  //# sourceMappingURL=setupRoutes.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAcjD,OAAO,EAAuD,KAAK,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AA2cvI,wBAAgB,iBAAiB,CAAC,IAAI,CAAC,EAAE;IACvC,oFAAoF;IACpF,cAAc,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;IACvE,4DAA4D;IAC5D,sBAAsB,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,KAAK,OAAO,CAAC,2BAA2B,CAAC,CAAC;IAClF,8DAA8D;IAC9D,cAAc,CAAC,EAAE,OAAO,CAAC;CAC1B,GAAG;IACF,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,mBAAmB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACpE,aAAa,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC9D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,oBAAoB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACrE,yBAAyB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC3E,CAsaA;AAmCD,iEAAiE;AACjE,MAAM,MAAM,iBAAiB,GAAG,SAAS,GAAG,gBAAgB,GAAG,QAAQ,CAAC;AAOxE;;;;;;GAMG;AACH,wBAAsB,wBAAwB,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,SAAY,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAsB3G;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAsB,0BAA0B,CAC9C,IAAI,SAAY,EAChB,kBAAkB,GAAE,CAAC,MAAM,EAAE,MAAM,KAAK,MAAM,GAAG,IAAoB,GACpE,OAAO,CAAC,IAAI,CAAC,CA4Bf;AAED;;;;;;GAMG;AACH,wBAAsB,YAAY,CAAC,IAAI,SAAY,GAAG,OAAO,CAAC,OAAO,CAAC,CAgBrE;AAyBD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAsB,iBAAiB,CAAC,IAAI,SAAY,EAAE,cAAc,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CA4ClG;AAoBD;;;;;;;;GAQG;AACH,wBAAsB,yBAAyB,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,kBAAkB,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CA4C/H;AAUD,wBAAsB,0BAA0B,CAC9C,MAAM,EAAE,MAAM,EACd,OAAO,CAAC,EAAE,MAAM,EAChB,kBAAkB,CAAC,EAAE,MAAM,GAC1B,OAAO,CAAC,MAAM,CAAC,CAmCjB"}
1
+ {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"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"}