feat: Global hooks approval modal with plan/question/permission modes
- Add PermissionRequest and Stop approval hooks to local Claude config - Unify PermissionApproval into multi-mode card (permission, plan, question) - Support allowAlways, deny-with-reason, and AskUserQuestion answering - Add cross-process broadcast fallback (HTTP to sync server) - Fix approval scripts to default to .claude/debug/ for local agent
This commit is contained in:
@@ -151,6 +151,16 @@
|
|||||||
"timeout": 5000
|
"timeout": 5000
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"matcher": ".*",
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "powershell -NoProfile -File hooks/approval-permission.ps1",
|
||||||
|
"timeout": 130000
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"Notification": [
|
"Notification": [
|
||||||
@@ -174,6 +184,15 @@
|
|||||||
"timeout": 10000
|
"timeout": 10000
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hooks": [
|
||||||
|
{
|
||||||
|
"type": "command",
|
||||||
|
"command": "powershell -NoProfile -File hooks/approval-plan.ps1",
|
||||||
|
"timeout": 130000
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ watch(totalPending, (val) => {
|
|||||||
v-for="perm in group.permissions"
|
v-for="perm in group.permissions"
|
||||||
:key="perm.requestId"
|
:key="perm.requestId"
|
||||||
:request="perm"
|
:request="perm"
|
||||||
@respond="respondPermission"
|
@respond="(id, decision, reason) => respondPermission(id, decision, reason)"
|
||||||
/>
|
/>
|
||||||
<PlanApproval
|
<PlanApproval
|
||||||
v-for="plan in group.plans"
|
v-for="plan in group.plans"
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { ref, computed } from 'vue'
|
||||||
import type { HooksApprovalPermissionRequest } from '@/types/hooks-approval'
|
import type { HooksApprovalPermissionRequest } from '@/types/hooks-approval'
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
@@ -6,16 +7,130 @@ const props = defineProps<{
|
|||||||
}>()
|
}>()
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
respond: [requestId: string, decision: 'allow' | 'deny']
|
respond: [requestId: string, decision: string, reason?: string]
|
||||||
}>()
|
}>()
|
||||||
|
|
||||||
function formatInput(input: unknown): string {
|
// ── Mode detection ──
|
||||||
if (!input) return ''
|
type CardMode = 'permission' | 'plan' | 'question'
|
||||||
if (typeof input === 'string') return input
|
|
||||||
|
const mode = computed<CardMode>(() => {
|
||||||
|
if (props.request.tool_name === 'ExitPlanMode') return 'plan'
|
||||||
|
if (props.request.tool_name === 'AskUserQuestion') return 'question'
|
||||||
|
return 'permission'
|
||||||
|
})
|
||||||
|
|
||||||
|
const input = computed(() => {
|
||||||
|
if (!props.request.tool_input) return null
|
||||||
|
return props.request.tool_input as Record<string, unknown>
|
||||||
|
})
|
||||||
|
|
||||||
|
// ── Plan mode ──
|
||||||
|
const planText = computed(() => {
|
||||||
|
if (mode.value !== 'plan' || !input.value) return ''
|
||||||
|
return (input.value.plan as string) || ''
|
||||||
|
})
|
||||||
|
|
||||||
|
const showPlanEditor = ref(false)
|
||||||
|
const planEditText = ref('')
|
||||||
|
|
||||||
|
function handlePlanEdit() {
|
||||||
|
if (!showPlanEditor.value) {
|
||||||
|
showPlanEditor.value = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
emit('respond', props.request.requestId, 'deny', planEditText.value || 'Continue with modifications.')
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Question mode ──
|
||||||
|
interface QuestionOption {
|
||||||
|
label: string
|
||||||
|
description?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
interface QuestionItem {
|
||||||
|
question: string
|
||||||
|
header?: string
|
||||||
|
options?: QuestionOption[]
|
||||||
|
multiSelect?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const questions = computed<QuestionItem[]>(() => {
|
||||||
|
if (mode.value !== 'question' || !input.value) return []
|
||||||
|
const qs = input.value.questions
|
||||||
|
if (Array.isArray(qs)) return qs as QuestionItem[]
|
||||||
|
return []
|
||||||
|
})
|
||||||
|
|
||||||
|
// Track selected options per question (keyed by question text)
|
||||||
|
const selectedAnswers = ref<Record<string, Set<string>>>({})
|
||||||
|
const customAnswers = ref<Record<string, string>>({})
|
||||||
|
const showCustom = ref<Record<string, boolean>>({})
|
||||||
|
|
||||||
|
function toggleOption(questionText: string, label: string, multiSelect?: boolean) {
|
||||||
|
if (!selectedAnswers.value[questionText]) {
|
||||||
|
selectedAnswers.value[questionText] = new Set()
|
||||||
|
}
|
||||||
|
const set = selectedAnswers.value[questionText]
|
||||||
|
if (multiSelect) {
|
||||||
|
if (set.has(label)) set.delete(label)
|
||||||
|
else set.add(label)
|
||||||
|
} else {
|
||||||
|
if (set.has(label)) set.clear()
|
||||||
|
else { set.clear(); set.add(label) }
|
||||||
|
}
|
||||||
|
// Trigger reactivity
|
||||||
|
selectedAnswers.value = { ...selectedAnswers.value }
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleCustom(questionText: string) {
|
||||||
|
showCustom.value = { ...showCustom.value, [questionText]: !showCustom.value[questionText] }
|
||||||
|
}
|
||||||
|
|
||||||
|
function submitAnswers() {
|
||||||
|
// Build answers object: { "question text": "selected label" }
|
||||||
|
const answers: Record<string, string> = {}
|
||||||
|
for (const q of questions.value) {
|
||||||
|
const selected = selectedAnswers.value[q.question]
|
||||||
|
const custom = customAnswers.value[q.question]
|
||||||
|
if (custom?.trim()) {
|
||||||
|
answers[q.question] = custom.trim()
|
||||||
|
} else if (selected?.size) {
|
||||||
|
answers[q.question] = Array.from(selected).join(', ')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Deny the tool (so Claude gets the answer as the deny message)
|
||||||
|
// The answer goes as a structured JSON string
|
||||||
|
const answerText = JSON.stringify({ answers })
|
||||||
|
emit('respond', props.request.requestId, 'deny', answerText)
|
||||||
|
}
|
||||||
|
|
||||||
|
function allowQuestion() {
|
||||||
|
// Let AskUserQuestion run normally (shows in terminal)
|
||||||
|
emit('respond', props.request.requestId, 'allow')
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Permission mode ──
|
||||||
|
const showFreeResponse = ref(false)
|
||||||
|
const freeText = ref('')
|
||||||
|
|
||||||
|
function handleFreeResponse() {
|
||||||
|
if (!showFreeResponse.value) {
|
||||||
|
showFreeResponse.value = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (freeText.value.trim()) {
|
||||||
|
emit('respond', props.request.requestId, 'deny', freeText.value.trim())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Common ──
|
||||||
|
function formatInput(inp: unknown): string {
|
||||||
|
if (!inp) return ''
|
||||||
|
if (typeof inp === 'string') return inp
|
||||||
try {
|
try {
|
||||||
return JSON.stringify(input, null, 2)
|
return JSON.stringify(inp, null, 2)
|
||||||
} catch {
|
} catch {
|
||||||
return String(input)
|
return String(inp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -28,8 +143,123 @@ function elapsed(): string {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="permission-card">
|
<!-- ═══════ PLAN MODE (ExitPlanMode) ═══════ -->
|
||||||
<div class="card-header">
|
<div v-if="mode === 'plan'" class="card plan-card">
|
||||||
|
<div class="card-header plan-header">
|
||||||
|
<span class="card-icon">
|
||||||
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/>
|
||||||
|
<polyline points="14 2 14 8 20 8"/>
|
||||||
|
<line x1="16" y1="13" x2="8" y2="13"/>
|
||||||
|
<line x1="16" y1="17" x2="8" y2="17"/>
|
||||||
|
</svg>
|
||||||
|
</span>
|
||||||
|
<span class="card-label">Plan Approval</span>
|
||||||
|
<span class="card-elapsed">{{ elapsed() }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="card-body">
|
||||||
|
<div v-if="planText" class="plan-text">
|
||||||
|
<pre class="plan-pre">{{ planText }}</pre>
|
||||||
|
</div>
|
||||||
|
<div v-else class="plan-empty">Claude is waiting for plan approval</div>
|
||||||
|
<Transition name="expand">
|
||||||
|
<div v-if="showPlanEditor" class="edit-section">
|
||||||
|
<textarea v-model="planEditText" class="edit-textarea" placeholder="Add instructions or feedback..." rows="3"></textarea>
|
||||||
|
</div>
|
||||||
|
</Transition>
|
||||||
|
</div>
|
||||||
|
<div class="card-actions">
|
||||||
|
<button class="btn btn-approve" @click="emit('respond', request.requestId, 'allow')">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5"><polyline points="20 6 9 17 4 12"/></svg>
|
||||||
|
Approve
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-edit" @click="handlePlanEdit">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7"/>
|
||||||
|
<path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z"/>
|
||||||
|
</svg>
|
||||||
|
{{ showPlanEditor ? 'Send' : 'Edit & Continue' }}
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-reject" @click="emit('respond', request.requestId, 'deny')">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/>
|
||||||
|
</svg>
|
||||||
|
Reject
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ═══════ QUESTION MODE (AskUserQuestion) ═══════ -->
|
||||||
|
<div v-else-if="mode === 'question'" class="card question-card">
|
||||||
|
<div class="card-header question-header">
|
||||||
|
<span class="card-icon">
|
||||||
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<circle cx="12" cy="12" r="10"/>
|
||||||
|
<path d="M9.09 9a3 3 0 0 1 5.83 1c0 2-3 3-3 3"/>
|
||||||
|
<line x1="12" y1="17" x2="12.01" y2="17"/>
|
||||||
|
</svg>
|
||||||
|
</span>
|
||||||
|
<span class="card-label">Question</span>
|
||||||
|
<span class="card-elapsed">{{ elapsed() }}</span>
|
||||||
|
</div>
|
||||||
|
<div class="card-body">
|
||||||
|
<div v-for="(q, qi) in questions" :key="qi" class="question-block">
|
||||||
|
<div class="question-text">
|
||||||
|
<span v-if="q.header" class="question-tag">{{ q.header }}</span>
|
||||||
|
{{ q.question }}
|
||||||
|
<span v-if="q.multiSelect" class="multi-hint">(multiple)</span>
|
||||||
|
</div>
|
||||||
|
<div v-if="q.options?.length" class="options-grid">
|
||||||
|
<button
|
||||||
|
v-for="opt in q.options"
|
||||||
|
:key="opt.label"
|
||||||
|
:class="['option-btn', { selected: selectedAnswers[q.question]?.has(opt.label) }]"
|
||||||
|
@click="toggleOption(q.question, opt.label, q.multiSelect)"
|
||||||
|
>
|
||||||
|
<span class="option-label">{{ opt.label }}</span>
|
||||||
|
<span v-if="opt.description" class="option-desc">{{ opt.description }}</span>
|
||||||
|
</button>
|
||||||
|
<button :class="['option-btn other', { selected: showCustom[q.question] }]" @click="toggleCustom(q.question)">
|
||||||
|
<span class="option-label">Other</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<Transition name="expand">
|
||||||
|
<div v-if="showCustom[q.question] || !q.options?.length" class="custom-input">
|
||||||
|
<textarea
|
||||||
|
v-model="customAnswers[q.question]"
|
||||||
|
class="edit-textarea"
|
||||||
|
:placeholder="q.options?.length ? 'Custom answer...' : 'Type your answer...'"
|
||||||
|
rows="2"
|
||||||
|
></textarea>
|
||||||
|
</div>
|
||||||
|
</Transition>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="card-actions">
|
||||||
|
<button class="btn btn-approve" @click="submitAnswers">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<line x1="22" y1="2" x2="11" y2="13"/><polygon points="22 2 15 22 11 13 2 9 22 2"/>
|
||||||
|
</svg>
|
||||||
|
Answer
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-allow-terminal" @click="allowQuestion">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<polyline points="4 17 10 11 4 5"/><line x1="12" y1="19" x2="20" y2="19"/>
|
||||||
|
</svg>
|
||||||
|
Ask in Terminal
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-reject" @click="emit('respond', request.requestId, 'deny')">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
|
<line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/>
|
||||||
|
</svg>
|
||||||
|
Dismiss
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ═══════ PERMISSION MODE (generic) ═══════ -->
|
||||||
|
<div v-else class="card permission-card">
|
||||||
|
<div class="card-header perm-header">
|
||||||
<span class="card-icon">
|
<span class="card-icon">
|
||||||
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
<path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"/>
|
<path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"/>
|
||||||
@@ -38,7 +268,6 @@ function elapsed(): string {
|
|||||||
<span class="card-label">Permission Request</span>
|
<span class="card-label">Permission Request</span>
|
||||||
<span class="card-elapsed">{{ elapsed() }}</span>
|
<span class="card-elapsed">{{ elapsed() }}</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="info-row" v-if="request.tool_name">
|
<div class="info-row" v-if="request.tool_name">
|
||||||
<span class="info-label">Tool</span>
|
<span class="info-label">Tool</span>
|
||||||
@@ -52,19 +281,34 @@ function elapsed(): string {
|
|||||||
<span class="info-label">Input</span>
|
<span class="info-label">Input</span>
|
||||||
<pre class="input-pre">{{ formatInput(request.tool_input) }}</pre>
|
<pre class="input-pre">{{ formatInput(request.tool_input) }}</pre>
|
||||||
</div>
|
</div>
|
||||||
|
<Transition name="expand">
|
||||||
|
<div v-if="showFreeResponse" class="edit-section">
|
||||||
|
<textarea v-model="freeText" class="edit-textarea" placeholder="Deny with a custom message..." rows="2"></textarea>
|
||||||
|
</div>
|
||||||
|
</Transition>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<button class="btn btn-allow" @click="emit('respond', request.requestId, 'allow')">
|
<button class="btn btn-allow" @click="emit('respond', request.requestId, 'allow')">
|
||||||
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5"><polyline points="20 6 9 17 4 12"/></svg>
|
||||||
<polyline points="20 6 9 17 4 12"/>
|
|
||||||
</svg>
|
|
||||||
Allow
|
Allow
|
||||||
</button>
|
</button>
|
||||||
|
<button class="btn btn-allow-always" @click="emit('respond', request.requestId, 'allowAlways')">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"/>
|
||||||
|
<polyline points="9 12 11 14 15 10"/>
|
||||||
|
</svg>
|
||||||
|
Always
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-free" @click="handleFreeResponse">
|
||||||
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
||||||
|
<path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7"/>
|
||||||
|
<path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z"/>
|
||||||
|
</svg>
|
||||||
|
{{ showFreeResponse ? 'Send' : 'Message' }}
|
||||||
|
</button>
|
||||||
<button class="btn btn-deny" @click="emit('respond', request.requestId, 'deny')">
|
<button class="btn btn-deny" @click="emit('respond', request.requestId, 'deny')">
|
||||||
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5">
|
||||||
<line x1="18" y1="6" x2="6" y2="18"/>
|
<line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/>
|
||||||
<line x1="6" y1="6" x2="18" y2="18"/>
|
|
||||||
</svg>
|
</svg>
|
||||||
Deny
|
Deny
|
||||||
</button>
|
</button>
|
||||||
@@ -73,9 +317,9 @@ function elapsed(): string {
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.permission-card {
|
/* ── Shared card base ── */
|
||||||
|
.card {
|
||||||
border: 1px solid var(--border-color);
|
border: 1px solid var(--border-color);
|
||||||
border-left: 3px solid #f59e0b;
|
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
background: var(--bg-secondary);
|
background: var(--bg-secondary);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
@@ -87,24 +331,17 @@ function elapsed(): string {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
padding: 0.5rem 0.75rem;
|
padding: 0.5rem 0.75rem;
|
||||||
background: rgba(245, 158, 11, 0.06);
|
|
||||||
border-bottom: 1px solid var(--border-color);
|
border-bottom: 1px solid var(--border-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
.card-icon {
|
|
||||||
color: #f59e0b;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.card-label {
|
.card-label {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: #f59e0b;
|
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
letter-spacing: 0.5px;
|
letter-spacing: 0.5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.card-icon { display: flex; align-items: center; }
|
||||||
.card-elapsed {
|
.card-elapsed {
|
||||||
margin-left: auto;
|
margin-left: auto;
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
@@ -119,25 +356,33 @@ function elapsed(): string {
|
|||||||
gap: 0.4rem;
|
gap: 0.4rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.info-row {
|
.card-actions {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
|
||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
|
padding: 0.5rem 0.75rem;
|
||||||
|
border-top: 1px solid var(--border-color);
|
||||||
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.info-label {
|
/* ── Permission (yellow) ── */
|
||||||
font-size: 11px;
|
.permission-card { border-left: 3px solid #f59e0b; }
|
||||||
color: var(--text-muted);
|
.perm-header { background: rgba(245, 158, 11, 0.06); }
|
||||||
font-weight: 500;
|
.perm-header .card-icon, .perm-header .card-label { color: #f59e0b; }
|
||||||
min-width: 40px;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.info-value {
|
/* ── Plan (purple) ── */
|
||||||
font-size: 13px;
|
.plan-card { border-left: 3px solid #8b5cf6; }
|
||||||
color: var(--text-primary);
|
.plan-header { background: rgba(139, 92, 246, 0.06); }
|
||||||
}
|
.plan-header .card-icon, .plan-header .card-label { color: #8b5cf6; }
|
||||||
|
|
||||||
|
/* ── Question (blue) ── */
|
||||||
|
.question-card { border-left: 3px solid #0ea5e9; }
|
||||||
|
.question-header { background: rgba(14, 165, 233, 0.06); }
|
||||||
|
.question-header .card-icon, .question-header .card-label { color: #0ea5e9; }
|
||||||
|
|
||||||
|
/* ── Info rows (permission) ── */
|
||||||
|
.info-row { display: flex; align-items: center; gap: 0.5rem; }
|
||||||
|
.info-label { font-size: 11px; color: var(--text-muted); font-weight: 500; min-width: 40px; flex-shrink: 0; }
|
||||||
|
.info-value { font-size: 13px; color: var(--text-primary); }
|
||||||
.tool-name {
|
.tool-name {
|
||||||
background: rgba(99, 102, 241, 0.1);
|
background: rgba(99, 102, 241, 0.1);
|
||||||
color: var(--accent, #6366f1);
|
color: var(--accent, #6366f1);
|
||||||
@@ -147,12 +392,7 @@ function elapsed(): string {
|
|||||||
font-family: 'SF Mono', 'Fira Code', monospace;
|
font-family: 'SF Mono', 'Fira Code', monospace;
|
||||||
}
|
}
|
||||||
|
|
||||||
.input-block {
|
.input-block { display: flex; flex-direction: column; gap: 0.25rem; }
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 0.25rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.input-pre {
|
.input-pre {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0.5rem;
|
padding: 0.5rem;
|
||||||
@@ -168,13 +408,81 @@ function elapsed(): string {
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.card-actions {
|
/* ── Plan body ── */
|
||||||
display: flex;
|
.plan-text { max-height: 250px; overflow-y: auto; }
|
||||||
gap: 0.5rem;
|
.plan-pre {
|
||||||
padding: 0.5rem 0.75rem;
|
margin: 0;
|
||||||
border-top: 1px solid var(--border-color);
|
padding: 0.5rem;
|
||||||
|
background: var(--bg-primary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-family: 'SF Mono', 'Fira Code', monospace;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
white-space: pre-wrap;
|
||||||
|
word-break: break-word;
|
||||||
|
line-height: 1.5;
|
||||||
}
|
}
|
||||||
|
.plan-empty { font-size: 13px; color: var(--text-muted); font-style: italic; padding: 0.5rem 0; }
|
||||||
|
|
||||||
|
/* ── Question body ── */
|
||||||
|
.question-block { display: flex; flex-direction: column; gap: 0.5rem; }
|
||||||
|
.question-block + .question-block { margin-top: 0.75rem; padding-top: 0.75rem; border-top: 1px solid var(--border-color); }
|
||||||
|
.question-text { font-size: 13px; color: var(--text-primary); font-weight: 500; line-height: 1.4; }
|
||||||
|
.question-tag {
|
||||||
|
display: inline-block;
|
||||||
|
background: rgba(14, 165, 233, 0.12);
|
||||||
|
color: #0ea5e9;
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 600;
|
||||||
|
padding: 0.1rem 0.35rem;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin-right: 0.3rem;
|
||||||
|
vertical-align: middle;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.3px;
|
||||||
|
}
|
||||||
|
.multi-hint { font-size: 11px; color: var(--text-muted); font-weight: 400; }
|
||||||
|
|
||||||
|
.options-grid { display: flex; flex-direction: column; gap: 0.35rem; }
|
||||||
|
.option-btn {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.15rem;
|
||||||
|
padding: 0.5rem 0.65rem;
|
||||||
|
background: var(--bg-primary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
text-align: left;
|
||||||
|
transition: all 0.15s;
|
||||||
|
}
|
||||||
|
.option-btn:hover { border-color: #0ea5e9; background: rgba(14, 165, 233, 0.04); }
|
||||||
|
.option-btn.selected { border-color: #0ea5e9; background: rgba(14, 165, 233, 0.1); }
|
||||||
|
.option-btn.other { border-style: dashed; }
|
||||||
|
.option-label { font-size: 13px; font-weight: 600; color: var(--text-primary); }
|
||||||
|
.option-desc { font-size: 11px; color: var(--text-muted); line-height: 1.3; }
|
||||||
|
|
||||||
|
.custom-input { margin-top: 0.25rem; }
|
||||||
|
|
||||||
|
/* ── Shared edit/textarea ── */
|
||||||
|
.edit-section { margin-top: 0.25rem; }
|
||||||
|
.edit-textarea {
|
||||||
|
width: 100%;
|
||||||
|
padding: 0.5rem;
|
||||||
|
background: var(--bg-primary);
|
||||||
|
border: 1px solid var(--border-color);
|
||||||
|
border-radius: 6px;
|
||||||
|
color: var(--text-primary);
|
||||||
|
font-size: 12px;
|
||||||
|
font-family: 'SF Mono', 'Fira Code', monospace;
|
||||||
|
resize: vertical;
|
||||||
|
outline: none;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
.edit-textarea:focus { border-color: var(--accent, #6366f1); }
|
||||||
|
|
||||||
|
/* ── Buttons ── */
|
||||||
.btn {
|
.btn {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -188,23 +496,34 @@ function elapsed(): string {
|
|||||||
transition: all 0.15s;
|
transition: all 0.15s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn-allow {
|
.btn-allow { background: #22c55e; color: white; }
|
||||||
background: #22c55e;
|
.btn-allow:hover { background: #16a34a; }
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.btn-allow:hover {
|
.btn-allow-always { background: #0ea5e9; color: white; }
|
||||||
background: #16a34a;
|
.btn-allow-always:hover { background: #0284c7; }
|
||||||
}
|
|
||||||
|
|
||||||
.btn-deny {
|
.btn-allow-terminal { background: #64748b; color: white; }
|
||||||
background: #ef4444;
|
.btn-allow-terminal:hover { background: #475569; }
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.btn-deny:hover {
|
.btn-approve { background: #22c55e; color: white; }
|
||||||
background: #dc2626;
|
.btn-approve:hover { background: #16a34a; }
|
||||||
}
|
|
||||||
|
.btn-edit { background: #8b5cf6; color: white; }
|
||||||
|
.btn-edit:hover { background: #7c3aed; }
|
||||||
|
|
||||||
|
.btn-free { background: #8b5cf6; color: white; }
|
||||||
|
.btn-free:hover { background: #7c3aed; }
|
||||||
|
|
||||||
|
.btn-deny { background: #ef4444; color: white; }
|
||||||
|
.btn-deny:hover { background: #dc2626; }
|
||||||
|
|
||||||
|
.btn-reject { background: #ef4444; color: white; }
|
||||||
|
.btn-reject:hover { background: #dc2626; }
|
||||||
|
|
||||||
|
/* ── Transitions ── */
|
||||||
|
.expand-enter-active, .expand-leave-active { transition: all 0.2s ease; overflow: hidden; }
|
||||||
|
.expand-enter-from, .expand-leave-to { opacity: 0; max-height: 0; }
|
||||||
|
.expand-enter-to, .expand-leave-from { max-height: 200px; }
|
||||||
|
|
||||||
@keyframes slideIn {
|
@keyframes slideIn {
|
||||||
from { opacity: 0; transform: translateY(-8px); }
|
from { opacity: 0; transform: translateY(-8px); }
|
||||||
|
|||||||
@@ -155,13 +155,13 @@ async function fetchPending() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function respondPermission(requestId: string, decision: 'allow' | 'deny') {
|
async function respondPermission(requestId: string, decision: string, reason?: string) {
|
||||||
console.log(`[GlobalApproval] Responding permission ${requestId}: ${decision}`)
|
console.log(`[GlobalApproval] Responding permission ${requestId}: ${decision}${reason ? ' reason=' + reason : ''}`)
|
||||||
try {
|
try {
|
||||||
await fetch('/api/hooks-approval/respond', {
|
await fetch('/api/hooks-approval/respond', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify({ requestId, decision })
|
body: JSON.stringify({ requestId, decision, reason })
|
||||||
})
|
})
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error('[GlobalApproval] Failed to respond permission:', e)
|
console.error('[GlobalApproval] Failed to respond permission:', e)
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
# Long-poll hooks-approval for PermissionRequest decisions
|
# Long-poll hooks-approval for PermissionRequest decisions
|
||||||
# Reads hook stdin, POSTs to backend, waits up to 125s for UI response
|
# Reads hook stdin, POSTs to backend, waits up to 125s for UI response
|
||||||
# Returns hookSpecificOutput JSON per Claude Code PermissionRequest docs
|
# Returns hookSpecificOutput JSON per Claude Code PermissionRequest docs
|
||||||
param([string]$agent = "ejecutor")
|
param([string]$agent = "")
|
||||||
$logFile = "$PSScriptRoot/../.claude-$agent/debug/hooks.log"
|
if ($agent) { $logFile = "$PSScriptRoot/../.claude-$agent/debug/hooks.log" }
|
||||||
|
else { $logFile = "$PSScriptRoot/../.claude/debug/hooks.log"; $agent = "local" }
|
||||||
$ts = Get-Date -Format "HH:mm:ss.fff"
|
$ts = Get-Date -Format "HH:mm:ss.fff"
|
||||||
$b = [Console]::In.ReadToEnd()
|
$b = [Console]::In.ReadToEnd()
|
||||||
Add-Content $logFile "[$ts] [PERM] Hook fired for agent=$agent stdin_len=$($b.Length)"
|
Add-Content $logFile "[$ts] [PERM] Hook fired for agent=$agent stdin_len=$($b.Length)"
|
||||||
|
|||||||
@@ -3,8 +3,9 @@
|
|||||||
# Returns Stop decision JSON per Claude Code docs:
|
# Returns Stop decision JSON per Claude Code docs:
|
||||||
# { "decision": "block", "reason": "..." } to continue implementing
|
# { "decision": "block", "reason": "..." } to continue implementing
|
||||||
# {} or empty to let Claude stop
|
# {} or empty to let Claude stop
|
||||||
param([string]$agent = "ejecutor")
|
param([string]$agent = "")
|
||||||
$logFile = "$PSScriptRoot/../.claude-$agent/debug/hooks.log"
|
if ($agent) { $logFile = "$PSScriptRoot/../.claude-$agent/debug/hooks.log" }
|
||||||
|
else { $logFile = "$PSScriptRoot/../.claude/debug/hooks.log"; $agent = "local" }
|
||||||
$ts = Get-Date -Format "HH:mm:ss.fff"
|
$ts = Get-Date -Format "HH:mm:ss.fff"
|
||||||
$b = [Console]::In.ReadToEnd()
|
$b = [Console]::In.ReadToEnd()
|
||||||
Add-Content $logFile "[$ts] [PLAN] Hook fired for agent=$agent stdin_len=$($b.Length)"
|
Add-Content $logFile "[$ts] [PLAN] Hook fired for agent=$agent stdin_len=$($b.Length)"
|
||||||
|
|||||||
@@ -196,14 +196,14 @@ export async function handleHooksApprovalRespond(req: Request): Promise<Response
|
|||||||
if (req.method !== 'POST') return null
|
if (req.method !== 'POST') return null
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const body = await req.json() as { requestId: string; decision: 'allow' | 'deny' }
|
const body = await req.json() as { requestId: string; decision: string; reason?: string }
|
||||||
|
|
||||||
if (!body.requestId || !body.decision) {
|
if (!body.requestId || !body.decision) {
|
||||||
return errorResponse('Missing requestId or decision', 400)
|
return errorResponse('Missing requestId or decision', 400)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!['allow', 'deny'].includes(body.decision)) {
|
if (!['allow', 'allowAlways', 'deny'].includes(body.decision)) {
|
||||||
return errorResponse('Decision must be "allow" or "deny"', 400)
|
return errorResponse('Decision must be "allow", "allowAlways", or "deny"', 400)
|
||||||
}
|
}
|
||||||
|
|
||||||
const pending = pendingRequests.get(body.requestId)
|
const pending = pendingRequests.get(body.requestId)
|
||||||
@@ -212,30 +212,42 @@ export async function handleHooksApprovalRespond(req: Request): Promise<Response
|
|||||||
return errorResponse('Permission request not found or expired', 404)
|
return errorResponse('Permission request not found or expired', 404)
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(`[HooksApproval] Responding permission ${body.requestId}: ${body.decision}`)
|
console.log(`[HooksApproval] Responding permission ${body.requestId}: ${body.decision}${body.reason ? ' reason=' + body.reason : ''}`)
|
||||||
clearTimeout(pending.timer)
|
clearTimeout(pending.timer)
|
||||||
pendingRequests.delete(body.requestId)
|
pendingRequests.delete(body.requestId)
|
||||||
|
|
||||||
// Build hookSpecificOutput per PermissionRequest docs:
|
// Build hookSpecificOutput per PermissionRequest docs
|
||||||
// hookSpecificOutput.decision.behavior = "allow" | "deny"
|
let hookOutput: Record<string, unknown>
|
||||||
const hookOutput: Record<string, unknown> = body.decision === 'allow'
|
|
||||||
? {
|
if (body.decision === 'allow') {
|
||||||
hookSpecificOutput: {
|
hookOutput = {
|
||||||
hookEventName: 'PermissionRequest',
|
hookSpecificOutput: {
|
||||||
decision: {
|
hookEventName: 'PermissionRequest',
|
||||||
behavior: 'allow'
|
decision: { behavior: 'allow' }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
} else if (body.decision === 'allowAlways') {
|
||||||
: {
|
// Allow + add tool to allowed permissions for session
|
||||||
hookSpecificOutput: {
|
const toolName = pending.payload.tool_name as string || '*'
|
||||||
hookEventName: 'PermissionRequest',
|
hookOutput = {
|
||||||
decision: {
|
hookSpecificOutput: {
|
||||||
behavior: 'deny',
|
hookEventName: 'PermissionRequest',
|
||||||
message: 'Denied via UI'
|
decision: { behavior: 'allow' },
|
||||||
}
|
allowedPermissions: [{ tool_name: toolName }]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// deny — with optional custom message
|
||||||
|
hookOutput = {
|
||||||
|
hookSpecificOutput: {
|
||||||
|
hookEventName: 'PermissionRequest',
|
||||||
|
decision: {
|
||||||
|
behavior: 'deny',
|
||||||
|
message: body.reason || 'Denied via UI'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pending.resolve(hookOutput)
|
pending.resolve(hookOutput)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user