@dollhousemcp/mcp-server 2.0.28 → 2.0.30

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 (39) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/generated/version.d.ts +2 -2
  3. package/dist/generated/version.js +3 -3
  4. package/dist/services/BuildInfoService.d.ts +9 -1
  5. package/dist/services/BuildInfoService.d.ts.map +1 -1
  6. package/dist/services/BuildInfoService.js +37 -3
  7. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
  8. package/dist/tools/portfolio/submitToPortfolioTool.js +4 -3
  9. package/dist/utils/permissionHookInstallers.d.ts +27 -0
  10. package/dist/utils/permissionHookInstallers.d.ts.map +1 -0
  11. package/dist/utils/permissionHookInstallers.js +465 -0
  12. package/dist/utils/permissionHookShared.d.ts +126 -0
  13. package/dist/utils/permissionHookShared.d.ts.map +1 -0
  14. package/dist/utils/permissionHookShared.js +388 -0
  15. package/dist/utils/permissionHookStatus.d.ts +10 -0
  16. package/dist/utils/permissionHookStatus.d.ts.map +1 -0
  17. package/dist/utils/permissionHookStatus.js +256 -0
  18. package/dist/utils/permissionHooks.d.ts +3 -73
  19. package/dist/utils/permissionHooks.d.ts.map +1 -1
  20. package/dist/utils/permissionHooks.js +4 -780
  21. package/dist/web/public/permissions.js +10 -0
  22. package/dist/web/public/setup.js +36 -0
  23. package/dist/web/routes/healthRoutes.d.ts +3 -0
  24. package/dist/web/routes/healthRoutes.d.ts.map +1 -1
  25. package/dist/web/routes/healthRoutes.js +22 -2
  26. package/dist/web/routes/permissionRoutes.d.ts +1 -0
  27. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  28. package/dist/web/routes/permissionRoutes.js +28 -7
  29. package/dist/web/routes/setupRoutes.d.ts +5 -1
  30. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  31. package/dist/web/routes/setupRoutes.js +28 -14
  32. package/dist/web/server.d.ts.map +1 -1
  33. package/dist/web/server.js +12 -1
  34. package/package.json +3 -1
  35. package/scripts/permission-hook-config.sh +67 -0
  36. package/scripts/pretooluse-dollhouse.sh +42 -13
  37. package/scripts/pretooluse-vscode.sh +23 -10
  38. package/scripts/pretooluse-windsurf.sh +6 -6
  39. package/server.json +2 -2
@@ -225,6 +225,16 @@
225
225
  } else if (data.permissionPromptActive) {
226
226
  hookDot.dataset.status = 'active';
227
227
  hookLabel.textContent = 'Prompt tool active';
228
+ } else if (data.hookNeedsRepair) {
229
+ hookDot.dataset.status = 'warning';
230
+ hookLabel.textContent = data.hookHost
231
+ ? `Hook needs repair (${data.hookHost})`
232
+ : 'Hook needs repair';
233
+ } else if (data.hookAutoRepaired) {
234
+ hookDot.dataset.status = 'active';
235
+ hookLabel.textContent = data.hookHost
236
+ ? `Hook refreshed (${data.hookHost})`
237
+ : 'Hook refreshed';
228
238
  } else if (data.hookInstalled) {
229
239
  hookDot.dataset.status = 'active';
230
240
  hookLabel.textContent = data.hookHost ? `Hook installed (${data.hookHost})` : 'Hook installed';
@@ -680,6 +680,13 @@ codex_hooks = true`;
680
680
  const updatePermissionInstallButton = (btn, detected) => {
681
681
  if (!btn || btn.classList.contains('is-success')) return;
682
682
 
683
+ if (detected?.hookNeedsRepair) {
684
+ btn.textContent = 'Repair hooks';
685
+ btn.disabled = false;
686
+ btn.classList.remove('is-match');
687
+ return;
688
+ }
689
+
683
690
  if (detected?.hookInstalled) {
684
691
  btn.textContent = 'Permissions enabled';
685
692
  btn.disabled = true;
@@ -1119,7 +1126,30 @@ codex_hooks = true`;
1119
1126
  return PERMISSION_SUPPORT_MATRIX[platformId];
1120
1127
  };
1121
1128
 
1129
+ const getHookRepairStatusCopy = (support, detected) => {
1130
+ if (detected?.hookNeedsRepair) {
1131
+ return {
1132
+ tone: 'warning',
1133
+ titleText: `${support.label} hook files need repair.`,
1134
+ messageText: 'DollhouseMCP detected stale local hook assets. Use Configure Now below to rewrite them, or reload the local server so the automatic repair pass can run again.',
1135
+ };
1136
+ }
1137
+
1138
+ if (detected?.hookAutoRepaired) {
1139
+ return {
1140
+ tone: 'info',
1141
+ titleText: `${support.label} hook files were refreshed automatically.`,
1142
+ messageText: 'The installed local hook assets were updated to match this release. Restart the client if it is already running.',
1143
+ };
1144
+ }
1145
+
1146
+ return null;
1147
+ };
1148
+
1122
1149
  const getFullNativePermissionStatusCopy = (support, detected) => {
1150
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1151
+ if (repairCopy) return repairCopy;
1152
+
1123
1153
  if (detected?.hookInstalled) {
1124
1154
  return {
1125
1155
  tone: 'info',
@@ -1145,6 +1175,9 @@ codex_hooks = true`;
1145
1175
 
1146
1176
  const getPartialPermissionStatusCopy = (support, detected) => {
1147
1177
  const activationLabel = support.label === 'Codex' ? 'Bash guardrails' : 'permission hooks';
1178
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1179
+ if (repairCopy) return repairCopy;
1180
+
1148
1181
  if (detected?.hookInstalled) {
1149
1182
  return {
1150
1183
  tone: 'info',
@@ -1185,6 +1218,9 @@ codex_hooks = true`;
1185
1218
  };
1186
1219
 
1187
1220
  const getManualPermissionStatusCopy = (support, detected) => {
1221
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1222
+ if (repairCopy) return repairCopy;
1223
+
1188
1224
  if (detected?.hookAssetsPrepared) {
1189
1225
  return {
1190
1226
  tone: 'info',
@@ -6,11 +6,14 @@
6
6
  import { Router } from 'express';
7
7
  import type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';
8
8
  import type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';
9
+ import { type PermissionHookAuditSummary } from '../../utils/permissionHooks.js';
9
10
  export interface HealthRoutesOptions {
10
11
  memorySink: MemoryLogSink;
11
12
  metricsSink?: MemoryMetricsSink;
12
13
  logClientCount: () => number;
13
14
  metricsClientCount: () => number;
15
+ permissionHooksHomeDir?: string;
16
+ getPermissionHookAuditSummaryFn?: (homeDir: string) => Promise<PermissionHookAuditSummary>;
14
17
  }
15
18
  export declare function createHealthRoutes(options: HealthRoutesOptions): Router;
16
19
  //# sourceMappingURL=healthRoutes.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"healthRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/healthRoutes.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,0CAA0C,CAAC;AAElF,MAAM,WAAW,mBAAmB;IAClC,UAAU,EAAE,aAAa,CAAC;IAC1B,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,cAAc,EAAE,MAAM,MAAM,CAAC;IAC7B,kBAAkB,EAAE,MAAM,MAAM,CAAC;CAClC;AAED,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,mBAAmB,GAAG,MAAM,CAyBvE"}
1
+ {"version":3,"file":"healthRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/healthRoutes.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAGjC,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,0CAA0C,CAAC;AAClF,OAAO,EAGL,KAAK,0BAA0B,EAChC,MAAM,gCAAgC,CAAC;AAExC,MAAM,WAAW,mBAAmB;IAClC,UAAU,EAAE,aAAa,CAAC;IAC1B,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,cAAc,EAAE,MAAM,MAAM,CAAC;IAC7B,kBAAkB,EAAE,MAAM,MAAM,CAAC;IACjC,sBAAsB,CAAC,EAAE,MAAM,CAAC;IAChC,+BAA+B,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,CAAC,0BAA0B,CAAC,CAAC;CAC5F;AAED,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,mBAAmB,GAAG,MAAM,CA6CvE"}
@@ -4,10 +4,12 @@
4
4
  * GET /api/health — Aggregated health: uptime, sink stats, SSE client counts
5
5
  */
6
6
  import { Router } from 'express';
7
+ import { homedir } from 'node:os';
8
+ import { getPermissionHookAuditSummary, summarizePermissionHookHealth, } from '../../utils/permissionHooks.js';
7
9
  export function createHealthRoutes(options) {
8
10
  const router = Router();
9
11
  const startTime = Date.now();
10
- router.get('/health', (_req, res) => {
12
+ router.get('/health', async (_req, res) => {
11
13
  const health = {
12
14
  status: 'ok',
13
15
  uptime: Math.floor((Date.now() - startTime) / 1000),
@@ -22,8 +24,26 @@ export function createHealthRoutes(options) {
22
24
  sseClients: options.metricsClientCount(),
23
25
  };
24
26
  }
27
+ try {
28
+ const permissionHookAudit = await (options.getPermissionHookAuditSummaryFn ?? getPermissionHookAuditSummary)(options.permissionHooksHomeDir ?? homedir());
29
+ health['permissionHooks'] = {
30
+ ...summarizePermissionHookHealth(permissionHookAudit),
31
+ installedHosts: permissionHookAudit.installedHosts,
32
+ currentHosts: permissionHookAudit.currentHosts,
33
+ needsRepairHosts: permissionHookAudit.needsRepairHosts,
34
+ lastStartupRepair: permissionHookAudit.lastStartupRepair,
35
+ };
36
+ }
37
+ catch (error) {
38
+ const message = error instanceof Error ? error.message : String(error);
39
+ health['status'] = 'degraded';
40
+ health['permissionHooks'] = {
41
+ status: 'error',
42
+ message: `Failed to audit permission hook status: ${message}`,
43
+ };
44
+ }
25
45
  res.json(health);
26
46
  });
27
47
  return router;
28
48
  }
29
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVhbHRoUm91dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3dlYi9yb3V0ZXMvaGVhbHRoUm91dGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFFSCxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBWWpDLE1BQU0sVUFBVSxrQkFBa0IsQ0FBQyxPQUE0QjtJQUM3RCxNQUFNLE1BQU0sR0FBRyxNQUFNLEVBQUUsQ0FBQztJQUN4QixNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsR0FBRyxFQUFFLENBQUM7SUFFN0IsTUFBTSxDQUFDLEdBQUcsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxJQUFhLEVBQUUsR0FBYSxFQUFFLEVBQUU7UUFDckQsTUFBTSxNQUFNLEdBQTRCO1lBQ3RDLE1BQU0sRUFBRSxJQUFJO1lBQ1osTUFBTSxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLEdBQUcsU0FBUyxDQUFDLEdBQUcsSUFBSSxDQUFDO1lBQ25ELElBQUksRUFBRTtnQkFDSixLQUFLLEVBQUUsT0FBTyxDQUFDLFVBQVUsQ0FBQyxRQUFRLEVBQUU7Z0JBQ3BDLFVBQVUsRUFBRSxPQUFPLENBQUMsY0FBYyxFQUFFO2FBQ3JDO1NBQ0YsQ0FBQztRQUVGLElBQUksT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ3hCLE1BQU0sQ0FBQyxTQUFTLENBQUMsR0FBRztnQkFDbEIsS0FBSyxFQUFFLE9BQU8sQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFO2dCQUNyQyxVQUFVLEVBQUUsT0FBTyxDQUFDLGtCQUFrQixFQUFFO2FBQ3pDLENBQUM7UUFDSixDQUFDO1FBRUQsR0FBRyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNuQixDQUFDLENBQUMsQ0FBQztJQUVILE9BQU8sTUFBTSxDQUFDO0FBQ2hCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEhlYWx0aCBBUEkgcm91dGUgZm9yIHRoZSB1bmlmaWVkIGRldiBjb25zb2xlLlxuICpcbiAqIEdFVCAvYXBpL2hlYWx0aCDigJQgQWdncmVnYXRlZCBoZWFsdGg6IHVwdGltZSwgc2luayBzdGF0cywgU1NFIGNsaWVudCBjb3VudHNcbiAqL1xuXG5pbXBvcnQgeyBSb3V0ZXIgfSBmcm9tICdleHByZXNzJztcbmltcG9ydCB0eXBlIHsgUmVxdWVzdCwgUmVzcG9uc2UgfSBmcm9tICdleHByZXNzJztcbmltcG9ydCB0eXBlIHsgTWVtb3J5TG9nU2luayB9IGZyb20gJy4uLy4uL2xvZ2dpbmcvc2lua3MvTWVtb3J5TG9nU2luay5qcyc7XG5pbXBvcnQgdHlwZSB7IE1lbW9yeU1ldHJpY3NTaW5rIH0gZnJvbSAnLi4vLi4vbWV0cmljcy9zaW5rcy9NZW1vcnlNZXRyaWNzU2luay5qcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgSGVhbHRoUm91dGVzT3B0aW9ucyB7XG4gIG1lbW9yeVNpbms6IE1lbW9yeUxvZ1Npbms7XG4gIG1ldHJpY3NTaW5rPzogTWVtb3J5TWV0cmljc1Npbms7XG4gIGxvZ0NsaWVudENvdW50OiAoKSA9PiBudW1iZXI7XG4gIG1ldHJpY3NDbGllbnRDb3VudDogKCkgPT4gbnVtYmVyO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlSGVhbHRoUm91dGVzKG9wdGlvbnM6IEhlYWx0aFJvdXRlc09wdGlvbnMpOiBSb3V0ZXIge1xuICBjb25zdCByb3V0ZXIgPSBSb3V0ZXIoKTtcbiAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKTtcblxuICByb3V0ZXIuZ2V0KCcvaGVhbHRoJywgKF9yZXE6IFJlcXVlc3QsIHJlczogUmVzcG9uc2UpID0+IHtcbiAgICBjb25zdCBoZWFsdGg6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge1xuICAgICAgc3RhdHVzOiAnb2snLFxuICAgICAgdXB0aW1lOiBNYXRoLmZsb29yKChEYXRlLm5vdygpIC0gc3RhcnRUaW1lKSAvIDEwMDApLFxuICAgICAgbG9nczoge1xuICAgICAgICBzdGF0czogb3B0aW9ucy5tZW1vcnlTaW5rLmdldFN0YXRzKCksXG4gICAgICAgIHNzZUNsaWVudHM6IG9wdGlvbnMubG9nQ2xpZW50Q291bnQoKSxcbiAgICAgIH0sXG4gICAgfTtcblxuICAgIGlmIChvcHRpb25zLm1ldHJpY3NTaW5rKSB7XG4gICAgICBoZWFsdGhbJ21ldHJpY3MnXSA9IHtcbiAgICAgICAgc3RhdHM6IG9wdGlvbnMubWV0cmljc1NpbmsuZ2V0U3RhdHMoKSxcbiAgICAgICAgc3NlQ2xpZW50czogb3B0aW9ucy5tZXRyaWNzQ2xpZW50Q291bnQoKSxcbiAgICAgIH07XG4gICAgfVxuXG4gICAgcmVzLmpzb24oaGVhbHRoKTtcbiAgfSk7XG5cbiAgcmV0dXJuIHJvdXRlcjtcbn1cbiJdfQ==
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVhbHRoUm91dGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3dlYi9yb3V0ZXMvaGVhbHRoUm91dGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFFSCxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRWpDLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFHbEMsT0FBTyxFQUNMLDZCQUE2QixFQUM3Qiw2QkFBNkIsR0FFOUIsTUFBTSxnQ0FBZ0MsQ0FBQztBQVd4QyxNQUFNLFVBQVUsa0JBQWtCLENBQUMsT0FBNEI7SUFDN0QsTUFBTSxNQUFNLEdBQUcsTUFBTSxFQUFFLENBQUM7SUFDeEIsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO0lBRTdCLE1BQU0sQ0FBQyxHQUFHLENBQUMsU0FBUyxFQUFFLEtBQUssRUFBRSxJQUFhLEVBQUUsR0FBYSxFQUFFLEVBQUU7UUFDM0QsTUFBTSxNQUFNLEdBQTRCO1lBQ3RDLE1BQU0sRUFBRSxJQUFJO1lBQ1osTUFBTSxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLEdBQUcsU0FBUyxDQUFDLEdBQUcsSUFBSSxDQUFDO1lBQ25ELElBQUksRUFBRTtnQkFDSixLQUFLLEVBQUUsT0FBTyxDQUFDLFVBQVUsQ0FBQyxRQUFRLEVBQUU7Z0JBQ3BDLFVBQVUsRUFBRSxPQUFPLENBQUMsY0FBYyxFQUFFO2FBQ3JDO1NBQ0YsQ0FBQztRQUVGLElBQUksT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ3hCLE1BQU0sQ0FBQyxTQUFTLENBQUMsR0FBRztnQkFDbEIsS0FBSyxFQUFFLE9BQU8sQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFO2dCQUNyQyxVQUFVLEVBQUUsT0FBTyxDQUFDLGtCQUFrQixFQUFFO2FBQ3pDLENBQUM7UUFDSixDQUFDO1FBRUQsSUFBSSxDQUFDO1lBQ0gsTUFBTSxtQkFBbUIsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDLCtCQUErQixJQUFJLDZCQUE2QixDQUFDLENBQzFHLE9BQU8sQ0FBQyxzQkFBc0IsSUFBSSxPQUFPLEVBQUUsQ0FDNUMsQ0FBQztZQUNGLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHO2dCQUMxQixHQUFHLDZCQUE2QixDQUFDLG1CQUFtQixDQUFDO2dCQUNyRCxjQUFjLEVBQUUsbUJBQW1CLENBQUMsY0FBYztnQkFDbEQsWUFBWSxFQUFFLG1CQUFtQixDQUFDLFlBQVk7Z0JBQzlDLGdCQUFnQixFQUFFLG1CQUFtQixDQUFDLGdCQUFnQjtnQkFDdEQsaUJBQWlCLEVBQUUsbUJBQW1CLENBQUMsaUJBQWlCO2FBQ3pELENBQUM7UUFDSixDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE1BQU0sT0FBTyxHQUFHLEtBQUssWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUN2RSxNQUFNLENBQUMsUUFBUSxDQUFDLEdBQUcsVUFBVSxDQUFDO1lBQzlCLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHO2dCQUMxQixNQUFNLEVBQUUsT0FBTztnQkFDZixPQUFPLEVBQUUsMkNBQTJDLE9BQU8sRUFBRTthQUM5RCxDQUFDO1FBQ0osQ0FBQztRQUVELEdBQUcsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDbkIsQ0FBQyxDQUFDLENBQUM7SUFFSCxPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBIZWFsdGggQVBJIHJvdXRlIGZvciB0aGUgdW5pZmllZCBkZXYgY29uc29sZS5cbiAqXG4gKiBHRVQgL2FwaS9oZWFsdGgg4oCUIEFnZ3JlZ2F0ZWQgaGVhbHRoOiB1cHRpbWUsIHNpbmsgc3RhdHMsIFNTRSBjbGllbnQgY291bnRzXG4gKi9cblxuaW1wb3J0IHsgUm91dGVyIH0gZnJvbSAnZXhwcmVzcyc7XG5pbXBvcnQgdHlwZSB7IFJlcXVlc3QsIFJlc3BvbnNlIH0gZnJvbSAnZXhwcmVzcyc7XG5pbXBvcnQgeyBob21lZGlyIH0gZnJvbSAnbm9kZTpvcyc7XG5pbXBvcnQgdHlwZSB7IE1lbW9yeUxvZ1NpbmsgfSBmcm9tICcuLi8uLi9sb2dnaW5nL3NpbmtzL01lbW9yeUxvZ1NpbmsuanMnO1xuaW1wb3J0IHR5cGUgeyBNZW1vcnlNZXRyaWNzU2luayB9IGZyb20gJy4uLy4uL21ldHJpY3Mvc2lua3MvTWVtb3J5TWV0cmljc1NpbmsuanMnO1xuaW1wb3J0IHtcbiAgZ2V0UGVybWlzc2lvbkhvb2tBdWRpdFN1bW1hcnksXG4gIHN1bW1hcml6ZVBlcm1pc3Npb25Ib29rSGVhbHRoLFxuICB0eXBlIFBlcm1pc3Npb25Ib29rQXVkaXRTdW1tYXJ5LFxufSBmcm9tICcuLi8uLi91dGlscy9wZXJtaXNzaW9uSG9va3MuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEhlYWx0aFJvdXRlc09wdGlvbnMge1xuICBtZW1vcnlTaW5rOiBNZW1vcnlMb2dTaW5rO1xuICBtZXRyaWNzU2luaz86IE1lbW9yeU1ldHJpY3NTaW5rO1xuICBsb2dDbGllbnRDb3VudDogKCkgPT4gbnVtYmVyO1xuICBtZXRyaWNzQ2xpZW50Q291bnQ6ICgpID0+IG51bWJlcjtcbiAgcGVybWlzc2lvbkhvb2tzSG9tZURpcj86IHN0cmluZztcbiAgZ2V0UGVybWlzc2lvbkhvb2tBdWRpdFN1bW1hcnlGbj86IChob21lRGlyOiBzdHJpbmcpID0+IFByb21pc2U8UGVybWlzc2lvbkhvb2tBdWRpdFN1bW1hcnk+O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlSGVhbHRoUm91dGVzKG9wdGlvbnM6IEhlYWx0aFJvdXRlc09wdGlvbnMpOiBSb3V0ZXIge1xuICBjb25zdCByb3V0ZXIgPSBSb3V0ZXIoKTtcbiAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKTtcblxuICByb3V0ZXIuZ2V0KCcvaGVhbHRoJywgYXN5bmMgKF9yZXE6IFJlcXVlc3QsIHJlczogUmVzcG9uc2UpID0+IHtcbiAgICBjb25zdCBoZWFsdGg6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0ge1xuICAgICAgc3RhdHVzOiAnb2snLFxuICAgICAgdXB0aW1lOiBNYXRoLmZsb29yKChEYXRlLm5vdygpIC0gc3RhcnRUaW1lKSAvIDEwMDApLFxuICAgICAgbG9nczoge1xuICAgICAgICBzdGF0czogb3B0aW9ucy5tZW1vcnlTaW5rLmdldFN0YXRzKCksXG4gICAgICAgIHNzZUNsaWVudHM6IG9wdGlvbnMubG9nQ2xpZW50Q291bnQoKSxcbiAgICAgIH0sXG4gICAgfTtcblxuICAgIGlmIChvcHRpb25zLm1ldHJpY3NTaW5rKSB7XG4gICAgICBoZWFsdGhbJ21ldHJpY3MnXSA9IHtcbiAgICAgICAgc3RhdHM6IG9wdGlvbnMubWV0cmljc1NpbmsuZ2V0U3RhdHMoKSxcbiAgICAgICAgc3NlQ2xpZW50czogb3B0aW9ucy5tZXRyaWNzQ2xpZW50Q291bnQoKSxcbiAgICAgIH07XG4gICAgfVxuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHBlcm1pc3Npb25Ib29rQXVkaXQgPSBhd2FpdCAob3B0aW9ucy5nZXRQZXJtaXNzaW9uSG9va0F1ZGl0U3VtbWFyeUZuID8/IGdldFBlcm1pc3Npb25Ib29rQXVkaXRTdW1tYXJ5KShcbiAgICAgICAgb3B0aW9ucy5wZXJtaXNzaW9uSG9va3NIb21lRGlyID8/IGhvbWVkaXIoKSxcbiAgICAgICk7XG4gICAgICBoZWFsdGhbJ3Blcm1pc3Npb25Ib29rcyddID0ge1xuICAgICAgICAuLi5zdW1tYXJpemVQZXJtaXNzaW9uSG9va0hlYWx0aChwZXJtaXNzaW9uSG9va0F1ZGl0KSxcbiAgICAgICAgaW5zdGFsbGVkSG9zdHM6IHBlcm1pc3Npb25Ib29rQXVkaXQuaW5zdGFsbGVkSG9zdHMsXG4gICAgICAgIGN1cnJlbnRIb3N0czogcGVybWlzc2lvbkhvb2tBdWRpdC5jdXJyZW50SG9zdHMsXG4gICAgICAgIG5lZWRzUmVwYWlySG9zdHM6IHBlcm1pc3Npb25Ib29rQXVkaXQubmVlZHNSZXBhaXJIb3N0cyxcbiAgICAgICAgbGFzdFN0YXJ0dXBSZXBhaXI6IHBlcm1pc3Npb25Ib29rQXVkaXQubGFzdFN0YXJ0dXBSZXBhaXIsXG4gICAgICB9O1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zdCBtZXNzYWdlID0gZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpO1xuICAgICAgaGVhbHRoWydzdGF0dXMnXSA9ICdkZWdyYWRlZCc7XG4gICAgICBoZWFsdGhbJ3Blcm1pc3Npb25Ib29rcyddID0ge1xuICAgICAgICBzdGF0dXM6ICdlcnJvcicsXG4gICAgICAgIG1lc3NhZ2U6IGBGYWlsZWQgdG8gYXVkaXQgcGVybWlzc2lvbiBob29rIHN0YXR1czogJHttZXNzYWdlfWAsXG4gICAgICB9O1xuICAgIH1cblxuICAgIHJlcy5qc29uKGhlYWx0aCk7XG4gIH0pO1xuXG4gIHJldHVybiByb3V0ZXI7XG59XG4iXX0=
@@ -14,6 +14,7 @@ import type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';
14
14
  */
15
15
  export interface RegisterPermissionRoutesOptions {
16
16
  homeDir?: string;
17
+ autoRepairHookAssets?: boolean;
17
18
  }
18
19
  export declare function registerPermissionRoutes(router: Router, handler: MCPAQLHandler, options?: RegisterPermissionRoutesOptions): void;
19
20
  //# sourceMappingURL=permissionRoutes.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"permissionRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAG1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;AA8T7E;;;GAGG;AACH,MAAM,WAAW,+BAA+B;IAC9C,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,MAAM,EACd,OAAO,EAAE,aAAa,EACtB,OAAO,GAAE,+BAAoC,GAC5C,IAAI,CA+NN"}
1
+ {"version":3,"file":"permissionRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAG1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;AA8V7E;;;GAGG;AACH,MAAM,WAAW,+BAA+B;IAC9C,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,oBAAoB,CAAC,EAAE,OAAO,CAAC;CAChC;AAED,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,MAAM,EACd,OAAO,EAAE,aAAa,EACtB,OAAO,GAAE,+BAAoC,GAC5C,IAAI,CA6PN"}
@@ -11,7 +11,7 @@ import { homedir } from 'node:os';
11
11
  import { logger } from '../../utils/logger.js';
12
12
  import { formatPermissionResponse } from '../../handlers/mcp-aql/evaluatePermission.js';
13
13
  import { ensureLatestPortFile } from '../portDiscovery.js';
14
- import { getPermissionHookStatusAsync } from '../../utils/permissionHooks.js';
14
+ import { getLastPermissionHookStartupRepairSummary, getPermissionHookStatusAsync, reconcilePermissionHookStatus, summarizePermissionHookHealth, } from '../../utils/permissionHooks.js';
15
15
  import { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';
16
16
  import { PERMISSION_AUTHORITY_HOSTS, PERMISSION_AUTHORITY_MODES, readPermissionAuthorityState, setPermissionAuthorityMode, } from '../../utils/permissionAuthority.js';
17
17
  const PERMISSION_ROUTE_RATE_LIMIT_REQUESTS = 120;
@@ -212,10 +212,10 @@ async function selfHealLatestPermissionPortFile(port) {
212
212
  });
213
213
  }
214
214
  }
215
- async function resolveInstalledPermissionAuthorityHosts(homeDir, authorityState) {
215
+ async function resolveInstalledPermissionAuthorityHosts(homeDir, authorityState, autoRepairHookAssets) {
216
216
  const installedStatuses = await Promise.all(PERMISSION_AUTHORITY_HOSTS.map(async (host) => ({
217
217
  host,
218
- status: await getPermissionHookStatusAsync(homeDir, host),
218
+ status: await reconcilePermissionHookStatus(host, { homeDir, autoRepair: autoRepairHookAssets }),
219
219
  })));
220
220
  const installedHosts = installedStatuses
221
221
  .filter(({ status }) => status.installed)
@@ -225,6 +225,7 @@ async function resolveInstalledPermissionAuthorityHosts(homeDir, authorityState)
225
225
  }
226
226
  export function registerPermissionRoutes(router, handler, options = {}) {
227
227
  const authorityHomeDir = options.homeDir ?? homedir();
228
+ const autoRepairHookAssets = options.autoRepairHookAssets ?? process.env.NODE_ENV !== 'test';
228
229
  const decisionTracker = createPermissionDecisionTracker();
229
230
  /**
230
231
  * POST /api/evaluate_permission
@@ -305,7 +306,11 @@ export function registerPermissionRoutes(router, handler, options = {}) {
305
306
  }
306
307
  const data = opResult.data;
307
308
  const authorityState = await readPermissionAuthorityState(authorityHomeDir);
308
- const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);
309
+ const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState, autoRepairHookAssets);
310
+ const hookHost = typeof data.hookHost === 'string' ? data.hookHost : undefined;
311
+ const hookStatus = hookHost
312
+ ? await reconcilePermissionHookStatus(hookHost, { homeDir: authorityHomeDir, autoRepair: autoRepairHookAssets })
313
+ : await getPermissionHookStatusAsync(authorityHomeDir);
309
314
  const elements = normalizePolicyElements((data.elements || []));
310
315
  const denyPatterns = data.combinedDenyPatterns ?? [];
311
316
  const allowPatterns = data.combinedAllowPatterns ?? [];
@@ -313,6 +318,15 @@ export function registerPermissionRoutes(router, handler, options = {}) {
313
318
  const denyOperations = data.combinedDenyOperations ?? [];
314
319
  const allowOperations = data.combinedAllowOperations ?? [];
315
320
  const confirmOperations = data.combinedConfirmOperations ?? [];
321
+ const hookStartupRepair = getLastPermissionHookStartupRepairSummary();
322
+ const hookHealthHost = hookStatus.host ?? hookHost ?? 'managed-host';
323
+ const hookHealth = summarizePermissionHookHealth({
324
+ installedHosts: hookStatus.installed || hookStatus.assetsPrepared ? [hookHealthHost] : [],
325
+ currentHosts: hookStatus.assetsCurrent ? [hookHealthHost] : [],
326
+ repairedHosts: hookStatus.autoRepaired ? [hookHealthHost] : [],
327
+ needsRepairHosts: hookStatus.needsRepair ? [hookHealthHost] : [],
328
+ lastStartupRepair: hookStartupRepair,
329
+ });
316
330
  res.json({
317
331
  ...(sessionId ? { sessionId } : {}),
318
332
  activeElementCount: data.activeElementCount,
@@ -329,8 +343,15 @@ export function registerPermissionRoutes(router, handler, options = {}) {
329
343
  elements,
330
344
  knownSessions: extractKnownPolicySessions(elements),
331
345
  permissionPromptActive: data.permissionPromptActive,
332
- hookInstalled: data.hookInstalled,
346
+ hookInstalled: hookStatus.installed,
333
347
  hookHost: data.hookHost,
348
+ hookAssetsPrepared: hookStatus.assetsPrepared,
349
+ hookAssetsCurrent: hookStatus.assetsCurrent,
350
+ hookAutoRepaired: hookStatus.autoRepaired,
351
+ hookNeedsRepair: hookStatus.needsRepair,
352
+ hookRepairError: hookStatus.repairError,
353
+ hookHealth,
354
+ hookStartupRepair,
334
355
  authority: authorityState,
335
356
  authoritySupportedHosts: installedAuthorityHosts,
336
357
  authoritySupportedModes: [...PERMISSION_AUTHORITY_MODES],
@@ -349,7 +370,7 @@ export function registerPermissionRoutes(router, handler, options = {}) {
349
370
  router.get('/permissions/authority', async (_req, res) => {
350
371
  try {
351
372
  const authorityState = await readPermissionAuthorityState(authorityHomeDir);
352
- const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState);
373
+ const installedAuthorityHosts = await resolveInstalledPermissionAuthorityHosts(authorityHomeDir, authorityState, autoRepairHookAssets);
353
374
  res.json({
354
375
  ...authorityState,
355
376
  supportedHosts: installedAuthorityHosts,
@@ -425,4 +446,4 @@ function normalizeAuthorityMode(value) {
425
446
  function asStringArray(value) {
426
447
  return Array.isArray(value) ? value.filter((entry) => typeof entry === 'string') : [];
427
448
  }
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"]}
449
+ //# 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 {\n  getLastPermissionHookStartupRepairSummary,\n  getPermissionHookStatusAsync,\n  reconcilePermissionHookStatus,\n  summarizePermissionHookHealth,\n} 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\ntype PermissionDecisionTargetKey =\n  | 'file_path'\n  | 'path'\n  | 'url'\n  | 'pattern'\n  | 'query'\n  | 'element_name'\n  | 'request_id';\n\ninterface PermissionDecisionInput extends Record<string, unknown> {\n  command?: string;\n  file_path?: string;\n  path?: string;\n  url?: string;\n  pattern?: string;\n  query?: string;\n  element_name?: string;\n  request_id?: string;\n}\n\ninterface PermissionDecisionTargetDescriptor {\n  key: PermissionDecisionTargetKey;\n  label: 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: PermissionDecisionInput,\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: PermissionDecisionInput,\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: PermissionDecisionTargetDescriptor[] = [\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: PermissionDecisionInput,\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  autoRepairHookAssets: boolean,\n): Promise<PermissionAuthorityHost[]> {\n  const installedStatuses = await Promise.all(\n    PERMISSION_AUTHORITY_HOSTS.map(async (host) => ({\n      host,\n      status: await reconcilePermissionHookStatus(host, { homeDir, autoRepair: autoRepairHookAssets }),\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  autoRepairHookAssets?: boolean;\n}\n\nexport function registerPermissionRoutes(\n  router: Router,\n  handler: MCPAQLHandler,\n  options: RegisterPermissionRoutesOptions = {},\n): void {\n  const authorityHomeDir = options.homeDir ?? homedir();\n  const autoRepairHookAssets = options.autoRepairHookAssets ?? process.env.NODE_ENV !== 'test';\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?: PermissionDecisionInput;\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(\n        authorityHomeDir,\n        authorityState,\n        autoRepairHookAssets,\n      );\n      const hookHost = typeof data.hookHost === 'string' ? data.hookHost : undefined;\n      const hookStatus = hookHost\n        ? await reconcilePermissionHookStatus(hookHost, { homeDir: authorityHomeDir, autoRepair: autoRepairHookAssets })\n        : await getPermissionHookStatusAsync(authorityHomeDir);\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      const hookStartupRepair = getLastPermissionHookStartupRepairSummary();\n      const hookHealthHost = hookStatus.host ?? hookHost ?? 'managed-host';\n      const hookHealth = summarizePermissionHookHealth({\n        installedHosts: hookStatus.installed || hookStatus.assetsPrepared ? [hookHealthHost] : [],\n        currentHosts: hookStatus.assetsCurrent ? [hookHealthHost] : [],\n        repairedHosts: hookStatus.autoRepaired ? [hookHealthHost] : [],\n        needsRepairHosts: hookStatus.needsRepair ? [hookHealthHost] : [],\n        lastStartupRepair: hookStartupRepair,\n      });\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: hookStatus.installed,\n        hookHost: data.hookHost,\n        hookAssetsPrepared: hookStatus.assetsPrepared,\n        hookAssetsCurrent: hookStatus.assetsCurrent,\n        hookAutoRepaired: hookStatus.autoRepaired,\n        hookNeedsRepair: hookStatus.needsRepair,\n        hookRepairError: hookStatus.repairError,\n        hookHealth,\n        hookStartupRepair,\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(\n        authorityHomeDir,\n        authorityState,\n        autoRepairHookAssets,\n      );\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"]}
@@ -8,7 +8,7 @@
8
8
  * and command arguments are hardcoded — no user-supplied shell input.
9
9
  */
10
10
  import type { Request, Response } from 'express';
11
- import { type InstallPermissionHookResult } from '../../utils/permissionHooks.js';
11
+ import { type InstallPermissionHookResult, type PermissionHookStatus } from '../../utils/permissionHooks.js';
12
12
  /**
13
13
  * Create setup handlers (Express 5 compatible — plain handler functions, not Router).
14
14
  */
@@ -31,6 +31,10 @@ export declare function createSetupRoutes(opts?: {
31
31
  _runInstallMcp?: (client: string, version?: string) => Promise<string>;
32
32
  /** Override permission hook installer. For testing only. */
33
33
  _installPermissionHook?: (client: string) => Promise<InstallPermissionHookResult>;
34
+ /** Override permission hook status reconciler. For testing only. */
35
+ _reconcilePermissionHookStatus?: (client: string) => Promise<PermissionHookStatus>;
36
+ /** Enable automatic hook asset repair during detect. Defaults off in tests. */
37
+ _autoRepairPermissionHooksOnDetect?: boolean;
34
38
  /** Skip the sliding-window rate limiter. For testing only. */
35
39
  _skipRateLimit?: boolean;
36
40
  }): {
@@ -1 +1 @@
1
- {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"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"}