@aria-cli/tools 1.0.12 → 1.0.14

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 (233) hide show
  1. package/dist/index.js +378 -70
  2. package/dist/network-runtime/index.js +8 -12
  3. package/dist-cjs/index.js +400 -435
  4. package/dist-cjs/network-runtime/index.js +8 -172
  5. package/package.json +8 -6
  6. package/dist/.tsbuildinfo +0 -1
  7. package/dist/ask-user-interaction.js +0 -22
  8. package/dist/cache/web-cache.js +0 -66
  9. package/dist/definitions/arion.js +0 -104
  10. package/dist/definitions/browser/browser.js +0 -418
  11. package/dist/definitions/browser/index.js +0 -4
  12. package/dist/definitions/browser/pw-downloads.js +0 -114
  13. package/dist/definitions/browser/pw-interactions.js +0 -199
  14. package/dist/definitions/browser/pw-responses.js +0 -76
  15. package/dist/definitions/browser/pw-session.js +0 -310
  16. package/dist/definitions/browser/pw-shared.js +0 -66
  17. package/dist/definitions/browser/pw-snapshot.js +0 -301
  18. package/dist/definitions/browser/pw-state.js +0 -62
  19. package/dist/definitions/browser/types.js +0 -4
  20. package/dist/definitions/code-intelligence.js +0 -470
  21. package/dist/definitions/core.js +0 -109
  22. package/dist/definitions/delegation.js +0 -512
  23. package/dist/definitions/deploy.js +0 -65
  24. package/dist/definitions/filesystem.js +0 -196
  25. package/dist/definitions/frg.js +0 -63
  26. package/dist/definitions/index.js +0 -20
  27. package/dist/definitions/memory.js +0 -123
  28. package/dist/definitions/messaging.js +0 -625
  29. package/dist/definitions/meta.js +0 -349
  30. package/dist/definitions/network.js +0 -159
  31. package/dist/definitions/outlook.js +0 -277
  32. package/dist/definitions/patch/apply-patch.js +0 -184
  33. package/dist/definitions/patch/fuzzy-match.js +0 -166
  34. package/dist/definitions/patch/index.js +0 -1
  35. package/dist/definitions/patch/patch-parser.js +0 -207
  36. package/dist/definitions/patch/sandbox-paths.js +0 -105
  37. package/dist/definitions/process/index.js +0 -4
  38. package/dist/definitions/process/process-registry.js +0 -213
  39. package/dist/definitions/process/process.js +0 -386
  40. package/dist/definitions/process/pty-keys.js +0 -254
  41. package/dist/definitions/process/session-slug.js +0 -142
  42. package/dist/definitions/quip.js +0 -195
  43. package/dist/definitions/search.js +0 -60
  44. package/dist/definitions/session-history.js +0 -69
  45. package/dist/definitions/shell.js +0 -181
  46. package/dist/definitions/slack.js +0 -180
  47. package/dist/definitions/web.js +0 -109
  48. package/dist/executors/apply-patch.js +0 -901
  49. package/dist/executors/arion.js +0 -119
  50. package/dist/executors/code-intelligence.js +0 -882
  51. package/dist/executors/deploy.js +0 -848
  52. package/dist/executors/filesystem.js +0 -1122
  53. package/dist/executors/frg-freshness.js +0 -576
  54. package/dist/executors/frg.js +0 -298
  55. package/dist/executors/index.js +0 -46
  56. package/dist/executors/learning-meta.js +0 -1146
  57. package/dist/executors/lsp-client.js +0 -296
  58. package/dist/executors/memory.js +0 -750
  59. package/dist/executors/meta.js +0 -220
  60. package/dist/executors/process-registry.js +0 -465
  61. package/dist/executors/pty-session-store.js +0 -30
  62. package/dist/executors/pty.js +0 -271
  63. package/dist/executors/restart.js +0 -119
  64. package/dist/executors/search-freshness.js +0 -195
  65. package/dist/executors/search-types.js +0 -52
  66. package/dist/executors/search.js +0 -66
  67. package/dist/executors/self-diagnose.js +0 -398
  68. package/dist/executors/session-history.js +0 -283
  69. package/dist/executors/shell-safety.js +0 -473
  70. package/dist/executors/shell.js +0 -954
  71. package/dist/executors/utils.js +0 -33
  72. package/dist/executors/web.js +0 -542
  73. package/dist/extraction/content-extraction.js +0 -235
  74. package/dist/extraction/index.js +0 -4
  75. package/dist/headless-control-contract.js +0 -967
  76. package/dist/local-control-http-auth.js +0 -2
  77. package/dist/mcp/client.js +0 -181
  78. package/dist/mcp/connection.js +0 -480
  79. package/dist/mcp/index.js +0 -10
  80. package/dist/mcp/jsonrpc.js +0 -144
  81. package/dist/mcp/types.js +0 -7
  82. package/dist/network-control-adapter.js +0 -72
  83. package/dist/network-runtime/address-types.js +0 -165
  84. package/dist/network-runtime/db-owner-fencing.js +0 -69
  85. package/dist/network-runtime/delivery-receipts.js +0 -267
  86. package/dist/network-runtime/direct-endpoint-authority.js +0 -25
  87. package/dist/network-runtime/local-control-contract.js +0 -627
  88. package/dist/network-runtime/node-store-contract.js +0 -34
  89. package/dist/network-runtime/pair-route-contract.js +0 -77
  90. package/dist/network-runtime/peer-capabilities.js +0 -28
  91. package/dist/network-runtime/peer-principal-ref.js +0 -12
  92. package/dist/network-runtime/peer-state-machine.js +0 -121
  93. package/dist/network-runtime/protocol-schemas.js +0 -205
  94. package/dist/network-runtime/runtime-bootstrap-contract.js +0 -60
  95. package/dist/outlook/desktop-session.js +0 -279
  96. package/dist/policy.js +0 -149
  97. package/dist/providers/brave.js +0 -62
  98. package/dist/providers/duckduckgo.js +0 -176
  99. package/dist/providers/exa.js +0 -63
  100. package/dist/providers/firecrawl.js +0 -55
  101. package/dist/providers/index.js +0 -7
  102. package/dist/providers/jina.js +0 -49
  103. package/dist/providers/router.js +0 -96
  104. package/dist/providers/search-provider.js +0 -32
  105. package/dist/providers/tavily.js +0 -54
  106. package/dist/quip/desktop-session.js +0 -317
  107. package/dist/registry/index.js +0 -1
  108. package/dist/registry/registry.js +0 -756
  109. package/dist/runtime-socket-local-control-client.js +0 -330
  110. package/dist/security/dns-normalization.js +0 -19
  111. package/dist/security/dns-pinning.js +0 -123
  112. package/dist/security/external-content.js +0 -91
  113. package/dist/security/ssrf.js +0 -181
  114. package/dist/slack/desktop-session.js +0 -324
  115. package/dist/tool-factory.js +0 -47
  116. package/dist/types.js +0 -7
  117. package/dist/utils/retry.js +0 -132
  118. package/dist/utils/safe-parse-json.js +0 -160
  119. package/dist/utils/url.js +0 -19
  120. package/dist-cjs/.tsbuildinfo +0 -1
  121. package/dist-cjs/ask-user-interaction.js +0 -27
  122. package/dist-cjs/cache/web-cache.js +0 -70
  123. package/dist-cjs/definitions/arion.js +0 -107
  124. package/dist-cjs/definitions/browser/browser.js +0 -421
  125. package/dist-cjs/definitions/browser/index.js +0 -8
  126. package/dist-cjs/definitions/browser/pw-downloads.js +0 -117
  127. package/dist-cjs/definitions/browser/pw-interactions.js +0 -213
  128. package/dist-cjs/definitions/browser/pw-responses.js +0 -84
  129. package/dist-cjs/definitions/browser/pw-session.js +0 -326
  130. package/dist-cjs/definitions/browser/pw-shared.js +0 -72
  131. package/dist-cjs/definitions/browser/pw-snapshot.js +0 -307
  132. package/dist-cjs/definitions/browser/pw-state.js +0 -70
  133. package/dist-cjs/definitions/browser/types.js +0 -5
  134. package/dist-cjs/definitions/code-intelligence.js +0 -473
  135. package/dist-cjs/definitions/core.js +0 -133
  136. package/dist-cjs/definitions/delegation.js +0 -515
  137. package/dist-cjs/definitions/deploy.js +0 -68
  138. package/dist-cjs/definitions/filesystem.js +0 -199
  139. package/dist-cjs/definitions/frg.js +0 -66
  140. package/dist-cjs/definitions/index.js +0 -43
  141. package/dist-cjs/definitions/memory.js +0 -126
  142. package/dist-cjs/definitions/messaging.js +0 -631
  143. package/dist-cjs/definitions/meta.js +0 -352
  144. package/dist-cjs/definitions/network.js +0 -162
  145. package/dist-cjs/definitions/outlook.js +0 -280
  146. package/dist-cjs/definitions/patch/apply-patch.js +0 -191
  147. package/dist-cjs/definitions/patch/fuzzy-match.js +0 -172
  148. package/dist-cjs/definitions/patch/index.js +0 -5
  149. package/dist-cjs/definitions/patch/patch-parser.js +0 -215
  150. package/dist-cjs/definitions/patch/sandbox-paths.js +0 -113
  151. package/dist-cjs/definitions/process/index.js +0 -8
  152. package/dist-cjs/definitions/process/process-registry.js +0 -231
  153. package/dist-cjs/definitions/process/process.js +0 -389
  154. package/dist-cjs/definitions/process/pty-keys.js +0 -259
  155. package/dist-cjs/definitions/process/session-slug.js +0 -145
  156. package/dist-cjs/definitions/quip.js +0 -198
  157. package/dist-cjs/definitions/search.js +0 -63
  158. package/dist-cjs/definitions/session-history.js +0 -72
  159. package/dist-cjs/definitions/shell.js +0 -184
  160. package/dist-cjs/definitions/slack.js +0 -183
  161. package/dist-cjs/definitions/web.js +0 -112
  162. package/dist-cjs/executors/apply-patch.js +0 -938
  163. package/dist-cjs/executors/arion.js +0 -125
  164. package/dist-cjs/executors/code-intelligence.js +0 -925
  165. package/dist-cjs/executors/deploy.js +0 -869
  166. package/dist-cjs/executors/filesystem.js +0 -1167
  167. package/dist-cjs/executors/frg-freshness.js +0 -627
  168. package/dist-cjs/executors/frg.js +0 -334
  169. package/dist-cjs/executors/index.js +0 -143
  170. package/dist-cjs/executors/learning-meta.js +0 -1165
  171. package/dist-cjs/executors/lsp-client.js +0 -310
  172. package/dist-cjs/executors/memory.js +0 -796
  173. package/dist-cjs/executors/meta.js +0 -226
  174. package/dist-cjs/executors/process-registry.js +0 -469
  175. package/dist-cjs/executors/pty-session-store.js +0 -34
  176. package/dist-cjs/executors/pty.js +0 -312
  177. package/dist-cjs/executors/restart.js +0 -155
  178. package/dist-cjs/executors/search-freshness.js +0 -234
  179. package/dist-cjs/executors/search-types.js +0 -56
  180. package/dist-cjs/executors/search.js +0 -102
  181. package/dist-cjs/executors/self-diagnose.js +0 -434
  182. package/dist-cjs/executors/session-history.js +0 -320
  183. package/dist-cjs/executors/shell-safety.js +0 -478
  184. package/dist-cjs/executors/shell.js +0 -1001
  185. package/dist-cjs/executors/utils.js +0 -73
  186. package/dist-cjs/executors/web.js +0 -547
  187. package/dist-cjs/extraction/content-extraction.js +0 -243
  188. package/dist-cjs/extraction/index.js +0 -8
  189. package/dist-cjs/headless-control-contract.js +0 -972
  190. package/dist-cjs/local-control-http-auth.js +0 -5
  191. package/dist-cjs/mcp/client.js +0 -185
  192. package/dist-cjs/mcp/connection.js +0 -484
  193. package/dist-cjs/mcp/index.js +0 -30
  194. package/dist-cjs/mcp/jsonrpc.js +0 -148
  195. package/dist-cjs/mcp/types.js +0 -8
  196. package/dist-cjs/network-control-adapter.js +0 -77
  197. package/dist-cjs/network-runtime/address-types.js +0 -168
  198. package/dist-cjs/network-runtime/db-owner-fencing.js +0 -76
  199. package/dist-cjs/network-runtime/delivery-receipts.js +0 -276
  200. package/dist-cjs/network-runtime/direct-endpoint-authority.js +0 -29
  201. package/dist-cjs/network-runtime/local-control-contract.js +0 -633
  202. package/dist-cjs/network-runtime/node-store-contract.js +0 -38
  203. package/dist-cjs/network-runtime/pair-route-contract.js +0 -80
  204. package/dist-cjs/network-runtime/peer-capabilities.js +0 -37
  205. package/dist-cjs/network-runtime/peer-principal-ref.js +0 -15
  206. package/dist-cjs/network-runtime/peer-state-machine.js +0 -129
  207. package/dist-cjs/network-runtime/protocol-schemas.js +0 -212
  208. package/dist-cjs/network-runtime/runtime-bootstrap-contract.js +0 -63
  209. package/dist-cjs/outlook/desktop-session.js +0 -318
  210. package/dist-cjs/policy.js +0 -155
  211. package/dist-cjs/providers/brave.js +0 -66
  212. package/dist-cjs/providers/duckduckgo.js +0 -180
  213. package/dist-cjs/providers/exa.js +0 -67
  214. package/dist-cjs/providers/firecrawl.js +0 -59
  215. package/dist-cjs/providers/index.js +0 -17
  216. package/dist-cjs/providers/jina.js +0 -53
  217. package/dist-cjs/providers/router.js +0 -100
  218. package/dist-cjs/providers/search-provider.js +0 -36
  219. package/dist-cjs/providers/tavily.js +0 -58
  220. package/dist-cjs/quip/desktop-session.js +0 -353
  221. package/dist-cjs/registry/index.js +0 -6
  222. package/dist-cjs/registry/registry.js +0 -761
  223. package/dist-cjs/runtime-socket-local-control-client.js +0 -367
  224. package/dist-cjs/security/dns-normalization.js +0 -22
  225. package/dist-cjs/security/dns-pinning.js +0 -160
  226. package/dist-cjs/security/external-content.js +0 -95
  227. package/dist-cjs/security/ssrf.js +0 -221
  228. package/dist-cjs/slack/desktop-session.js +0 -366
  229. package/dist-cjs/tool-factory.js +0 -50
  230. package/dist-cjs/types.js +0 -8
  231. package/dist-cjs/utils/retry.js +0 -169
  232. package/dist-cjs/utils/safe-parse-json.js +0 -164
  233. package/dist-cjs/utils/url.js +0 -23
@@ -1,478 +0,0 @@
1
- "use strict";
2
- /**
3
- * @aria/tools - Shell command risk classifier
4
- *
5
- * Statically classifies shell commands into risk tiers to gate execution:
6
- * - "safe" : read-only, execute immediately without approval
7
- * - "moderate" : requires runtime policy handling (approval, allowlist, or autorun)
8
- * - "blocked" : catastrophic, hard-denied — never execute
9
- */
10
- Object.defineProperty(exports, "__esModule", { value: true });
11
- exports.BLOCKED_PATTERNS = void 0;
12
- exports.classifyCommand = classifyCommand;
13
- exports.classifyExecInvocation = classifyExecInvocation;
14
- /** Read-only single-word commands that never modify state. */
15
- const SAFE_SINGLE = new Set([
16
- "ls",
17
- "cat",
18
- "head",
19
- "tail",
20
- "wc",
21
- "file",
22
- "stat",
23
- "grep",
24
- "rg",
25
- "find",
26
- "which",
27
- "whereis",
28
- "echo",
29
- "date",
30
- "whoami",
31
- "pwd",
32
- "printenv",
33
- "uname",
34
- "hostname",
35
- ]);
36
- /** Read-only multi-word command prefixes (order: longest match first). */
37
- const SAFE_MULTI = [
38
- "git stash list",
39
- "git status",
40
- "git log",
41
- "git diff",
42
- "git show",
43
- "git blame",
44
- "git branch",
45
- "git remote",
46
- "git tag",
47
- "node --version",
48
- "npm --version",
49
- "pnpm --version",
50
- "python --version",
51
- "pnpm list",
52
- "npm list",
53
- ];
54
- /** Patterns that are unconditionally blocked — catastrophic risk. */
55
- exports.BLOCKED_PATTERNS = [
56
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*\/(?:\s|$)/, // rm targeting filesystem root (/)
57
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*\/\*(?:\s|$)/, // rm targeting root wildcard (/*)
58
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*\.(?:\s|$)/, // rm . (current dir wipe, not ./subdir)
59
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*~(?:[a-zA-Z]\w*)?(?:\/\*)?(?:\s|$)/, // rm ~ (bare home), ~/* (home wildcard), ~user (other user home)
60
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*\$HOME\b/, // rm with $HOME variable
61
- /rm\s+(?:--?[A-Za-z0-9-]+\s+)*\*(?:\s|$)/, // rm with bare wildcard (rm -rf *)
62
- />\s*\/dev\/(?:sd[a-z]|nvme\d+|vd[a-z])\b/, // write to block devices
63
- /mkfs/, // format filesystems
64
- /dd\s+.*(?:if=|of=)/, // raw disk reads/writes
65
- /chmod\s+(?:-R\s+)?777\b/, // world-writable permissions
66
- /curl[\s\S]*\|\s*(ba)?sh/, // pipe-to-shell (including newline-obfuscated variants)
67
- /wget[\s\S]*\|\s*(ba)?sh/, // pipe-to-shell (including newline-obfuscated variants)
68
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*eval\b/, // shell eval injection
69
- // Inline execution (sh -c, python -c, node -e, etc.) downgraded to moderate.
70
- // These are legitimate developer operations — dangerous payloads are still
71
- // caught by other blocked patterns (rm /, curl|sh, fork bombs, etc.).
72
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*shutdown\b/, // system shutdown
73
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*reboot\b/, // system reboot
74
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*halt\b/, // system halt
75
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*init\s+0\b/, // init runlevel poweroff
76
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*systemctl\s+(?:poweroff|halt|reboot)\b/, // systemctl power controls
77
- // kill downgraded to moderate — legitimate process management (kill PID, kill -0)
78
- // is a normal developer operation. Catastrophic kill (kill -9 1) is still caught
79
- // by the PID-1 pattern below. See shell-safety.test.ts for coverage.
80
- /(?:^|[;&|]\s*|\$\(|`|\()\s*(?:(?:env|command)\s+)*kill\s+(?:-\d+\s+|-[A-Z]+\s+)*\b1\b/, // kill PID 1 (init) — catastrophic
81
- // ${...} parameter expansion downgraded to moderate — standard bash operations
82
- // like ${VAR}, ${#VAR} (length), ${VAR:-default} are common developer patterns.
83
- // Truly dangerous expansions (${VAR:=$(cmd)}) are caught by other patterns
84
- // (eval, curl|sh, etc.) or by the subshell/backtick check in classifyCommand.
85
- /:\(\)\{\s*:\|:&\s*\};:/, // fork bomb
86
- /\bsudo\b/, // privilege escalation
87
- /git\s+push\s+.*--force(?!-with-lease)\b/, // force push (allow --force-with-lease)
88
- /git\s+push(?:\s+-[A-Za-z]*f[A-Za-z]*\b|\s+.*\s-[A-Za-z]*f[A-Za-z]*\b)/, // short force flags (-f, -uf, etc.)
89
- /git\s+reset\s+--hard/, // hard reset
90
- ];
91
- /**
92
- * Returns true if the raw command text matches any blocked pattern.
93
- */
94
- function isBlocked(raw) {
95
- const withoutHeredocs = stripHeredocBodies(raw);
96
- const withoutQuotedLiterals = stripSingleAndDoubleQuotedLiterals(withoutHeredocs);
97
- return exports.BLOCKED_PATTERNS.some((re) => re.test(withoutQuotedLiterals));
98
- }
99
- /**
100
- * Strip heredoc bodies so that blocked-pattern checks don't fire on
101
- * data content inside heredocs. Supports both quoted and unquoted
102
- * delimiters: `<< 'EOF'`, `<< "EOF"`, `<< EOF`, `<<-EOF`.
103
- *
104
- * Only the body between the delimiter lines is replaced with spaces;
105
- * the shell command on the `<<` line and the closing delimiter are
106
- * preserved so other pattern checks still apply to the command itself.
107
- */
108
- function stripHeredocBodies(raw) {
109
- // Match << (optional dash) then optional quotes around the delimiter word
110
- const heredocRe = /<<-?\s*(?:'([^']+)'|"([^"]+)"|(\w+))/g;
111
- let result = raw;
112
- let match;
113
- // Collect all heredoc markers first, then strip from the end backwards
114
- // so index positions remain valid.
115
- const markers = [];
116
- while ((match = heredocRe.exec(raw)) !== null) {
117
- const delimiter = match[1] ?? match[2] ?? match[3] ?? "";
118
- if (!delimiter)
119
- continue;
120
- // Body starts after the next newline following the << marker
121
- const afterMarker = raw.indexOf("\n", match.index);
122
- if (afterMarker === -1)
123
- continue;
124
- const bodyStart = afterMarker + 1;
125
- // Find the closing delimiter: must be on its own line (with optional
126
- // leading whitespace for <<- heredocs).
127
- const closingRe = new RegExp(`^\\s*${delimiter.replace(/[.*+?^${}()|[\]\\]/g, "\\$&")}\\s*$`, "m");
128
- const bodySlice = raw.slice(bodyStart);
129
- const closingMatch = closingRe.exec(bodySlice);
130
- if (!closingMatch)
131
- continue;
132
- const bodyEnd = bodyStart + closingMatch.index;
133
- markers.push({ delimiter, bodyStart, bodyEnd });
134
- }
135
- // Strip bodies from last to first to preserve indices
136
- for (let i = markers.length - 1; i >= 0; i--) {
137
- const { bodyStart, bodyEnd } = markers[i];
138
- const body = result.slice(bodyStart, bodyEnd);
139
- result = result.slice(0, bodyStart) + body.replace(/[^\n]/g, " ") + result.slice(bodyEnd);
140
- }
141
- return result;
142
- }
143
- /**
144
- * Remove single-quoted and double-quoted literal text from a command so
145
- * blocked-pattern checks don't fire on plain quoted prose like:
146
- * echo "please do not run rm -rf /"
147
- */
148
- function stripSingleAndDoubleQuotedLiterals(raw) {
149
- let result = "";
150
- let inSingle = false;
151
- let inDouble = false;
152
- let escaped = false;
153
- for (let i = 0; i < raw.length; i++) {
154
- const ch = raw[i];
155
- if (inSingle) {
156
- if (ch === "'") {
157
- inSingle = false;
158
- result += " ";
159
- }
160
- else {
161
- result += " ";
162
- }
163
- continue;
164
- }
165
- if (inDouble) {
166
- if (escaped) {
167
- escaped = false;
168
- result += " ";
169
- continue;
170
- }
171
- if (ch === "\\") {
172
- escaped = true;
173
- result += " ";
174
- continue;
175
- }
176
- if (ch === '"') {
177
- inDouble = false;
178
- result += " ";
179
- }
180
- else {
181
- result += " ";
182
- }
183
- continue;
184
- }
185
- if (ch === "'") {
186
- inSingle = true;
187
- result += " ";
188
- continue;
189
- }
190
- if (ch === '"') {
191
- inDouble = true;
192
- result += " ";
193
- continue;
194
- }
195
- result += ch;
196
- }
197
- return result;
198
- }
199
- /**
200
- * Strip an optional absolute-path prefix from a token
201
- * so `/usr/bin/rm` is treated the same as `rm`.
202
- */
203
- function stripPathPrefix(token) {
204
- const i = token.lastIndexOf("/");
205
- return i === -1 ? token : token.slice(i + 1);
206
- }
207
- function hasUnquotedSubshellOrBacktick(command) {
208
- let inSingle = false;
209
- let inDouble = false;
210
- let escaped = false;
211
- for (let i = 0; i < command.length; i++) {
212
- const ch = command[i];
213
- const next = command[i + 1];
214
- if (inSingle) {
215
- if (ch === "'")
216
- inSingle = false;
217
- continue;
218
- }
219
- if (inDouble) {
220
- if (escaped) {
221
- escaped = false;
222
- continue;
223
- }
224
- if (ch === "\\") {
225
- escaped = true;
226
- continue;
227
- }
228
- if (ch === '"') {
229
- inDouble = false;
230
- continue;
231
- }
232
- if (ch === "`" || (ch === "$" && next === "(")) {
233
- return true;
234
- }
235
- continue;
236
- }
237
- if (escaped) {
238
- escaped = false;
239
- continue;
240
- }
241
- if (ch === "\\") {
242
- escaped = true;
243
- continue;
244
- }
245
- if (ch === "'") {
246
- inSingle = true;
247
- continue;
248
- }
249
- if (ch === '"') {
250
- inDouble = true;
251
- continue;
252
- }
253
- if (ch === "`" || (ch === "$" && next === "(")) {
254
- return true;
255
- }
256
- }
257
- return false;
258
- }
259
- function splitTopLevelChain(command) {
260
- const parts = [];
261
- let current = "";
262
- let inSingle = false;
263
- let inDouble = false;
264
- let escaped = false;
265
- for (let i = 0; i < command.length; i++) {
266
- const ch = command[i];
267
- const next = command[i + 1];
268
- if (inSingle) {
269
- current += ch;
270
- if (ch === "'")
271
- inSingle = false;
272
- continue;
273
- }
274
- if (inDouble) {
275
- current += ch;
276
- if (escaped) {
277
- escaped = false;
278
- continue;
279
- }
280
- if (ch === "\\") {
281
- escaped = true;
282
- continue;
283
- }
284
- if (ch === '"')
285
- inDouble = false;
286
- continue;
287
- }
288
- if (escaped) {
289
- current += ch;
290
- escaped = false;
291
- continue;
292
- }
293
- if (ch === "\\") {
294
- current += ch;
295
- escaped = true;
296
- continue;
297
- }
298
- if (ch === "'") {
299
- current += ch;
300
- inSingle = true;
301
- continue;
302
- }
303
- if (ch === '"') {
304
- current += ch;
305
- inDouble = true;
306
- continue;
307
- }
308
- const isSeparator = ch === ";" ||
309
- ch === "\n" ||
310
- ch === "\r" ||
311
- (ch === "&" && next === "&") ||
312
- (ch === "|" && next === "|");
313
- if (isSeparator) {
314
- parts.push(current.trim());
315
- current = "";
316
- if ((ch === "&" || ch === "|") && next === ch) {
317
- i += 1;
318
- }
319
- continue;
320
- }
321
- current += ch;
322
- }
323
- parts.push(current.trim());
324
- return parts;
325
- }
326
- function splitTopLevelPipes(command) {
327
- const parts = [];
328
- let current = "";
329
- let inSingle = false;
330
- let inDouble = false;
331
- let escaped = false;
332
- for (let i = 0; i < command.length; i++) {
333
- const ch = command[i];
334
- const next = command[i + 1];
335
- const prev = i > 0 ? command[i - 1] : "";
336
- if (inSingle) {
337
- current += ch;
338
- if (ch === "'")
339
- inSingle = false;
340
- continue;
341
- }
342
- if (inDouble) {
343
- current += ch;
344
- if (escaped) {
345
- escaped = false;
346
- continue;
347
- }
348
- if (ch === "\\") {
349
- escaped = true;
350
- continue;
351
- }
352
- if (ch === '"')
353
- inDouble = false;
354
- continue;
355
- }
356
- if (escaped) {
357
- current += ch;
358
- escaped = false;
359
- continue;
360
- }
361
- if (ch === "\\") {
362
- current += ch;
363
- escaped = true;
364
- continue;
365
- }
366
- if (ch === "'") {
367
- current += ch;
368
- inSingle = true;
369
- continue;
370
- }
371
- if (ch === '"') {
372
- current += ch;
373
- inDouble = true;
374
- continue;
375
- }
376
- if (ch === "|" && next !== "|" && prev !== "|") {
377
- parts.push(current.trim());
378
- current = "";
379
- continue;
380
- }
381
- current += ch;
382
- }
383
- parts.push(current.trim());
384
- return parts;
385
- }
386
- /**
387
- * Matches shell output redirection operators that WRITE to the filesystem.
388
- * Excludes 2>&1 (stderr-to-stdout merge) which is read-only.
389
- *
390
- * Matches: >, >>, 2> (not followed by &), &>
391
- * Does NOT match: 2>&1, <, <<
392
- */
393
- const REDIRECTION_RE = /(?:>>|(?:^|[^2])>(?!&)|2>(?!&)|&>)/;
394
- /**
395
- * Determine whether a single simple command (no pipes/chains) is safe.
396
- * Returns true only when the command prefix is in the safe lists
397
- * AND the segment contains no output redirection.
398
- */
399
- function isSegmentSafe(segment) {
400
- const trimmed = segment.trim();
401
- if (trimmed === "")
402
- return false;
403
- // Output redirection makes any command non-safe
404
- if (REDIRECTION_RE.test(trimmed))
405
- return false;
406
- // Multi-word safe match first (e.g. "git status")
407
- for (const prefix of SAFE_MULTI) {
408
- if (trimmed === prefix || trimmed.startsWith(prefix + " "))
409
- return true;
410
- }
411
- // Single-word: first token, path-stripped
412
- const firstToken = trimmed.split(/\s+/)[0] ?? "";
413
- return SAFE_SINGLE.has(stripPathPrefix(firstToken));
414
- }
415
- /**
416
- * Classify a shell command's risk level.
417
- *
418
- * - "safe" — skip approval, execute immediately
419
- * - "moderate" — handle via runtime policy (approval, allowlist, or autorun)
420
- * - "blocked" — never execute, return error immediately
421
- */
422
- function classifyCommand(command) {
423
- const trimmed = command.trim();
424
- // Empty / whitespace-only — can't determine intent
425
- if (trimmed === "")
426
- return "moderate";
427
- // 0. Strip heredoc bodies once, upfront, so every downstream check
428
- // (whole-command, per-segment, per-pipe) operates on the sanitised text.
429
- // This prevents heredoc *data* from triggering blocked patterns.
430
- const stripped = stripHeredocBodies(trimmed);
431
- // 1. Check blocked patterns on the entire raw command
432
- if (isBlocked(stripped))
433
- return "blocked";
434
- // 2. Subshells / backticks — can't statically analyze safely.
435
- // Check before splitting to avoid quote-unaware false positives.
436
- if (hasUnquotedSubshellOrBacktick(stripped))
437
- return "moderate";
438
- // 3. Split on chain operators (&&, ||, ;, newline) outside quoted strings.
439
- const chainSegments = splitTopLevelChain(stripped);
440
- for (const seg of chainSegments) {
441
- if (isBlocked(seg))
442
- return "blocked";
443
- }
444
- // 4. Split each chain segment on top-level pipes and check.
445
- const allSegments = [];
446
- for (const seg of chainSegments) {
447
- const piped = splitTopLevelPipes(seg);
448
- for (const p of piped) {
449
- if (isBlocked(p))
450
- return "blocked";
451
- allSegments.push(p);
452
- }
453
- }
454
- // 5. Env-var assignment prefix (VAR=val command)
455
- if (/^[A-Za-z_]\w*=/.test(trimmed))
456
- return "moderate";
457
- // 6. Check if ALL segments are safe (filter empty segments from chain splitting)
458
- const nonEmpty = allSegments.filter((s) => s.trim() !== "");
459
- if (nonEmpty.length > 0 && nonEmpty.every(isSegmentSafe))
460
- return "safe";
461
- // 7. Default — needs approval
462
- return "moderate";
463
- }
464
- const EXPLICIT_SHELLS = new Set(["sh", "bash", "zsh", "ksh", "dash", "ash", "fish"]);
465
- /**
466
- * Classify argv-based process execution (spawn/exec) with shell-aware behavior.
467
- *
468
- * When callers explicitly invoke a shell interpreter with `-c`, classify the
469
- * payload command text itself. This preserves catastrophic-command blocking
470
- * while avoiding false positives that would block every `bash -c ...` call.
471
- */
472
- function classifyExecInvocation(program, args = []) {
473
- const shellName = stripPathPrefix(program).toLowerCase();
474
- if (EXPLICIT_SHELLS.has(shellName) && args[0] === "-c" && typeof args[1] === "string") {
475
- return classifyCommand(args[1]);
476
- }
477
- return classifyCommand([program, ...args].join(" "));
478
- }