API: /api/v1/campaigns (paginated list), /api/v1/campaigns/{uuid}
(soft-merge chain follow), /api/v1/campaigns/{uuid}/identities
(member identities), and /api/v1/campaigns/events (SSE under
campaign.> + JWT-via-?token=, snapshot-on-connect). Mirror of the
identity router; same auth, same shape, same OpenAPI tags pattern.
Frontend: CampaignDetail.tsx page (same visual vocabulary as
IdentityDetail), useCampaignStream hook (mirror of
useIdentityStream), /campaigns/:id route, IdentityDetail's
CAMPAIGN badge becomes clickable and navigates to the campaign.
useIdentityStream now listens for identity.campaign.assigned so
the badge appears live without a manual refresh.
255 lines
9.2 KiB
Python
255 lines
9.2 KiB
Python
"""Tests for the campaign-clustering read API.
|
|
|
|
Mirrors :mod:`tests.web.test_api_identities` for the layer above.
|
|
The campaign clusterer is a separate worker; these tests cover the
|
|
read-only API which ships in the same wave. Empty-table behaviour,
|
|
soft-merge resolution, and pagination forwarding are the headline
|
|
cases.
|
|
"""
|
|
from datetime import datetime, timezone
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
import pytest
|
|
from fastapi import HTTPException
|
|
|
|
|
|
def _campaign_row(
|
|
uuid: str = "c-uuid-1",
|
|
merged_into_uuid: str | None = None,
|
|
identity_count: int = 0,
|
|
) -> dict:
|
|
now = datetime(2026, 4, 26, tzinfo=timezone.utc).isoformat()
|
|
return {
|
|
"uuid": uuid,
|
|
"schema_version": 1,
|
|
"first_seen_at": None,
|
|
"last_seen_at": None,
|
|
"created_at": now,
|
|
"updated_at": now,
|
|
"confidence": None,
|
|
"identity_count": identity_count,
|
|
"ja3_hashes": None,
|
|
"hassh_hashes": None,
|
|
"payload_simhashes": None,
|
|
"c2_endpoints": None,
|
|
"merged_into_uuid": merged_into_uuid,
|
|
"notes": None,
|
|
}
|
|
|
|
|
|
def _identity_row(uuid: str, campaign_id: str | None) -> dict:
|
|
return {
|
|
"uuid": uuid,
|
|
"schema_version": 1,
|
|
"campaign_id": campaign_id,
|
|
"merged_into_uuid": None,
|
|
}
|
|
|
|
|
|
# ─── GET /campaigns ──────────────────────────────────────────────────────────
|
|
|
|
|
|
class TestListCampaigns:
|
|
@pytest.mark.asyncio
|
|
async def test_empty_table_returns_zero_total(self):
|
|
from decnet.web.router.campaigns.api_list_campaigns import list_campaigns
|
|
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaigns.repo"
|
|
) as mock_repo:
|
|
mock_repo.list_campaigns = AsyncMock(return_value=[])
|
|
mock_repo.count_campaigns = AsyncMock(return_value=0)
|
|
|
|
result = await list_campaigns(
|
|
limit=50, offset=0, user={"uuid": "u", "role": "viewer"}
|
|
)
|
|
|
|
assert result == {"total": 0, "limit": 50, "offset": 0, "data": []}
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_returns_seeded_data(self):
|
|
from decnet.web.router.campaigns.api_list_campaigns import list_campaigns
|
|
|
|
rows = [_campaign_row(f"c-{n}") for n in range(3)]
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaigns.repo"
|
|
) as mock_repo:
|
|
mock_repo.list_campaigns = AsyncMock(return_value=rows)
|
|
mock_repo.count_campaigns = AsyncMock(return_value=3)
|
|
|
|
result = await list_campaigns(
|
|
limit=50, offset=0, user={"uuid": "u", "role": "viewer"}
|
|
)
|
|
|
|
assert result["total"] == 3
|
|
assert [r["uuid"] for r in result["data"]] == ["c-0", "c-1", "c-2"]
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_pagination_args_forwarded(self):
|
|
from decnet.web.router.campaigns.api_list_campaigns import list_campaigns
|
|
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaigns.repo"
|
|
) as mock_repo:
|
|
mock_repo.list_campaigns = AsyncMock(return_value=[])
|
|
mock_repo.count_campaigns = AsyncMock(return_value=0)
|
|
|
|
await list_campaigns(
|
|
limit=10, offset=20, user={"uuid": "u", "role": "viewer"}
|
|
)
|
|
|
|
mock_repo.list_campaigns.assert_awaited_once_with(limit=10, offset=20)
|
|
|
|
|
|
# ─── GET /campaigns/{uuid} ───────────────────────────────────────────────────
|
|
|
|
|
|
class TestGetCampaignDetail:
|
|
@pytest.mark.asyncio
|
|
async def test_404_on_missing_uuid(self):
|
|
from decnet.web.router.campaigns.api_get_campaign_detail import (
|
|
get_campaign_detail,
|
|
)
|
|
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_get_campaign_detail.repo"
|
|
) as mock_repo:
|
|
mock_repo.get_campaign_by_uuid = AsyncMock(return_value=None)
|
|
|
|
with pytest.raises(HTTPException) as exc:
|
|
await get_campaign_detail(
|
|
uuid="ghost", user={"uuid": "u", "role": "viewer"}
|
|
)
|
|
assert exc.value.status_code == 404
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_returns_campaign_with_live_identity_count(self):
|
|
from decnet.web.router.campaigns.api_get_campaign_detail import (
|
|
get_campaign_detail,
|
|
)
|
|
|
|
campaign = _campaign_row("c-real", identity_count=2)
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_get_campaign_detail.repo"
|
|
) as mock_repo:
|
|
mock_repo.get_campaign_by_uuid = AsyncMock(return_value=campaign)
|
|
mock_repo.count_identities_for_campaign = AsyncMock(return_value=5)
|
|
|
|
result = await get_campaign_detail(
|
|
uuid="c-real", user={"uuid": "u", "role": "viewer"}
|
|
)
|
|
|
|
assert result["uuid"] == "c-real"
|
|
assert result["identity_count_live"] == 5
|
|
assert result["identity_count"] == 2
|
|
|
|
|
|
# ─── GET /campaigns/{uuid}/identities ────────────────────────────────────────
|
|
|
|
|
|
class TestListCampaignIdentities:
|
|
@pytest.mark.asyncio
|
|
async def test_404_when_campaign_missing(self):
|
|
from decnet.web.router.campaigns.api_list_campaign_identities import (
|
|
list_campaign_identities,
|
|
)
|
|
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaign_identities.repo"
|
|
) as mock_repo:
|
|
mock_repo.get_campaign_by_uuid = AsyncMock(return_value=None)
|
|
|
|
with pytest.raises(HTTPException) as exc:
|
|
await list_campaign_identities(
|
|
uuid="ghost", limit=50, offset=0,
|
|
user={"uuid": "u", "role": "viewer"},
|
|
)
|
|
assert exc.value.status_code == 404
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_returns_identities_for_existing_campaign(self):
|
|
from decnet.web.router.campaigns.api_list_campaign_identities import (
|
|
list_campaign_identities,
|
|
)
|
|
|
|
campaign = _campaign_row("c-real")
|
|
idents = [
|
|
_identity_row("i-1", "c-real"),
|
|
_identity_row("i-2", "c-real"),
|
|
]
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaign_identities.repo"
|
|
) as mock_repo:
|
|
mock_repo.get_campaign_by_uuid = AsyncMock(return_value=campaign)
|
|
mock_repo.list_identities_for_campaign = AsyncMock(return_value=idents)
|
|
mock_repo.count_identities_for_campaign = AsyncMock(return_value=2)
|
|
|
|
result = await list_campaign_identities(
|
|
uuid="c-real", limit=50, offset=0,
|
|
user={"uuid": "u", "role": "viewer"},
|
|
)
|
|
|
|
assert result["total"] == 2
|
|
assert [r["uuid"] for r in result["data"]] == ["i-1", "i-2"]
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_merged_uuid_resolves_to_winners_identities(self):
|
|
"""Soft-merged campaigns: identities are listed under the winner."""
|
|
from decnet.web.router.campaigns.api_list_campaign_identities import (
|
|
list_campaign_identities,
|
|
)
|
|
|
|
winner = _campaign_row("c-winner")
|
|
with patch(
|
|
"decnet.web.router.campaigns.api_list_campaign_identities.repo"
|
|
) as mock_repo:
|
|
mock_repo.get_campaign_by_uuid = AsyncMock(return_value=winner)
|
|
mock_repo.list_identities_for_campaign = AsyncMock(return_value=[])
|
|
mock_repo.count_identities_for_campaign = AsyncMock(return_value=0)
|
|
|
|
await list_campaign_identities(
|
|
uuid="c-loser", limit=50, offset=0,
|
|
user={"uuid": "u", "role": "viewer"},
|
|
)
|
|
|
|
mock_repo.list_identities_for_campaign.assert_awaited_once_with(
|
|
"c-winner", limit=50, offset=0,
|
|
)
|
|
|
|
|
|
# ─── Repo-level integration ──────────────────────────────────────────────────
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_repo_methods_against_empty_schema(tmp_path):
|
|
from decnet.web.db.factory import get_repository
|
|
|
|
repo = get_repository(db_path=str(tmp_path / "campaigns.db"))
|
|
await repo.initialize()
|
|
|
|
assert await repo.list_campaigns(limit=50, offset=0) == []
|
|
assert await repo.count_campaigns() == 0
|
|
assert await repo.get_campaign_by_uuid("anything") is None
|
|
assert await repo.list_identities_for_campaign("anything") == []
|
|
assert await repo.count_identities_for_campaign("anything") == 0
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_repo_follows_campaign_merge_chain(tmp_path):
|
|
from decnet.web.db.factory import get_repository
|
|
|
|
repo = get_repository(db_path=str(tmp_path / "merge.db"))
|
|
await repo.initialize()
|
|
await repo.create_campaign({"uuid": "winner-uuid"})
|
|
await repo.create_campaign(
|
|
{"uuid": "loser-uuid", "merged_into_uuid": "winner-uuid"}
|
|
)
|
|
|
|
resolved = await repo.get_campaign_by_uuid("loser-uuid")
|
|
assert resolved is not None
|
|
assert resolved["uuid"] == "winner-uuid"
|
|
|
|
direct = await repo.get_campaign_by_uuid("winner-uuid")
|
|
assert direct["uuid"] == "winner-uuid"
|
|
assert direct["merged_into_uuid"] is None
|