"""Tests for /learn — open-ended skill distillation. Covers the shared prompt builder (agent.learn_prompt.build_learn_prompt) and the slash-command registry wiring. /learn has no engine and no model tool: it builds a standards-guided prompt that the live agent runs as a normal turn, so these are the load-bearing behavior contracts. """ from agent.learn_prompt import build_learn_prompt, _AUTHORING_STANDARDS class TestBuildLearnPrompt: def test_embeds_the_user_request_verbatim(self): req = "the REST client in ~/projects/acme-sdk, focus on auth" prompt = build_learn_prompt(req) assert req in prompt def test_always_includes_the_authoring_standards(self): # The standards are what make distilled skills match house style; # they must travel with every prompt regardless of input. for req in ["", "a url https://x/y", "what we just did"]: assert _AUTHORING_STANDARDS in build_learn_prompt(req) def test_instructs_saving_via_skill_manage_not_a_raw_file(self): prompt = build_learn_prompt("learn the thing") assert "skill_manage" in prompt def test_references_gather_tools_for_open_ended_sourcing(self): # Open-ended sourcing relies on the agent's own tools, named so it # knows dirs/URLs/conversation/paste all route through existing tools. prompt = build_learn_prompt("learn from somewhere") for tool in ("read_file", "search_files", "web_extract"): assert tool in prompt def test_empty_request_falls_back_to_the_conversation(self): # Bare /learn should distill "what we just did", not error. prompt = build_learn_prompt("") assert "conversation" in prompt.lower() # And still carries the standards + save instruction. assert "skill_manage" in prompt def test_whitespace_only_request_is_treated_as_empty(self): assert build_learn_prompt(" \n ") == build_learn_prompt("") def test_description_length_rule_is_in_the_standards(self): # The single most-violated rule must be explicit in the prompt. assert "60" in _AUTHORING_STANDARDS class TestLearnRegistryWiring: def test_learn_is_registered_and_resolves(self): from hermes_cli.commands import resolve_command cmd = resolve_command("learn") assert cmd is not None assert cmd.name == "learn" def test_learn_is_in_tools_and_skills_category(self): from hermes_cli.commands import resolve_command assert resolve_command("learn").category == "Tools & Skills" def test_learn_works_on_the_gateway(self): # /learn must reach the gateway runner (it's a both-surfaces command), # not be CLI-only. from hermes_cli.commands import GATEWAY_KNOWN_COMMANDS assert "learn" in GATEWAY_KNOWN_COMMANDS def test_learn_is_not_cli_only(self): from hermes_cli.commands import resolve_command assert not resolve_command("learn").cli_only