Centralize the skill → slash command registration that Telegram already had
in commands.py so Discord uses the exact same priority system, filtering,
and cap enforcement:
1. Core/built-in commands (never trimmed)
2. Plugin commands (never trimmed)
3. Skill commands (fill remaining slots, alphabetical, only tier trimmed)
Changes:
hermes_cli/commands.py:
- Rename _TG_NAME_LIMIT → _CMD_NAME_LIMIT (32 chars shared by both platforms)
- Rename _clamp_telegram_names → _clamp_command_names (generic)
- Extract _collect_gateway_skill_entries() — shared plugin + skill
collection with platform filtering, name sanitization, description
truncation, and cap enforcement
- Refactor telegram_menu_commands() to use the shared helper
- Add discord_skill_commands() that returns (name, desc, cmd_key) triples
- Preserve _sanitize_telegram_name() for Telegram-specific name cleaning
gateway/platforms/discord.py:
- Call discord_skill_commands() from _register_slash_commands()
- Create app_commands.Command per skill entry with cmd_key callback
- Respect 100-command global Discord limit
- Log warning when skills are skipped due to cap
Backward-compat aliases preserved for _TG_NAME_LIMIT and
_clamp_telegram_names.
Tests: 9 new tests (7 Discord + 2 backward-compat), 98 total pass.
Inspired by PR #5498 (sprmn24). Closes #5480.
992 lines
39 KiB
Python
992 lines
39 KiB
Python
"""Tests for the central command registry and autocomplete."""
|
|
|
|
from prompt_toolkit.completion import CompleteEvent
|
|
from prompt_toolkit.document import Document
|
|
|
|
from hermes_cli.commands import (
|
|
COMMAND_REGISTRY,
|
|
COMMANDS,
|
|
COMMANDS_BY_CATEGORY,
|
|
CommandDef,
|
|
GATEWAY_KNOWN_COMMANDS,
|
|
SUBCOMMANDS,
|
|
SlashCommandAutoSuggest,
|
|
SlashCommandCompleter,
|
|
_CMD_NAME_LIMIT,
|
|
_TG_NAME_LIMIT,
|
|
_clamp_command_names,
|
|
_clamp_telegram_names,
|
|
_sanitize_telegram_name,
|
|
discord_skill_commands,
|
|
gateway_help_lines,
|
|
resolve_command,
|
|
slack_subcommand_map,
|
|
telegram_bot_commands,
|
|
telegram_menu_commands,
|
|
)
|
|
|
|
|
|
def _completions(completer: SlashCommandCompleter, text: str):
|
|
return list(
|
|
completer.get_completions(
|
|
Document(text=text),
|
|
CompleteEvent(completion_requested=True),
|
|
)
|
|
)
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# CommandDef registry tests
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestCommandRegistry:
|
|
def test_registry_is_nonempty(self):
|
|
assert len(COMMAND_REGISTRY) > 30
|
|
|
|
def test_every_entry_is_commanddef(self):
|
|
for entry in COMMAND_REGISTRY:
|
|
assert isinstance(entry, CommandDef), f"Unexpected type: {type(entry)}"
|
|
|
|
def test_no_duplicate_canonical_names(self):
|
|
names = [cmd.name for cmd in COMMAND_REGISTRY]
|
|
assert len(names) == len(set(names)), f"Duplicate names: {[n for n in names if names.count(n) > 1]}"
|
|
|
|
def test_no_alias_collides_with_canonical_name(self):
|
|
"""An alias must not shadow another command's canonical name."""
|
|
canonical_names = {cmd.name for cmd in COMMAND_REGISTRY}
|
|
for cmd in COMMAND_REGISTRY:
|
|
for alias in cmd.aliases:
|
|
if alias in canonical_names:
|
|
# reset -> new is intentional (reset IS an alias for new)
|
|
target = next(c for c in COMMAND_REGISTRY if c.name == alias)
|
|
# This should only happen if the alias points to the same entry
|
|
assert resolve_command(alias).name == cmd.name or alias == cmd.name, \
|
|
f"Alias '{alias}' of '{cmd.name}' shadows canonical '{target.name}'"
|
|
|
|
def test_every_entry_has_valid_category(self):
|
|
valid_categories = {"Session", "Configuration", "Tools & Skills", "Info", "Exit"}
|
|
for cmd in COMMAND_REGISTRY:
|
|
assert cmd.category in valid_categories, f"{cmd.name} has invalid category '{cmd.category}'"
|
|
|
|
def test_cli_only_and_gateway_only_are_mutually_exclusive(self):
|
|
for cmd in COMMAND_REGISTRY:
|
|
assert not (cmd.cli_only and cmd.gateway_only), \
|
|
f"{cmd.name} cannot be both cli_only and gateway_only"
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# resolve_command tests
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestResolveCommand:
|
|
def test_canonical_name_resolves(self):
|
|
assert resolve_command("help").name == "help"
|
|
assert resolve_command("background").name == "background"
|
|
|
|
def test_alias_resolves_to_canonical(self):
|
|
assert resolve_command("bg").name == "background"
|
|
assert resolve_command("reset").name == "new"
|
|
assert resolve_command("q").name == "quit"
|
|
assert resolve_command("exit").name == "quit"
|
|
assert resolve_command("gateway").name == "platforms"
|
|
assert resolve_command("set-home").name == "sethome"
|
|
assert resolve_command("reload_mcp").name == "reload-mcp"
|
|
|
|
def test_leading_slash_stripped(self):
|
|
assert resolve_command("/help").name == "help"
|
|
assert resolve_command("/bg").name == "background"
|
|
|
|
def test_unknown_returns_none(self):
|
|
assert resolve_command("nonexistent") is None
|
|
assert resolve_command("") is None
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Derived dicts (backwards compat)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestDerivedDicts:
|
|
def test_commands_dict_excludes_gateway_only(self):
|
|
"""gateway_only commands should NOT appear in the CLI COMMANDS dict."""
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.gateway_only:
|
|
assert f"/{cmd.name}" not in COMMANDS, \
|
|
f"gateway_only command /{cmd.name} should not be in COMMANDS"
|
|
|
|
def test_commands_dict_includes_all_cli_commands(self):
|
|
for cmd in COMMAND_REGISTRY:
|
|
if not cmd.gateway_only:
|
|
assert f"/{cmd.name}" in COMMANDS, \
|
|
f"/{cmd.name} missing from COMMANDS dict"
|
|
|
|
def test_commands_dict_includes_aliases(self):
|
|
assert "/bg" in COMMANDS
|
|
assert "/reset" in COMMANDS
|
|
assert "/q" in COMMANDS
|
|
assert "/exit" in COMMANDS
|
|
assert "/reload_mcp" in COMMANDS
|
|
assert "/gateway" in COMMANDS
|
|
|
|
def test_commands_by_category_covers_all_categories(self):
|
|
registry_categories = {cmd.category for cmd in COMMAND_REGISTRY if not cmd.gateway_only}
|
|
assert set(COMMANDS_BY_CATEGORY.keys()) == registry_categories
|
|
|
|
def test_every_command_has_nonempty_description(self):
|
|
for cmd, desc in COMMANDS.items():
|
|
assert isinstance(desc, str) and len(desc) > 0, f"{cmd} has empty description"
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Gateway helpers
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestGatewayKnownCommands:
|
|
def test_excludes_cli_only_without_config_gate(self):
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.cli_only and not cmd.gateway_config_gate:
|
|
assert cmd.name not in GATEWAY_KNOWN_COMMANDS, \
|
|
f"cli_only command '{cmd.name}' should not be in GATEWAY_KNOWN_COMMANDS"
|
|
|
|
def test_includes_config_gated_cli_only(self):
|
|
"""Commands with gateway_config_gate are always in GATEWAY_KNOWN_COMMANDS."""
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.gateway_config_gate:
|
|
assert cmd.name in GATEWAY_KNOWN_COMMANDS, \
|
|
f"config-gated command '{cmd.name}' should be in GATEWAY_KNOWN_COMMANDS"
|
|
|
|
def test_includes_gateway_commands(self):
|
|
for cmd in COMMAND_REGISTRY:
|
|
if not cmd.cli_only:
|
|
assert cmd.name in GATEWAY_KNOWN_COMMANDS
|
|
for alias in cmd.aliases:
|
|
assert alias in GATEWAY_KNOWN_COMMANDS
|
|
|
|
def test_bg_alias_in_gateway(self):
|
|
assert "bg" in GATEWAY_KNOWN_COMMANDS
|
|
assert "background" in GATEWAY_KNOWN_COMMANDS
|
|
|
|
def test_is_frozenset(self):
|
|
assert isinstance(GATEWAY_KNOWN_COMMANDS, frozenset)
|
|
|
|
|
|
class TestGatewayHelpLines:
|
|
def test_returns_nonempty_list(self):
|
|
lines = gateway_help_lines()
|
|
assert len(lines) > 10
|
|
|
|
def test_excludes_cli_only_commands_without_config_gate(self):
|
|
lines = gateway_help_lines()
|
|
joined = "\n".join(lines)
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.cli_only and not cmd.gateway_config_gate:
|
|
assert f"`/{cmd.name}" not in joined, \
|
|
f"cli_only command /{cmd.name} should not be in gateway help"
|
|
|
|
def test_includes_alias_note_for_bg(self):
|
|
lines = gateway_help_lines()
|
|
bg_line = [l for l in lines if "/background" in l]
|
|
assert len(bg_line) == 1
|
|
assert "/bg" in bg_line[0]
|
|
|
|
|
|
class TestTelegramBotCommands:
|
|
def test_returns_list_of_tuples(self):
|
|
cmds = telegram_bot_commands()
|
|
assert len(cmds) > 10
|
|
for name, desc in cmds:
|
|
assert isinstance(name, str)
|
|
assert isinstance(desc, str)
|
|
|
|
def test_no_hyphens_in_command_names(self):
|
|
"""Telegram does not support hyphens in command names."""
|
|
for name, _ in telegram_bot_commands():
|
|
assert "-" not in name, f"Telegram command '{name}' contains a hyphen"
|
|
|
|
def test_all_names_valid_telegram_chars(self):
|
|
"""Telegram requires: lowercase a-z, 0-9, underscores only."""
|
|
import re
|
|
tg_valid = re.compile(r"^[a-z0-9_]+$")
|
|
for name, _ in telegram_bot_commands():
|
|
assert tg_valid.match(name), f"Invalid Telegram command name: {name!r}"
|
|
|
|
def test_excludes_cli_only_without_config_gate(self):
|
|
names = {name for name, _ in telegram_bot_commands()}
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.cli_only and not cmd.gateway_config_gate:
|
|
tg_name = cmd.name.replace("-", "_")
|
|
assert tg_name not in names
|
|
|
|
|
|
class TestSlackSubcommandMap:
|
|
def test_returns_dict(self):
|
|
mapping = slack_subcommand_map()
|
|
assert isinstance(mapping, dict)
|
|
assert len(mapping) > 10
|
|
|
|
def test_values_are_slash_prefixed(self):
|
|
for key, val in slack_subcommand_map().items():
|
|
assert val.startswith("/"), f"Slack mapping for '{key}' should start with /"
|
|
|
|
def test_includes_aliases(self):
|
|
mapping = slack_subcommand_map()
|
|
assert "bg" in mapping
|
|
assert "reset" in mapping
|
|
|
|
def test_excludes_cli_only_without_config_gate(self):
|
|
mapping = slack_subcommand_map()
|
|
for cmd in COMMAND_REGISTRY:
|
|
if cmd.cli_only and not cmd.gateway_config_gate:
|
|
assert cmd.name not in mapping
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Config-gated gateway commands
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestGatewayConfigGate:
|
|
"""Tests for the gateway_config_gate mechanism on CommandDef."""
|
|
|
|
def test_verbose_has_config_gate(self):
|
|
cmd = resolve_command("verbose")
|
|
assert cmd is not None
|
|
assert cmd.cli_only is True
|
|
assert cmd.gateway_config_gate == "display.tool_progress_command"
|
|
|
|
def test_verbose_in_gateway_known_commands(self):
|
|
"""Config-gated commands are always recognized by the gateway."""
|
|
assert "verbose" in GATEWAY_KNOWN_COMMANDS
|
|
|
|
def test_config_gate_excluded_from_help_when_off(self, tmp_path, monkeypatch):
|
|
"""When the config gate is falsy, the command should not appear in help."""
|
|
# Write a config with the gate off (default)
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: false\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
lines = gateway_help_lines()
|
|
joined = "\n".join(lines)
|
|
assert "`/verbose" not in joined
|
|
|
|
def test_config_gate_included_in_help_when_on(self, tmp_path, monkeypatch):
|
|
"""When the config gate is truthy, the command should appear in help."""
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: true\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
lines = gateway_help_lines()
|
|
joined = "\n".join(lines)
|
|
assert "`/verbose" in joined
|
|
|
|
def test_config_gate_excluded_from_telegram_when_off(self, tmp_path, monkeypatch):
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: false\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
names = {name for name, _ in telegram_bot_commands()}
|
|
assert "verbose" not in names
|
|
|
|
def test_config_gate_included_in_telegram_when_on(self, tmp_path, monkeypatch):
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: true\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
names = {name for name, _ in telegram_bot_commands()}
|
|
assert "verbose" in names
|
|
|
|
def test_config_gate_excluded_from_slack_when_off(self, tmp_path, monkeypatch):
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: false\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
mapping = slack_subcommand_map()
|
|
assert "verbose" not in mapping
|
|
|
|
def test_config_gate_included_in_slack_when_on(self, tmp_path, monkeypatch):
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text("display:\n tool_progress_command: true\n")
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
mapping = slack_subcommand_map()
|
|
assert "verbose" in mapping
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Autocomplete (SlashCommandCompleter)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestSlashCommandCompleter:
|
|
# -- basic prefix completion -----------------------------------------
|
|
|
|
def test_builtin_prefix_completion_uses_shared_registry(self):
|
|
completions = _completions(SlashCommandCompleter(), "/re")
|
|
texts = {item.text for item in completions}
|
|
|
|
assert "reset" in texts
|
|
assert "retry" in texts
|
|
assert "reload-mcp" in texts
|
|
|
|
def test_builtin_completion_display_meta_shows_description(self):
|
|
completions = _completions(SlashCommandCompleter(), "/help")
|
|
assert len(completions) == 1
|
|
assert completions[0].display_meta_text == "Show available commands"
|
|
|
|
# -- exact-match trailing space --------------------------------------
|
|
|
|
def test_exact_match_completion_adds_trailing_space(self):
|
|
completions = _completions(SlashCommandCompleter(), "/help")
|
|
|
|
assert [item.text for item in completions] == ["help "]
|
|
|
|
def test_partial_match_does_not_add_trailing_space(self):
|
|
completions = _completions(SlashCommandCompleter(), "/hel")
|
|
|
|
assert [item.text for item in completions] == ["help"]
|
|
|
|
# -- non-slash input returns nothing ---------------------------------
|
|
|
|
def test_no_completions_for_non_slash_input(self):
|
|
assert _completions(SlashCommandCompleter(), "help") == []
|
|
|
|
def test_no_completions_for_empty_input(self):
|
|
assert _completions(SlashCommandCompleter(), "") == []
|
|
|
|
# -- skill commands via provider ------------------------------------
|
|
|
|
def test_skill_commands_are_completed_from_provider(self):
|
|
completer = SlashCommandCompleter(
|
|
skill_commands_provider=lambda: {
|
|
"/gif-search": {"description": "Search for GIFs across providers"},
|
|
}
|
|
)
|
|
|
|
completions = _completions(completer, "/gif")
|
|
|
|
assert len(completions) == 1
|
|
assert completions[0].text == "gif-search"
|
|
assert completions[0].display_text == "/gif-search"
|
|
assert completions[0].display_meta_text == "⚡ Search for GIFs across providers"
|
|
|
|
def test_skill_exact_match_adds_trailing_space(self):
|
|
completer = SlashCommandCompleter(
|
|
skill_commands_provider=lambda: {
|
|
"/gif-search": {"description": "Search for GIFs"},
|
|
}
|
|
)
|
|
|
|
completions = _completions(completer, "/gif-search")
|
|
|
|
assert len(completions) == 1
|
|
assert completions[0].text == "gif-search "
|
|
|
|
def test_no_skill_provider_means_no_skill_completions(self):
|
|
"""Default (None) provider should not blow up or add completions."""
|
|
completer = SlashCommandCompleter()
|
|
completions = _completions(completer, "/gif")
|
|
# /gif doesn't match any builtin command
|
|
assert completions == []
|
|
|
|
def test_skill_provider_exception_is_swallowed(self):
|
|
"""A broken provider should not crash autocomplete."""
|
|
completer = SlashCommandCompleter(
|
|
skill_commands_provider=lambda: (_ for _ in ()).throw(RuntimeError("boom")),
|
|
)
|
|
# Should return builtin matches only, no crash
|
|
completions = _completions(completer, "/he")
|
|
texts = {item.text for item in completions}
|
|
assert "help" in texts
|
|
|
|
def test_skill_description_truncated_at_50_chars(self):
|
|
long_desc = "A" * 80
|
|
completer = SlashCommandCompleter(
|
|
skill_commands_provider=lambda: {
|
|
"/long-skill": {"description": long_desc},
|
|
}
|
|
)
|
|
completions = _completions(completer, "/long")
|
|
assert len(completions) == 1
|
|
meta = completions[0].display_meta_text
|
|
# "⚡ " prefix + 50 chars + "..."
|
|
assert meta == f"⚡ {'A' * 50}..."
|
|
|
|
def test_skill_missing_description_uses_fallback(self):
|
|
completer = SlashCommandCompleter(
|
|
skill_commands_provider=lambda: {
|
|
"/no-desc": {},
|
|
}
|
|
)
|
|
completions = _completions(completer, "/no-desc")
|
|
assert len(completions) == 1
|
|
assert "Skill command" in completions[0].display_meta_text
|
|
|
|
|
|
# ── SUBCOMMANDS extraction ──────────────────────────────────────────────
|
|
|
|
|
|
class TestSubcommands:
|
|
def test_explicit_subcommands_extracted(self):
|
|
"""Commands with explicit subcommands on CommandDef are extracted."""
|
|
assert "/prompt" in SUBCOMMANDS
|
|
assert "clear" in SUBCOMMANDS["/prompt"]
|
|
|
|
def test_reasoning_has_subcommands(self):
|
|
assert "/reasoning" in SUBCOMMANDS
|
|
subs = SUBCOMMANDS["/reasoning"]
|
|
assert "high" in subs
|
|
assert "show" in subs
|
|
assert "hide" in subs
|
|
|
|
def test_voice_has_subcommands(self):
|
|
assert "/voice" in SUBCOMMANDS
|
|
assert "on" in SUBCOMMANDS["/voice"]
|
|
assert "off" in SUBCOMMANDS["/voice"]
|
|
|
|
def test_cron_has_subcommands(self):
|
|
assert "/cron" in SUBCOMMANDS
|
|
assert "list" in SUBCOMMANDS["/cron"]
|
|
assert "add" in SUBCOMMANDS["/cron"]
|
|
|
|
def test_commands_without_subcommands_not_in_dict(self):
|
|
"""Plain commands should not appear in SUBCOMMANDS."""
|
|
assert "/help" not in SUBCOMMANDS
|
|
assert "/quit" not in SUBCOMMANDS
|
|
assert "/clear" not in SUBCOMMANDS
|
|
|
|
|
|
# ── Subcommand tab completion ───────────────────────────────────────────
|
|
|
|
|
|
class TestSubcommandCompletion:
|
|
def test_subcommand_completion_after_space(self):
|
|
"""Typing '/reasoning ' then Tab should show subcommands."""
|
|
completions = _completions(SlashCommandCompleter(), "/reasoning ")
|
|
texts = {c.text for c in completions}
|
|
assert "high" in texts
|
|
assert "show" in texts
|
|
|
|
def test_subcommand_prefix_filters(self):
|
|
"""Typing '/reasoning sh' should only show 'show'."""
|
|
completions = _completions(SlashCommandCompleter(), "/reasoning sh")
|
|
texts = {c.text for c in completions}
|
|
assert texts == {"show"}
|
|
|
|
def test_subcommand_exact_match_suppressed(self):
|
|
"""Typing the full subcommand shouldn't re-suggest it."""
|
|
completions = _completions(SlashCommandCompleter(), "/reasoning show")
|
|
texts = {c.text for c in completions}
|
|
assert "show" not in texts
|
|
|
|
def test_no_subcommands_for_plain_command(self):
|
|
"""Commands without subcommands yield nothing after space."""
|
|
completions = _completions(SlashCommandCompleter(), "/help ")
|
|
assert completions == []
|
|
|
|
|
|
# ── Ghost text (SlashCommandAutoSuggest) ────────────────────────────────
|
|
|
|
|
|
def _suggestion(text: str, completer=None) -> str | None:
|
|
"""Get ghost text suggestion for given input."""
|
|
suggest = SlashCommandAutoSuggest(completer=completer)
|
|
doc = Document(text=text)
|
|
|
|
class FakeBuffer:
|
|
pass
|
|
|
|
result = suggest.get_suggestion(FakeBuffer(), doc)
|
|
return result.text if result else None
|
|
|
|
|
|
class TestGhostText:
|
|
def test_command_name_suggestion(self):
|
|
"""/he → 'lp'"""
|
|
assert _suggestion("/he") == "lp"
|
|
|
|
def test_command_name_suggestion_reasoning(self):
|
|
"""/rea → 'soning'"""
|
|
assert _suggestion("/rea") == "soning"
|
|
|
|
def test_no_suggestion_for_complete_command(self):
|
|
assert _suggestion("/help") is None
|
|
|
|
def test_subcommand_suggestion(self):
|
|
"""/reasoning h → 'igh'"""
|
|
assert _suggestion("/reasoning h") == "igh"
|
|
|
|
def test_subcommand_suggestion_show(self):
|
|
"""/reasoning sh → 'ow'"""
|
|
assert _suggestion("/reasoning sh") == "ow"
|
|
|
|
def test_no_suggestion_for_non_slash(self):
|
|
assert _suggestion("hello") is None
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Telegram command name sanitization
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
class TestSanitizeTelegramName:
|
|
"""Tests for _sanitize_telegram_name() — Telegram requires [a-z0-9_] only."""
|
|
|
|
def test_hyphens_replaced_with_underscores(self):
|
|
assert _sanitize_telegram_name("my-skill-name") == "my_skill_name"
|
|
|
|
def test_plus_sign_stripped(self):
|
|
"""Regression: skill name 'Jellyfin + Jellystat 24h Summary'."""
|
|
assert _sanitize_telegram_name("jellyfin-+-jellystat-24h-summary") == "jellyfin_jellystat_24h_summary"
|
|
|
|
def test_slash_stripped(self):
|
|
"""Regression: skill name 'Sonarr v3/v4 API Integration'."""
|
|
assert _sanitize_telegram_name("sonarr-v3/v4-api-integration") == "sonarr_v3v4_api_integration"
|
|
|
|
def test_uppercase_lowercased(self):
|
|
assert _sanitize_telegram_name("MyCommand") == "mycommand"
|
|
|
|
def test_dots_and_special_chars_stripped(self):
|
|
assert _sanitize_telegram_name("skill.v2@beta!") == "skillv2beta"
|
|
|
|
def test_consecutive_underscores_collapsed(self):
|
|
assert _sanitize_telegram_name("a---b") == "a_b"
|
|
assert _sanitize_telegram_name("a-+-b") == "a_b"
|
|
|
|
def test_leading_trailing_underscores_stripped(self):
|
|
assert _sanitize_telegram_name("-leading") == "leading"
|
|
assert _sanitize_telegram_name("trailing-") == "trailing"
|
|
assert _sanitize_telegram_name("-both-") == "both"
|
|
|
|
def test_digits_preserved(self):
|
|
assert _sanitize_telegram_name("skill-24h") == "skill_24h"
|
|
|
|
def test_empty_after_sanitization(self):
|
|
assert _sanitize_telegram_name("+++") == ""
|
|
|
|
def test_spaces_only_becomes_empty(self):
|
|
assert _sanitize_telegram_name(" ") == ""
|
|
|
|
def test_already_valid(self):
|
|
assert _sanitize_telegram_name("valid_name_123") == "valid_name_123"
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Telegram command name clamping (32-char limit)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
class TestClampTelegramNames:
|
|
"""Tests for _clamp_telegram_names() — 32-char enforcement + collision."""
|
|
|
|
def test_short_names_unchanged(self):
|
|
entries = [("help", "Show help"), ("status", "Show status")]
|
|
result = _clamp_telegram_names(entries, set())
|
|
assert result == entries
|
|
|
|
def test_long_name_truncated(self):
|
|
long = "a" * 40
|
|
result = _clamp_telegram_names([(long, "desc")], set())
|
|
assert len(result) == 1
|
|
assert result[0][0] == "a" * _TG_NAME_LIMIT
|
|
assert result[0][1] == "desc"
|
|
|
|
def test_collision_with_reserved_gets_digit_suffix(self):
|
|
# The truncated form collides with a reserved name
|
|
prefix = "x" * _TG_NAME_LIMIT
|
|
long_name = "x" * 40
|
|
result = _clamp_telegram_names([(long_name, "d")], reserved={prefix})
|
|
assert len(result) == 1
|
|
name = result[0][0]
|
|
assert len(name) == _TG_NAME_LIMIT
|
|
assert name == "x" * (_TG_NAME_LIMIT - 1) + "0"
|
|
|
|
def test_collision_between_entries_gets_incrementing_digits(self):
|
|
# Two long names that truncate to the same 32-char prefix
|
|
base = "y" * 40
|
|
entries = [(base + "_alpha", "d1"), (base + "_beta", "d2")]
|
|
result = _clamp_telegram_names(entries, set())
|
|
assert len(result) == 2
|
|
assert result[0][0] == "y" * _TG_NAME_LIMIT
|
|
assert result[1][0] == "y" * (_TG_NAME_LIMIT - 1) + "0"
|
|
|
|
def test_collision_with_reserved_and_entries_skips_taken_digits(self):
|
|
prefix = "z" * _TG_NAME_LIMIT
|
|
digit0 = "z" * (_TG_NAME_LIMIT - 1) + "0"
|
|
# Reserve both the plain truncation and digit-0
|
|
reserved = {prefix, digit0}
|
|
long_name = "z" * 50
|
|
result = _clamp_telegram_names([(long_name, "d")], reserved)
|
|
assert len(result) == 1
|
|
assert result[0][0] == "z" * (_TG_NAME_LIMIT - 1) + "1"
|
|
|
|
def test_all_digits_exhausted_drops_entry(self):
|
|
prefix = "w" * _TG_NAME_LIMIT
|
|
# Reserve the plain truncation + all 10 digit slots
|
|
reserved = {prefix} | {"w" * (_TG_NAME_LIMIT - 1) + str(d) for d in range(10)}
|
|
long_name = "w" * 50
|
|
result = _clamp_telegram_names([(long_name, "d")], reserved)
|
|
assert result == []
|
|
|
|
def test_exact_32_chars_not_truncated(self):
|
|
name = "a" * _TG_NAME_LIMIT
|
|
result = _clamp_telegram_names([(name, "desc")], set())
|
|
assert result[0][0] == name
|
|
|
|
def test_duplicate_short_name_deduplicated(self):
|
|
entries = [("foo", "d1"), ("foo", "d2")]
|
|
result = _clamp_telegram_names(entries, set())
|
|
assert len(result) == 1
|
|
assert result[0] == ("foo", "d1")
|
|
|
|
|
|
class TestTelegramMenuCommands:
|
|
"""Integration: telegram_menu_commands enforces the 32-char limit."""
|
|
|
|
def test_all_names_within_limit(self):
|
|
menu, _ = telegram_menu_commands(max_commands=100)
|
|
for name, _desc in menu:
|
|
assert 1 <= len(name) <= _TG_NAME_LIMIT, (
|
|
f"Command '{name}' is {len(name)} chars (limit {_TG_NAME_LIMIT})"
|
|
)
|
|
|
|
def test_excludes_telegram_disabled_skills(self, tmp_path, monkeypatch):
|
|
"""Skills disabled for telegram should not appear in the menu."""
|
|
from unittest.mock import patch, MagicMock
|
|
|
|
# Set up a config with a telegram-specific disabled list
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text(
|
|
"skills:\n"
|
|
" platform_disabled:\n"
|
|
" telegram:\n"
|
|
" - my-disabled-skill\n"
|
|
)
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
# Mock get_skill_commands to return two skills
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/my-disabled-skill": {
|
|
"name": "my-disabled-skill",
|
|
"description": "Should be hidden",
|
|
"skill_md_path": f"{fake_skills_dir}/my-disabled-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/my-disabled-skill",
|
|
},
|
|
"/my-enabled-skill": {
|
|
"name": "my-enabled-skill",
|
|
"description": "Should be visible",
|
|
"skill_md_path": f"{fake_skills_dir}/my-enabled-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/my-enabled-skill",
|
|
},
|
|
}
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
menu, hidden = telegram_menu_commands(max_commands=100)
|
|
|
|
menu_names = {n for n, _ in menu}
|
|
assert "my_enabled_skill" in menu_names
|
|
assert "my_disabled_skill" not in menu_names
|
|
|
|
def test_special_chars_in_skill_names_sanitized(self, tmp_path, monkeypatch):
|
|
"""Skills with +, /, or other special chars produce valid Telegram names."""
|
|
from unittest.mock import patch
|
|
import re
|
|
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/jellyfin-+-jellystat-24h-summary": {
|
|
"name": "Jellyfin + Jellystat 24h Summary",
|
|
"description": "Test",
|
|
"skill_md_path": f"{fake_skills_dir}/jellyfin/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/jellyfin",
|
|
},
|
|
"/sonarr-v3/v4-api": {
|
|
"name": "Sonarr v3/v4 API",
|
|
"description": "Test",
|
|
"skill_md_path": f"{fake_skills_dir}/sonarr/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/sonarr",
|
|
},
|
|
}
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
menu, _ = telegram_menu_commands(max_commands=100)
|
|
|
|
# Every name must match Telegram's [a-z0-9_] requirement
|
|
tg_valid = re.compile(r"^[a-z0-9_]+$")
|
|
for name, _ in menu:
|
|
assert tg_valid.match(name), f"Invalid Telegram command name: {name!r}"
|
|
|
|
def test_empty_sanitized_names_excluded(self, tmp_path, monkeypatch):
|
|
"""Skills whose names sanitize to empty string are silently dropped."""
|
|
from unittest.mock import patch
|
|
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/+++": {
|
|
"name": "+++",
|
|
"description": "All special chars",
|
|
"skill_md_path": f"{fake_skills_dir}/bad/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/bad",
|
|
},
|
|
"/valid-skill": {
|
|
"name": "valid-skill",
|
|
"description": "Normal skill",
|
|
"skill_md_path": f"{fake_skills_dir}/valid/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/valid",
|
|
},
|
|
}
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
menu, _ = telegram_menu_commands(max_commands=100)
|
|
|
|
menu_names = {n for n, _ in menu}
|
|
# The valid skill should be present, the empty one should not
|
|
assert "valid_skill" in menu_names
|
|
# No empty string in menu names
|
|
assert "" not in menu_names
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Backward-compat aliases
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestBackwardCompatAliases:
|
|
"""The renamed constants/functions still exist under the old names."""
|
|
|
|
def test_tg_name_limit_alias(self):
|
|
assert _TG_NAME_LIMIT == _CMD_NAME_LIMIT == 32
|
|
|
|
def test_clamp_telegram_names_is_clamp_command_names(self):
|
|
assert _clamp_telegram_names is _clamp_command_names
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Discord skill command registration
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestDiscordSkillCommands:
|
|
"""Tests for discord_skill_commands() — centralized skill registration."""
|
|
|
|
def test_returns_skill_entries(self, tmp_path, monkeypatch):
|
|
"""Skills under SKILLS_DIR (not .hub) should be returned."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/gif-search": {
|
|
"name": "gif-search",
|
|
"description": "Search for GIFs",
|
|
"skill_md_path": f"{fake_skills_dir}/gif-search/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/gif-search",
|
|
},
|
|
"/code-review": {
|
|
"name": "code-review",
|
|
"description": "Review code changes",
|
|
"skill_md_path": f"{fake_skills_dir}/code-review/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/code-review",
|
|
},
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, hidden = discord_skill_commands(
|
|
max_slots=50, reserved_names=set(),
|
|
)
|
|
|
|
names = {n for n, _d, _k in entries}
|
|
assert "gif-search" in names
|
|
assert "code-review" in names
|
|
assert hidden == 0
|
|
# Verify cmd_key is preserved for handler callbacks
|
|
keys = {k for _n, _d, k in entries}
|
|
assert "/gif-search" in keys
|
|
assert "/code-review" in keys
|
|
|
|
def test_names_allow_hyphens(self, tmp_path, monkeypatch):
|
|
"""Discord names should keep hyphens (unlike Telegram's _ sanitization)."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/my-cool-skill": {
|
|
"name": "my-cool-skill",
|
|
"description": "A cool skill",
|
|
"skill_md_path": f"{fake_skills_dir}/my-cool-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/my-cool-skill",
|
|
},
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, _ = discord_skill_commands(
|
|
max_slots=50, reserved_names=set(),
|
|
)
|
|
|
|
assert entries[0][0] == "my-cool-skill" # hyphens preserved
|
|
|
|
def test_cap_enforcement(self, tmp_path, monkeypatch):
|
|
"""Entries beyond max_slots should be hidden."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
f"/skill-{i:03d}": {
|
|
"name": f"skill-{i:03d}",
|
|
"description": f"Skill {i}",
|
|
"skill_md_path": f"{fake_skills_dir}/skill-{i:03d}/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/skill-{i:03d}",
|
|
}
|
|
for i in range(20)
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, hidden = discord_skill_commands(
|
|
max_slots=5, reserved_names=set(),
|
|
)
|
|
|
|
assert len(entries) == 5
|
|
assert hidden == 15
|
|
|
|
def test_excludes_discord_disabled_skills(self, tmp_path, monkeypatch):
|
|
"""Skills disabled for discord should not appear."""
|
|
from unittest.mock import patch
|
|
|
|
config_file = tmp_path / "config.yaml"
|
|
config_file.write_text(
|
|
"skills:\n"
|
|
" platform_disabled:\n"
|
|
" discord:\n"
|
|
" - secret-skill\n"
|
|
)
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/secret-skill": {
|
|
"name": "secret-skill",
|
|
"description": "Should not appear",
|
|
"skill_md_path": f"{fake_skills_dir}/secret-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/secret-skill",
|
|
},
|
|
"/public-skill": {
|
|
"name": "public-skill",
|
|
"description": "Should appear",
|
|
"skill_md_path": f"{fake_skills_dir}/public-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/public-skill",
|
|
},
|
|
}
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, _ = discord_skill_commands(
|
|
max_slots=50, reserved_names=set(),
|
|
)
|
|
|
|
names = {n for n, _d, _k in entries}
|
|
assert "secret-skill" not in names
|
|
assert "public-skill" in names
|
|
|
|
def test_reserved_names_not_overwritten(self, tmp_path, monkeypatch):
|
|
"""Skills whose names collide with built-in commands should be skipped."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
fake_cmds = {
|
|
"/status": {
|
|
"name": "status",
|
|
"description": "Skill that collides with built-in",
|
|
"skill_md_path": f"{fake_skills_dir}/status/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/status",
|
|
},
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, _ = discord_skill_commands(
|
|
max_slots=50, reserved_names={"status"},
|
|
)
|
|
|
|
names = {n for n, _d, _k in entries}
|
|
assert "status" not in names
|
|
|
|
def test_description_truncated_at_100_chars(self, tmp_path, monkeypatch):
|
|
"""Descriptions exceeding 100 chars should be truncated."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
long_desc = "x" * 150
|
|
fake_cmds = {
|
|
"/verbose-skill": {
|
|
"name": "verbose-skill",
|
|
"description": long_desc,
|
|
"skill_md_path": f"{fake_skills_dir}/verbose-skill/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/verbose-skill",
|
|
},
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, _ = discord_skill_commands(
|
|
max_slots=50, reserved_names=set(),
|
|
)
|
|
|
|
assert len(entries[0][1]) == 100
|
|
assert entries[0][1].endswith("...")
|
|
|
|
def test_all_names_within_32_chars(self, tmp_path, monkeypatch):
|
|
"""All returned names must respect the 32-char Discord limit."""
|
|
from unittest.mock import patch
|
|
|
|
fake_skills_dir = str(tmp_path / "skills")
|
|
long_name = "a" * 50
|
|
fake_cmds = {
|
|
f"/{long_name}": {
|
|
"name": long_name,
|
|
"description": "Long name skill",
|
|
"skill_md_path": f"{fake_skills_dir}/{long_name}/SKILL.md",
|
|
"skill_dir": f"{fake_skills_dir}/{long_name}",
|
|
},
|
|
}
|
|
monkeypatch.setenv("HERMES_HOME", str(tmp_path))
|
|
(tmp_path / "skills").mkdir(exist_ok=True)
|
|
with (
|
|
patch("agent.skill_commands.get_skill_commands", return_value=fake_cmds),
|
|
patch("tools.skills_tool.SKILLS_DIR", tmp_path / "skills"),
|
|
):
|
|
entries, _ = discord_skill_commands(
|
|
max_slots=50, reserved_names=set(),
|
|
)
|
|
|
|
for name, _d, _k in entries:
|
|
assert len(name) <= _CMD_NAME_LIMIT, (
|
|
f"Name '{name}' is {len(name)} chars (limit {_CMD_NAME_LIMIT})"
|
|
)
|