mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-24 10:52:21 +00:00
Plugins could observe session/tool/approval lifecycle but had no way to observe kanban task transitions. Adds three observer hooks fired by the board's claim/complete/block transitions: - kanban_task_claimed (dispatcher process, before worker spawn) - kanban_task_completed (worker process, carries summary) - kanban_task_blocked (worker process, carries reason) Each fires AFTER the DB write txn commits, so a plugin observes durable state and a slow/hanging callback can never hold the SQLite write lock. All firing is best-effort: a raising hook is logged and swallowed and never breaks a board transition. profile_name is resolved from HERMES_HOME so dispatcher- and worker-side hooks carry the right profile. Requested by @Smithangshu on Discord.
135 lines
4.4 KiB
Python
135 lines
4.4 KiB
Python
"""Tests for kanban lifecycle plugin hooks.
|
|
|
|
Verifies that claim/complete/block transitions fire the
|
|
kanban_task_claimed / kanban_task_completed / kanban_task_blocked plugin
|
|
hooks AFTER the board DB change is committed, with the documented kwargs,
|
|
and that a misbehaving hook callback never breaks the transition.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from hermes_cli import kanban_db as kb
|
|
from hermes_cli.plugins import VALID_HOOKS, get_plugin_manager
|
|
|
|
|
|
@pytest.fixture
|
|
def kanban_home(tmp_path, monkeypatch):
|
|
home = tmp_path / ".hermes"
|
|
home.mkdir()
|
|
monkeypatch.setenv("HERMES_HOME", str(home))
|
|
monkeypatch.setattr(Path, "home", lambda: tmp_path)
|
|
kb.init_db()
|
|
return home
|
|
|
|
|
|
@pytest.fixture
|
|
def captured_hooks(monkeypatch):
|
|
"""Register capturing callbacks for the three kanban lifecycle hooks.
|
|
|
|
Patches the plugin manager's _hooks dict directly (the same registry
|
|
invoke_hook reads) and restores it afterward.
|
|
"""
|
|
mgr = get_plugin_manager()
|
|
events: list[tuple[str, dict]] = []
|
|
saved = {k: list(v) for k, v in mgr._hooks.items()}
|
|
for hook in ("kanban_task_claimed", "kanban_task_completed", "kanban_task_blocked"):
|
|
mgr._hooks.setdefault(hook, []).append(
|
|
lambda _h=hook, **kw: events.append((_h, kw))
|
|
)
|
|
try:
|
|
yield events
|
|
finally:
|
|
mgr._hooks = saved
|
|
|
|
|
|
def test_hooks_are_registered_as_valid():
|
|
"""The three lifecycle hook names are part of VALID_HOOKS."""
|
|
assert "kanban_task_claimed" in VALID_HOOKS
|
|
assert "kanban_task_completed" in VALID_HOOKS
|
|
assert "kanban_task_blocked" in VALID_HOOKS
|
|
|
|
|
|
def test_claim_fires_hook(kanban_home, captured_hooks):
|
|
conn = kb.connect()
|
|
try:
|
|
tid = kb.create_task(conn, title="t", assignee="worker")
|
|
claimed = kb.claim_task(conn, tid)
|
|
assert claimed is not None
|
|
finally:
|
|
conn.close()
|
|
fired = [e for e in captured_hooks if e[0] == "kanban_task_claimed"]
|
|
assert len(fired) == 1
|
|
kw = fired[0][1]
|
|
assert kw["task_id"] == tid
|
|
assert kw["assignee"] == "worker"
|
|
assert "profile_name" in kw
|
|
assert kw["run_id"] is not None
|
|
|
|
|
|
def test_complete_fires_hook_with_summary(kanban_home, captured_hooks):
|
|
conn = kb.connect()
|
|
try:
|
|
tid = kb.create_task(conn, title="t", assignee="worker")
|
|
kb.claim_task(conn, tid)
|
|
assert kb.complete_task(conn, tid, summary="all done")
|
|
finally:
|
|
conn.close()
|
|
fired = [e for e in captured_hooks if e[0] == "kanban_task_completed"]
|
|
assert len(fired) == 1
|
|
kw = fired[0][1]
|
|
assert kw["task_id"] == tid
|
|
assert kw["summary"] == "all done"
|
|
assert kw["assignee"] == "worker"
|
|
|
|
|
|
def test_block_fires_hook_with_reason(kanban_home, captured_hooks):
|
|
conn = kb.connect()
|
|
try:
|
|
tid = kb.create_task(conn, title="t", assignee="worker")
|
|
kb.claim_task(conn, tid)
|
|
assert kb.block_task(conn, tid, reason="needs human")
|
|
finally:
|
|
conn.close()
|
|
fired = [e for e in captured_hooks if e[0] == "kanban_task_blocked"]
|
|
assert len(fired) == 1
|
|
kw = fired[0][1]
|
|
assert kw["task_id"] == tid
|
|
assert kw["reason"] == "needs human"
|
|
|
|
|
|
def test_no_hook_on_failed_transition(kanban_home, captured_hooks):
|
|
"""complete_task on an unclaimed/nonexistent task fires no hook."""
|
|
conn = kb.connect()
|
|
try:
|
|
# Completing a task that doesn't exist returns False without firing.
|
|
assert kb.complete_task(conn, "t_doesnotexist", summary="x") is False
|
|
finally:
|
|
conn.close()
|
|
assert [e for e in captured_hooks if e[0] == "kanban_task_completed"] == []
|
|
|
|
|
|
def test_misbehaving_hook_does_not_break_transition(kanban_home, monkeypatch):
|
|
"""A hook callback that raises must not break the board transition."""
|
|
mgr = get_plugin_manager()
|
|
saved = {k: list(v) for k, v in mgr._hooks.items()}
|
|
|
|
def _boom(**kw):
|
|
raise RuntimeError("plugin exploded")
|
|
|
|
mgr._hooks.setdefault("kanban_task_completed", []).append(_boom)
|
|
try:
|
|
conn = kb.connect()
|
|
try:
|
|
tid = kb.create_task(conn, title="t", assignee="worker")
|
|
kb.claim_task(conn, tid)
|
|
# Despite the raising hook, completion succeeds and persists.
|
|
assert kb.complete_task(conn, tid, summary="ok") is True
|
|
assert kb.get_task(conn, tid).status == "done"
|
|
finally:
|
|
conn.close()
|
|
finally:
|
|
mgr._hooks = saved
|