Commit 08fe14c
* feat: implement command hook system (#32)
Add extensibility hook system matching Claude Code's design. Hooks let
users run shell commands at tool lifecycle points (PreToolUse, PostToolUse,
PostToolUseFailure) to enforce policies, audit actions, or modify inputs.
- 3 core abstractions: HookEvent (sealed), HookResult (sealed), HookExecutor
- HookConfig/HookMatcher/HookRegistry for config-driven hook resolution
- CommandHookExecutor: ProcessBuilder with JSON stdin/stdout, exit code
semantics (0=proceed, 2=block, other=non-blocking error), timeout
- PreToolUse hooks run before permission check; can block or modify input
- PostToolUse/PostToolUseFailure hooks fire async on virtual threads
- Config loaded from ~/.aceclaw/config.json and project config (appending)
- 24 new tests across 3 test classes (unit + E2E integration)
* fix: read stdout/stderr concurrently to prevent pipe deadlock on Linux
Process pipe buffers are finite (~64KB on Linux). If the hook command
writes more than the buffer capacity before we start reading, it blocks
on write while we block on waitFor() — a classic deadlock. Fix by
draining stdout and stderr on virtual threads concurrently with waitFor.
* fix: start stdout/stderr readers before stdin write to prevent data loss
On fast CI runners, the hook process may exit before stdin write completes.
If the process doesn't read stdin, the write throws broken pipe IOException
which previously aborted the entire executeOne() method — stdout/stderr
reader threads were never started, so the hook's output was lost.
Fix: start reader threads FIRST, then write stdin with a separate
try-catch so broken pipe is non-fatal. This ensures we always capture
the process output regardless of stdin write success.
* refactor: address CodeRabbit review feedback on PR #41
- Convert HookMatcherFormat/HookConfigFormat from mutable classes to records
- Use session's projectPath for hook cwd instead of daemon's workingDir
- Replace Thread.sleep with bounded polling in HookIntegrationTest
- Add clarifying comment on exit-0 deny test contract
* fix: guard against null in hook merge and PostToolUseFailure
- AceClawConfig: skip null/empty hook lists during config merge to prevent NPE
- StreamingAgentHandler: ensure non-null error string for PostToolUseFailure
so firePostHookAsync dispatches the correct event type
---------
Co-authored-by: Xinhua Gu <xinhua.gu@arz.at>
1 parent fa976fe commit 08fe14c
13 files changed
Lines changed: 1746 additions & 11 deletions
File tree
- aceclaw-core/src/main/java/dev/aceclaw/core/agent
- aceclaw-daemon/src
- main/java/dev/aceclaw/daemon
- test/java/dev/aceclaw/daemon
Lines changed: 65 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
| 8 | + | |
| 9 | + | |
| 10 | + | |
| 11 | + | |
| 12 | + | |
| 13 | + | |
| 14 | + | |
| 15 | + | |
| 16 | + | |
| 17 | + | |
| 18 | + | |
| 19 | + | |
| 20 | + | |
| 21 | + | |
| 22 | + | |
| 23 | + | |
| 24 | + | |
| 25 | + | |
| 26 | + | |
| 27 | + | |
| 28 | + | |
| 29 | + | |
| 30 | + | |
| 31 | + | |
| 32 | + | |
| 33 | + | |
| 34 | + | |
| 35 | + | |
| 36 | + | |
| 37 | + | |
| 38 | + | |
| 39 | + | |
| 40 | + | |
| 41 | + | |
| 42 | + | |
| 43 | + | |
| 44 | + | |
| 45 | + | |
| 46 | + | |
| 47 | + | |
| 48 | + | |
| 49 | + | |
| 50 | + | |
| 51 | + | |
| 52 | + | |
| 53 | + | |
| 54 | + | |
| 55 | + | |
| 56 | + | |
| 57 | + | |
| 58 | + | |
| 59 | + | |
| 60 | + | |
| 61 | + | |
| 62 | + | |
| 63 | + | |
| 64 | + | |
| 65 | + | |
Lines changed: 21 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
| 8 | + | |
| 9 | + | |
| 10 | + | |
| 11 | + | |
| 12 | + | |
| 13 | + | |
| 14 | + | |
| 15 | + | |
| 16 | + | |
| 17 | + | |
| 18 | + | |
| 19 | + | |
| 20 | + | |
| 21 | + | |
Lines changed: 44 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
| 8 | + | |
| 9 | + | |
| 10 | + | |
| 11 | + | |
| 12 | + | |
| 13 | + | |
| 14 | + | |
| 15 | + | |
| 16 | + | |
| 17 | + | |
| 18 | + | |
| 19 | + | |
| 20 | + | |
| 21 | + | |
| 22 | + | |
| 23 | + | |
| 24 | + | |
| 25 | + | |
| 26 | + | |
| 27 | + | |
| 28 | + | |
| 29 | + | |
| 30 | + | |
| 31 | + | |
| 32 | + | |
| 33 | + | |
| 34 | + | |
| 35 | + | |
| 36 | + | |
| 37 | + | |
| 38 | + | |
| 39 | + | |
| 40 | + | |
| 41 | + | |
| 42 | + | |
| 43 | + | |
| 44 | + | |
Lines changed: 42 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
8 | 8 | | |
9 | 9 | | |
10 | 10 | | |
| 11 | + | |
| 12 | + | |
| 13 | + | |
11 | 14 | | |
12 | 15 | | |
13 | 16 | | |
| |||
62 | 65 | | |
63 | 66 | | |
64 | 67 | | |
| 68 | + | |
65 | 69 | | |
66 | 70 | | |
67 | 71 | | |
| |||
260 | 264 | | |
261 | 265 | | |
262 | 266 | | |
| 267 | + | |
| 268 | + | |
| 269 | + | |
| 270 | + | |
| 271 | + | |
| 272 | + | |
| 273 | + | |
| 274 | + | |
263 | 275 | | |
264 | 276 | | |
265 | 277 | | |
| |||
364 | 376 | | |
365 | 377 | | |
366 | 378 | | |
| 379 | + | |
| 380 | + | |
| 381 | + | |
| 382 | + | |
| 383 | + | |
| 384 | + | |
| 385 | + | |
| 386 | + | |
| 387 | + | |
| 388 | + | |
| 389 | + | |
| 390 | + | |
| 391 | + | |
| 392 | + | |
| 393 | + | |
367 | 394 | | |
368 | 395 | | |
369 | 396 | | |
| |||
425 | 452 | | |
426 | 453 | | |
427 | 454 | | |
| 455 | + | |
428 | 456 | | |
| 457 | + | |
| 458 | + | |
| 459 | + | |
| 460 | + | |
| 461 | + | |
| 462 | + | |
| 463 | + | |
| 464 | + | |
| 465 | + | |
| 466 | + | |
| 467 | + | |
| 468 | + | |
| 469 | + | |
| 470 | + | |
429 | 471 | | |
Lines changed: 13 additions & 1 deletion
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
306 | 306 | | |
307 | 307 | | |
308 | 308 | | |
309 | | - | |
| 309 | + | |
| 310 | + | |
| 311 | + | |
| 312 | + | |
| 313 | + | |
| 314 | + | |
| 315 | + | |
| 316 | + | |
| 317 | + | |
| 318 | + | |
| 319 | + | |
| 320 | + | |
| 321 | + | |
310 | 322 | | |
311 | 323 | | |
312 | 324 | | |
| |||
0 commit comments