mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-30 06:41:51 +00:00
Reported by @LikiusInik in Discord: on Termux only 3 built-in skills
appeared and /gh-pr-workflow + every other slash-skill from
github/productivity/mlops was missing.
Root cause: skill_matches_platform() compares sys.platform.startswith()
against the skill's platforms list. Termux is a Linux userland on
Android, but Python 3.13+ reports sys.platform == "android" instead of
"linux" — so the ~60 built-in skills tagged platforms:[linux,macos,
windows] (github-pr-workflow, google-workspace, github-auth,
huggingface-hub, etc.) all got filtered out at the listing step in
tools/skills_tool.py:_find_all_skills and never appeared as /slash
commands or in skill_view.
Fix: when is_termux() detects we're running inside Termux, accept
"linux" platform tags regardless of whether sys.platform is "linux"
(pre-3.13) or "android" (3.13+). Also accept explicit
platforms:[termux] / [android] tags. macOS-only and Windows-only
skills correctly remain excluded.
E2E (simulated TERMUX_VERSION=set + sys.platform="android"):
Before: _find_all_skills() returned ~3 skills.
After: _find_all_skills() returns 84 skills including
github-pr-workflow, google-workspace, github-auth,
huggingface-hub. Apple-only skills remain excluded.
Non-Termux Linux/macOS/Windows behavior unchanged (verified).
Tests: tests/agent/test_skill_utils.py — 9 new cases covering
android-as-Termux, the [linux,macos,windows] case, macOS-only
exclusion, explicit termux/android tags, non-Termux Android safety,
and unchanged behavior on real Linux/macOS.
199 lines
7.4 KiB
Python
199 lines
7.4 KiB
Python
"""Tests for agent/skill_utils.py."""
|
|
|
|
from unittest.mock import patch
|
|
|
|
from agent.skill_utils import (
|
|
extract_skill_conditions,
|
|
iter_skill_index_files,
|
|
skill_matches_platform,
|
|
)
|
|
|
|
|
|
def test_metadata_as_dict_with_hermes():
|
|
"""Normal case: metadata is a dict containing hermes keys."""
|
|
frontmatter = {
|
|
"metadata": {
|
|
"hermes": {
|
|
"fallback_for_toolsets": ["toolset_a"],
|
|
"requires_toolsets": ["toolset_b"],
|
|
"fallback_for_tools": ["tool_x"],
|
|
"requires_tools": ["tool_y"],
|
|
}
|
|
}
|
|
}
|
|
result = extract_skill_conditions(frontmatter)
|
|
assert result["fallback_for_toolsets"] == ["toolset_a"]
|
|
assert result["requires_toolsets"] == ["toolset_b"]
|
|
assert result["fallback_for_tools"] == ["tool_x"]
|
|
assert result["requires_tools"] == ["tool_y"]
|
|
|
|
|
|
def test_metadata_as_string_does_not_crash():
|
|
"""Bug case: metadata is a non-dict truthy value (e.g. a YAML string)."""
|
|
frontmatter = {"metadata": "some text"}
|
|
result = extract_skill_conditions(frontmatter)
|
|
assert result == {
|
|
"fallback_for_toolsets": [],
|
|
"requires_toolsets": [],
|
|
"fallback_for_tools": [],
|
|
"requires_tools": [],
|
|
}
|
|
|
|
|
|
def test_metadata_as_none():
|
|
"""metadata key is present but set to null/None."""
|
|
frontmatter = {"metadata": None}
|
|
result = extract_skill_conditions(frontmatter)
|
|
assert result == {
|
|
"fallback_for_toolsets": [],
|
|
"requires_toolsets": [],
|
|
"fallback_for_tools": [],
|
|
"requires_tools": [],
|
|
}
|
|
|
|
|
|
def test_metadata_missing_entirely():
|
|
"""metadata key is absent from frontmatter."""
|
|
frontmatter = {"name": "my-skill", "description": "Does stuff."}
|
|
result = extract_skill_conditions(frontmatter)
|
|
assert result == {
|
|
"fallback_for_toolsets": [],
|
|
"requires_toolsets": [],
|
|
"fallback_for_tools": [],
|
|
"requires_tools": [],
|
|
}
|
|
|
|
|
|
def test_iter_skill_index_files_prunes_dependency_dirs(tmp_path):
|
|
real = tmp_path / "real-skill"
|
|
real.mkdir()
|
|
(real / "SKILL.md").write_text("---\nname: real-skill\n---\n", encoding="utf-8")
|
|
|
|
nested = (
|
|
tmp_path
|
|
/ "bring"
|
|
/ "scripts"
|
|
/ ".venv"
|
|
/ "lib"
|
|
/ "python3.13"
|
|
/ "site-packages"
|
|
/ "typer"
|
|
/ ".agents"
|
|
/ "skills"
|
|
/ "typer"
|
|
)
|
|
nested.mkdir(parents=True)
|
|
(nested / "SKILL.md").write_text("---\nname: typer\n---\n", encoding="utf-8")
|
|
|
|
node_module = (
|
|
tmp_path
|
|
/ "web-skill"
|
|
/ "node_modules"
|
|
/ "dep"
|
|
/ ".agents"
|
|
/ "skills"
|
|
/ "dep"
|
|
)
|
|
node_module.mkdir(parents=True)
|
|
(node_module / "SKILL.md").write_text("---\nname: dep\n---\n", encoding="utf-8")
|
|
|
|
found = list(iter_skill_index_files(tmp_path, "SKILL.md"))
|
|
|
|
assert found == [real / "SKILL.md"]
|
|
|
|
|
|
# ── skill_matches_platform on Termux ──────────────────────────────────────
|
|
|
|
|
|
class TestSkillMatchesPlatformTermux:
|
|
"""Termux is Linux userland on Android. Skills tagged platforms:[linux]
|
|
must load there regardless of whether Python reports sys.platform as
|
|
"linux" (pre-3.13) or "android" (3.13+). Reported by user @LikiusInik
|
|
in May 2026 — only 3 built-in skills appeared on Termux because every
|
|
github/productivity/mlops skill is tagged platforms:[linux,macos,windows]
|
|
and sys.platform=="android" did not start with "linux".
|
|
"""
|
|
|
|
def test_no_platforms_field_matches_everywhere(self):
|
|
# Backward-compat default — skills without a platforms tag load
|
|
# on any OS, Termux included.
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform({}) is True
|
|
assert skill_matches_platform({"name": "foo"}) is True
|
|
|
|
def test_linux_skill_loads_on_termux_android_platform(self):
|
|
# Python 3.13+ on Termux reports sys.platform == "android".
|
|
fm = {"platforms": ["linux"]}
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform(fm) is True
|
|
|
|
def test_linux_macos_windows_skill_loads_on_termux(self):
|
|
# The common "[linux, macos, windows]" tag used by github-*,
|
|
# productivity, mlops, etc.
|
|
fm = {"platforms": ["linux", "macos", "windows"]}
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform(fm) is True
|
|
|
|
def test_linux_skill_loads_on_termux_linux_platform(self):
|
|
# Pre-3.13 Termux reports sys.platform == "linux" already — this
|
|
# works without the Termux escape hatch but must still pass.
|
|
fm = {"platforms": ["linux"]}
|
|
with patch("agent.skill_utils.sys.platform", "linux"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform(fm) is True
|
|
|
|
def test_macos_only_skill_still_excluded_on_termux(self):
|
|
# macOS-only skills (apple-notes, imessage, ...) should NOT load
|
|
# on Termux. The Termux fallback only widens platforms:[linux,...].
|
|
fm = {"platforms": ["macos"]}
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform(fm) is False
|
|
|
|
def test_windows_only_skill_still_excluded_on_termux(self):
|
|
fm = {"platforms": ["windows"]}
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform(fm) is False
|
|
|
|
def test_explicit_termux_or_android_tag_matches(self):
|
|
# Skills can also opt in explicitly via platforms:[termux] or
|
|
# platforms:[android] — both should match a Termux session.
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=True
|
|
):
|
|
assert skill_matches_platform({"platforms": ["termux"]}) is True
|
|
assert skill_matches_platform({"platforms": ["android"]}) is True
|
|
|
|
def test_non_termux_android_does_not_widen(self):
|
|
# If we're somehow on a plain Android Python (not Termux), don't
|
|
# silently load Linux skills — Termux is the supported environment.
|
|
fm = {"platforms": ["linux"]}
|
|
with patch("agent.skill_utils.sys.platform", "android"), patch(
|
|
"agent.skill_utils.is_termux", return_value=False
|
|
):
|
|
assert skill_matches_platform(fm) is False
|
|
|
|
def test_linux_skill_on_real_linux_unaffected(self):
|
|
# The non-Termux Linux path must not change.
|
|
fm = {"platforms": ["linux"]}
|
|
with patch("agent.skill_utils.sys.platform", "linux"), patch(
|
|
"agent.skill_utils.is_termux", return_value=False
|
|
):
|
|
assert skill_matches_platform(fm) is True
|
|
|
|
def test_macos_skill_on_real_macos_unaffected(self):
|
|
fm = {"platforms": ["macos"]}
|
|
with patch("agent.skill_utils.sys.platform", "darwin"), patch(
|
|
"agent.skill_utils.is_termux", return_value=False
|
|
):
|
|
assert skill_matches_platform(fm) is True
|