125 lines
5.2 KiB
Python
125 lines
5.2 KiB
Python
|
|
"""Functional tests for CLI entry points via Typer's CliRunner.
|
||
|
|
|
||
|
|
Each test invokes the real CLI command. Ollama is not running, so
|
||
|
|
commands that need inference will fail gracefully — and that's a valid
|
||
|
|
user scenario we want to verify.
|
||
|
|
"""
|
||
|
|
|
||
|
|
import pytest
|
||
|
|
|
||
|
|
|
||
|
|
# ── timmy CLI ─────────────────────────────────────────────────────────────────
|
||
|
|
|
||
|
|
|
||
|
|
class TestTimmyCLI:
|
||
|
|
"""Tests the `timmy` command (chat, think, status)."""
|
||
|
|
|
||
|
|
def test_status_runs(self, timmy_runner):
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["status"])
|
||
|
|
# Ollama is offline, so this should either:
|
||
|
|
# - Print an error about Ollama being unreachable, OR
|
||
|
|
# - Exit non-zero
|
||
|
|
# Either way, the CLI itself shouldn't crash with an unhandled exception.
|
||
|
|
# The exit code tells us if the command ran at all.
|
||
|
|
assert result.exit_code is not None
|
||
|
|
|
||
|
|
def test_chat_requires_message(self, timmy_runner):
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["chat"])
|
||
|
|
# Missing required argument
|
||
|
|
assert result.exit_code != 0
|
||
|
|
assert "Missing argument" in result.output or "Usage" in result.output
|
||
|
|
|
||
|
|
def test_think_requires_topic(self, timmy_runner):
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["think"])
|
||
|
|
assert result.exit_code != 0
|
||
|
|
assert "Missing argument" in result.output or "Usage" in result.output
|
||
|
|
|
||
|
|
def test_chat_with_message_runs(self, timmy_runner):
|
||
|
|
"""Chat with a real message — Ollama offline means graceful failure."""
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["chat", "hello"])
|
||
|
|
# Will fail because Ollama isn't running, but the CLI should handle it
|
||
|
|
assert result.exit_code is not None
|
||
|
|
|
||
|
|
def test_backend_flag_accepted(self, timmy_runner):
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["status", "--backend", "ollama"])
|
||
|
|
assert result.exit_code is not None
|
||
|
|
|
||
|
|
def test_help_text(self, timmy_runner):
|
||
|
|
runner, app = timmy_runner
|
||
|
|
result = runner.invoke(app, ["--help"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "Timmy" in result.output or "sovereign" in result.output.lower()
|
||
|
|
|
||
|
|
|
||
|
|
# ── timmy-serve CLI ───────────────────────────────────────────────────────────
|
||
|
|
|
||
|
|
|
||
|
|
class TestTimmyServeCLI:
|
||
|
|
"""Tests the `timmy-serve` command (start, invoice, status)."""
|
||
|
|
|
||
|
|
def test_start_dry_run(self, serve_runner):
|
||
|
|
"""--dry-run should print config and exit cleanly."""
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["start", "--dry-run"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "Starting Timmy Serve" in result.output
|
||
|
|
assert "Dry run" in result.output or "dry run" in result.output
|
||
|
|
|
||
|
|
def test_start_dry_run_custom_port(self, serve_runner):
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["start", "--dry-run", "--port", "9999"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "9999" in result.output
|
||
|
|
|
||
|
|
def test_start_dry_run_custom_price(self, serve_runner):
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["start", "--dry-run", "--price", "500"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "500" in result.output
|
||
|
|
|
||
|
|
def test_invoice_creates_real_invoice(self, serve_runner):
|
||
|
|
"""Create a real Lightning invoice via the mock backend."""
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["invoice", "--amount", "200", "--memo", "test invoice"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "Invoice created" in result.output
|
||
|
|
assert "200" in result.output
|
||
|
|
assert "Payment hash" in result.output or "payment_hash" in result.output.lower()
|
||
|
|
|
||
|
|
def test_status_shows_earnings(self, serve_runner):
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["status"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "Total invoices" in result.output or "invoices" in result.output.lower()
|
||
|
|
assert "sats" in result.output.lower()
|
||
|
|
|
||
|
|
def test_help_text(self, serve_runner):
|
||
|
|
runner, app = serve_runner
|
||
|
|
result = runner.invoke(app, ["--help"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "Serve" in result.output or "Lightning" in result.output
|
||
|
|
|
||
|
|
|
||
|
|
# ── self-tdd CLI ──────────────────────────────────────────────────────────────
|
||
|
|
|
||
|
|
|
||
|
|
class TestSelfTddCLI:
|
||
|
|
"""Tests the `self-tdd` command (watch)."""
|
||
|
|
|
||
|
|
def test_help_text(self, tdd_runner):
|
||
|
|
runner, app = tdd_runner
|
||
|
|
result = runner.invoke(app, ["--help"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "watchdog" in result.output.lower() or "test" in result.output.lower()
|
||
|
|
|
||
|
|
def test_watch_help(self, tdd_runner):
|
||
|
|
runner, app = tdd_runner
|
||
|
|
result = runner.invoke(app, ["watch", "--help"])
|
||
|
|
assert result.exit_code == 0
|
||
|
|
assert "interval" in result.output.lower()
|