aider_run multi-provider credentials (tools/aider.py):
- _resolve_credentials() — general credential resolver; replaces the previous
OpenRouter-only injection; resolution priority: Anthropic model hint → explicit
host_label → model prefix (openrouter/*, groq/*, deepseek/*, …) → OpenRouter
default → Anthropic API key → any keyed cloud host → local/generic host
- _host_flags() — generates --api-key slug=key for known cloud providers (OpenRouter,
OpenAI, Groq, Together, Fireworks, X.ai, DeepSeek, Mistral); generates
--openai-api-base + --openai-api-key for generic/local hosts (Open WebUI, Ollama);
appends /api suffix for openwebui host_type; auto-prefixes model with 'openai/'
for generic endpoints when model has no / prefix
- Anthropic API keys from providers.anthropic.credentials (not a host entry)
- host_label param added to aider_run and FunctionDeclaration — pick a configured
host by partial label match (e.g. 'OpenRouter', 'Local', 'scott-lt-i7-rtx')
- 16 unit tests for _resolve_credentials covering all resolution paths
main.py: move @app.get("/health") before app.include_router(ui.router) — the
/{username} catch-all in ui.router was swallowing the /health path
Test suite: 37 pre-existing failures → 182/182 passing
- test_tools.py: _task_list() missing priority arg (6 callsites); cron ID regex
c_\w+ → c_[\w-]+ (token_urlsafe includes '-', causing intermittent truncation)
- test_webhooks.py: rewritten for per-user channel config architecture —
patch routers.nextcloud_talk/google_chat.get_user_channels instead of removed
settings fields; corrected endpoints /webhook/nextcloud/scott and
/channels/google-chat/scott; non-empty cfg dicts so falsy-guard passes
- test_health.py: test_unknown_route_404 now uses 3-segment path (/{u}/{p}/x)
since single-segment paths hit the /{username} UI catch-all
- test_api_files.py: removed '../config.py' from not-in-allowed test (ASGI
normalizes it to /config.py which hits /{username} catch-all, not files router)
- test_security.py: same webhook patch target fix; per-user endpoint URLs
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
273 lines
10 KiB
Python
273 lines
10 KiB
Python
"""
|
|
Unit tests for Inara's internal tools — scratch, tasks, cron.
|
|
|
|
These test the sync implementation functions directly, using a temp directory.
|
|
No HTTP, no LLM calls.
|
|
"""
|
|
import json
|
|
import pytest
|
|
from pathlib import Path
|
|
from unittest.mock import patch
|
|
|
|
|
|
@pytest.fixture
|
|
def persona_dir(tmp_path) -> Path:
|
|
"""A temp persona directory pre-populated with empty tool files."""
|
|
d = tmp_path / "inara"
|
|
d.mkdir()
|
|
(d / "SCRATCH.md").write_text("")
|
|
(d / "TASKS.json").write_text("[]")
|
|
(d / "CRONS.json").write_text("[]")
|
|
(d / "REMINDERS.md").write_text("")
|
|
return d
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def patch_persona_path(persona_dir):
|
|
"""
|
|
Route all persona_path() calls to the temp dir.
|
|
Each tool does `from persona import persona_path`, so we must patch
|
|
the name in each module's namespace, not just in persona itself.
|
|
"""
|
|
with (
|
|
patch("tools.scratch.persona_path", return_value=persona_dir),
|
|
patch("tools.tasks.persona_path", return_value=persona_dir),
|
|
patch("tools.cron.persona_path", return_value=persona_dir),
|
|
patch("cron_runner._persona_path", return_value=persona_dir),
|
|
):
|
|
yield
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Scratch
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestScratch:
|
|
def test_read_empty(self):
|
|
from tools.scratch import _scratch_read
|
|
result = _scratch_read()
|
|
assert "empty" in result.lower()
|
|
|
|
def test_write_and_read(self, persona_dir):
|
|
from tools.scratch import _scratch_write, _scratch_read
|
|
_scratch_write("# My notes\n\nSome content here.")
|
|
content = _scratch_read()
|
|
assert "My notes" in content
|
|
assert "Some content here" in content
|
|
|
|
def test_append_adds_section(self, persona_dir):
|
|
from tools.scratch import _scratch_write, _scratch_append, _scratch_read
|
|
_scratch_write("# Existing")
|
|
_scratch_append("New section content", heading="Section A")
|
|
content = _scratch_read()
|
|
assert "Existing" in content
|
|
assert "Section A" in content
|
|
assert "New section content" in content
|
|
|
|
def test_append_auto_heading(self, persona_dir):
|
|
from tools.scratch import _scratch_append, _scratch_read
|
|
_scratch_append("Content without explicit heading")
|
|
content = _scratch_read()
|
|
assert "UTC" in content # auto heading includes timestamp
|
|
|
|
def test_clear(self, persona_dir):
|
|
from tools.scratch import _scratch_write, _scratch_clear, _scratch_read
|
|
_scratch_write("Some content")
|
|
_scratch_clear()
|
|
assert "empty" in _scratch_read().lower()
|
|
|
|
def test_write_strips_trailing_whitespace(self, persona_dir):
|
|
from tools.scratch import _scratch_write, _scratch_read
|
|
_scratch_write("Content \n\n\n")
|
|
content = (persona_dir / "SCRATCH.md").read_text()
|
|
assert content.endswith("\n")
|
|
assert not content.endswith("\n\n")
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Tasks
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestTasks:
|
|
def _mk(self, title, description=None, priority="normal"):
|
|
from tools.tasks import _task_create
|
|
return _task_create(title, description, priority)
|
|
|
|
def _id(self, result: str) -> str:
|
|
import re
|
|
m = re.search(r't_\w+', result)
|
|
assert m, f"No task ID in: {result}"
|
|
return m.group()
|
|
|
|
def test_list_empty(self):
|
|
from tools.tasks import _task_list
|
|
assert "No tasks" in _task_list(status=None, priority=None)
|
|
|
|
def test_create_and_list(self):
|
|
from tools.tasks import _task_list
|
|
self._mk("Buy coffee", description="Dark roast", priority="high")
|
|
result = _task_list(status=None, priority=None)
|
|
assert "Buy coffee" in result
|
|
assert "[high]" in result
|
|
|
|
def test_create_bad_priority_defaults_to_normal(self):
|
|
from tools.tasks import _task_list
|
|
self._mk("Test task", priority="urgent") # invalid — becomes "normal"
|
|
result = _task_list(status=None, priority=None)
|
|
assert "Test task" in result
|
|
assert "[normal]" not in result # normal priority not shown in brackets
|
|
|
|
def test_update_status(self):
|
|
from tools.tasks import _task_update, _task_list
|
|
tid = self._id(self._mk("Work item"))
|
|
_task_update(tid, status="in_progress", title=None, description=None, priority=None)
|
|
assert "Work item" in _task_list(status="in_progress", priority=None)
|
|
|
|
def test_complete(self):
|
|
from tools.tasks import _task_complete, _task_list
|
|
tid = self._id(self._mk("Finish this"))
|
|
_task_complete(tid)
|
|
assert "Finish this" in _task_list(status="done", priority=None)
|
|
assert "Finish this" not in _task_list(status="todo", priority=None)
|
|
|
|
def test_filter_by_status(self):
|
|
from tools.tasks import _task_list
|
|
self._mk("A task")
|
|
assert "A task" in _task_list(status="todo", priority=None)
|
|
assert "A task" not in _task_list(status="done", priority=None)
|
|
|
|
def test_update_unknown_id(self):
|
|
from tools.tasks import _task_update
|
|
result = _task_update("t_doesnotexist", status="done",
|
|
title=None, description=None, priority=None)
|
|
assert "not found" in result.lower()
|
|
|
|
def test_persistence(self, persona_dir):
|
|
"""Tasks survive between _load() calls (written to TASKS.json)."""
|
|
self._mk("Persistent task")
|
|
data = json.loads((persona_dir / "TASKS.json").read_text())
|
|
assert any(t["title"] == "Persistent task" for t in data)
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# Cron
|
|
# ---------------------------------------------------------------------------
|
|
|
|
class TestCronRunner:
|
|
def test_parse_hourly(self):
|
|
from cron_runner import parse_schedule
|
|
assert parse_schedule("hourly") == {"minute": 0}
|
|
|
|
def test_parse_daily_default(self):
|
|
from cron_runner import parse_schedule
|
|
r = parse_schedule("daily")
|
|
assert r["hour"] == 9
|
|
assert r["minute"] == 0
|
|
|
|
def test_parse_daily_custom_time(self):
|
|
from cron_runner import parse_schedule
|
|
r = parse_schedule("daily:14:30")
|
|
assert r["hour"] == 14
|
|
assert r["minute"] == 30
|
|
|
|
def test_parse_weekly(self):
|
|
from cron_runner import parse_schedule
|
|
r = parse_schedule("weekly:mon")
|
|
assert r["day_of_week"] == "mon"
|
|
assert r["hour"] == 9
|
|
|
|
def test_parse_weekly_with_time(self):
|
|
from cron_runner import parse_schedule
|
|
r = parse_schedule("weekly:fri:17:00")
|
|
assert r["day_of_week"] == "fri"
|
|
assert r["hour"] == 17
|
|
assert r["minute"] == 0
|
|
|
|
def test_parse_full_day_names(self):
|
|
from cron_runner import parse_schedule
|
|
assert parse_schedule("weekly:monday")["day_of_week"] == "mon"
|
|
assert parse_schedule("weekly:friday")["day_of_week"] == "fri"
|
|
|
|
def test_parse_unknown_schedule(self):
|
|
from cron_runner import parse_schedule
|
|
with pytest.raises(ValueError, match="Unrecognised"):
|
|
parse_schedule("every-tuesday")
|
|
|
|
def test_parse_bad_dow(self):
|
|
from cron_runner import parse_schedule
|
|
with pytest.raises(ValueError, match="day of week"):
|
|
parse_schedule("weekly:funday")
|
|
|
|
def test_parse_bad_time_non_integer(self):
|
|
from cron_runner import parse_schedule
|
|
with pytest.raises((ValueError, Exception)):
|
|
parse_schedule("daily:noon") # non-integer — parse error
|
|
|
|
|
|
class TestCronTools:
|
|
def test_list_empty(self):
|
|
from tools.cron import _cron_list
|
|
result = _cron_list()
|
|
assert "No crons" in result
|
|
|
|
def test_add_and_list(self):
|
|
from tools.cron import _cron_add, _cron_list
|
|
with patch("tools.cron._scheduler_add"):
|
|
r = _cron_add("Morning reminder", "daily:09:00", "remind", "Check in.")
|
|
assert "c_" in r
|
|
listing = _cron_list()
|
|
assert "Morning reminder" in listing
|
|
assert "daily:09:00" in listing
|
|
|
|
def test_add_bad_schedule(self):
|
|
from tools.cron import _cron_add
|
|
r = _cron_add("Bad job", "every-day", "remind", "Hello")
|
|
assert "Bad schedule" in r
|
|
|
|
def test_add_bad_type(self):
|
|
from tools.cron import _cron_add
|
|
r = _cron_add("Bad job", "daily", "email", "Hello")
|
|
assert "Bad type" in r
|
|
|
|
def _extract_id(self, result: str) -> str:
|
|
import re
|
|
# token_urlsafe can include '-'; use [\w-]+ to capture the full ID
|
|
m = re.search(r'c_[\w-]+', result)
|
|
assert m, f"No cron ID in: {result}"
|
|
return m.group()
|
|
|
|
def test_remove(self):
|
|
from tools.cron import _cron_add, _cron_remove, _cron_list
|
|
with patch("tools.cron._scheduler_add"), patch("tools.cron._scheduler_remove"):
|
|
r = _cron_add("To remove", "hourly", "note", "Tick")
|
|
cron_id = self._extract_id(r)
|
|
result = _cron_remove(cron_id)
|
|
assert "Removed" in result
|
|
assert "To remove" not in _cron_list()
|
|
|
|
def test_remove_unknown(self):
|
|
from tools.cron import _cron_remove
|
|
result = _cron_remove("c_doesnotexist")
|
|
assert "Not found" in result
|
|
|
|
def test_toggle_pause_resume(self):
|
|
from tools.cron import _cron_add, _cron_toggle, _cron_list
|
|
with patch("tools.cron._scheduler_add"), patch("tools.cron._scheduler_pause"), patch("tools.cron._scheduler_resume"):
|
|
r = _cron_add("Toggleable", "daily", "note", "Content")
|
|
cron_id = self._extract_id(r)
|
|
|
|
result = _cron_toggle(cron_id)
|
|
assert "Paused" in result
|
|
assert "PAUSED" in _cron_list()
|
|
|
|
result = _cron_toggle(cron_id)
|
|
assert "Resumed" in result
|
|
assert "enabled" in _cron_list()
|
|
|
|
def test_reminders_clear(self, persona_dir):
|
|
from tools.cron import _reminders_clear
|
|
(persona_dir / "REMINDERS.md").write_text("## Some reminder\n\nContent")
|
|
result = _reminders_clear()
|
|
assert "cleared" in result.lower()
|
|
assert (persona_dir / "REMINDERS.md").read_text() == ""
|