Files
ArchiveBox/archivebox/tests/test_cli_server.py
Nick Sweeting b749b26c5d wip
2026-03-23 03:58:32 -07:00

170 lines
5.7 KiB
Python

#!/usr/bin/env python3
"""
Tests for archivebox server command.
Verify server can start (basic smoke tests only, no full server testing).
"""
import os
import subprocess
import sys
from unittest.mock import Mock
def test_sqlite_connections_use_explicit_30_second_busy_timeout():
from archivebox.core.settings import SQLITE_CONNECTION_OPTIONS
assert SQLITE_CONNECTION_OPTIONS["OPTIONS"]["timeout"] == 30
assert "PRAGMA busy_timeout = 30000;" in SQLITE_CONNECTION_OPTIONS["OPTIONS"]["init_command"]
def test_server_shows_usage_info(tmp_path, process):
"""Test that server command shows usage or starts."""
os.chdir(tmp_path)
# Just check that the command is recognized
# We won't actually start a full server in tests
result = subprocess.run(
["archivebox", "server", "--help"],
capture_output=True,
text=True,
timeout=10,
)
assert result.returncode == 0
assert "server" in result.stdout.lower() or "http" in result.stdout.lower()
def test_server_init_flag(tmp_path, process):
"""Test that --init flag runs init before starting server."""
os.chdir(tmp_path)
# Check init flag is recognized
result = subprocess.run(
["archivebox", "server", "--help"],
capture_output=True,
text=True,
timeout=10,
)
assert result.returncode == 0
assert "--init" in result.stdout or "init" in result.stdout.lower()
def test_runner_worker_uses_current_interpreter():
"""The supervised runner should use the active Python environment, not PATH."""
from archivebox.workers.supervisord_util import RUNNER_WORKER
assert RUNNER_WORKER["command"] == f"{sys.executable} -m archivebox run --daemon"
def test_reload_workers_use_current_interpreter_and_supervisord_managed_runner():
from archivebox.workers.supervisord_util import RUNNER_WATCH_WORKER, RUNSERVER_WORKER
runserver = RUNSERVER_WORKER("127.0.0.1", "8000", reload=True, pidfile="/tmp/runserver.pid")
watcher = RUNNER_WATCH_WORKER("/tmp/runserver.pid")
assert runserver["name"] == "worker_runserver"
assert runserver["command"] == f"{sys.executable} -m archivebox manage runserver 127.0.0.1:8000"
assert 'ARCHIVEBOX_RUNSERVER="1"' in runserver["environment"]
assert 'ARCHIVEBOX_AUTORELOAD="1"' in runserver["environment"]
assert 'ARCHIVEBOX_RUNSERVER_PIDFILE="/tmp/runserver.pid"' in runserver["environment"]
assert watcher["name"] == "worker_runner_watch"
assert watcher["command"] == f"{sys.executable} -m archivebox manage runner_watch --pidfile=/tmp/runserver.pid"
def test_stop_existing_background_runner_cleans_up_and_stops_orchestrators():
from archivebox.cli.archivebox_server import stop_existing_background_runner
runner_a = Mock()
runner_a.kill_tree = Mock()
runner_a.terminate = Mock()
runner_b = Mock()
runner_b.kill_tree = Mock(side_effect=RuntimeError("boom"))
runner_b.terminate = Mock()
process_model = Mock()
process_model.StatusChoices.RUNNING = "running"
process_model.TypeChoices.ORCHESTRATOR = "orchestrator"
queryset = Mock()
queryset.order_by.return_value = [runner_a, runner_b]
process_model.objects.filter.return_value = queryset
supervisor = Mock()
stop_worker = Mock()
log = Mock()
stopped = stop_existing_background_runner(
machine=Mock(),
process_model=process_model,
supervisor=supervisor,
stop_worker_fn=stop_worker,
log=log,
)
assert stopped == 2
assert process_model.cleanup_stale_running.call_count == 2
stop_worker.assert_any_call(supervisor, "worker_runner")
stop_worker.assert_any_call(supervisor, "worker_runner_watch")
runner_a.kill_tree.assert_called_once_with(graceful_timeout=2.0)
runner_b.terminate.assert_called_once_with(graceful_timeout=2.0)
log.assert_called_once()
def test_stop_existing_server_workers_takes_over_same_runserver_port(monkeypatch):
from archivebox.cli.archivebox_server import stop_existing_server_workers
supervisor = Mock()
supervisor.getProcessInfo.side_effect = lambda name: {
"worker_runserver": {"statename": "RUNNING"},
"worker_daphne": {"statename": "STOPPED"},
}.get(name, None)
stop_worker = Mock()
log = Mock()
monkeypatch.setattr(
"archivebox.cli.archivebox_server._read_supervisor_worker_command",
lambda worker_name: f"{sys.executable} -m archivebox manage runserver 0.0.0.0:8000" if worker_name == "worker_runserver" else "",
)
stopped = stop_existing_server_workers(
supervisor=supervisor,
stop_worker_fn=stop_worker,
host="0.0.0.0",
port="8000",
log=log,
)
assert stopped == 1
stop_worker.assert_called_once_with(supervisor, "worker_runserver")
log.assert_called_once()
def test_stop_existing_server_workers_leaves_different_port_running(monkeypatch):
from archivebox.cli.archivebox_server import stop_existing_server_workers
supervisor = Mock()
supervisor.getProcessInfo.side_effect = lambda name: {
"worker_runserver": {"statename": "RUNNING"},
"worker_daphne": {"statename": "STOPPED"},
}.get(name, None)
stop_worker = Mock()
log = Mock()
monkeypatch.setattr(
"archivebox.cli.archivebox_server._read_supervisor_worker_command",
lambda worker_name: f"{sys.executable} -m archivebox manage runserver 127.0.0.1:9000" if worker_name == "worker_runserver" else "",
)
stopped = stop_existing_server_workers(
supervisor=supervisor,
stop_worker_fn=stop_worker,
host="0.0.0.0",
port="8000",
log=log,
)
assert stopped == 0
stop_worker.assert_not_called()
log.assert_not_called()