mirror of
https://github.com/AstrBotDevs/AstrBot
synced 2026-07-01 01:10:21 +08:00
* refactor: migrate to fastapi * structure refactor * fix: pyright fix * refactor: improve error handling and public messages in plugin services * feat(api): refactor API client integration and enhance request handling - Updated API client configuration to use a dedicated HTTP client. - Introduced utility functions for generating options, queries, and form data for API requests. - Refactored multiple API methods to utilize the new utility functions for improved consistency and readability. - Renamed types for clarity and updated import statements accordingly. feat(docs): add script to update OpenAPI JSON from YAML spec - Created a Python script to convert OpenAPI YAML specification to JSON format. - The script supports customizable input and output paths. - Ensured the script handles directory creation for output paths and validates the YAML structure. * fix * feat(auth): implement rate limiting for v1 login endpoint and enhance request handling * Refactor dashboard API routers to use legacy_router for backward compatibility - Changed all instances of dashboard_router to legacy_router across multiple API modules including platform, plugins, providers, sessions, skills, stats, subagents, t2i, tools, updates, and asgi_runtime. - Updated route definitions to ensure existing endpoints remain functional under the new router structure. - Introduced support for Quart request context in asgi_runtime to enhance compatibility with existing Quart-based plugins. - Added a test case to validate the functionality of the new Quart request context handling in plugin extensions. * chore: remove cli test * fix: update dashboard tests for fastapi migration * chore: satisfy ruff checks * fix: update openapi api key scopes * fix: sync config scope chip selection * fix: restore quart dependency * docs: clarify quart plugin api compatibility * docs: update openapi scope documentation * fix: use singular skill openapi scope * fix: hide update service exception details * fix: address fastapi review comments * fix: address dashboard review findings * docs: revert unrelated package deployment changes * docs: update agent api generation guidance * feat: add plugin page web api helpers * docs: add plugin page bridge demo * fix: type plugin upload files * fix: stabilize plugin page uploads * fix: type plugin web request proxy * docs: remove plugin page docs example * fix: authenticate plugin page SSE bridge
157 lines
5.1 KiB
Python
157 lines
5.1 KiB
Python
from __future__ import annotations
|
|
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from astrbot.core.skills.skill_manager import SkillManager
|
|
|
|
|
|
def _write_skill(root: Path, name: str, description: str) -> None:
|
|
skill_dir = root / name
|
|
skill_dir.mkdir(parents=True, exist_ok=True)
|
|
skill_dir.joinpath("SKILL.md").write_text(
|
|
f"---\ndescription: {description}\n---\n# {name}\n",
|
|
encoding="utf-8",
|
|
)
|
|
|
|
|
|
def test_list_skills_merges_local_and_sandbox_cache(monkeypatch, tmp_path: Path):
|
|
data_dir = tmp_path / "data"
|
|
temp_dir = tmp_path / "temp"
|
|
skills_root = tmp_path / "skills"
|
|
data_dir.mkdir(parents=True, exist_ok=True)
|
|
temp_dir.mkdir(parents=True, exist_ok=True)
|
|
skills_root.mkdir(parents=True, exist_ok=True)
|
|
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_data_path",
|
|
lambda: str(data_dir),
|
|
)
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_temp_path",
|
|
lambda: str(temp_dir),
|
|
)
|
|
|
|
mgr = SkillManager(skills_root=str(skills_root))
|
|
_write_skill(skills_root, "custom-local", "local description")
|
|
|
|
mgr.set_sandbox_skills_cache(
|
|
[
|
|
{
|
|
"name": "python-sandbox",
|
|
"description": "ship built-in",
|
|
"path": "/app/skills/python-sandbox/SKILL.md",
|
|
},
|
|
{
|
|
"name": "custom-local",
|
|
"description": "should be ignored by local override",
|
|
"path": "skills/custom-local/SKILL.md",
|
|
},
|
|
]
|
|
)
|
|
|
|
skills = mgr.list_skills(runtime="sandbox")
|
|
by_name = {item.name: item for item in skills}
|
|
|
|
assert sorted(by_name) == ["custom-local", "python-sandbox"]
|
|
assert by_name["custom-local"].description == "local description"
|
|
assert by_name["custom-local"].path == "skills/custom-local/SKILL.md"
|
|
assert by_name["python-sandbox"].description == "ship built-in"
|
|
assert by_name["python-sandbox"].path == "/app/skills/python-sandbox/SKILL.md"
|
|
|
|
|
|
def test_sandbox_cached_skill_respects_active_and_display_path(
|
|
monkeypatch,
|
|
tmp_path: Path,
|
|
):
|
|
data_dir = tmp_path / "data"
|
|
temp_dir = tmp_path / "temp"
|
|
skills_root = tmp_path / "skills"
|
|
data_dir.mkdir(parents=True, exist_ok=True)
|
|
temp_dir.mkdir(parents=True, exist_ok=True)
|
|
skills_root.mkdir(parents=True, exist_ok=True)
|
|
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_data_path",
|
|
lambda: str(data_dir),
|
|
)
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_temp_path",
|
|
lambda: str(temp_dir),
|
|
)
|
|
|
|
mgr = SkillManager(skills_root=str(skills_root))
|
|
mgr.set_sandbox_skills_cache(
|
|
[
|
|
{
|
|
"name": "browser-automation",
|
|
"description": "gull built-in",
|
|
"path": "/app/skills/browser-automation/SKILL.md",
|
|
}
|
|
]
|
|
)
|
|
|
|
all_skills = mgr.list_skills(
|
|
runtime="sandbox",
|
|
active_only=False,
|
|
show_sandbox_path=False,
|
|
)
|
|
assert len(all_skills) == 1
|
|
assert all_skills[0].path == "/app/skills/browser-automation/SKILL.md"
|
|
|
|
with pytest.raises(PermissionError):
|
|
mgr.set_skill_active("browser-automation", False)
|
|
|
|
active_skills = mgr.list_skills(runtime="sandbox", active_only=True)
|
|
assert len(active_skills) == 1
|
|
assert active_skills[0].name == "browser-automation"
|
|
|
|
|
|
def test_sandbox_and_local_path_resolution_with_show_sandbox_path_false(
|
|
monkeypatch,
|
|
tmp_path: Path,
|
|
):
|
|
data_dir = tmp_path / "data"
|
|
temp_dir = tmp_path / "temp"
|
|
skills_root = tmp_path / "skills"
|
|
data_dir.mkdir(parents=True, exist_ok=True)
|
|
temp_dir.mkdir(parents=True, exist_ok=True)
|
|
skills_root.mkdir(parents=True, exist_ok=True)
|
|
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_data_path",
|
|
lambda: str(data_dir),
|
|
)
|
|
monkeypatch.setattr(
|
|
"astrbot.core.skills.skill_manager.get_astrbot_temp_path",
|
|
lambda: str(temp_dir),
|
|
)
|
|
|
|
mgr = SkillManager(skills_root=str(skills_root))
|
|
_write_skill(skills_root, "custom-local", "local description")
|
|
mgr.set_sandbox_skills_cache(
|
|
[
|
|
{
|
|
"name": "custom-local",
|
|
"description": "cached description should be overridden",
|
|
"path": "/app/skills/custom-local/SKILL.md",
|
|
},
|
|
{
|
|
"name": "python-sandbox",
|
|
"description": "ship built-in",
|
|
"path": "/app/skills/python-sandbox/SKILL.md",
|
|
},
|
|
]
|
|
)
|
|
|
|
skills = mgr.list_skills(runtime="sandbox", show_sandbox_path=False)
|
|
by_name = {item.name: item for item in skills}
|
|
|
|
assert sorted(by_name) == ["custom-local", "python-sandbox"]
|
|
assert by_name["custom-local"].description == "local description"
|
|
local_skill_path = Path(by_name["custom-local"].path)
|
|
assert local_skill_path.is_relative_to(skills_root)
|
|
assert local_skill_path == skills_root / "custom-local" / "SKILL.md"
|
|
assert by_name["python-sandbox"].path == "/app/skills/python-sandbox/SKILL.md"
|