mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-13 09:01:54 +00:00
Product rename across every surface: module/file names (blueprint_catalog, tools/blueprints, blueprint_cmd), slash command /cron-recipe -> /blueprint (alias /bp), dashboard API /api/cron/blueprints, desktop deep-link hermes://blueprint/<key>, docs catalog page + extract script, and the skill frontmatter block metadata.hermes.blueprint. No behavior change.
188 lines
6 KiB
Python
188 lines
6 KiB
Python
"""Tests for the blueprints layer (skill frontmatter <-> cron automation bridge).
|
|
|
|
A blueprint is a skill with a metadata.hermes.blueprint block. These verify parsing,
|
|
the create-job bridge, and the export round-trip without touching the real
|
|
cron store.
|
|
"""
|
|
|
|
import sys
|
|
from pathlib import Path
|
|
from unittest.mock import patch
|
|
|
|
import pytest
|
|
|
|
from tools.blueprints import (
|
|
BlueprintError,
|
|
BlueprintSpec,
|
|
create_blueprint_job,
|
|
export_blueprint,
|
|
parse_blueprint,
|
|
blueprint_spec_for_installed,
|
|
)
|
|
|
|
|
|
BLUEPRINT_SKILL = """---
|
|
name: morning-brief
|
|
description: Summarize unread email and calendar every morning.
|
|
version: 1.0.0
|
|
metadata:
|
|
hermes:
|
|
tags: [blueprint, email]
|
|
blueprint:
|
|
schedule: "0 8 * * *"
|
|
deliver: telegram
|
|
prompt: "Summarize my unread email and today's calendar."
|
|
---
|
|
|
|
# Morning Brief
|
|
|
|
Every morning, gather unread email and the day's calendar and send a digest.
|
|
"""
|
|
|
|
PLAIN_SKILL = """---
|
|
name: not-a-blueprint
|
|
description: Just a regular skill.
|
|
metadata:
|
|
hermes:
|
|
tags: [misc]
|
|
---
|
|
|
|
# Not a blueprint
|
|
"""
|
|
|
|
MALFORMED_BLUEPRINT = """---
|
|
name: broken
|
|
description: Blueprint with no schedule.
|
|
metadata:
|
|
hermes:
|
|
blueprint:
|
|
deliver: origin
|
|
---
|
|
|
|
# Broken
|
|
"""
|
|
|
|
|
|
class TestParseBlueprint:
|
|
def test_parses_full_blueprint(self):
|
|
spec = parse_blueprint(BLUEPRINT_SKILL)
|
|
assert spec is not None
|
|
assert spec.skill_name == "morning-brief"
|
|
assert spec.schedule == "0 8 * * *"
|
|
assert spec.deliver == "telegram"
|
|
assert spec.prompt is not None and spec.prompt.startswith("Summarize")
|
|
|
|
def test_plain_skill_is_not_a_blueprint(self):
|
|
assert parse_blueprint(PLAIN_SKILL) is None
|
|
|
|
def test_no_frontmatter_is_not_a_blueprint(self):
|
|
assert parse_blueprint("just some text, no frontmatter") is None
|
|
|
|
def test_missing_schedule_raises(self):
|
|
with pytest.raises(BlueprintError):
|
|
parse_blueprint(MALFORMED_BLUEPRINT)
|
|
|
|
def test_blueprint_not_mapping_raises(self):
|
|
bad = "---\nname: x\nmetadata:\n hermes:\n blueprint: not-a-dict\n---\n\nbody"
|
|
with pytest.raises(BlueprintError):
|
|
parse_blueprint(bad)
|
|
|
|
def test_deliver_defaults_to_origin(self):
|
|
skill = (
|
|
"---\nname: r\ndescription: d\nmetadata:\n hermes:\n"
|
|
' blueprint:\n schedule: "every 1h"\n---\n\nbody'
|
|
)
|
|
spec = parse_blueprint(skill)
|
|
assert spec is not None
|
|
assert spec.deliver == "origin"
|
|
|
|
|
|
class TestBlueprintSpecForInstalled:
|
|
def test_finds_and_parses_installed_blueprint(self, tmp_path):
|
|
skills_dir = tmp_path / "skills"
|
|
rec_dir = skills_dir / "productivity" / "morning-brief"
|
|
rec_dir.mkdir(parents=True)
|
|
(rec_dir / "SKILL.md").write_text(BLUEPRINT_SKILL, encoding="utf-8")
|
|
|
|
with patch("tools.skills_hub.SKILLS_DIR", skills_dir):
|
|
spec = blueprint_spec_for_installed("morning-brief")
|
|
assert spec is not None
|
|
assert spec.schedule == "0 8 * * *"
|
|
|
|
def test_missing_skill_returns_none(self, tmp_path):
|
|
skills_dir = tmp_path / "skills"
|
|
skills_dir.mkdir()
|
|
with patch("tools.skills_hub.SKILLS_DIR", skills_dir):
|
|
assert blueprint_spec_for_installed("nope") is None
|
|
|
|
def test_plain_skill_returns_none(self, tmp_path):
|
|
skills_dir = tmp_path / "skills"
|
|
d = skills_dir / "misc" / "not-a-blueprint"
|
|
d.mkdir(parents=True)
|
|
(d / "SKILL.md").write_text(PLAIN_SKILL, encoding="utf-8")
|
|
with patch("tools.skills_hub.SKILLS_DIR", skills_dir):
|
|
assert blueprint_spec_for_installed("not-a-blueprint") is None
|
|
|
|
|
|
class TestCreateBlueprintJob:
|
|
def test_bridges_to_create_job(self):
|
|
spec = parse_blueprint(BLUEPRINT_SKILL)
|
|
assert spec is not None
|
|
captured = {}
|
|
|
|
def fake_create_job(**kwargs):
|
|
captured.update(kwargs)
|
|
return {"id": "abc123", **kwargs}
|
|
|
|
with patch("cron.jobs.create_job", fake_create_job):
|
|
job = create_blueprint_job(spec, origin={"platform": "telegram"})
|
|
|
|
assert captured["schedule"] == "0 8 * * *"
|
|
assert captured["skills"] == ["morning-brief"]
|
|
assert captured["deliver"] == "telegram"
|
|
assert captured["prompt"].startswith("Summarize")
|
|
assert job["id"] == "abc123"
|
|
|
|
|
|
class TestExportBlueprint:
|
|
def test_round_trips_job_to_skill_md(self):
|
|
job = {
|
|
"name": "My Morning Brief",
|
|
"schedule_display": "0 8 * * *",
|
|
"skills": ["morning-brief"],
|
|
"deliver": "telegram",
|
|
"prompt": "Summarize my unread email.",
|
|
}
|
|
md = export_blueprint(job, "# Morning Brief\n\nDoes the morning digest.")
|
|
# The exported SKILL.md must itself parse back as a blueprint.
|
|
spec = parse_blueprint(md)
|
|
assert spec is not None
|
|
assert spec.schedule == "0 8 * * *"
|
|
assert spec.deliver == "telegram"
|
|
# Name is sanitized to a valid skill identifier.
|
|
assert spec.skill_name == "my-morning-brief"
|
|
|
|
def test_export_has_blueprint_tag(self):
|
|
job = {"name": "x", "schedule_display": "every 2h", "skills": ["x"]}
|
|
md = export_blueprint(job, "body")
|
|
assert "blueprint" in md
|
|
assert "automation" in md
|
|
|
|
def test_export_interval_job_without_display(self):
|
|
# Regression: parse_schedule stores interval periods as "minutes" —
|
|
# exporting a job with only the parsed schedule dict must round-trip
|
|
# the real interval, not fall back to the daily default.
|
|
job = {
|
|
"name": "poller",
|
|
"schedule": {"kind": "interval", "minutes": 30},
|
|
"skills": ["poller"],
|
|
}
|
|
md = export_blueprint(job, "body")
|
|
spec = parse_blueprint(md)
|
|
assert spec is not None
|
|
assert spec.schedule == "every 30m"
|
|
|
|
job["schedule"] = {"kind": "interval", "minutes": 120}
|
|
spec = parse_blueprint(export_blueprint(job, "body"))
|
|
assert spec is not None
|
|
assert spec.schedule == "every 2h"
|