summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorDax Raad <[email protected]>2026-01-01 21:01:00 -0500
committerDax Raad <[email protected]>2026-01-01 21:01:00 -0500
commit963f407062ccde868b6abb5a21178dea861bc4ca (patch)
tree312f604d4ecc5abd41b28ad2ed40788cef652e3a
parent4f1ef9391061621c7f0a844894d210ee2a289e89 (diff)
downloadopencode-963f407062ccde868b6abb5a21178dea861bc4ca.tar.gz
opencode-963f407062ccde868b6abb5a21178dea861bc4ca.zip
tui: improve permission error handling and evaluation logic
-rw-r--r--.opencode/opencode.jsonc5
-rw-r--r--packages/opencode/src/cli/cmd/tui/routes/session/index.tsx2
-rw-r--r--packages/opencode/src/permission/next.ts31
3 files changed, 24 insertions, 14 deletions
diff --git a/.opencode/opencode.jsonc b/.opencode/opencode.jsonc
index ad9925767..6008ab9bc 100644
--- a/.opencode/opencode.jsonc
+++ b/.opencode/opencode.jsonc
@@ -10,6 +10,11 @@
"options": {},
},
},
+ "permission": {
+ "bash": {
+ "ls foo": "ask",
+ },
+ },
"mcp": {
"context7": {
"type": "remote",
diff --git a/packages/opencode/src/cli/cmd/tui/routes/session/index.tsx b/packages/opencode/src/cli/cmd/tui/routes/session/index.tsx
index 94ba05614..4b3b67a31 100644
--- a/packages/opencode/src/cli/cmd/tui/routes/session/index.tsx
+++ b/packages/opencode/src/cli/cmd/tui/routes/session/index.tsx
@@ -1406,7 +1406,7 @@ function InlineTool(props: { icon: string; complete: any; pending: string; child
const error = createMemo(() => (props.part.state.status === "error" ? props.part.state.error : undefined))
- const denied = createMemo(() => error()?.includes("rejected permission"))
+ const denied = createMemo(() => error()?.includes("rejected permission") || error()?.includes("specified a rule"))
return (
<box
diff --git a/packages/opencode/src/permission/next.ts b/packages/opencode/src/permission/next.ts
index 4f7d831e7..e823e7429 100644
--- a/packages/opencode/src/permission/next.ts
+++ b/packages/opencode/src/permission/next.ts
@@ -121,10 +121,11 @@ export namespace PermissionNext {
const s = await state()
const { ruleset, ...request } = input
for (const pattern of request.patterns ?? []) {
- const action = evaluate(request.permission, pattern, ruleset, s.approved)
- log.info("evaluated", { permission: request.permission, pattern, action })
- if (action === "deny") throw new RejectedError()
- if (action === "ask") {
+ const rule = evaluate(request.permission, pattern, ruleset, s.approved)
+ log.info("evaluated", { permission: request.permission, pattern, action: rule })
+ if (rule.action === "deny")
+ throw new AutoRejectedError(ruleset.filter((r) => Wildcard.match(request.permission, r.permission)))
+ if (rule.action === "ask") {
const id = input.id ?? Identifier.ascending("permission")
return new Promise<void>((resolve, reject) => {
const info: Request = {
@@ -139,7 +140,7 @@ export namespace PermissionNext {
Bus.publish(Event.Asked, info)
})
}
- if (action === "allow") continue
+ if (rule.action === "allow") continue
}
},
)
@@ -195,7 +196,7 @@ export namespace PermissionNext {
for (const [id, pending] of Object.entries(s.pending)) {
if (pending.info.sessionID !== sessionID) continue
const ok = pending.info.patterns.every(
- (pattern) => evaluate(pending.info.permission, pattern, s.approved) === "allow",
+ (pattern) => evaluate(pending.info.permission, pattern, s.approved).action === "allow",
)
if (!ok) continue
delete s.pending[id]
@@ -215,13 +216,13 @@ export namespace PermissionNext {
},
)
- export function evaluate(permission: string, pattern: string, ...rulesets: Ruleset[]): Action {
+ export function evaluate(permission: string, pattern: string, ...rulesets: Ruleset[]): Rule {
const merged = merge(...rulesets)
log.info("evaluate", { permission, pattern, ruleset: merged })
const match = merged.findLast(
(rule) => Wildcard.match(permission, rule.permission) && Wildcard.match(pattern, rule.pattern),
)
- return match?.action ?? "ask"
+ return match ?? { action: "allow", permission, pattern: "*" }
}
const EDIT_TOOLS = ["edit", "write", "patch", "multiedit"]
@@ -230,7 +231,7 @@ export namespace PermissionNext {
const result = new Set<string>()
for (const tool of tools) {
const permission = EDIT_TOOLS.includes(tool) ? "edit" : tool
- if (evaluate(permission, "*", ruleset) === "deny") {
+ if (evaluate(permission, "*", ruleset).action === "deny") {
result.add(tool)
}
}
@@ -238,11 +239,15 @@ export namespace PermissionNext {
}
export class RejectedError extends Error {
- constructor(public readonly reason?: string) {
+ constructor() {
+ super(`The user rejected permission to use this specific tool call. You may try again with different parameters.`)
+ }
+ }
+
+ export class AutoRejectedError extends Error {
+ constructor(public readonly ruleset: Ruleset) {
super(
- reason !== undefined
- ? reason
- : `The user rejected permission to use this specific tool call. You may try again with different parameters.`,
+ `The user has specified a rule which prevents you from using this specific tool call. Here are some of the relevant rules ${JSON.stringify(ruleset)}`,
)
}
}