@dollhousemcp/mcp-server 2.0.17 → 2.0.18
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +6 -0
- package/dist/elements/BaseElement.d.ts +1 -0
- package/dist/elements/BaseElement.d.ts.map +1 -1
- package/dist/elements/BaseElement.js +7 -1
- package/dist/elements/agents/AgentManager.js +2 -2
- package/dist/elements/base/ElementFileOperations.js +2 -2
- package/dist/elements/ensembles/EnsembleManager.js +3 -3
- package/dist/elements/memories/MemoryManager.js +2 -2
- package/dist/elements/skills/SkillManager.js +2 -2
- package/dist/elements/templates/TemplateManager.js +2 -2
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
- package/dist/handlers/ElementCRUDHandler.js +3 -2
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts +1 -0
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/MCPAQLHandler.js +35 -14
- package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts +9 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.js +63 -4
- package/dist/handlers/strategies/AgentActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/AgentActivationStrategy.js +5 -1
- package/dist/handlers/strategies/BaseActivationStrategy.d.ts +1 -0
- package/dist/handlers/strategies/BaseActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/BaseActivationStrategy.js +15 -1
- package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/EnsembleActivationStrategy.js +5 -1
- package/dist/handlers/strategies/MemoryActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/MemoryActivationStrategy.js +5 -1
- package/dist/handlers/strategies/PersonaActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/PersonaActivationStrategy.js +5 -1
- package/dist/handlers/strategies/SkillActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/SkillActivationStrategy.js +5 -1
- package/dist/handlers/strategies/TemplateActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/TemplateActivationStrategy.js +7 -2
- package/dist/persona/PersonaElement.js +2 -2
- package/dist/services/SerializationService.d.ts.map +1 -1
- package/dist/services/SerializationService.js +7 -1
- package/dist/types/elements/IElement.d.ts +9 -0
- package/dist/types/elements/IElement.d.ts.map +1 -1
- package/dist/types/elements/IElement.js +1 -1
- package/dist/web/public/permissions.css +190 -2
- package/dist/web/public/permissions.js +171 -30
- package/dist/web/public/setup.js +131 -60
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +77 -5
- package/dist/web/routes/setupRoutes.d.ts.map +1 -1
- package/dist/web/routes/setupRoutes.js +16 -2
- package/package.json +1 -1
- package/scripts/pretooluse-dollhouse.sh +39 -1
- package/server.json +2 -2
package/dist/web/public/setup.js
CHANGED
|
@@ -18,13 +18,13 @@
|
|
|
18
18
|
const PLATFORMS = [
|
|
19
19
|
// Claude Desktop & Claude Code panels are handwritten in HTML (unique structure)
|
|
20
20
|
{ id: 'claude-desktop', rootKey: 'mcpServers' },
|
|
21
|
-
{ id: 'claude-code', rootKey: 'mcpServers', cli: 'claude',
|
|
21
|
+
{ id: 'claude-code', rootKey: 'mcpServers', cli: 'claude', hookCommand: `bash ${HOOK_BASE_SCRIPT_PATH}`, hookConfigPath: '<code>~/.claude/settings.json</code>' },
|
|
22
22
|
// These panels are generated from this data by renderGeneratedPanels()
|
|
23
|
-
{ id: 'cursor', rootKey: 'mcpServers', installClient: 'cursor', openClient: 'cursor', configPath: '<code>.cursor/mcp.json</code> in your project, or <code>~/.cursor/mcp.json</code> for all projects', hint: 'Or configure via Settings > MCP Servers in the Cursor UI.',
|
|
24
|
-
{ id: 'vscode', rootKey: 'servers', installClient: 'vscode', configPath: '<code>.vscode/mcp.json</code> in your workspace', hint: 'VS Code uses <code>"servers"</code>, not <code>"mcpServers"</code>.',
|
|
25
|
-
{ id: 'codex', rootKey: 'mcpServers', installClient: 'codex', openClient: 'codex', cli: 'codex', toml: true, tomlPath: '<code>~/.codex/config.toml</code> (Codex uses TOML, not JSON)',
|
|
26
|
-
{ id: 'gemini', rootKey: 'mcpServers', installClient: 'gemini-cli', openClient: 'gemini-cli', cli: 'gemini', configPath: '<code>~/.gemini/settings.json</code> or <code>.gemini/settings.json</code> in your project',
|
|
27
|
-
{ id: 'windsurf', rootKey: 'mcpServers', installClient: 'windsurf', openClient: 'windsurf', configPath: '<code>~/.codeium/windsurf/mcp_config.json</code>', hint: 'Or click the MCPs icon in the Cascade panel > Configure.',
|
|
23
|
+
{ id: 'cursor', rootKey: 'mcpServers', installClient: 'cursor', openClient: 'cursor', configPath: '<code>.cursor/mcp.json</code> in your project, or <code>~/.cursor/mcp.json</code> for all projects', hint: 'Or configure via Settings > MCP Servers in the Cursor UI.', hookCommand: `bash ${HOOKS_DIR}/pretooluse-cursor.sh`, hookConfigPath: '<code>.cursor/hooks.json</code> in your project, or <code>~/.cursor/hooks.json</code> for all projects' },
|
|
24
|
+
{ id: 'vscode', rootKey: 'servers', installClient: 'vscode', configPath: '<code>.vscode/mcp.json</code> in your workspace', hint: 'VS Code uses <code>"servers"</code>, not <code>"mcpServers"</code>.', hookCommand: `bash ${HOOKS_DIR}/pretooluse-vscode.sh`, hookConfigPath: '<code>~/.copilot/hooks/dollhouse-permissions.json</code> plus <code>chat.hookFilesLocations</code> in VS Code user settings' },
|
|
25
|
+
{ id: 'codex', rootKey: 'mcpServers', installClient: 'codex', openClient: 'codex', cli: 'codex', toml: true, tomlPath: '<code>~/.codex/config.toml</code> (Codex uses TOML, not JSON)', hookCommand: `bash ${HOOKS_DIR}/pretooluse-codex.sh`, hookConfigPath: '<code>~/.codex/hooks.json</code> and <code>~/.codex/config.toml</code>' },
|
|
26
|
+
{ id: 'gemini', rootKey: 'mcpServers', installClient: 'gemini-cli', openClient: 'gemini-cli', cli: 'gemini', configPath: '<code>~/.gemini/settings.json</code> or <code>.gemini/settings.json</code> in your project', hookCommand: `bash ${HOOKS_DIR}/pretooluse-gemini.sh`, hookConfigPath: '<code>~/.gemini/settings.json</code> or <code>.gemini/settings.json</code> in your project' },
|
|
27
|
+
{ id: 'windsurf', rootKey: 'mcpServers', installClient: 'windsurf', openClient: 'windsurf', configPath: '<code>~/.codeium/windsurf/mcp_config.json</code>', hint: 'Or click the MCPs icon in the Cascade panel > Configure.', hookCommand: `bash ${HOOKS_DIR}/pretooluse-windsurf.sh`, hookConfigPath: '<code>~/.codeium/windsurf/hooks.json</code> or <code>.windsurf/hooks.json</code> in your project' },
|
|
28
28
|
{ id: 'cline', rootKey: 'mcpServers', installClient: 'cline', openClient: 'cline', configPath: 'Cline stores MCP servers in <code>cline_mcp_settings.json</code> inside its extension settings. Use Configure Now or open the file directly.' },
|
|
29
29
|
{ id: 'lmstudio', rootKey: 'mcpServers', installClient: 'lmstudio', openClient: 'lmstudio', configPath: '<code>~/.lmstudio/mcp.json</code> (or open via Program tab > Install > Edit mcp.json)', hint: 'Restart LM Studio after saving.' },
|
|
30
30
|
];
|
|
@@ -888,6 +888,7 @@ codex_hooks = true`;
|
|
|
888
888
|
'claude': 'claude-desktop',
|
|
889
889
|
'claude-code': 'claude-code',
|
|
890
890
|
'cursor': 'cursor',
|
|
891
|
+
'cline': 'cline',
|
|
891
892
|
'windsurf': 'windsurf',
|
|
892
893
|
'lmstudio': 'lmstudio',
|
|
893
894
|
'gemini-cli': 'gemini',
|
|
@@ -965,32 +966,29 @@ codex_hooks = true`;
|
|
|
965
966
|
}
|
|
966
967
|
};
|
|
967
968
|
|
|
968
|
-
const
|
|
969
|
-
'claude-desktop':
|
|
970
|
-
|
|
971
|
-
|
|
972
|
-
|
|
973
|
-
|
|
974
|
-
|
|
975
|
-
|
|
976
|
-
cline: 'Cline',
|
|
977
|
-
lmstudio: 'LM Studio',
|
|
978
|
-
};
|
|
979
|
-
|
|
980
|
-
const VERIFIED_PERMISSION_PLATFORMS = {
|
|
969
|
+
const PERMISSION_SUPPORT_MATRIX = {
|
|
970
|
+
'claude-desktop': {
|
|
971
|
+
label: 'Claude Desktop',
|
|
972
|
+
supportLevel: 'unsupported',
|
|
973
|
+
statusTag: 'coming soon',
|
|
974
|
+
badgeClass: 'unsupported',
|
|
975
|
+
limitation: 'Claude Desktop can host DollhouseMCP, but this release does not ship a native permissions setup flow for it yet.',
|
|
976
|
+
},
|
|
981
977
|
'claude-code': {
|
|
982
978
|
label: 'Claude Code',
|
|
979
|
+
supportLevel: 'full_native',
|
|
983
980
|
statusTag: 'claude code',
|
|
981
|
+
badgeClass: 'verified',
|
|
984
982
|
configPath: '<code>~/.claude/settings.json</code>',
|
|
985
983
|
scriptPath: HOOK_BASE_SCRIPT_PATH,
|
|
986
984
|
settingsBlock: CLAUDE_CODE_HOOK_SETTINGS,
|
|
985
|
+
limitation: 'Claude Code has the full native permission-hook path in this release.',
|
|
987
986
|
},
|
|
988
|
-
};
|
|
989
|
-
|
|
990
|
-
const PARTIAL_PERMISSION_PLATFORMS = {
|
|
991
987
|
gemini: {
|
|
992
988
|
label: 'Gemini CLI',
|
|
989
|
+
supportLevel: 'partial_native',
|
|
993
990
|
statusTag: 'allow / deny',
|
|
991
|
+
badgeClass: 'manual',
|
|
994
992
|
configPath: '<code>~/.gemini/settings.json</code> or <code>.gemini/settings.json</code> in your project',
|
|
995
993
|
scriptPath: `${HOOKS_DIR}/pretooluse-gemini.sh`,
|
|
996
994
|
settingsBlock: GEMINI_HOOK_SETTINGS,
|
|
@@ -998,7 +996,9 @@ codex_hooks = true`;
|
|
|
998
996
|
},
|
|
999
997
|
cursor: {
|
|
1000
998
|
label: 'Cursor',
|
|
999
|
+
supportLevel: 'partial_native',
|
|
1001
1000
|
statusTag: 'native hooks',
|
|
1001
|
+
badgeClass: 'manual',
|
|
1002
1002
|
configPath: '<code>.cursor/hooks.json</code> in your project, or <code>~/.cursor/hooks.json</code> for all projects',
|
|
1003
1003
|
scriptPath: `${HOOKS_DIR}/pretooluse-cursor.sh`,
|
|
1004
1004
|
settingsBlock: CURSOR_HOOK_SETTINGS,
|
|
@@ -1006,7 +1006,9 @@ codex_hooks = true`;
|
|
|
1006
1006
|
},
|
|
1007
1007
|
vscode: {
|
|
1008
1008
|
label: 'VS Code',
|
|
1009
|
+
supportLevel: 'partial_native',
|
|
1009
1010
|
statusTag: 'native hooks',
|
|
1011
|
+
badgeClass: 'manual',
|
|
1010
1012
|
configPath: '<code>~/.copilot/hooks/dollhouse-permissions.json</code> and VS Code user settings',
|
|
1011
1013
|
scriptPath: `${HOOKS_DIR}/pretooluse-vscode.sh`,
|
|
1012
1014
|
settingsBlock: VSCODE_HOOK_SETTINGS,
|
|
@@ -1017,7 +1019,9 @@ codex_hooks = true`;
|
|
|
1017
1019
|
},
|
|
1018
1020
|
windsurf: {
|
|
1019
1021
|
label: 'Windsurf',
|
|
1022
|
+
supportLevel: 'partial_native',
|
|
1020
1023
|
statusTag: 'allow / deny',
|
|
1024
|
+
badgeClass: 'manual',
|
|
1021
1025
|
configPath: '<code>~/.codeium/windsurf/hooks.json</code> or <code>.windsurf/hooks.json</code> in your project',
|
|
1022
1026
|
scriptPath: `${HOOKS_DIR}/pretooluse-windsurf.sh`,
|
|
1023
1027
|
settingsBlock: WINDSURF_HOOK_SETTINGS,
|
|
@@ -1025,20 +1029,46 @@ codex_hooks = true`;
|
|
|
1025
1029
|
},
|
|
1026
1030
|
codex: {
|
|
1027
1031
|
label: 'Codex',
|
|
1032
|
+
supportLevel: 'partial_native',
|
|
1028
1033
|
statusTag: 'bash only',
|
|
1034
|
+
badgeClass: 'manual',
|
|
1029
1035
|
configPath: '<code>~/.codex/hooks.json</code> and <code>~/.codex/config.toml</code>',
|
|
1030
1036
|
scriptPath: `${HOOKS_DIR}/pretooluse-codex.sh`,
|
|
1031
1037
|
settingsBlock: CODEX_HOOK_SETTINGS,
|
|
1032
1038
|
featureBlock: CODEX_HOOK_FEATURES_TOML,
|
|
1033
1039
|
limitation: 'Codex currently only supports native PreToolUse hooks for Bash, so this turns on Bash permission guardrails only.',
|
|
1034
1040
|
},
|
|
1041
|
+
cline: {
|
|
1042
|
+
label: 'Cline',
|
|
1043
|
+
supportLevel: 'mcp_only',
|
|
1044
|
+
statusTag: 'mcp only',
|
|
1045
|
+
badgeClass: 'manual',
|
|
1046
|
+
limitation: 'Cline MCP setup is supported here, but native permission-hook automation is still incomplete in this release.',
|
|
1047
|
+
},
|
|
1048
|
+
lmstudio: {
|
|
1049
|
+
label: 'LM Studio',
|
|
1050
|
+
supportLevel: 'mcp_only',
|
|
1051
|
+
statusTag: 'mcp only',
|
|
1052
|
+
badgeClass: 'manual',
|
|
1053
|
+
limitation: 'LM Studio MCP setup is supported here, but permission enforcement currently relies on LM Studio built-in confirmations or a future fallback adapter.',
|
|
1054
|
+
},
|
|
1055
|
+
};
|
|
1056
|
+
|
|
1057
|
+
const getPermissionSupport = (platformId, detected) => {
|
|
1058
|
+
if (detected?.support) {
|
|
1059
|
+
return {
|
|
1060
|
+
...PERMISSION_SUPPORT_MATRIX[platformId],
|
|
1061
|
+
supportLevel: detected.support.level || PERMISSION_SUPPORT_MATRIX[platformId]?.supportLevel,
|
|
1062
|
+
};
|
|
1063
|
+
}
|
|
1064
|
+
return PERMISSION_SUPPORT_MATRIX[platformId];
|
|
1035
1065
|
};
|
|
1036
1066
|
|
|
1037
|
-
const
|
|
1067
|
+
const getFullNativePermissionStatusCopy = (support, detected) => {
|
|
1038
1068
|
if (detected?.hookInstalled) {
|
|
1039
1069
|
return {
|
|
1040
1070
|
tone: 'info',
|
|
1041
|
-
titleText: `${
|
|
1071
|
+
titleText: `${support.label} permission enforcement is enabled.`,
|
|
1042
1072
|
messageText: 'No further changes are needed here unless you want to reinstall the hook settings.',
|
|
1043
1073
|
};
|
|
1044
1074
|
}
|
|
@@ -1046,44 +1076,60 @@ codex_hooks = true`;
|
|
|
1046
1076
|
if (detected?.installed) {
|
|
1047
1077
|
return {
|
|
1048
1078
|
tone: 'warning',
|
|
1049
|
-
titleText: `${
|
|
1050
|
-
messageText: `DollhouseMCP is configured as an MCP server. Use Configure Now below to also install the ${
|
|
1079
|
+
titleText: `${support.label} is connected for this client.`,
|
|
1080
|
+
messageText: `DollhouseMCP is configured as an MCP server. Use Configure Now below to also install the ${support.label} permission hook.`,
|
|
1051
1081
|
};
|
|
1052
1082
|
}
|
|
1053
1083
|
|
|
1054
1084
|
return {
|
|
1055
1085
|
tone: 'info',
|
|
1056
|
-
titleText: `${
|
|
1057
|
-
messageText: `First connect DollhouseMCP using Auto-updating or Pinned version, then use Configure Now below to install the ${
|
|
1086
|
+
titleText: `${support.label} permissions are not configured yet.`,
|
|
1087
|
+
messageText: `First connect DollhouseMCP using Auto-updating or Pinned version, then use Configure Now below to install the ${support.label} permission hook.`,
|
|
1058
1088
|
};
|
|
1059
1089
|
};
|
|
1060
1090
|
|
|
1061
|
-
const getPartialPermissionStatusCopy = (
|
|
1062
|
-
const activationLabel =
|
|
1091
|
+
const getPartialPermissionStatusCopy = (support, detected) => {
|
|
1092
|
+
const activationLabel = support.label === 'Codex' ? 'Bash guardrails' : 'permission hooks';
|
|
1063
1093
|
if (detected?.hookInstalled) {
|
|
1064
1094
|
return {
|
|
1065
1095
|
tone: 'info',
|
|
1066
|
-
titleText: `${
|
|
1067
|
-
messageText:
|
|
1096
|
+
titleText: `${support.label} ${activationLabel} are enabled.`,
|
|
1097
|
+
messageText: support.limitation,
|
|
1068
1098
|
};
|
|
1069
1099
|
}
|
|
1070
1100
|
|
|
1071
1101
|
if (detected?.installed) {
|
|
1072
1102
|
return {
|
|
1073
1103
|
tone: 'warning',
|
|
1074
|
-
titleText: `${
|
|
1075
|
-
messageText: `DollhouseMCP is configured as an MCP server. Use Configure Now below to turn on ${
|
|
1104
|
+
titleText: `${support.label} is connected for this client.`,
|
|
1105
|
+
messageText: `DollhouseMCP is configured as an MCP server. Use Configure Now below to turn on ${support.label}'s native ${activationLabel}.`,
|
|
1076
1106
|
};
|
|
1077
1107
|
}
|
|
1078
1108
|
|
|
1079
1109
|
return {
|
|
1080
1110
|
tone: 'info',
|
|
1081
|
-
titleText: `${
|
|
1082
|
-
messageText: `First connect DollhouseMCP using Auto-updating or Pinned version, then use Configure Now below to install ${
|
|
1111
|
+
titleText: `${support.label} ${activationLabel} are not configured yet.`,
|
|
1112
|
+
messageText: `First connect DollhouseMCP using Auto-updating or Pinned version, then use Configure Now below to install ${support.label}'s native ${activationLabel}.`,
|
|
1083
1113
|
};
|
|
1084
1114
|
};
|
|
1085
1115
|
|
|
1086
|
-
const
|
|
1116
|
+
const getMcpOnlyPermissionStatusCopy = (support, detected) => {
|
|
1117
|
+
if (detected?.installed) {
|
|
1118
|
+
return {
|
|
1119
|
+
tone: 'warning',
|
|
1120
|
+
titleText: `${support.label} is connected for this client.`,
|
|
1121
|
+
messageText: `${support.limitation} This release keeps that client in the MCP and fallback lane for now.`,
|
|
1122
|
+
};
|
|
1123
|
+
}
|
|
1124
|
+
|
|
1125
|
+
return {
|
|
1126
|
+
tone: 'info',
|
|
1127
|
+
titleText: `${support.label} supports MCP setup in this release.`,
|
|
1128
|
+
messageText: `${support.limitation} Use Auto-updating or Pinned version above to connect DollhouseMCP first.`,
|
|
1129
|
+
};
|
|
1130
|
+
};
|
|
1131
|
+
|
|
1132
|
+
const getManualPermissionStatusCopy = (support, detected) => {
|
|
1087
1133
|
if (detected?.hookAssetsPrepared) {
|
|
1088
1134
|
return {
|
|
1089
1135
|
tone: 'info',
|
|
@@ -1106,32 +1152,40 @@ codex_hooks = true`;
|
|
|
1106
1152
|
};
|
|
1107
1153
|
};
|
|
1108
1154
|
|
|
1109
|
-
const getUnsupportedPermissionStatusCopy = (
|
|
1155
|
+
const getUnsupportedPermissionStatusCopy = (support, detected) => ({
|
|
1110
1156
|
tone: detected?.installed ? 'warning' : 'neutral',
|
|
1111
|
-
titleText: `Permissions & security tools are unavailable for ${
|
|
1157
|
+
titleText: `Permissions & security tools are unavailable for ${support.label} right now.`,
|
|
1112
1158
|
messageText: detected?.installed
|
|
1113
1159
|
? 'DollhouseMCP is connected for this client, but this release does not include a supported permissions setup flow here yet.'
|
|
1114
|
-
:
|
|
1160
|
+
: support.limitation,
|
|
1115
1161
|
});
|
|
1116
1162
|
|
|
1117
1163
|
const getPermissionStatusCopy = (platformId, detected) => {
|
|
1118
|
-
const
|
|
1119
|
-
if (
|
|
1120
|
-
return
|
|
1164
|
+
const support = getPermissionSupport(platformId, detected);
|
|
1165
|
+
if (!support) {
|
|
1166
|
+
return getUnsupportedPermissionStatusCopy({
|
|
1167
|
+
label: 'this client',
|
|
1168
|
+
limitation: 'This release does not include a supported permissions setup flow for this client yet.',
|
|
1169
|
+
}, detected);
|
|
1121
1170
|
}
|
|
1122
1171
|
|
|
1123
|
-
|
|
1124
|
-
|
|
1125
|
-
return getPartialPermissionStatusCopy(partial, detected);
|
|
1172
|
+
if (support.supportLevel === 'full_native') {
|
|
1173
|
+
return getFullNativePermissionStatusCopy(support, detected);
|
|
1126
1174
|
}
|
|
1127
1175
|
|
|
1128
|
-
|
|
1129
|
-
|
|
1130
|
-
return getManualPermissionStatusCopy(detected);
|
|
1176
|
+
if (support.supportLevel === 'partial_native') {
|
|
1177
|
+
return getPartialPermissionStatusCopy(support, detected);
|
|
1131
1178
|
}
|
|
1132
1179
|
|
|
1133
|
-
|
|
1134
|
-
|
|
1180
|
+
if (support.supportLevel === 'mcp_only') {
|
|
1181
|
+
return getMcpOnlyPermissionStatusCopy(support, detected);
|
|
1182
|
+
}
|
|
1183
|
+
|
|
1184
|
+
if (support.supportLevel === 'manual') {
|
|
1185
|
+
return getManualPermissionStatusCopy(support, detected);
|
|
1186
|
+
}
|
|
1187
|
+
|
|
1188
|
+
return getUnsupportedPermissionStatusCopy(support, detected);
|
|
1135
1189
|
};
|
|
1136
1190
|
|
|
1137
1191
|
const updatePermissionStatus = (panel, platformId, detected) => {
|
|
@@ -1418,24 +1472,41 @@ codex_hooks = true`;
|
|
|
1418
1472
|
</div>`;
|
|
1419
1473
|
};
|
|
1420
1474
|
|
|
1475
|
+
const renderMcpOnlyPermissionSection = (support) => `<div class="setup-method setup-security-mode" data-setup-modes="permissions" hidden>
|
|
1476
|
+
<h3>Permissions & Security <span class="setup-support-badge setup-support-badge--${support.badgeClass}">${support.statusTag}</span></h3>
|
|
1477
|
+
<div class="setup-permission-status" data-state="info">
|
|
1478
|
+
<strong class="setup-permission-status-title"></strong>
|
|
1479
|
+
<p class="setup-permission-status-msg"></p>
|
|
1480
|
+
</div>
|
|
1481
|
+
<p class="setup-hint">${support.limitation}</p>
|
|
1482
|
+
</div>`;
|
|
1483
|
+
|
|
1421
1484
|
const renderPermissionSection = (p) => {
|
|
1422
|
-
const
|
|
1485
|
+
const support = getPermissionSupport(p.id);
|
|
1423
1486
|
const configPath = p.hookConfigPath || p.configPath || p.tomlPath || 'this client’s user configuration';
|
|
1424
1487
|
|
|
1425
|
-
if (
|
|
1426
|
-
return
|
|
1488
|
+
if (!support) {
|
|
1489
|
+
return '';
|
|
1490
|
+
}
|
|
1491
|
+
|
|
1492
|
+
if (support.supportLevel === 'full_native') {
|
|
1493
|
+
return renderVerifiedPermissionSection(p, support);
|
|
1494
|
+
}
|
|
1495
|
+
|
|
1496
|
+
if (support.supportLevel === 'partial_native') {
|
|
1497
|
+
return renderPartialPermissionSection(p, support);
|
|
1427
1498
|
}
|
|
1428
1499
|
|
|
1429
|
-
if (
|
|
1430
|
-
return
|
|
1500
|
+
if (support.supportLevel === 'mcp_only') {
|
|
1501
|
+
return renderMcpOnlyPermissionSection(support);
|
|
1431
1502
|
}
|
|
1432
1503
|
|
|
1433
|
-
if (
|
|
1504
|
+
if (support.supportLevel === 'manual') {
|
|
1434
1505
|
const platformName = p.id === 'gemini' ? 'gemini' : p.id;
|
|
1435
1506
|
const wrapperFilename = `pretooluse-${platformName}.sh`;
|
|
1436
1507
|
const wrapperScript = buildHookWrapperScript(platformName);
|
|
1437
1508
|
return `<div class="setup-method setup-security-mode" data-setup-modes="permissions" hidden>
|
|
1438
|
-
<h3>Permissions & Security <span class="setup-support-badge setup-support-badge
|
|
1509
|
+
<h3>Permissions & Security <span class="setup-support-badge setup-support-badge--${support.badgeClass}">${support.statusTag}</span></h3>
|
|
1439
1510
|
<div class="setup-permission-status" data-state="info">
|
|
1440
1511
|
<strong class="setup-permission-status-title"></strong>
|
|
1441
1512
|
<p class="setup-permission-status-msg"></p>
|
|
@@ -1453,7 +1524,7 @@ codex_hooks = true`;
|
|
|
1453
1524
|
}
|
|
1454
1525
|
|
|
1455
1526
|
return `<div class="setup-method setup-security-mode" data-setup-modes="permissions" hidden>
|
|
1456
|
-
<h3>Permissions & Security <span class="setup-support-badge setup-support-badge
|
|
1527
|
+
<h3>Permissions & Security <span class="setup-support-badge setup-support-badge--${support.badgeClass}">${support.statusTag}</span></h3>
|
|
1457
1528
|
<div class="setup-permission-status" data-state="neutral">
|
|
1458
1529
|
<strong class="setup-permission-status-title"></strong>
|
|
1459
1530
|
<p class="setup-permission-status-msg"></p>
|
|
@@ -1467,7 +1538,7 @@ codex_hooks = true`;
|
|
|
1467
1538
|
|
|
1468
1539
|
intro.innerHTML = `<div class="setup-permissions-note">
|
|
1469
1540
|
<strong>Permissions & Security</strong>
|
|
1470
|
-
<p>Use this mode to turn on permission enforcement for supported clients. Claude Code is fully guided in this release
|
|
1541
|
+
<p>Use this mode to turn on permission enforcement for supported clients. Claude Code is fully guided in this release. Gemini CLI, Cursor, VS Code, Windsurf, and Codex have native partial support, while Cline and LM Studio stay in the MCP and fallback lane for now. Other clients will be marked as coming soon.</p>
|
|
1471
1542
|
</div>`;
|
|
1472
1543
|
};
|
|
1473
1544
|
|
|
@@ -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;AAE1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;
|
|
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;AAE1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;AA+Q7E;;;GAGG;AACH,wBAAgB,wBAAwB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,aAAa,GAAG,IAAI,CA0IrF"}
|
|
@@ -13,6 +13,8 @@ import { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.j
|
|
|
13
13
|
const PERMISSION_ROUTE_RATE_LIMIT_REQUESTS = 120;
|
|
14
14
|
const PERMISSION_ROUTE_RATE_LIMIT_WINDOW_MS = 60_000;
|
|
15
15
|
const DECISION_BUFFER_SIZE = 200;
|
|
16
|
+
const CLAUDE_COMPATIBLE_HOOK_PLATFORMS = new Set(['claude_code', 'vscode']);
|
|
17
|
+
const NORMALIZABLE_PERMISSION_DECISIONS = new Set(['allow', 'deny', 'ask']);
|
|
16
18
|
/** Extract a string field from a record, trying multiple keys in order */
|
|
17
19
|
function extractString(obj, keys, fallback) {
|
|
18
20
|
for (const key of keys) {
|
|
@@ -48,11 +50,71 @@ function extractReason(result) {
|
|
|
48
50
|
}
|
|
49
51
|
return extractString(result, ['reason', 'message'], '');
|
|
50
52
|
}
|
|
53
|
+
function shouldNormalizeClaudeHook(platform) {
|
|
54
|
+
return platform !== undefined && CLAUDE_COMPATIBLE_HOOK_PLATFORMS.has(platform);
|
|
55
|
+
}
|
|
56
|
+
function normalizePermissionResponseForPlatform(platform, input, result) {
|
|
57
|
+
if (!shouldNormalizeClaudeHook(platform)) {
|
|
58
|
+
return result;
|
|
59
|
+
}
|
|
60
|
+
const nested = result.hookSpecificOutput;
|
|
61
|
+
if (nested && typeof nested === 'object' && !Array.isArray(nested)) {
|
|
62
|
+
const nestedDecision = nested.permissionDecision;
|
|
63
|
+
if (typeof nestedDecision === 'string') {
|
|
64
|
+
return result;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
const decision = extractDecision(result);
|
|
68
|
+
if (NORMALIZABLE_PERMISSION_DECISIONS.has(decision)) {
|
|
69
|
+
return formatPermissionResponse(decision, platform, input, extractReason(result));
|
|
70
|
+
}
|
|
71
|
+
return formatPermissionResponse('allow', platform, input);
|
|
72
|
+
}
|
|
73
|
+
function buildDecisionDetails(toolName, input, result, platform) {
|
|
74
|
+
const details = [];
|
|
75
|
+
if (platform) {
|
|
76
|
+
details.push({ label: 'Platform', value: platform, monospace: true });
|
|
77
|
+
}
|
|
78
|
+
if (toolName === 'Bash' && typeof input.command === 'string' && input.command !== '') {
|
|
79
|
+
details.push({ label: 'Command', value: input.command, monospace: true });
|
|
80
|
+
}
|
|
81
|
+
const targetDescriptors = [
|
|
82
|
+
{ key: 'file_path', label: 'File', monospace: true },
|
|
83
|
+
{ key: 'path', label: 'Path', monospace: true },
|
|
84
|
+
{ key: 'url', label: 'URL' },
|
|
85
|
+
{ key: 'pattern', label: 'Pattern', monospace: true },
|
|
86
|
+
{ key: 'query', label: 'Query' },
|
|
87
|
+
{ key: 'element_name', label: 'Element', monospace: true },
|
|
88
|
+
{ key: 'request_id', label: 'Request', monospace: true },
|
|
89
|
+
];
|
|
90
|
+
let target;
|
|
91
|
+
let targetLabel;
|
|
92
|
+
for (const descriptor of targetDescriptors) {
|
|
93
|
+
const value = input[descriptor.key];
|
|
94
|
+
if (typeof value !== 'string' || value === '') {
|
|
95
|
+
continue;
|
|
96
|
+
}
|
|
97
|
+
target = value;
|
|
98
|
+
targetLabel = descriptor.label;
|
|
99
|
+
details.push({ label: descriptor.label, value, monospace: descriptor.monospace });
|
|
100
|
+
break;
|
|
101
|
+
}
|
|
102
|
+
const matchedPattern = extractString(result, ['matched_pattern', 'matchedPattern'], '');
|
|
103
|
+
if (matchedPattern !== '') {
|
|
104
|
+
details.push({ label: 'Matched Pattern', value: matchedPattern, monospace: true });
|
|
105
|
+
}
|
|
106
|
+
const policySource = extractString(result, ['policy_source', 'policySource'], '');
|
|
107
|
+
if (policySource !== '') {
|
|
108
|
+
details.push({ label: 'Policy Source', value: policySource, monospace: true });
|
|
109
|
+
}
|
|
110
|
+
return { target, targetLabel, details };
|
|
111
|
+
}
|
|
51
112
|
function createPermissionDecisionTracker(bufferSize = DECISION_BUFFER_SIZE) {
|
|
52
113
|
const recentDecisions = [];
|
|
53
114
|
let decisionCounter = 0;
|
|
54
115
|
return {
|
|
55
|
-
trackDecision(sessionId, toolName, input, result) {
|
|
116
|
+
trackDecision(sessionId, toolName, input, result, platform) {
|
|
117
|
+
const detailState = buildDecisionDetails(toolName, input, result, platform);
|
|
56
118
|
const entry = {
|
|
57
119
|
id: `d-${++decisionCounter}`,
|
|
58
120
|
timestamp: new Date().toISOString(),
|
|
@@ -61,6 +123,10 @@ function createPermissionDecisionTracker(bufferSize = DECISION_BUFFER_SIZE) {
|
|
|
61
123
|
command: toolName === 'Bash' && typeof input?.command === 'string' ? input.command : undefined,
|
|
62
124
|
decision: extractDecision(result),
|
|
63
125
|
reason: extractReason(result),
|
|
126
|
+
platform,
|
|
127
|
+
target: detailState.target,
|
|
128
|
+
targetLabel: detailState.targetLabel,
|
|
129
|
+
details: detailState.details,
|
|
64
130
|
};
|
|
65
131
|
recentDecisions.unshift(entry);
|
|
66
132
|
if (recentDecisions.length > bufferSize) {
|
|
@@ -92,6 +158,8 @@ function normalizePolicyElements(elements) {
|
|
|
92
158
|
allowRules: mergeRuleArrays(element.allowPatterns, element.allowOperations),
|
|
93
159
|
confirmRules: mergeRuleArrays(element.confirmPatterns, element.confirmOperations),
|
|
94
160
|
denyRules: mergeRuleArrays(element.denyPatterns, element.denyOperations),
|
|
161
|
+
invalidGatekeeperPolicy: !!element.invalidGatekeeperPolicy,
|
|
162
|
+
invalidGatekeeperMessage: typeof element.invalidGatekeeperMessage === 'string' ? element.invalidGatekeeperMessage : undefined,
|
|
95
163
|
}));
|
|
96
164
|
}
|
|
97
165
|
function resolveElementName(element) {
|
|
@@ -171,11 +239,14 @@ export function registerPermissionRoutes(router, handler) {
|
|
|
171
239
|
res.json(formatPermissionResponse('allow', platform, input || {})); // fail open
|
|
172
240
|
return;
|
|
173
241
|
}
|
|
174
|
-
const
|
|
242
|
+
const rawResult = opResult.data;
|
|
243
|
+
const responseData = normalizePermissionResponseForPlatform(platform, input || {}, rawResult);
|
|
244
|
+
const trackedResult = { ...rawResult, ...responseData };
|
|
245
|
+
const decision = extractDecision(responseData);
|
|
175
246
|
logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);
|
|
176
247
|
// Track decision for live dashboard feed
|
|
177
|
-
decisionTracker.trackDecision(session_id, tool_name, input || {},
|
|
178
|
-
res.json(
|
|
248
|
+
decisionTracker.trackDecision(session_id, tool_name, input || {}, trackedResult, platform);
|
|
249
|
+
res.json(responseData);
|
|
179
250
|
}
|
|
180
251
|
catch (err) {
|
|
181
252
|
const elapsedMs = Date.now() - startMs;
|
|
@@ -231,6 +302,7 @@ export function registerPermissionRoutes(router, handler) {
|
|
|
231
302
|
hookInstalled: data.hookInstalled,
|
|
232
303
|
hookHost: data.hookHost,
|
|
233
304
|
enforcementReady: data.enforcementReady,
|
|
305
|
+
invalidPolicyElementCount: data.invalidPolicyElementCount ?? 0,
|
|
234
306
|
advisory: data.advisory,
|
|
235
307
|
recentDecisions: decisionTracker.getRecentDecisions(),
|
|
236
308
|
});
|
|
@@ -241,4 +313,4 @@ export function registerPermissionRoutes(router, handler) {
|
|
|
241
313
|
}
|
|
242
314
|
});
|
|
243
315
|
}
|
|
244
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,OAAmB,MAAM,SAAS,CAAC;AAC1C,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AAE/C,OAAO,EAAE,wBAAwB,EAAE,MAAM,8CAA8C,CAAC;AAExF,OAAO,EAAE,wBAAwB,EAAE,MAAM,yCAAyC,CAAC;AAqBnF,MAAM,oCAAoC,GAAG,GAAG,CAAC;AACjD,MAAM,qCAAqC,GAAG,MAAM,CAAC;AACrD,MAAM,oBAAoB,GAAG,GAAG,CAAC;AAOjC,0EAA0E;AAC1E,SAAS,aAAa,CAAC,GAA4B,EAAE,IAAc,EAAE,QAAgB;IACnF,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,GAAG,GAAG,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC;QACvB,IAAI,OAAO,GAAG,KAAK,QAAQ;YAAE,OAAO,GAAG,CAAC;IAC1C,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,SAAS,eAAe,CAAC,MAA+B;IACtD,MAAM,MAAM,GAAG,MAAM,CAAC,kBAAkB,CAAC;IACzC,IAAI,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;QACnE,MAAM,cAAc,GAAI,MAAkC,CAAC,kBAAkB,CAAC;QAC9E,IAAI,OAAO,cAAc,KAAK,QAAQ;YAAE,OAAO,cAAc,CAAC;IAChE,CAAC;IAED,IAAI,OAAO,MAAM,CAAC,UAAU,KAAK,QAAQ;QAAE,OAAO,MAAM,CAAC,UAAU,CAAC;IACpE,IAAI,OAAO,MAAM,CAAC,QAAQ,KAAK,QAAQ;QAAE,OAAO,MAAM,CAAC,QAAQ,CAAC;IAChE,IAAI,OAAO,MAAM,CAAC,QAAQ,KAAK,QAAQ;QAAE,OAAO,MAAM,CAAC,QAAQ,CAAC;IAChE,IAAI,OAAO,MAAM,CAAC,OAAO,KAAK,SAAS;QAAE,OAAO,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;IAClF,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,aAAa,CAAC,MAA+B;IACpD,MAAM,MAAM,GAAG,MAAM,CAAC,kBAAkB,CAAC;IACzC,IAAI,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;QACnE,MAAM,YAAY,GAAI,MAAkC,CAAC,wBAAwB,CAAC;QAClF,IAAI,OAAO,YAAY,KAAK,QAAQ;YAAE,OAAO,YAAY,CAAC;IAC5D,CAAC;IAED,OAAO,aAAa,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,CAAC;AAC1D,CAAC;AAED,SAAS,+BAA+B,CAAC,UAAU,GAAG,oBAAoB;IACxE,MAAM,eAAe,GAAyB,EAAE,CAAC;IACjD,IAAI,eAAe,GAAG,CAAC,CAAC;IAExB,OAAO;QACL,aAAa,CAAC,SAA6B,EAAE,QAAgB,EAAE,KAA8B,EAAE,MAA+B;YAC5H,MAAM,KAAK,GAAuB;gBAChC,EAAE,EAAE,KAAK,EAAE,eAAe,EAAE;gBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;gBACnC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBAC/C,SAAS,EAAE,QAAQ;gBACnB,OAAO,EAAE,QAAQ,KAAK,MAAM,IAAI,OAAO,KAAK,EAAE,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS;gBAC9F,QAAQ,EAAE,eAAe,CAAC,MAAM,CAAC;gBACjC,MAAM,EAAE,aAAa,CAAC,MAAM,CAAC;aAC9B,CAAC;YACF,eAAe,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YAC/B,IAAI,eAAe,CAAC,MAAM,GAAG,UAAU,EAAE,CAAC;gBACxC,eAAe,CAAC,MAAM,GAAG,UAAU,CAAC;YACtC,CAAC;QACH,CAAC;QACD,kBAAkB;YAChB,OAAO,eAAe,CAAC;QACzB,CAAC;KACF,CAAC;AACJ,CAAC;AAED,SAAS,eAAe,CAAC,GAAG,OAAkB;IAC5C,MAAM,MAAM,GAAG,IAAI,GAAG,EAAU,CAAC;IACjC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;QAC7B,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC;YAAE,SAAS;QACrC,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,EAAE,EAAE,CAAC;gBAC9C,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YACpB,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;AAC5B,CAAC;AAED,SAAS,uBAAuB,CAAC,QAAwC;IACvE,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAChC,GAAG,OAAO;QACV,YAAY,EAAE,kBAAkB,CAAC,OAAO,CAAC;QACzC,UAAU,EAAE,eAAe,CAAC,OAAO,CAAC,aAAa,EAAE,OAAO,CAAC,eAAe,CAAC;QAC3E,YAAY,EAAE,eAAe,CAAC,OAAO,CAAC,eAAe,EAAE,OAAO,CAAC,iBAAiB,CAAC;QACjF,SAAS,EAAE,eAAe,CAAC,OAAO,CAAC,YAAY,EAAE,OAAO,CAAC,cAAc,CAAC;KACzE,CAAC,CAAC,CAAC;AACN,CAAC;AAED,SAAS,kBAAkB,CAAC,OAAgC;IAC1D,IAAI,OAAO,OAAO,CAAC,YAAY,KAAK,QAAQ;QAAE,OAAO,OAAO,CAAC,YAAY,CAAC;IAC1E,IAAI,OAAO,OAAO,CAAC,IAAI,KAAK,QAAQ;QAAE,OAAO,OAAO,CAAC,IAAI,CAAC;IAC1D,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,kEAAkE;AAClE,SAAS,cAAc,CAAC,OAAgB;IACtC,IAAI,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC;QAAE,OAAO,OAAO,CAAC,CAAC,CAAC,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC;IAC3F,OAAO,OAA+D,CAAC;AACzE,CAAC;AAED,SAAS,0BAA0B,CAAC,QAAwC;IAC1E,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;IAC/B,MAAM,aAAa,GAAyB,EAAE,CAAC;IAE/C,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,UAAU,GAAG,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,CAAC;QAC/E,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;YACnC,IAAI,OAAO,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,EAAE,IAAI,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC7E,SAAS;YACX,CAAC;YAED,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACpB,aAAa,CAAC,IAAI,CAAC;gBACjB,SAAS;gBACT,WAAW,EAAE,SAAS;gBACtB,MAAM,EAAE,QAAQ;aACjB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;AAC9E,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,wBAAwB,CAAC,MAAc,EAAE,OAAsB;IAC7E,MAAM,eAAe,GAAG,+BAA+B,EAAE,CAAC;IAC1D;;;;;OAKG;IACH,MAAM,iBAAiB,GAAG,IAAI,wBAAwB,CACpD,oCAAoC,EACpC,qCAAqC,CACtC,CAAC;IACF,MAAM,CAAC,IAAI,CAAC,sBAAsB,EAAE,OAAO,CAAC,IAAI,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QACrE,MAAM,IAAI,GAAG,GAAG,CAAC,IAKhB,CAAC;QACF,MAAM,QAAQ,GAAG,OAAO,IAAI,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC;QAEpG,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,EAAE,CAAC;YACpC,GAAG,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,0BAA0B;YACrF,OAAO;QACT,CAAC;QAED,4EAA4E;QAC5E,MAAM,SAAS,GAAG,OAAO,IAAI,CAAC,SAAS,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACnG,MAAM,UAAU,GAAG,OAAO,IAAI,CAAC,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACtG,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;QAEzB,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,GAAG,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,yBAAyB;YAC7F,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC3B,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,cAAc,CAAC,MAAM,OAAO,CAAC,UAAU,CAAC;gBACvD,SAAS,EAAE,qBAAqB;gBAChC,MAAM,EAAE;oBACN,SAAS;oBACT,KAAK,EAAE,KAAK,IAAI,EAAE;oBAClB,QAAQ;oBACR,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;iBACtC;aACF,CAAC,CAAC,CAAC;YACJ,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC;YAEvC,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;gBACtB,MAAM,CAAC,IAAI,CAAC,+CAA+C,SAAS,QAAQ,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC;gBAC9F,GAAG,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,YAAY;gBAChF,OAAO;YACT,CAAC;YAED,MAAM,QAAQ,GAAG,eAAe,CAAC,QAAQ,CAAC,IAA+B,CAAC,CAAC;YAC3E,MAAM,CAAC,KAAK,CAAC,wCAAwC,SAAS,MAAM,QAAQ,KAAK,SAAS,KAAK,CAAC,CAAC;YAEjG,yCAAyC;YACzC,eAAe,CAAC,aAAa,CAAC,UAAU,EAAE,SAAS,EAAE,KAAK,IAAI,EAAE,EAAE,QAAQ,CAAC,IAA+B,CAAC,CAAC;YAE5G,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QAC1B,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC;YACvC,MAAM,CAAC,KAAK,CAAC,8CAA8C,SAAS,MAAM,EAAE,GAAG,CAAC,CAAC;YACjF,GAAG,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,YAAY;QAClF,CAAC;IACH,CAAC,CAAC,CAAC;IAEH;;;;OAIG;IACH,MAAM,CAAC,GAAG,CAAC,qBAAqB,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QACnD,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,OAAO,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC,KAAK,QAAQ,IAAI,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC;gBACpF,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC;gBACxB,CAAC,CAAC,SAAS,CAAC;YAEd,MAAM,QAAQ,GAAG,cAAc,CAAC,MAAM,OAAO,CAAC,UAAU,CAAC;gBACvD,SAAS,EAAE,4BAA4B;gBACvC,MAAM,EAAE;oBACN,eAAe,EAAE,WAAW;oBAC5B,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;iBAChD;aACF,CAAC,CAAC,CAAC;YAEJ,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;gBACtB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,QAAQ,CAAC,KAAK,IAAI,wBAAwB,EAAE,CAAC,CAAC;gBAC5E,OAAO;YACT,CAAC;YAED,MAAM,IAAI,GAAG,QAAQ,CAAC,IAA+B,CAAC;YACtD,MAAM,QAAQ,GAAG,uBAAuB,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAmC,CAAC,CAAC;YAElG,MAAM,YAAY,GAAI,IAAI,CAAC,oBAA6C,IAAI,EAAE,CAAC;YAC/E,MAAM,aAAa,GAAI,IAAI,CAAC,qBAA8C,IAAI,EAAE,CAAC;YACjF,MAAM,eAAe,GAAI,IAAI,CAAC,uBAAgD,IAAI,EAAE,CAAC;YACrF,MAAM,cAAc,GAAI,IAAI,CAAC,sBAA+C,IAAI,EAAE,CAAC;YACnF,MAAM,eAAe,GAAI,IAAI,CAAC,uBAAgD,IAAI,EAAE,CAAC;YACrF,MAAM,iBAAiB,GAAI,IAAI,CAAC,yBAAkD,IAAI,EAAE,CAAC;YAEzF,GAAG,CAAC,IAAI,CAAC;gBACP,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBACnC,kBAAkB,EAAE,IAAI,CAAC,kBAAkB;gBAC3C,YAAY,EAAE,IAAI,CAAC,YAAY;gBAC/B,YAAY;gBACZ,aAAa;gBACb,eAAe;gBACf,cAAc;gBACd,eAAe;gBACf,iBAAiB;gBACjB,SAAS,EAAE,eAAe,CAAC,YAAY,EAAE,cAAc,CAAC;gBACxD,UAAU,EAAE,eAAe,CAAC,aAAa,EAAE,eAAe,CAAC;gBAC3D,YAAY,EAAE,eAAe,CAAC,eAAe,EAAE,iBAAiB,CAAC;gBACjE,QAAQ;gBACR,aAAa,EAAE,0BAA0B,CAAC,QAAQ,CAAC;gBACnD,sBAAsB,EAAE,IAAI,CAAC,sBAAsB;gBACnD,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,gBAAgB,EAAE,IAAI,CAAC,gBAAgB;gBACvC,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,eAAe,EAAE,eAAe,CAAC,kBAAkB,EAAE;aACtD,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,2CAA2C,EAAE,GAAG,CAAC,CAAC;YAC/D,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iCAAiC,EAAE,CAAC,CAAC;QACrE,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC","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}\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(sessionId: string | undefined, toolName: string, input: Record<string, unknown>, result: Record<string, unknown>): void;\n  getRecentDecisions(): PermissionDecision[];\n}\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 createPermissionDecisionTracker(bufferSize = DECISION_BUFFER_SIZE): PermissionDecisionTracker {\n  const recentDecisions: PermissionDecision[] = [];\n  let decisionCounter = 0;\n\n  return {\n    trackDecision(sessionId: string | undefined, toolName: string, input: Record<string, unknown>, result: Record<string, unknown>): void {\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      };\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  }));\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 decision = extractDecision(opResult.data as Record<string, unknown>);\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 || {}, opResult.data as Record<string, unknown>);\n\n      res.json(opResult.data);\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        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"]}
|
|
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"]}
|