mirror of
https://github.com/zvx-echo6/central.git
synced 2026-05-22 02:24:38 +02:00
feat(gui): implement first-run setup wizard (1b-8)
Add a 5-step setup wizard that replaces the single-step /setup: 1. Create Operator - create initial operator account 2. System Settings - configure map tile URL and attribution 3. API Keys - optionally add API keys for adapters 4. Configure Adapters - enable/disable adapters with region picker 5. Finish Setup - review and complete setup Key changes: - Update middleware to handle wizard URL structure and step routing - Add wizard routes for each step with proper auth checks - Create new templates using base_wizard.html for consistent styling - Add audit events for system.update and setup.complete - Update tests for new middleware behavior Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
parent
96ec88883c
commit
62116ca6a4
12 changed files with 1840 additions and 27 deletions
|
|
@ -12,8 +12,8 @@ class TestSetupGateMiddleware:
|
|||
"""Tests for SetupGateMiddleware."""
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_allows_setup_route_when_incomplete(self):
|
||||
"""SetupGateMiddleware allows /setup when setup_complete=False."""
|
||||
async def test_allows_setup_subpath_when_incomplete(self):
|
||||
"""SetupGateMiddleware allows /setup/operator when setup_complete=False."""
|
||||
mock_pool = MagicMock()
|
||||
mock_conn = MagicMock()
|
||||
mock_conn.fetchrow = AsyncMock(return_value={"setup_complete": False})
|
||||
|
|
@ -21,6 +21,31 @@ class TestSetupGateMiddleware:
|
|||
mock_conn.__aexit__ = AsyncMock()
|
||||
mock_pool.acquire = MagicMock(return_value=mock_conn)
|
||||
|
||||
with patch("central.gui.middleware.get_pool", return_value=mock_pool):
|
||||
app = FastAPI()
|
||||
|
||||
@app.get("/setup/operator")
|
||||
async def setup_operator():
|
||||
return {"message": "operator"}
|
||||
|
||||
app.add_middleware(SetupGateMiddleware)
|
||||
client = TestClient(app)
|
||||
|
||||
response = client.get("/setup/operator")
|
||||
assert response.status_code == 200
|
||||
assert response.json() == {"message": "operator"}
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_redirects_setup_base_to_wizard_step(self):
|
||||
"""SetupGateMiddleware redirects /setup to wizard step when incomplete."""
|
||||
mock_pool = MagicMock()
|
||||
mock_conn = MagicMock()
|
||||
mock_conn.fetchrow = AsyncMock(return_value={"setup_complete": False})
|
||||
mock_conn.fetchval = AsyncMock(return_value=0) # No operators
|
||||
mock_conn.__aenter__ = AsyncMock(return_value=mock_conn)
|
||||
mock_conn.__aexit__ = AsyncMock()
|
||||
mock_pool.acquire = MagicMock(return_value=mock_conn)
|
||||
|
||||
with patch("central.gui.middleware.get_pool", return_value=mock_pool):
|
||||
app = FastAPI()
|
||||
|
||||
|
|
@ -28,12 +53,16 @@ class TestSetupGateMiddleware:
|
|||
async def setup():
|
||||
return {"message": "setup"}
|
||||
|
||||
@app.get("/setup/operator")
|
||||
async def setup_operator():
|
||||
return {"message": "operator"}
|
||||
|
||||
app.add_middleware(SetupGateMiddleware)
|
||||
client = TestClient(app)
|
||||
client = TestClient(app, follow_redirects=False)
|
||||
|
||||
response = client.get("/setup")
|
||||
assert response.status_code == 200
|
||||
assert response.json() == {"message": "setup"}
|
||||
assert response.status_code == 302
|
||||
assert response.headers["location"] == "/setup/operator"
|
||||
|
||||
@pytest.mark.asyncio
|
||||
async def test_allows_health_when_incomplete(self):
|
||||
|
|
@ -135,7 +164,7 @@ class TestSetupGateMiddleware:
|
|||
|
||||
@pytest.mark.asyncio
|
||||
async def test_redirects_setup_when_complete(self):
|
||||
"""SetupGateMiddleware redirects /setup to / when setup_complete=True."""
|
||||
"""SetupGateMiddleware redirects /setup/* to / when setup_complete=True."""
|
||||
mock_pool = MagicMock()
|
||||
mock_conn = MagicMock()
|
||||
mock_conn.fetchrow = AsyncMock(return_value={"setup_complete": True})
|
||||
|
|
@ -154,9 +183,18 @@ class TestSetupGateMiddleware:
|
|||
async def setup():
|
||||
return {"message": "setup"}
|
||||
|
||||
@app.get("/setup/operator")
|
||||
async def setup_operator():
|
||||
return {"message": "operator"}
|
||||
|
||||
app.add_middleware(SetupGateMiddleware)
|
||||
client = TestClient(app, follow_redirects=False)
|
||||
|
||||
# Both /setup and /setup/operator should redirect to /
|
||||
response = client.get("/setup")
|
||||
assert response.status_code == 302
|
||||
assert response.headers["location"] == "/"
|
||||
|
||||
response = client.get("/setup/operator")
|
||||
assert response.status_code == 302
|
||||
assert response.headers["location"] == "/"
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue