Files
python-tdd/src/apps/epic/tests/integrated/test_views.py
Disco DeDisco a4adf9664b room scroll-of-events: table-hex aperture binary scroll-snaps to the room provenance feed — TDD
From Role Select onwards, scrolling DOWN in the table-hex aperture swaps the entire hex view for the room's GameEvent feed (mirrors my_sky's wheel<->form; scroll-snap-stop:always = no partial scroll). Reuses the Billscroll query + core/_partials/_scroll.html so the feed renders identically.

room.html: #id_room_aperture wraps .room-hex-pane (existing .room-shell + the _table_positions strip moved INSIDE so the circles scroll away with the hex) + .room-scroll-pane (includes new _room_scroll.html); 'is-scrollable' added iff table_status set.

_room_scroll.html (new) = the DRY seam for my_sea; includes the shared scroll partial + a tiny dots-animation script (no scroll-position persistence). room_view adds events/viewer/scroll_position (same query as billboard.views.scroll).

_room.scss: .room-aperture + .room-pane (height:100%, not min-height); .is-scrollable engages scroll-snap-type:y mandatory + per-pane scroll-snap-align:start & scroll-snap-stop:always; .room-scroll-pane styles #id_drama_scroll + .scroll-buffer { margin-top:auto } (pure-CSS bottom-pin).

Trap: the aperture & panes set NO z-index/transform/opacity/filter -> NO stacking context, so the position strip's z-130 still resolves in the root context, above the gate/sig overlays (z-100/120). Verified by gatekeeper FTs (token drop + circle/modal layering).

Deferred INDEFINITELY (user): rising-game-cost + max room membership + max simultaneous CARTE slots + in-slot token combinations — until CARTE is anything but a secret type of Trinket.

Tests: RoomScrollOfEventsTest (5 ITs — aperture wraps hex+scroll panes, is-scrollable from Role Select, feed renders + scoped to room, no scroll pane in gate phase); functional_tests/test_game_room_scroll.py (2 FTs — computed scroll-snap props + scroll-down reveals the feed). 551 epic ITs/UTs green; 2 new FTs green; gatekeeper (token-drop + circle layering) + role-select (card fan) FTs green.

[[project-room-scroll-of-events]] [[project-position-circle-tooltips]]

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-01 18:32:57 -04:00

3357 lines
146 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

from datetime import timedelta
from unittest.mock import ANY, patch
from django.test import TestCase
from django.urls import reverse
from django.utils import timezone
from apps.drama.models import GameEvent, Note
from apps.lyric.models import Token, User
from apps.epic.models import (
Character, DeckVariant, GateSlot, Room, RoomInvite, SigReservation, TableSeat, TarotCard,
)
class RoomCreationViewTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="founder@test.io")
self.client.force_login(self.user)
def test_post_creates_room_and_redirects_to_gatekeeper(self):
response = self.client.post(
reverse("epic:create_room"),
data={"name": "Test Room"},
)
room = Room.objects.get(owner=self.user)
self.assertRedirects(
response, reverse(
"epic:gatekeeper",
args=[room.id],
)
)
def test_post_requires_login(self):
self.client.logout()
response = self.client.post(
reverse("epic:create_room"),
data={"name": "Test Room"},
)
def test_create_room_get_redirects_to_gameboard(self):
response = self.client.get(reverse("epic:create_room"))
self.assertRedirects(response, "/gameboard/")
def test_create_room_records_welcome_event_with_no_actor(self):
"""First scroll log on a fresh room is a system-authored welcome —
not a user action, so actor=None. The visible greeting is the
ROOM_CREATED event's `to_prose` ("Welcome to <name>!")."""
self.client.post(
reverse("epic:create_room"),
data={"name": "Welcoming Room"},
)
room = Room.objects.get(owner=self.user)
event = room.events.first()
self.assertIsNotNone(event, "no ROOM_CREATED event recorded")
self.assertEqual(event.verb, GameEvent.ROOM_CREATED)
self.assertIsNone(event.actor, "welcome line must be system-authored")
def test_create_room_welcome_event_renders_welcome_prose(self):
self.client.post(
reverse("epic:create_room"),
data={"name": "Greenroom"},
)
room = Room.objects.get(owner=self.user)
event = room.events.first()
self.assertEqual(event.to_prose(), "Welcome to Greenroom!")
class MyGamesContextTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="gamer@example.com")
self.client.force_login(self.user)
def test_gameboard_context_includes_owned_rooms(self):
room = Room.objects.create(name="Durango", owner=self.user)
response = self.client.get("/gameboard/")
self.assertIn(room, response.context["my_games"])
def test_gameboard_context_includes_rooms_with_filled_slot(self):
other = User.objects.create(email="friend@example.com")
room = Room.objects.create(name="Their Room", owner=other)
slot = room.gate_slots.get(slot_number=2)
slot.gamer = self.user
slot.status = "FILLED"
slot.save()
response = self.client.get("/gameboard/")
self.assertIn(room, response.context["my_games"])
class GateStatusViewTest(TestCase):
def setUp(self):
self.owner = User.objects.create(email="founder@test.io")
self.client.force_login(self.owner)
self.room = Room.objects.create(name="Test Room", owner=self.owner)
def test_gate_status_returns_launch_btn_when_open(self):
self.room.gate_status = Room.OPEN
self.room.save()
response = self.client.get(reverse("epic:gate_status", kwargs={"room_id": self.room.id}))
self.assertEqual(response.status_code, 200)
self.assertContains(response, "launch-game-btn")
def test_gate_status_returns_partial_when_gathering(self):
response = self.client.get(
reverse("epic:gate_status", kwargs={"room_id": self.room.id})
)
self.assertEqual(response.status_code, 200)
self.assertContains(response, "gate-modal")
class DropTokenViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
def test_drop_token_reserves_lowest_empty_slot(self):
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
slot = self.room.gate_slots.get(slot_number=1)
self.assertEqual(slot.status, GateSlot.RESERVED)
self.assertEqual(slot.gamer, self.gamer)
def test_drop_token_skips_already_filled_slots(self):
other = User.objects.create(email="other@test.io")
slot1 = self.room.gate_slots.get(slot_number=1)
slot1.gamer = other
slot1.status = GateSlot.FILLED
slot1.save()
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
slot2 = self.room.gate_slots.get(slot_number=2)
self.assertEqual(slot2.status, GateSlot.RESERVED)
self.assertEqual(slot2.gamer, self.gamer)
def test_drop_token_blocked_when_another_slot_reserved(self):
other = User.objects.create(email="other@test.io")
slot1 = self.room.gate_slots.get(slot_number=1)
slot1.gamer = other
slot1.status = GateSlot.RESERVED
slot1.reserved_at = timezone.now()
slot1.save()
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
# Slot 2 should remain EMPTY — lock held by other user
slot2 = self.room.gate_slots.get(slot_number=2)
self.assertEqual(slot2.status, GateSlot.EMPTY)
def test_drop_token_blocked_when_user_already_has_filled_slot(self):
slot1 = self.room.gate_slots.get(slot_number=1)
slot1.gamer = self.gamer
slot1.status = GateSlot.FILLED
slot1.save()
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
slot2 = self.room.gate_slots.get(slot_number=2)
self.assertEqual(slot2.status, GateSlot.EMPTY)
def test_drop_token_sets_reserved_at(self):
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
slot = self.room.gate_slots.get(slot_number=1)
self.assertIsNotNone(slot.reserved_at)
def test_drop_token_redirects_to_gatekeeper(self):
response = self.client.post(
reverse("epic:drop_token", kwargs={"room_id": self.room.id})
)
self.assertRedirects(
response, reverse("epic:gatekeeper", args=[self.room.id])
)
def test_carte_drop_sets_current_room(self):
carte = Token.objects.create(user=self.gamer, token_type=Token.CARTE)
self.client.post(
reverse("epic:drop_token", kwargs={"room_id": self.room.id}),
data={"token_id": carte.pk},
)
carte.refresh_from_db()
self.assertEqual(carte.current_room, self.room)
def test_carte_drop_unequips_trinket(self):
carte = Token.objects.create(user=self.gamer, token_type=Token.CARTE)
self.gamer.equipped_trinket = carte
self.gamer.save(update_fields=["equipped_trinket"])
self.client.post(
reverse("epic:drop_token", kwargs={"room_id": self.room.id}),
data={"token_id": carte.pk},
)
self.gamer.refresh_from_db()
self.assertIsNone(self.gamer.equipped_trinket)
def test_carte_drop_rejected_when_already_in_different_room(self):
other_room = Room.objects.create(name="Other Room", owner=self.gamer)
carte = Token.objects.create(
user=self.gamer, token_type=Token.CARTE, current_room=other_room,
)
response = self.client.post(
reverse("epic:drop_token", kwargs={"room_id": self.room.id}),
data={"token_id": carte.pk},
)
self.assertEqual(response.status_code, 409)
carte.refresh_from_db()
self.assertEqual(carte.current_room, other_room) # unchanged
class ConfirmTokenViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.gamer
self.slot.status = GateSlot.RESERVED
self.slot.reserved_at = timezone.now()
self.slot.save()
Token.objects.create(user=self.gamer, token_type=Token.FREE)
def test_confirm_marks_slot_filled(self):
self.client.post(
reverse("epic:confirm_token", kwargs={"room_id": self.room.id})
)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.FILLED)
def test_confirm_sets_gate_open_when_all_slots_filled(self):
# Fill slots 26 via ORM
for i in range(2, 7):
other = User.objects.create(email=f"g{i}@test.io")
s = self.room.gate_slots.get(slot_number=i)
s.gamer = other
s.status = GateSlot.FILLED
s.save()
self.client.post(
reverse("epic:confirm_token", kwargs={"room_id": self.room.id})
)
self.room.refresh_from_db()
self.assertEqual(self.room.gate_status, Room.OPEN)
def test_confirm_redirects_to_gatekeeper(self):
response = self.client.post(
reverse("epic:confirm_token", kwargs={"room_id": self.room.id})
)
self.assertRedirects(
response, reverse("epic:gatekeeper", args=[self.room.id])
)
def test_confirm_does_nothing_without_reserved_slot(self):
self.slot.status = GateSlot.EMPTY
self.slot.gamer = None
self.slot.save()
self.client.post(
reverse("epic:confirm_token", kwargs={"room_id": self.room.id})
)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
class ReturnTokenViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.gamer
self.slot.status = GateSlot.RESERVED
self.slot.reserved_at = timezone.now()
self.slot.save()
def test_return_clears_reserved_slot(self):
self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
self.assertIsNone(self.slot.gamer)
self.assertIsNone(self.slot.reserved_at)
def test_return_after_confirm_clears_filled_slot(self):
self.slot.status = GateSlot.FILLED
self.slot.save()
self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
self.assertIsNone(self.slot.gamer)
def test_return_redirects_to_gatekeeper(self):
response = self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
self.assertRedirects(
response, reverse("epic:gatekeeper", args=[self.room.id])
)
def test_return_restores_coin_token(self):
coin = Token.objects.get(user=self.gamer, token_type=Token.COIN)
coin.current_room = self.room
coin.next_ready_at = timezone.now() + timedelta(days=7)
coin.save()
self.slot.status = GateSlot.FILLED
self.slot.debited_token_type = Token.COIN
self.slot.save()
self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
coin.refresh_from_db()
self.assertIsNone(coin.current_room)
self.assertIsNone(coin.next_ready_at)
def test_return_restores_free_token(self):
Token.objects.filter(user=self.gamer, token_type=Token.FREE).delete()
expires = timezone.now() + timedelta(days=3)
self.slot.status = GateSlot.FILLED
self.slot.debited_token_type = Token.FREE
self.slot.debited_token_expires_at = expires
self.slot.save()
self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
restored = Token.objects.filter(user=self.gamer, token_type=Token.FREE).first()
self.assertIsNotNone(restored)
self.assertEqual(restored.expires_at, expires)
def test_return_restores_tithe_token(self):
self.slot.status = GateSlot.FILLED
self.slot.debited_token_type = Token.TITHE
self.slot.save()
self.client.post(
reverse("epic:return_token", kwargs={"room_id": self.room.id})
)
self.assertTrue(
Token.objects.filter(user=self.gamer, token_type=Token.TITHE).exists()
)
class DropTokenAvailabilityViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
self.other_room = Room.objects.create(name="Other Room", owner=owner)
self.coin = Token.objects.get(user=self.gamer, token_type=Token.COIN)
def test_drop_reserves_slot_when_tokens_available(self):
self.client.post(reverse("epic:drop_token", kwargs={"room_id": self.room.id}))
slot = self.room.gate_slots.get(slot_number=1)
self.assertEqual(slot.status, GateSlot.RESERVED)
# token not debited yet — that happens at confirm
self.coin.refresh_from_db()
self.assertIsNone(self.coin.current_room)
def test_drop_returns_402_when_all_tokens_depleted(self):
self.coin.current_room = self.other_room
self.coin.save()
Token.objects.filter(user=self.gamer, token_type=Token.FREE).delete()
response = self.client.post(
reverse("epic:drop_token", kwargs={"room_id": self.room.id})
)
self.assertEqual(response.status_code, 402)
class ConfirmTokenPriorityViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
self.other_room = Room.objects.create(name="Other Room", owner=owner)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.gamer
self.slot.status = GateSlot.RESERVED
self.slot.reserved_at = timezone.now()
self.slot.save()
self.coin = Token.objects.get(user=self.gamer, token_type=Token.COIN)
def test_confirm_leases_coin_to_room(self):
self.client.post(reverse("epic:confirm_token", kwargs={"room_id": self.room.id}))
self.coin.refresh_from_db()
self.assertEqual(self.coin.current_room, self.room)
self.assertTrue(Token.objects.filter(pk=self.coin.pk).exists())
def test_confirm_uses_free_token_when_coin_in_use(self):
self.coin.current_room = self.other_room
self.coin.save()
self.client.post(reverse("epic:confirm_token", kwargs={"room_id": self.room.id}))
self.assertEqual(
Token.objects.filter(user=self.gamer, token_type=Token.FREE).count(), 0
)
self.coin.refresh_from_db()
self.assertEqual(self.coin.current_room, self.other_room)
def test_confirm_uses_tithe_when_free_tokens_exhausted(self):
self.coin.current_room = self.other_room
self.coin.save()
Token.objects.filter(user=self.gamer, token_type=Token.FREE).delete()
tithe = Token.objects.create(user=self.gamer, token_type=Token.TITHE)
self.client.post(reverse("epic:confirm_token", kwargs={"room_id": self.room.id}))
self.assertFalse(Token.objects.filter(pk=tithe.pk).exists())
def test_pass_not_consumed_and_coin_not_leased(self):
"""Equipped PASS picked over the still-equipped-by-default COIN —
confirm leaves PASS untouched + doesn't lease the COIN (PASS is
never-consumed; COIN stays free for a future room). The equip
slot is the precondition; DON-ing PASS swaps it in for the auto-
equipped COIN."""
self.gamer.is_staff = True
self.gamer.save()
pass_token = Token.objects.create(user=self.gamer, token_type=Token.PASS)
self.gamer.equipped_trinket = pass_token
self.gamer.save(update_fields=["equipped_trinket"])
self.client.post(reverse("epic:confirm_token", kwargs={"room_id": self.room.id}))
self.assertTrue(Token.objects.filter(pk=pass_token.pk).exists())
self.coin.refresh_from_db()
self.assertIsNone(self.coin.current_room)
class RoleSelectRenderingTest(TestCase):
def setUp(self):
self.founder = User.objects.create(email="founder@test.io")
self.room = Room.objects.create(name="Test Room", owner=self.founder)
self.gamers = [self.founder]
for i in range(2, 7):
self.gamers.append(User.objects.create(email=f"g{i}@test.io"))
for i, gamer in enumerate(self.gamers, start=1):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.save()
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
for i, gamer in enumerate(self.gamers, start=1):
TableSeat.objects.create(room=self.room, gamer=gamer, slot_number=i)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_room_view_includes_card_stack_when_role_select(self):
self.client.force_login(self.founder)
response = self.client.get(
self.url
)
self.assertContains(response, "card-stack")
def test_card_stack_eligible_for_slot1_gamer(self):
self.client.force_login(self.founder)
response = self.client.get(
self.url
)
self.assertContains(response, 'data-state="eligible"')
def test_card_stack_ineligible_for_slot2_gamer(self):
self.client.force_login(self.gamers[1])
response = self.client.get(
self.url
)
self.assertContains(response, 'data-state="ineligible"')
def test_card_stack_ineligible_shows_fa_ban(self):
self.client.force_login(self.gamers[1])
response = self.client.get(
self.url
)
self.assertContains(response, "fa-ban")
def test_card_stack_eligible_omits_fa_ban(self):
self.client.force_login(self.founder)
response = self.client.get(
self.url
)
# Seat ban icons carry "position-status-icon"; card-stack ban does not.
# Assert the bare "fa-solid fa-ban" (card-stack form) is absent.
self.assertNotContains(response, 'class="fa-solid fa-ban"')
def test_gatekeeper_overlay_absent_when_role_select(self):
self.client.force_login(self.founder)
response = self.client.get(
self.url
)
self.assertNotContains(response, "gate-overlay")
def test_tray_wrap_has_role_select_phase_class(self):
# Tray handle hidden until gamer confirms a role pick
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertContains(response, 'id="id_tray_wrap" class="role-select-phase"')
def test_tray_absent_during_gatekeeper_phase(self):
# Tray must not render before the gamer occupies a seat
room = Room.objects.create(name="Gate Room", owner=self.founder)
self.client.force_login(self.founder)
response = self.client.get(
reverse("epic:gatekeeper", kwargs={"room_id": room.id})
)
self.assertNotContains(response, 'id="id_tray_wrap"')
def test_six_table_seats_rendered(self):
self.client.force_login(self.founder)
response = self.client.get(
self.url
)
self.assertContains(response, "table-seat", count=6)
def test_table_seats_never_active_on_load(self):
# Seat glow is JS-only (during tray animation); never server-rendered
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertNotContains(response, 'class="table-seat active"')
def test_assigned_seat_renders_role_confirmed_class(self):
# A seat with a role already picked must load as role-confirmed (opaque chair)
self.gamers[0].refresh_from_db()
seat = self.room.table_seats.get(slot_number=1)
seat.role = "PC"
seat.save()
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertContains(response, 'table-seat role-confirmed')
def test_unassigned_seat_lacks_role_confirmed_class(self):
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertNotContains(response, 'table-seat role-confirmed')
def test_assigned_slot_circle_renders_role_assigned_class(self):
# Slot 1 circle hidden because 1 role was assigned (count-based, not role-label-based)
seat = self.room.table_seats.get(slot_number=1)
seat.role = "PC"
seat.save()
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertContains(response, 'gate-slot filled role-assigned')
def test_slot_circle_hides_by_count_not_role_label(self):
# Gamer in slot 1 picks NC (not PC) — slot 1 circle must still hide, not slot 2's
seat = self.room.table_seats.get(slot_number=1)
seat.role = "NC"
seat.save()
self.client.force_login(self.founder)
response = self.client.get(self.url)
content = response.content.decode()
import re
# Template renders class before data-slot; capture both orderings
circles = re.findall(r'class="([^"]*gate-slot[^"]*)"[^>]*data-slot="(\d)"', content)
slot1_classes = next((cls for cls, slot in circles if slot == "1"), "")
slot2_classes = next((cls for cls, slot in circles if slot == "2"), "")
self.assertIn("role-assigned", slot1_classes)
self.assertNotIn("role-assigned", slot2_classes)
def test_unassigned_slot_circle_lacks_role_assigned_class(self):
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertNotContains(response, 'role-assigned')
def test_position_strip_rendered_during_role_select(self):
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertContains(response, "position-strip")
def test_position_strip_has_six_gate_slots(self):
self.client.force_login(self.founder)
response = self.client.get(self.url)
self.assertContains(response, "gate-slot", count=6)
def test_card_stack_has_data_user_slots_for_eligible_gamer(self):
self.client.force_login(self.founder) # founder is slot 1 only
response = self.client.get(
self.url
)
self.assertContains(response, 'data-user-slots="1"')
def test_card_stack_has_data_user_slots_for_ineligible_gamer(self):
self.client.force_login(self.gamers[1]) # slot 2 gamer
response = self.client.get(
self.url
)
self.assertContains(response, 'data-user-slots="2"')
def test_assigned_seat_renders_check_icon(self):
seat = self.room.table_seats.get(slot_number=1)
seat.role = "PC"
seat.save()
self.client.force_login(self.founder)
response = self.client.get(self.url)
content = response.content.decode()
# The PC seat should have fa-circle-check, not fa-ban
pc_seat_start = content.index('data-role="PC"')
pc_seat_chunk = content[pc_seat_start:pc_seat_start + 300]
self.assertIn("fa-circle-check", pc_seat_chunk)
self.assertNotIn("fa-ban", pc_seat_chunk)
def test_unassigned_seat_renders_ban_icon(self):
# slot 2's role is still null
self.client.force_login(self.founder)
response = self.client.get(self.url)
content = response.content.decode()
nc_seat_start = content.index('data-role="NC"')
nc_seat_chunk = content[nc_seat_start:nc_seat_start + 300]
self.assertIn("fa-ban", nc_seat_chunk)
self.assertNotIn("fa-circle-check", nc_seat_chunk)
def _circle_start(content, slot_number):
"""Index of the gate-slot circle's opening `<div` for the given slot.
Scopes to `.gate-slot` (room.html also renders `.table-seat` data-slot=N
elements first, so a bare data-slot search would hit the seat, not the
circle)."""
needle = f'data-slot="{slot_number}"'
pos = 0
while True:
idx = content.index('<div class="gate-slot', pos)
end = content.index(">", idx)
if needle in content[idx:end]:
return idx
pos = end
def _circle_tag(content, slot_number):
"""Return the opening `<div ...>` tag of the gate-slot circle for the
given slot — class + every data-tt-* attr live on this one tag."""
idx = _circle_start(content, slot_number)
return content[idx:content.index(">", idx)]
class PositionTooltipRenderTest(TestCase):
"""Render-level coverage for the rich position-circle tooltip payload
(sprint 2026-06-02) — the fast IT counterpart to the Selenium
PositionTooltipTest in functional_tests/test_game_room_position_tooltips.py.
Exercised on the GATE VIEW (room_gate), which rendered no circles before
this sprint."""
def setUp(self):
from apps.epic.models import DeckVariant
self.viewer = User.objects.create(email="disco@test.io", username="disco")
self.deck, _ = DeckVariant.objects.get_or_create(
slug="earthman",
defaults={"name": "Earthman", "card_count": 106, "is_default": True},
)
self.viewer.equipped_deck = self.deck
self.viewer.save(update_fields=["equipped_deck"])
self.room = Room.objects.create(name="Whataburgher", owner=self.viewer)
self.gamers = [self.viewer]
for i in range(2, 7):
self.gamers.append(
User.objects.create(email=f"g{i}@test.io", username=f"g{i}")
)
for i, gamer in enumerate(self.gamers, start=1):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.filled_at = timezone.now()
slot.debited_token_type = Token.FREE
slot.save()
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
self.gate_url = reverse("epic:room_gate", kwargs={"room_id": self.room.id})
self.client.force_login(self.viewer)
def _gate_content(self):
return self.client.get(self.gate_url).content.decode()
def test_gate_view_renders_six_position_circles(self):
content = self._gate_content()
self.assertContains(self.client.get(self.gate_url), "position-strip")
self.assertEqual(content.count('class="gate-slot'), 6)
def test_own_slot_is_me_current_others_are_gamer(self):
content = self._gate_content()
self.assertIn("tt-pos-me-current", _circle_tag(content, 1))
slot2 = _circle_tag(content, 2)
self.assertIn("tt-pos-gamer", slot2)
self.assertNotIn("tt-pos-me", slot2)
def test_other_gamer_handle_in_title_not_email(self):
slot2 = _circle_tag(self._gate_content(), 2)
self.assertIn('data-tt-title="@g2"', slot2)
# No email field in the tooltip payload (user-spec).
self.assertNotIn("data-tt-email", slot2)
# Title carries the article — "the Earthman", not bare "Earthman".
self.assertIn('data-tt-description="the Earthman"', slot2)
def test_bud_occupant_carries_bud_class_and_shoptalk(self):
from apps.billboard.models import BudshipNote
amigo = self.gamers[1]
self.viewer.buds.add(amigo)
BudshipNote.objects.create(user=self.viewer, bud=amigo, shoptalk="met at the deli")
slot2 = _circle_tag(self._gate_content(), 2)
self.assertIn("tt-pos-bud", slot2)
self.assertIn('data-tt-shoptalk="met at the deli"', slot2)
def test_deposit_count_and_expiry_present(self):
slot2 = _circle_tag(self._gate_content(), 2)
self.assertIn('data-tt-tokens="1"', slot2)
# FREE-funded slot → the deposited-token list reads "Free Token".
self.assertIn('data-tt-token-types="Free Token"', slot2)
# Lowercase "expires <when>" (relative timescale), not an ISO/locale date.
self.assertIn('data-tt-expiry="expires ', slot2)
def test_seat_significator_rank_rides_the_circle(self):
sig = TarotCard.objects.create(
deck_variant=self.deck, slug="queen-of-brands-em",
arcana="MIDDLE", suit="BRANDS", number=13, name="Queen of Brands",
)
TableSeat.objects.create(
room=self.room, gamer=self.gamers[1], slot_number=2, significator=sig,
)
slot2 = _circle_tag(self._gate_content(), 2)
self.assertIn(f'data-tt-sign-rank="{sig.corner_rank}"', slot2)
def test_no_occupant_email_anywhere_in_page_source(self):
# The position circles render @handle (at_handle), never the raw login
# email — not in the tooltip payload AND not in the hidden .slot-gamer
# span. Assert on the FULL response, not just the circle's opening tag.
content = self._gate_content()
self.assertNotIn(self.gamers[1].email, content) # g2@test.io
self.assertNotIn(self.viewer.email, content) # disco@test.io
class PositionTooltipCarteRenderTest(TestCase):
"""CARTE-solo render contract: a single gamer owns all six slots — their
non-current circles read tt-pos-me-also + carry a ?seat=N switch href, and
the deposited count reflects the CARTE token's slots_claimed."""
def setUp(self):
from apps.epic.models import DeckVariant
self.viewer = User.objects.create(email="disco@test.io", username="disco")
self.deck, _ = DeckVariant.objects.get_or_create(
slug="earthman",
defaults={"name": "Earthman", "card_count": 106, "is_default": True},
)
self.viewer.equipped_deck = self.deck
self.viewer.save(update_fields=["equipped_deck"])
self.room = Room.objects.create(name="Carte Room", owner=self.viewer)
self.room.gate_slots.update(
gamer=self.viewer, status=GateSlot.FILLED,
filled_at=timezone.now(), debited_token_type=Token.CARTE,
)
Token.objects.create(
user=self.viewer, token_type=Token.CARTE,
current_room=self.room, slots_claimed=6,
)
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
self.client.force_login(self.viewer)
self.room_url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_each_carte_slot_costs_one_token(self):
# A CARTE covers each seat at cost 1 — the per-slot expenditure count
# is 1 (GateSlot.token_cost), NOT the CARTE slots_claimed high-water
# mark. The deposited-token list reads "Carte Blanche".
content = self.client.get(self.room_url).content.decode()
# One CARTE covers ALL six seats — every slot reads cost 1 + "Carte
# Blanche" (token combinations per slot aren't possible: a re-deposit
# ejects the slot's token), so there is no mixed CARTE+FREE spread.
for n in (1, 4, 6):
tag = _circle_tag(content, n)
self.assertIn('data-tt-tokens="1"', tag)
self.assertIn('data-tt-token-types="Carte Blanche"', tag)
def test_own_other_seat_is_me_also_with_switch_href(self):
content = self.client.get(self.room_url).content.decode()
slot4 = _circle_tag(content, 4)
self.assertIn("tt-pos-me-also", slot4)
# The switch anchor lands just after the opening tag.
idx = _circle_start(content, 4)
chunk = content[idx:idx + 800]
self.assertIn("seat=4", chunk)
def test_seat_param_previews_that_seat_in_card_stack(self):
# Un-assigned owned seats → ?seat=4 previews seat 4, which is not the
# table's current turn (slot 1) → card-stack ineligible (.fa-ban).
for n in range(1, 7):
TableSeat.objects.create(room=self.room, gamer=self.viewer, slot_number=n)
content = self.client.get(self.room_url + "?seat=4").content.decode()
self.assertIn('data-active-slot="4"', content)
self.assertIn('data-state="ineligible"', content)
def test_no_seat_param_keeps_canonical_active_slot(self):
for n in range(1, 7):
TableSeat.objects.create(room=self.room, gamer=self.viewer, slot_number=n)
content = self.client.get(self.room_url).content.decode()
# Default: the table's turn is slot 1, and the viewer owns it → eligible.
self.assertIn('data-active-slot="1"', content)
self.assertIn('data-state="eligible"', content)
def test_room_gate_renders_circles_but_no_carte_action_forms(self):
# The renewal gate-view shows the circles (for their hover tooltips)
# but is NOT a gather surface — no CARTE drop/release forms.
gate_url = reverse("epic:room_gate", kwargs={"room_id": self.room.id})
content = self.client.get(gate_url).content.decode()
self.assertIn("position-strip", content)
self.assertEqual(content.count('class="gate-slot'), 6)
self.assertNotIn("slot-release-btn", content)
self.assertNotIn("drop-token-btn", content)
class PickRolesViewTest(TestCase):
def setUp(self):
self.founder = User.objects.create(email="founder@test.io")
self.client.force_login(self.founder)
self.room = Room.objects.create(name="Test Room", owner=self.founder)
for i in range(1, 7):
gamer = self.founder if i == 1 else User.objects.create(email=f"g{i}@test.io")
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.save()
self.room.gate_status = Room.OPEN
self.room.save()
def test_pick_roles_transitions_room_to_role_select(self):
self.client.post(reverse("epic:pick_roles", kwargs={"room_id": self.room.id}))
self.room.refresh_from_db()
self.assertEqual(self.room.table_status, Room.ROLE_SELECT)
def test_pick_roles_creates_one_table_seat_per_filled_slot(self):
self.client.post(reverse("epic:pick_roles", kwargs={"room_id": self.room.id}))
self.assertEqual(TableSeat.objects.filter(room=self.room).count(), 6)
def test_pick_roles_table_seats_carry_gamer_and_slot_number(self):
self.client.post(reverse("epic:pick_roles", kwargs={"room_id": self.room.id}))
seat = TableSeat.objects.get(room=self.room, slot_number=1)
self.assertEqual(seat.gamer, self.founder)
def test_only_open_room_can_start_role_select(self):
self.room.gate_status = Room.GATHERING
self.room.save()
self.client.post(reverse("epic:pick_roles", kwargs={"room_id": self.room.id}))
self.room.refresh_from_db()
self.assertIsNone(self.room.table_status)
def test_pick_roles_requires_login(self):
self.client.logout()
response = self.client.post(
reverse("epic:pick_roles", kwargs={"room_id": self.room.id})
)
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_pick_roles_redirects_to_room(self):
response = self.client.post(
reverse("epic:pick_roles", kwargs={"room_id": self.room.id})
)
self.assertRedirects(
response, reverse("epic:room", args=[self.room.id])
)
def test_pick_roles_notifies_channel_layer(self):
with patch("apps.epic.views._notify_role_select_start") as mock_notify:
self.client.post(reverse("epic:pick_roles", kwargs={"room_id": self.room.id}))
mock_notify.assert_called_once_with(self.room.id)
def test_pick_roles_idempotent_no_duplicate_seats(self):
url = reverse("epic:pick_roles", kwargs={"room_id": self.room.id})
self.client.post(url)
self.client.post(url) # second call must be a no-op
self.assertEqual(TableSeat.objects.filter(room=self.room).count(), 6)
class SelectRoleViewTest(TestCase):
def setUp(self):
self.founder = User.objects.create(email="founder@test.io")
self.room = Room.objects.create(name="Test Room", owner=self.founder)
self.gamers = [self.founder]
for i in range(2, 7):
self.gamers.append(User.objects.create(email=f"g{i}@test.io"))
for i, gamer in enumerate(self.gamers, start=1):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.save()
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
for i, gamer in enumerate(self.gamers, start=1):
TableSeat.objects.create(room=self.room, gamer=gamer, slot_number=i)
self.client.force_login(self.founder)
def test_select_role_records_choice(self):
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
seat = TableSeat.objects.get(room=self.room, slot_number=1)
self.assertEqual(seat.role, "PC")
def test_select_role_wrong_turn_makes_no_change(self):
self.client.force_login(self.gamers[1]) # slot 2 — not their turn
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BC"},
)
seat = TableSeat.objects.get(room=self.room, slot_number=2)
self.assertIsNone(seat.role)
def test_turn_advances_after_selection(self):
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
next_active = TableSeat.objects.filter(
room=self.room, role__isnull=True
).order_by("slot_number").first()
self.assertEqual(next_active.slot_number, 2)
def test_all_selected_stays_role_select_status(self):
roles = ["PC", "BC", "SC", "AC", "NC"]
for i, role in enumerate(roles):
seat = TableSeat.objects.get(room=self.room, slot_number=i + 1)
seat.role = role
seat.save()
self.client.force_login(self.gamers[5]) # slot 6 — last
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "EC"},
)
self.room.refresh_from_db()
self.assertEqual(self.room.table_status, Room.ROLE_SELECT)
def test_select_role_notifies_turn_changed(self):
with patch("apps.epic.views._notify_turn_changed") as mock_notify:
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
mock_notify.assert_called_once_with(self.room.id)
def test_select_role_notifies_all_roles_filled_when_last(self):
roles = ["PC", "BC", "SC", "AC", "NC"]
for i, role in enumerate(roles):
seat = TableSeat.objects.get(room=self.room, slot_number=i + 1)
seat.role = role
seat.save()
self.client.force_login(self.gamers[5])
with patch("apps.epic.views._notify_all_roles_filled") as mock_notify:
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "EC"},
)
mock_notify.assert_called_once_with(self.room.id)
def test_select_role_assigns_equipped_deck_to_seat(self):
earthman = DeckVariant.objects.get(slug="earthman")
self.founder.equipped_deck = earthman
self.founder.save(update_fields=["equipped_deck"])
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
seat = TableSeat.objects.get(room=self.room, slot_number=1)
self.assertEqual(seat.deck_variant, earthman)
def test_select_role_no_deck_leaves_deck_variant_null(self):
self.founder.equipped_deck = None
self.founder.save(update_fields=["equipped_deck"])
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
seat = TableSeat.objects.get(room=self.room, slot_number=1)
self.assertIsNone(seat.deck_variant)
def test_select_role_unequips_deck_from_user(self):
earthman = DeckVariant.objects.get(slug="earthman")
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
self.founder.refresh_from_db()
self.assertIsNone(self.founder.equipped_deck)
def test_select_role_requires_login(self):
self.client.logout()
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_select_role_returns_ok(self):
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
self.assertEqual(response.status_code, 200)
def test_select_role_returns_409_for_duplicate_role(self):
TableSeat.objects.filter(room=self.room, slot_number=2).update(role="BC")
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BC"},
)
self.assertEqual(response.status_code, 409)
def test_select_role_redirects_when_not_role_select_phase(self):
self.room.table_status = None
self.room.save()
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
self.assertRedirects(
response, reverse("epic:gatekeeper", args=[self.room.id])
)
def test_select_role_redirects_for_invalid_role_code(self):
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BOGUS"},
)
self.assertRedirects(
response, reverse("epic:room", args=[self.room.id])
)
def test_same_gamer_cannot_double_pick_sequentially(self):
"""A second POST from the active gamer — after their role has been
saved — must redirect rather than assign a second role."""
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "PC"},
)
response = self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BC"},
)
self.assertRedirects(
response, reverse("epic:room", args=[self.room.id])
)
self.assertEqual(
TableSeat.objects.filter(room=self.room, role__isnull=False).count(), 1
)
class SelectRoleMultiSeatTest(TestCase):
"""Carte Blanche multi-seat: second role reuses the deck from the first seat."""
def setUp(self):
self.founder = User.objects.create(email="founder@test.io")
self.client.force_login(self.founder)
self.room = Room.objects.create(name="Test Room", owner=self.founder)
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
self.earthman = DeckVariant.objects.get(slug="earthman")
def test_second_role_inherits_deck_from_first_seat_in_room(self):
# Founder's first seat: PC already taken with deck assigned
TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=1,
role="PC", deck_variant=self.earthman,
)
# Deck unequipped after first role
self.founder.equipped_deck = None
self.founder.save(update_fields=["equipped_deck"])
# Founder's second seat (Carte Blanche): no role yet
second_seat = TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=2,
)
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BC"},
)
second_seat.refresh_from_db()
self.assertEqual(second_seat.deck_variant, self.earthman)
def test_second_role_does_not_unequip_again(self):
"""No-op unequip when deck was already cleared by the first role."""
TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=1,
role="PC", deck_variant=self.earthman,
)
self.founder.equipped_deck = None
self.founder.save(update_fields=["equipped_deck"])
TableSeat.objects.create(room=self.room, gamer=self.founder, slot_number=2)
self.client.post(
reverse("epic:select_role", kwargs={"room_id": self.room.id}),
data={"role": "BC"},
)
self.founder.refresh_from_db()
self.assertIsNone(self.founder.equipped_deck) # still None, not broken
# ── Return-trip Role-Select bug — CARTE user navigates away + back ───── #
#
# After the FIRST role pick, select_role() clears user.equipped_deck
# ("deck committed to room"). The room view's role-select context then
# passes `equipped_deck_id = user.equipped_deck_id` to the template,
# which sets `data-equipped-deck=""` and JS guards against role-select
# with "Equip card deck before Role select." → blocks a CARTE user from
# continuing to pick roles for their remaining seats. Fix: the context's
# `equipped_deck_id` should also accept the deck_variant of any seat the
# user already holds in this room (the deck IS in play — it's just
# committed to existing seats, not to user.equipped_deck).
def test_role_select_context_recovers_deck_id_from_existing_seat(self):
"""User cleared their equipped_deck after first role pick, but they
still have a seat in this room w. deck_variant set → context should
report that deck's id so the guard doesn't fire on return."""
TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=1,
role="PC", deck_variant=self.earthman,
)
# Slot 2 still needs role (CARTE user's next seat)
TableSeat.objects.create(room=self.room, gamer=self.founder, slot_number=2)
self.founder.equipped_deck = None
self.founder.save(update_fields=["equipped_deck"])
# Simulate filled gate slots so the room renders in role-select state
for i in (1, 2):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = self.founder
slot.status = GateSlot.FILLED
slot.save()
response = self.client.get(
reverse("epic:room", kwargs={"room_id": self.room.id})
)
self.assertEqual(
response.context["equipped_deck_id"], self.earthman.id,
"Returning CARTE user should see their in-play deck reflected in "
"the role-select context so the JS guard doesn't fire.",
)
def test_role_select_context_renders_data_equipped_deck_non_empty(self):
"""Template-level check — the rendered `data-equipped-deck` attribute
should be non-empty so the JS guard at role-select.js:165 lets the
fan open."""
TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=1,
role="PC", deck_variant=self.earthman,
)
TableSeat.objects.create(room=self.room, gamer=self.founder, slot_number=2)
self.founder.equipped_deck = None
self.founder.save(update_fields=["equipped_deck"])
for i in (1, 2):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = self.founder
slot.status = GateSlot.FILLED
slot.save()
response = self.client.get(
reverse("epic:room", kwargs={"room_id": self.room.id})
)
# Should NOT contain the empty-string version that triggers the guard.
self.assertNotContains(response, 'data-equipped-deck=""')
self.assertContains(response, f'data-equipped-deck="{self.earthman.id}"')
class RoomViewAllRolesFilledTest(TestCase):
"""Room view in ROLE_SELECT with all seats assigned shows SCAN SIGS button."""
def setUp(self):
import lxml.html
self.lxml = lxml.html
self.owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=self.owner)
self.room.table_status = Room.ROLE_SELECT
self.room.save()
all_roles = ["PC", "BC", "SC", "AC", "NC", "EC"]
for i, role in enumerate(all_roles, start=1):
user = User.objects.create(email=f"p{i}@test.io")
TableSeat.objects.create(room=self.room, gamer=user, slot_number=i, role=role)
self.client.force_login(self.owner)
def test_pick_sigs_btn_present_when_all_roles_filled(self):
response = self.client.get(reverse("epic:room", kwargs={"room_id": self.room.id}))
parsed = self.lxml.fromstring(response.content)
[_] = parsed.cssselect("#id_pick_sigs_btn")
self.assertEqual(parsed.cssselect(".card-stack"), [])
def test_pick_sigs_btn_hidden_during_role_select(self):
# Clear one role — still mid-pick, wrap must be hidden
TableSeat.objects.filter(room=self.room, slot_number=6).update(role=None)
response = self.client.get(reverse("epic:room", kwargs={"room_id": self.room.id}))
parsed = self.lxml.fromstring(response.content)
[wrap] = parsed.cssselect("#id_pick_sigs_wrap")
self.assertIn("display:none", wrap.get("style", "").replace(" ", ""))
class PickSigsViewTest(TestCase):
def setUp(self):
self.owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=self.owner)
self.room.table_status = Room.ROLE_SELECT
self.room.save()
all_roles = ["PC", "BC", "SC", "AC", "NC", "EC"]
for i, role in enumerate(all_roles, start=1):
user = User.objects.create(email=f"p{i}@test.io")
TableSeat.objects.create(room=self.room, gamer=user, slot_number=i, role=role)
self.client.force_login(self.owner)
self.url = reverse("epic:pick_sigs", kwargs={"room_id": self.room.id})
def test_pick_sigs_requires_login(self):
self.client.logout()
response = self.client.post(self.url)
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_pick_sigs_transitions_to_sig_select(self):
self.client.post(self.url)
self.room.refresh_from_db()
self.assertEqual(self.room.table_status, Room.SIG_SELECT)
def test_pick_sigs_redirects_to_room(self):
response = self.client.post(self.url)
self.assertRedirects(response, reverse("epic:room", args=[self.room.id]))
def test_pick_sigs_is_noop_if_not_role_select(self):
self.room.table_status = Room.SIG_SELECT
self.room.save()
self.client.post(self.url)
self.room.refresh_from_db()
self.assertEqual(self.room.table_status, Room.SIG_SELECT)
def test_pick_sigs_notifies_sig_select_started(self):
with patch("apps.epic.views._notify_sig_select_started") as mock_notify:
self.client.post(self.url)
mock_notify.assert_called_once_with(self.room.id)
class RoomActionsViewTest(TestCase):
def setUp(self):
self.owner = User.objects.create(email="owner@test.io")
self.gamer = User.objects.create(email="gamer@test.io")
self.room = Room.objects.create(name="Test Room", owner=self.owner)
self.slot = self.room.gate_slots.get(slot_number=2)
self.slot.gamer = self.gamer
self.slot.status = "FILLED"
self.slot.save()
RoomInvite.objects.create(
room=self.room, inviter=self.owner,
invitee_email=self.gamer.email
)
def test_owner_delete_removes_room(self):
self.client.force_login(self.owner)
self.client.post(reverse("epic:delete_room", kwargs={"room_id": self.room.id}))
self.assertFalse(Room.objects.filter(pk=self.room.pk).exists())
def test_non_owner_delete_does_not_remove_room(self):
self.client.force_login(self.gamer)
self.client.post(reverse("epic:delete_room", kwargs={"room_id": self.room.id}))
self.assertTrue(Room.objects.filter(pk=self.room.pk).exists())
def test_delete_redirects_to_gameboard(self):
self.client.force_login(self.owner)
response = self.client.post(
reverse("epic:delete_room", kwargs={"room_id": self.room.id})
)
self.assertRedirects(response, "/gameboard/")
def test_abandon_clears_slot(self):
self.client.force_login(self.gamer)
self.client.post(reverse("epic:abandon_room", kwargs={"room_id": self.room.id}))
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, "EMPTY")
self.assertIsNone(self.slot.gamer)
def test_abandon_deletes_pending_invite(self):
self.client.force_login(self.gamer)
self.client.post(reverse("epic:abandon_room", kwargs={"room_id": self.room.id}))
self.assertFalse(
RoomInvite.objects.filter(
room=self.room, invitee_email=self.gamer.email
).exists()
)
def test_abandon_redirects_to_gameboard(self):
self.client.force_login(self.gamer)
response = self.client.post(
reverse("epic:abandon_room", kwargs={"room_id": self.room.id})
)
self.assertRedirects(response, "/gameboard/")
class ReleaseSlotViewTest(TestCase):
def setUp(self):
self.gamer = User.objects.create(email="gamer@test.io")
self.client.force_login(self.gamer)
owner = User.objects.create(email="owner@test.io")
self.room = Room.objects.create(name="Test Room", owner=owner)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.gamer
self.slot.status = GateSlot.FILLED
self.slot.debited_token_type = Token.CARTE
self.slot.save()
def test_release_slot_downgrades_open_room_to_gathering(self):
self.room.gate_status = Room.OPEN
self.room.save()
self.client.post(
reverse("epic:release_slot", kwargs={"room_id": self.room.id}),
data={"slot_number": self.slot.slot_number},
)
self.room.refresh_from_db()
self.assertEqual(self.room.gate_status, Room.GATHERING)
# ── Significator Selection ────────────────────────────────────────────────────
SIG_SEAT_ORDER = ["PC", "NC", "EC", "SC", "AC", "BC"]
def _full_sig_setUp(test_case, role_order=None):
"""Populate test_case with a SIG_SELECT room; return (room, gamers, earthman, card_in_deck).
Uses get_or_create for DeckVariant — migration data persists in TestCase."""
if role_order is None:
role_order = SIG_SEAT_ORDER[:]
earthman, _ = DeckVariant.objects.get_or_create(
slug="earthman",
defaults={"name": "Earthman Deck", "card_count": 108, "is_default": True},
)
founder = User.objects.create(email="founder@test.io")
gamers = [founder]
for i in range(2, 7):
gamers.append(User.objects.create(email=f"g{i}@test.io"))
for gamer in gamers:
gamer.equipped_deck = earthman
gamer.save(update_fields=["equipped_deck"])
room = Room.objects.create(name="Sig Room", owner=founder)
for i, (gamer, role) in enumerate(zip(gamers, role_order), start=1):
slot = room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.save()
TableSeat.objects.create(
room=room, gamer=gamer, slot_number=i, role=role,
role_revealed=True, deck_variant=earthman,
)
room.gate_status = Room.OPEN
room.table_status = Room.SIG_SELECT
room.save()
card_in_deck = TarotCard.objects.get(
deck_variant=earthman, arcana="MIDDLE", suit="BRANDS", number=11
)
test_case.client.force_login(founder)
return room, gamers, earthman, card_in_deck
class SigReserveSoloCarteTest(TestCase):
"""CARTE solo: one gamer owns every seat, so each polarity group is
solo-owned — reserving commits the sig to the active (?seat) seat
immediately (no 3-gamer countdown can ever complete), and the sig is
per-seat, not per-gamer. The fast IT counterpart to
CarteSeatSwitchTest.test_carte_saves_a_significator_per_seat."""
def setUp(self):
from apps.epic.models import DeckVariant
self.earthman, _ = DeckVariant.objects.get_or_create(
slug="earthman",
defaults={"name": "Earthman Deck", "card_count": 108, "is_default": True},
)
self.viewer = User.objects.create(email="disco@test.io", username="disco")
self.viewer.equipped_deck = self.earthman
self.viewer.save(update_fields=["equipped_deck"])
self.room = Room.objects.create(name="Carte Sig", owner=self.viewer)
for i, role in enumerate(SIG_SEAT_ORDER, start=1):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = self.viewer
slot.status = GateSlot.FILLED
slot.save()
TableSeat.objects.create(
room=self.room, gamer=self.viewer, slot_number=i, role=role,
role_revealed=True, deck_variant=self.earthman,
)
self.room.gate_status = Room.OPEN
self.room.table_status = Room.SIG_SELECT
self.room.save()
self.client.force_login(self.viewer)
self.url = reverse("epic:sig_reserve", kwargs={"room_id": self.room.id})
# Two distinct levity court cards (PC + NC are both levity).
self.card_a = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=11)
self.card_b = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=12)
def test_solo_reserve_commits_significator_to_active_seat(self):
self.client.post(self.url + "?seat=1",
data={"card_id": self.card_a.id, "action": "reserve"})
seat1 = TableSeat.objects.get(room=self.room, slot_number=1)
self.assertEqual(seat1.significator_id, self.card_a.id)
def test_solo_sig_is_per_seat_not_per_gamer(self):
# Commit seat 1, NVM (frees the per-gamer row; the committed sig
# persists through release), then commit a different card on seat 2.
self.client.post(self.url + "?seat=1",
data={"card_id": self.card_a.id, "action": "reserve"})
self.client.post(self.url,
data={"card_id": self.card_a.id, "action": "release"})
self.client.post(self.url + "?seat=2",
data={"card_id": self.card_b.id, "action": "reserve"})
seat1 = TableSeat.objects.get(room=self.room, slot_number=1)
seat2 = TableSeat.objects.get(room=self.room, slot_number=2)
self.assertEqual(seat1.significator_id, self.card_a.id)
self.assertEqual(seat2.significator_id, self.card_b.id)
self.assertNotEqual(seat1.significator_id, seat2.significator_id)
class SigSelectRenderingTest(TestCase):
"""Gate view at SIG_SELECT renders the Significator deck."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_sig_deck_element_present(self):
response = self.client.get(self.url)
self.assertContains(response, "id_sig_deck")
def test_sig_deck_contains_16_sig_cards_by_default(self):
"""Without Note unlocks the deck shows only 16 court cards (no Nomad/Schizo)."""
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 16)
def test_nomad_note_adds_nomad_to_sig_deck(self):
Note.objects.create(user=self.gamers[0], slug="nomad", earned_at=timezone.now())
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 17)
def test_schizo_note_adds_schizo_to_sig_deck(self):
Note.objects.create(user=self.gamers[0], slug="schizo", earned_at=timezone.now())
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 17)
def test_super_nomad_note_also_unlocks_nomad(self):
Note.objects.create(user=self.gamers[0], slug="super-nomad", earned_at=timezone.now())
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 17)
def test_super_schizo_note_also_unlocks_schizo(self):
Note.objects.create(user=self.gamers[0], slug="super-schizo", earned_at=timezone.now())
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 17)
def test_both_notes_gives_18_sig_cards(self):
Note.objects.create(user=self.gamers[0], slug="nomad", earned_at=timezone.now())
Note.objects.create(user=self.gamers[0], slug="schizo", earned_at=timezone.now())
response = self.client.get(self.url)
self.assertEqual(response.content.decode().count('data-card-id='), 18)
def test_seats_rendered_in_pc_nc_ec_sc_ac_bc_order(self):
response = self.client.get(self.url)
content = response.content.decode()
positions = {role: content.find(f'data-role="{role}"') for role in SIG_SEAT_ORDER}
# Every role must appear
self.assertTrue(all(pos != -1 for pos in positions.values()))
# Rendered in canonical sequence
ordered = sorted(SIG_SEAT_ORDER, key=lambda r: positions[r])
self.assertEqual(ordered, SIG_SEAT_ORDER)
def test_sig_deck_not_present_during_role_select(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self.client.get(self.url)
self.assertNotContains(response, "id_sig_deck")
def test_sig_cards_render_keyword_data_attributes(self):
response = self.client.get(self.url)
content = response.content.decode()
self.assertIn("data-keywords-upright=", content)
self.assertIn("data-keywords-reversed=", content)
def test_sig_stat_block_structure_rendered(self):
response = self.client.get(self.url)
self.assertContains(response, "sig-stat-block")
self.assertContains(response, "spin-btn")
self.assertContains(response, "stat-face--upright")
self.assertContains(response, "stat-face--reversed")
def test_sig_cards_render_energies_operations_data_attributes(self):
response = self.client.get(self.url)
self.assertContains(response, "data-energies=")
self.assertContains(response, "data-operations=")
def test_sig_info_panel_structure_rendered(self):
response = self.client.get(self.url)
self.assertContains(response, "sig-info")
self.assertContains(response, "fyi-btn")
self.assertContains(response, "sig-info-effect")
self.assertContains(response, "sig-info-index")
self.assertContains(response, "fyi-prev")
self.assertContains(response, "fyi-next")
class SelectSigCardViewTest(TestCase):
"""select_sig view — records choice, enforces turn order, rejects bad input."""
def setUp(self):
self.room, self.gamers, self.earthman, self.card = _full_sig_setUp(self)
# Founder is slot 1, role=PC — active first in canonical order
self.url = reverse("epic:select_sig", kwargs={"room_id": self.room.id})
def _post(self, card_id=None, client=None):
c = client or self.client
return c.post(self.url, data={"card_id": card_id or self.card.id})
def test_select_sig_records_choice_on_active_seat(self):
self._post()
seat = TableSeat.objects.get(room=self.room, role="PC")
self.assertEqual(seat.significator, self.card)
def test_select_sig_returns_200(self):
response = self._post()
self.assertEqual(response.status_code, 200)
def test_select_sig_wrong_turn_makes_no_change(self):
# Gamer 2 is NC — not their turn yet
self.client.force_login(self.gamers[1])
self._post()
seat = TableSeat.objects.get(room=self.room, role="NC")
self.assertIsNone(seat.significator)
def test_select_sig_wrong_turn_returns_403(self):
self.client.force_login(self.gamers[1])
response = self._post()
self.assertEqual(response.status_code, 403)
def test_select_sig_card_not_in_deck_returns_400(self):
# Create a pip card (number=5) — not in the sig deck (only court 1114 + major 01)
other = TarotCard.objects.create(
deck_variant=self.earthman, arcana="MINOR", suit="BRANDS", number=5,
name="Five of Brands Test", slug="five-of-brands-test",
keywords_upright=[], keywords_reversed=[],
)
response = self._post(card_id=other.id)
self.assertEqual(response.status_code, 400)
def test_select_sig_card_already_taken_returns_409(self):
# Another seat already holds this card as their significator
nc_seat = TableSeat.objects.get(room=self.room, role="NC")
nc_seat.significator = self.card
nc_seat.save()
response = self._post()
self.assertEqual(response.status_code, 409)
def test_select_sig_advances_active_seat_to_nc(self):
self._post()
from apps.epic.models import active_sig_seat
seat = active_sig_seat(self.room)
self.assertEqual(seat.role, "NC")
def test_select_sig_notifies_ws(self):
with patch("apps.epic.views._notify_sig_selected") as mock_notify:
self._post()
mock_notify.assert_called_once()
def test_select_sig_requires_login(self):
self.client.logout()
response = self._post()
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_select_sig_wrong_phase_redirects(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self._post()
self.assertRedirects(
response, reverse("epic:room", args=[self.room.id])
)
def test_select_sig_last_choice_does_not_advance_to_none(self):
"""After all 6 significators chosen, active_sig_seat() is None —
no unhandled AttributeError in the view."""
cards = list(TarotCard.objects.filter(deck_variant=self.earthman, arcana="MINOR"))
seats_in_order = list(
TableSeat.objects.filter(room=self.room).order_by("slot_number")
)
# Assign all but the last (BC) manually
for seat, card in zip(seats_in_order[:-1], cards):
seat.significator = card
seat.save()
# BC gamer POSTs the final choice
bc_seat = TableSeat.objects.get(room=self.room, role="BC")
self.client.force_login(bc_seat.gamer)
last_card = TarotCard.objects.filter(
deck_variant=self.earthman, arcana="MAJOR", number=0
).first()
response = self.client.post(self.url, data={"card_id": last_card.id})
self.assertIn(response.status_code, (200, 302))
class ConfirmTokenRecordsSlotFilledTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="gamer@test.io")
self.client.force_login(self.user)
self.room = Room.objects.create(name="Test Room", owner=self.user)
self.token = Token.objects.create(user=self.user, token_type=Token.TITHE)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.user
self.slot.status = GateSlot.RESERVED
self.slot.reserved_at = timezone.now()
self.slot.save()
def test_confirm_token_records_slot_filled_event(self):
session = self.client.session
session["kit_token_id"] = str(self.token.id)
session.save()
self.client.post(reverse("epic:confirm_token", args=[self.room.id]))
event = GameEvent.objects.get(room=self.room, verb=GameEvent.SLOT_FILLED)
self.assertEqual(event.actor, self.user)
self.assertEqual(event.data["slot_number"], 1)
self.assertEqual(event.data["token_type"], Token.TITHE)
def test_no_event_recorded_if_no_reserved_slot(self):
self.slot.gamer = None
self.slot.status = GateSlot.EMPTY
self.slot.save()
self.client.post(reverse("epic:confirm_token", args=[self.room.id]))
self.assertEqual(GameEvent.objects.filter(verb=GameEvent.SLOT_FILLED).count(), 0)
class SelectRoleRecordsRoleSelectedTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="player@test.io")
self.client.force_login(self.user)
self.room = Room.objects.create(
name="Role Room", owner=self.user, table_status=Room.ROLE_SELECT
)
self.seat = TableSeat.objects.create(
room=self.room, gamer=self.user, slot_number=1
)
def test_select_role_records_role_selected_event(self):
self.client.post(
reverse("epic:select_role", args=[self.room.id]),
data={"role": "PC"},
)
event = GameEvent.objects.get(room=self.room, verb=GameEvent.ROLE_SELECTED)
self.assertEqual(event.actor, self.user)
self.assertEqual(event.data["role"], "PC")
self.assertEqual(event.data["slot_number"], 1)
def test_no_event_if_role_already_taken(self):
TableSeat.objects.create(room=self.room, gamer=self.user, slot_number=2, role="PC")
self.client.post(
reverse("epic:select_role", args=[self.room.id]),
data={"role": "PC"},
)
self.assertEqual(GameEvent.objects.filter(verb=GameEvent.ROLE_SELECTED).count(), 0)
# ── sig_reserve view ──────────────────────────────────────────────────────────
class SigReserveViewTest(TestCase):
"""sig_reserve — provisional card hold; OK/NVM flow."""
def setUp(self):
self.room, self.gamers, self.earthman, self.card = _full_sig_setUp(self)
# founder (gamers[0]) is PC — levity polarity
self.client.force_login(self.gamers[0])
self.url = reverse("epic:sig_reserve", kwargs={"room_id": self.room.id})
def _reserve(self, card_id=None, action="reserve", client=None):
c = client or self.client
return c.post(self.url, data={
"card_id": card_id or self.card.id,
"action": action,
})
# ── happy-path reserve ────────────────────────────────────────────────
def test_reserve_creates_sig_reservation(self):
self._reserve()
self.assertTrue(SigReservation.objects.filter(
room=self.room, gamer=self.gamers[0], card=self.card
).exists())
def test_reserve_returns_200(self):
response = self._reserve()
self.assertEqual(response.status_code, 200)
def test_reservation_has_correct_polarity(self):
self._reserve()
res = SigReservation.objects.get(room=self.room, gamer=self.gamers[0])
self.assertEqual(res.polarity, "levity")
def test_gravity_gamer_reservation_has_gravity_polarity(self):
# gamers[3] is SC (index 3 → role SC → but _full_sig_setUp uses SIG_SEAT_ORDER
# which assigns PC→NC→EC→SC→AC→BC, so slot 4 = SC, slot 5 = AC, slot 6 = BC)
# gamers[5] is BC → gravity
bc_client = self.client.__class__()
bc_client.force_login(self.gamers[5])
self._reserve(client=bc_client)
res = SigReservation.objects.get(room=self.room, gamer=self.gamers[5])
self.assertEqual(res.polarity, "gravity")
# ── conflict handling ─────────────────────────────────────────────────
def test_reserve_taken_card_same_polarity_returns_409(self):
# NC (gamers[1]) reserves the same card first — both are levity
nc_client = self.client.__class__()
nc_client.force_login(self.gamers[1])
self._reserve(client=nc_client)
# Now PC tries to grab the same card — should be blocked
response = self._reserve()
self.assertEqual(response.status_code, 409)
def test_reserve_taken_card_cross_polarity_succeeds(self):
# BC (gamers[5], gravity) reserves the same card — different polarity, allowed
bc_client = self.client.__class__()
bc_client.force_login(self.gamers[5])
self._reserve(client=bc_client)
response = self._reserve() # PC (levity) grabs same card
self.assertEqual(response.status_code, 200)
def test_reserve_different_card_while_holding_returns_409(self):
"""Cannot OK a different card while holding one — must NVM first."""
card_b = TarotCard.objects.filter(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=12
).first()
self._reserve() # PC grabs card A → 200
response = self._reserve(card_id=card_b.id) # tries card B → 409
self.assertEqual(response.status_code, 409)
# Original reservation still intact
reservations = SigReservation.objects.filter(room=self.room, gamer=self.gamers[0])
self.assertEqual(reservations.count(), 1)
self.assertEqual(reservations.first().card, self.card)
def test_reserve_same_card_again_is_idempotent(self):
"""Re-POSTing the same card while already holding it returns 200 (no-op)."""
self._reserve()
response = self._reserve() # same card again
self.assertEqual(response.status_code, 200)
self.assertEqual(
SigReservation.objects.filter(room=self.room, gamer=self.gamers[0]).count(), 1
)
def test_reserve_blocked_then_unblocked_after_release(self):
"""After NVM, a new card can be OK'd."""
card_b = TarotCard.objects.filter(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=12
).first()
self._reserve() # hold card A
self._reserve(action="release") # NVM
response = self._reserve(card_id=card_b.id) # now card B → 200
self.assertEqual(response.status_code, 200)
self.assertTrue(SigReservation.objects.filter(
room=self.room, gamer=self.gamers[0], card=card_b
).exists())
# ── release ───────────────────────────────────────────────────────────
def test_release_deletes_reservation(self):
self._reserve()
self._reserve(action="release")
self.assertFalse(SigReservation.objects.filter(
room=self.room, gamer=self.gamers[0]
).exists())
def test_release_returns_200(self):
self._reserve()
response = self._reserve(action="release")
self.assertEqual(response.status_code, 200)
def test_release_with_no_reservation_still_200(self):
"""NVM when nothing held is harmless."""
response = self._reserve(action="release")
self.assertEqual(response.status_code, 200)
def test_release_while_ready_records_sig_unready(self):
"""Releasing a ready reservation implicitly acts as WAIT NVM and records SIG_UNREADY."""
self._reserve()
res = SigReservation.objects.get(room=self.room, gamer=self.gamers[0])
res.ready = True
res.save()
self._reserve(action="release")
self.assertTrue(self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_UNREADY
).exists())
# ── guards ────────────────────────────────────────────────────────────
def test_reserve_non_post_returns_405(self):
response = self.client.get(self.url)
self.assertEqual(response.status_code, 405)
def test_reserve_requires_login(self):
self.client.logout()
response = self._reserve()
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_reserve_requires_seated_gamer(self):
outsider = User.objects.create(email="outsider@test.io")
outsider_client = self.client.__class__()
outsider_client.force_login(outsider)
response = self._reserve(client=outsider_client)
self.assertEqual(response.status_code, 403)
def test_reserve_wrong_phase_returns_400(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self._reserve()
self.assertEqual(response.status_code, 400)
def test_reserve_broadcasts_ws(self):
with patch("apps.epic.views._notify_sig_reserved") as mock_notify:
self._reserve()
mock_notify.assert_called_once()
def test_release_broadcasts_ws(self):
self._reserve()
with patch("apps.epic.views._notify_sig_reserved") as mock_notify:
self._reserve(action="release")
mock_notify.assert_called_once()
def test_release_broadcasts_card_id_so_second_browser_can_clear_it(self):
"""WS release event must include the card_id; otherwise the receiving
browser can't find the card element to remove .sig-reserved--own."""
self._reserve()
with patch("apps.epic.views._notify_sig_reserved") as mock_notify:
self._reserve(action="release")
args, kwargs = mock_notify.call_args
self.assertEqual(args[1], self.card.pk) # card_id must not be None
self.assertFalse(kwargs['reserved']) # reserved=False
# ── sig_ready view ────────────────────────────────────────────────────────────
def _make_levity_reservations(room, gamers, earthman, ready=False):
"""Create SigReservations for the three levity gamers (PC, NC, SC).
Returns the three reservations in PC→NC→SC order."""
cards = [
TarotCard.objects.get(deck_variant=earthman, arcana="MIDDLE", suit="BRANDS", number=n)
for n in (11, 12, 13)
]
roles = ["PC", "NC", "SC"]
# gamers[0]=PC, gamers[1]=NC, gamers[3]=SC
gamer_indices = [0, 1, 3]
reservations = []
for gamer_idx, role, card in zip(gamer_indices, roles, cards):
seat = TableSeat.objects.get(room=room, role=role)
res = SigReservation.objects.create(
room=room, gamer=gamers[gamer_idx], card=card,
role=role, polarity="levity", seat=seat, ready=ready,
)
reservations.append(res)
return reservations
class SigReadyViewTest(TestCase):
"""sig_ready — toggle ready/unready for the polarity-room countdown."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.reservations = _make_levity_reservations(self.room, self.gamers, self.earthman)
self.url = reverse("epic:sig_ready", kwargs={"room_id": self.room.id})
def _post(self, action="ready", seconds_remaining=None, client=None):
c = client or self.client
data = {"action": action}
if seconds_remaining is not None:
data["seconds_remaining"] = seconds_remaining
return c.post(self.url, data=data)
# ── guards ────────────────────────────────────────────────────────────
def test_sig_ready_non_post_returns_405(self):
response = self.client.get(self.url)
self.assertEqual(response.status_code, 405)
def test_sig_ready_requires_login(self):
self.client.logout()
response = self._post()
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_sig_ready_requires_seated_gamer(self):
outsider = User.objects.create(email="outsider@test.io")
outsider_client = self.client.__class__()
outsider_client.force_login(outsider)
response = self._post(client=outsider_client)
self.assertEqual(response.status_code, 403)
def test_sig_ready_wrong_phase_returns_400(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self._post()
self.assertEqual(response.status_code, 400)
def test_sig_ready_without_reservation_returns_400(self):
"""Can't go ready without an OK'd card."""
SigReservation.objects.filter(room=self.room, gamer=self.gamers[0]).delete()
response = self._post()
self.assertEqual(response.status_code, 400)
# ── happy-path ready ──────────────────────────────────────────────────
def test_sig_ready_sets_ready_true_on_reservation(self):
self._post(action="ready")
res = SigReservation.objects.get(room=self.room, gamer=self.gamers[0])
self.assertTrue(res.ready)
def test_sig_ready_returns_200(self):
response = self._post(action="ready")
self.assertEqual(response.status_code, 200)
def test_sig_ready_already_ready_is_idempotent(self):
"""Re-posting ready when already ready returns 200 without re-triggering countdown."""
self.reservations[0].ready = True
self.reservations[0].save()
response = self._post(action="ready")
self.assertEqual(response.status_code, 200)
# ── unready ──────────────────────────────────────────────────────────
def test_sig_unready_sets_ready_false(self):
self.reservations[0].ready = True
self.reservations[0].save()
self._post(action="unready")
res = SigReservation.objects.get(room=self.room, gamer=self.gamers[0])
self.assertFalse(res.ready)
def test_sig_unready_when_not_ready_is_harmless(self):
response = self._post(action="unready")
self.assertEqual(response.status_code, 200)
# ── countdown mechanics ───────────────────────────────────────────────
def test_sig_ready_broadcasts_countdown_start_when_all_three_polarity_ready(self):
"""When all three levity gamers are ready, countdown_start broadcasts."""
# Make NC and SC ready first
for res in self.reservations[1:]:
res.ready = True
res.save()
# PC (founder) goes ready — triggers all-three condition
with patch("apps.epic.views._notify_countdown_start") as mock_notify:
self._post(action="ready")
mock_notify.assert_called_once()
args = mock_notify.call_args[0]
self.assertIn("levity", args) # polarity in call
def test_sig_ready_does_not_broadcast_countdown_when_only_two_ready(self):
self.reservations[1].ready = True
self.reservations[1].save()
with patch("apps.epic.views._notify_countdown_start") as mock_notify:
self._post(action="ready")
mock_notify.assert_not_called()
def test_sig_unready_invalid_seconds_defaults_to_12(self):
"""Non-numeric seconds_remaining falls back to 12."""
self.reservations[0].ready = True
self.reservations[0].save()
self._post(action="unready", seconds_remaining="abc")
self.reservations[0].refresh_from_db()
self.assertEqual(self.reservations[0].countdown_remaining, 12)
def test_sig_unready_saves_seconds_remaining_on_all_polarity_reservations(self):
for res in self.reservations:
res.ready = True
res.save()
self._post(action="unready", seconds_remaining=7)
for res in self.reservations:
res.refresh_from_db()
self.assertEqual(res.countdown_remaining, 7)
def test_sig_unready_broadcasts_countdown_cancel(self):
for res in self.reservations:
res.ready = True
res.save()
with patch("apps.epic.views._notify_countdown_cancel") as mock_notify:
self._post(action="unready", seconds_remaining=7)
mock_notify.assert_called_once()
def test_sig_ready_uses_saved_seconds_for_countdown_restart(self):
"""If countdown_remaining is saved (e.g. 7), countdown_start sends 7 not 12."""
for res in self.reservations:
res.ready = True
res.countdown_remaining = 7
res.save()
# One unreadied; now goes ready again — all 3 ready → start from 7
self.reservations[0].ready = False
self.reservations[0].save()
with patch("apps.epic.views._notify_countdown_start") as mock_notify:
self._post(action="ready")
mock_notify.assert_called_once()
args, kwargs = mock_notify.call_args
seconds_sent = kwargs.get("seconds") or args[1]
self.assertEqual(seconds_sent, 7)
# ── sig_confirm view ──────────────────────────────────────────────────────────
def _make_gravity_reservations(room, gamers, earthman, ready=False):
"""Create SigReservations for the three gravity gamers (EC, AC, BC)."""
cards = [
TarotCard.objects.get(deck_variant=earthman, arcana="MIDDLE", suit="GRAILS", number=n)
for n in (11, 12, 13)
]
roles = ["EC", "AC", "BC"]
# gamers[2]=EC, gamers[4]=AC, gamers[5]=BC
gamer_indices = [2, 4, 5]
reservations = []
for gamer_idx, role, card in zip(gamer_indices, roles, cards):
seat = TableSeat.objects.get(room=room, role=role)
res = SigReservation.objects.create(
room=room, gamer=gamers[gamer_idx], card=card,
role=role, polarity="gravity", seat=seat, ready=ready,
)
reservations.append(res)
return reservations
class SigConfirmViewTest(TestCase):
"""sig_confirm — finalize polarity group once countdown reaches zero."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
# All three levity gamers are ready
self.lev_res = _make_levity_reservations(
self.room, self.gamers, self.earthman, ready=True
)
# founder (PC) is already logged in from _full_sig_setUp
self.url = reverse("epic:sig_confirm", kwargs={"room_id": self.room.id})
def _post(self, polarity="levity", client=None):
c = client or self.client
return c.post(self.url, data={"polarity": polarity})
# ── guards ────────────────────────────────────────────────────────────
def test_sig_confirm_non_post_returns_405(self):
response = self.client.get(self.url)
self.assertEqual(response.status_code, 405)
def test_sig_confirm_requires_login(self):
self.client.logout()
response = self._post()
self.assertEqual(response.status_code, 302)
self.assertIn("/accounts/login/", response.url)
def test_sig_confirm_requires_seated_gamer(self):
outsider = User.objects.create(email="outsider@test.io")
outsider_client = self.client.__class__()
outsider_client.force_login(outsider)
response = self._post(client=outsider_client)
self.assertEqual(response.status_code, 403)
def test_sig_confirm_wrong_phase_returns_400(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self._post()
self.assertEqual(response.status_code, 400)
def test_sig_confirm_not_all_polarity_ready_returns_400(self):
"""If any of the three in the polarity group isn't ready, reject."""
self.lev_res[1].ready = False
self.lev_res[1].save()
response = self._post()
self.assertEqual(response.status_code, 400)
# ── happy-path ────────────────────────────────────────────────────────
def test_sig_confirm_sets_significator_on_seats_from_reservations(self):
self._post()
for res in self.lev_res:
seat = TableSeat.objects.get(room=self.room, role=res.role)
self.assertEqual(seat.significator, res.card)
def test_sig_confirm_returns_200(self):
response = self._post()
self.assertEqual(response.status_code, 200)
def test_sig_confirm_broadcasts_polarity_room_done(self):
with patch("apps.epic.views._notify_polarity_room_done") as mock_notify:
self._post()
mock_notify.assert_called_once()
args = mock_notify.call_args[0]
self.assertIn("levity", args)
def test_sig_confirm_is_idempotent_if_significators_already_set(self):
"""Second call from another browser returns 200 without re-running logic."""
self._post()
response = self._post()
self.assertEqual(response.status_code, 200)
# ── both polarities done ──────────────────────────────────────────────
def test_sig_confirm_broadcasts_pick_sky_available_when_both_polarities_done(self):
"""After both levity and gravity confirm, pick_sky_available fires."""
# Pre-set gravity seats to already have significators (simulating earlier confirm)
grav_cards = [
TarotCard.objects.get(deck_variant=self.earthman, arcana="MIDDLE", suit="GRAILS", number=n)
for n in (11, 12, 13)
]
for role, card in zip(["EC", "AC", "BC"], grav_cards):
seat = TableSeat.objects.get(room=self.room, role=role)
seat.significator = card
seat.save()
with patch("apps.epic.views._notify_pick_sky_available") as mock_notify:
self._post(polarity="levity")
mock_notify.assert_called_once()
def test_sig_confirm_sets_room_to_sky_select_when_both_polarities_done(self):
grav_cards = [
TarotCard.objects.get(deck_variant=self.earthman, arcana="MIDDLE", suit="GRAILS", number=n)
for n in (11, 12, 13)
]
for role, card in zip(["EC", "AC", "BC"], grav_cards):
seat = TableSeat.objects.get(room=self.room, role=role)
seat.significator = card
seat.save()
self._post(polarity="levity")
self.room.refresh_from_db()
self.assertEqual(self.room.table_status, Room.SKY_SELECT)
def test_sig_confirm_does_not_broadcast_pick_sky_available_when_only_one_polarity_done(self):
with patch("apps.epic.views._notify_pick_sky_available") as mock_notify:
self._post(polarity="levity")
mock_notify.assert_not_called()
# ── SKY_SELECT rendering ──────────────────────────────────────────────────────
class PickSkyRenderingTest(TestCase):
"""Room page at SKY_SELECT renders CAST SKY btn and sig card in tray cell 2."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.room.table_status = Room.SKY_SELECT
self.room.save()
self.sig_card = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=11
)
pc_seat = TableSeat.objects.get(room=self.room, role="PC")
pc_seat.significator = self.sig_card
pc_seat.save()
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_pick_sky_btn_present_in_sky_select_phase(self):
response = self.client.get(self.url)
self.assertContains(response, "id_pick_sky_btn")
def test_tray_cell_2_contains_sig_card_icon_in_sky_select(self):
response = self.client.get(self.url)
self.assertContains(response, "tray-sig-card")
def test_pick_sky_btn_hidden_during_sig_select(self):
# Rendered hidden (display:none) so JS can reveal it on pick_sky_available WS event
self.room.table_status = Room.SIG_SELECT
self.room.save()
response = self.client.get(self.url)
self.assertContains(response, 'id="id_pick_sky_btn"')
self.assertContains(response, 'style="display:none"')
def test_sky_delete_clears_seat_character_and_returns_json(self):
"""POST epic:sky_delete clears any Character on the requesting gamer's
seat — both unconfirmed drafts AND confirmed ones (the latter case is
why un-saved-via-DEL data was rehydrating on refresh: a SAVE SKY click
confirms a Character, and only that seat's Character row is the durable
target the in-room DEL has to purge)."""
# Seed both a draft & a confirmed Character — DEL must clear them both
from apps.epic.models import Character
pc_seat = TableSeat.objects.get(room=self.room, role="PC")
# Confirmed (the SAVE SKY case)
confirmed = Character.objects.create(
seat=pc_seat,
chart_data={"planets": {"Sun": {"sign": "Gemini"}}},
confirmed_at=timezone.now(),
)
url = reverse("epic:sky_delete", kwargs={"room_id": self.room.id})
response = self.client.post(url)
self.assertEqual(response.status_code, 200)
self.assertJSONEqual(response.content, {"deleted": True})
self.assertFalse(
Character.objects.filter(seat=pc_seat, retired_at__isnull=True).exists(),
"Both draft and confirmed Characters on the seat should be gone",
)
def test_sky_delete_405_on_get(self):
url = reverse("epic:sky_delete", kwargs={"room_id": self.room.id})
self.assertEqual(self.client.get(url).status_code, 405)
def test_sky_delete_requires_seat_owner(self):
"""A gamer who isn't seated at this room can't purge another seat."""
outsider = User.objects.create(email="outsider@test.io")
self.client.force_login(outsider)
url = reverse("epic:sky_delete", kwargs={"room_id": self.room.id})
self.assertEqual(self.client.post(url).status_code, 403)
def test_sky_delete_does_not_touch_user_model(self):
"""In-room DEL targets the seat's Character, never the User-level
sky_chart_data. (The Dashsky / My Sky applet DEL is the one that
clears the user's saved sky.)"""
founder = self.gamers[0]
founder.sky_chart_data = {"planets": {"Sun": {"sign": "Gemini"}}}
founder.sky_birth_tz = "America/New_York"
founder.save()
url = reverse("epic:sky_delete", kwargs={"room_id": self.room.id})
self.client.post(url)
founder.refresh_from_db()
self.assertEqual(founder.sky_chart_data, {"planets": {"Sun": {"sign": "Gemini"}}})
self.assertEqual(founder.sky_birth_tz, "America/New_York")
def test_no_sky_delete_btn_in_blank_sky_select_modal(self):
"""A fresh CAST SKY modal (no preview wheel rendered yet) must not
carry the DEL btn — it would otherwise float in the empty wheel area
suggesting there's something to delete when the user has only seen
the form. The JS schedulePreview success handler is the contract that
injects the btn after the wheel paints — so the rendered HTML should
carry no <button id="id_sky_delete_btn"> markup. (The literal string
does still appear inside the inline <script> that does the injection,
so the assertion targets the rendered attribute syntax, not the bare
identifier.)"""
response = self.client.get(self.url)
self.assertNotContains(response, 'id="id_sky_delete_btn"')
# ── SEA_SELECT rendering ──────────────────────────────────────────────────────
class PickSeaRenderingTest(TestCase):
"""At SKY_SELECT, a confirmed Character swaps CAST SKY → DRAW SEA + sea overlay."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.room.table_status = Room.SKY_SELECT
self.room.save()
self.sig_card = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=11
)
self.pc_seat = TableSeat.objects.get(room=self.room, role="PC")
self.pc_seat.significator = self.sig_card
self.pc_seat.save()
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def _confirm_sky(self, seat=None):
target = seat or self.pc_seat
return Character.objects.create(seat=target, confirmed_at=timezone.now())
def test_sky_confirmed_false_when_no_character(self):
response = self.client.get(self.url)
self.assertFalse(response.context["sky_confirmed"])
def test_sky_confirmed_true_when_character_confirmed(self):
self._confirm_sky()
response = self.client.get(self.url)
self.assertTrue(response.context["sky_confirmed"])
def test_pick_sea_btn_shown_when_sky_confirmed(self):
self._confirm_sky()
response = self.client.get(self.url)
self.assertContains(response, "id_pick_sea_btn")
def test_pick_sky_btn_shown_when_sky_not_confirmed(self):
response = self.client.get(self.url)
self.assertContains(response, "id_pick_sky_btn")
def test_sea_overlay_included_when_sky_confirmed(self):
self._confirm_sky()
response = self.client.get(self.url)
self.assertContains(response, "id_sea_overlay")
def test_sea_overlay_select_defaults_to_waite_smith_for_levity(self):
self._confirm_sky()
response = self.client.get(self.url)
self.assertContains(response, "Celtic Cross, Waite-Smith")
def test_sea_overlay_select_defaults_to_escape_velocity_for_gravity(self):
ec_gamer = self.gamers[2] # EC — gravity
self.client.force_login(ec_gamer)
ec_seat = TableSeat.objects.get(room=self.room, role="EC")
self._confirm_sky(seat=ec_seat)
response = self.client.get(self.url)
self.assertContains(response, "Celtic Cross, Escape Velocity")
def test_user_polarity_in_context_at_sky_select(self):
response = self.client.get(self.url)
self.assertIn("user_polarity", response.context)
self.assertEqual(response.context["user_polarity"], "levity") # PC is levity
def test_my_tray_sig_falls_back_to_seat_when_char_sig_is_none(self):
"""Characters created before the sig-sync fix have significator=None; fall back to seat."""
Character.objects.create(
seat=self.pc_seat,
significator=None,
confirmed_at=timezone.now(),
)
response = self.client.get(self.url)
self.assertEqual(response.context["my_tray_sig"], self.sig_card)
def test_my_tray_sig_comes_from_character_significator_when_confirmed(self):
"""When sky_confirmed, my_tray_sig reads from Character.significator (not TableSeat)."""
char = Character.objects.create(
seat=self.pc_seat,
significator=self.sig_card,
confirmed_at=timezone.now(),
)
# Give the seat a *different* sig card so we can distinguish the sources
other_card = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="GRAILS", number=11
)
self.pc_seat.significator = other_card
self.pc_seat.save()
response = self.client.get(self.url)
self.assertEqual(response.context["my_tray_sig"], char.significator)
self.assertNotEqual(response.context["my_tray_sig"], other_card)
# ── select_role GET redirect ──────────────────────────────────────────────────
class SelectRoleGetRedirectTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="gamer@sr.io")
self.client.force_login(self.user)
self.room = Room.objects.create(name="R", owner=self.user)
self.room.table_status = Room.ROLE_SELECT
self.room.save()
def test_get_redirects_to_room(self):
response = self.client.get(reverse("epic:select_role", kwargs={"room_id": self.room.id}))
self.assertRedirects(response, reverse("epic:room", kwargs={"room_id": self.room.id}),
fetch_redirect_response=False)
# ── sig_reserve / sig_ready / sig_confirm / select_sig helpers ────────────────
def _make_sig_room(owner, *extra_gamers):
room = Room.objects.create(name="SR", owner=owner)
seat_map = {}
gamers = [owner] + list(extra_gamers)
roles = ["PC", "NC", "EC", "SC", "AC", "BC"]
for i, (gamer, role) in enumerate(zip(gamers, roles), start=1):
seat = TableSeat.objects.create(room=room, gamer=gamer, slot_number=i, role=role)
seat_map[role] = seat
room.table_status = Room.SIG_SELECT
room.save()
return room, seat_map
class SelectSigViewTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="pc@selsig.io")
self.client.force_login(self.user)
self.room, self.seats = _make_sig_room(self.user)
def test_non_post_redirects(self):
response = self.client.get(reverse("epic:select_sig", kwargs={"room_id": self.room.id}))
self.assertEqual(response.status_code, 302)
def test_nonexistent_card_returns_400(self):
response = self.client.post(
reverse("epic:select_sig", kwargs={"room_id": self.room.id}),
{"card_id": "99999999"},
)
self.assertEqual(response.status_code, 400)
# ── sky_preview (epic) ──────────────────────────────────────────────────────
class SkyPreviewViewTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="pc@sky.io")
self.client.force_login(self.user)
self.room, _ = _make_sig_room(self.user)
self.room.table_status = Room.SKY_SELECT
self.room.save()
self.url = reverse("epic:sky_preview", kwargs={"room_id": self.room.id})
def test_missing_params_returns_400(self):
response = self.client.get(self.url, {"date": "1990-06-15"})
self.assertEqual(response.status_code, 400)
def test_non_numeric_lat_returns_400(self):
response = self.client.get(self.url, {"date": "1990-06-15", "lat": "abc", "lon": "0"})
self.assertEqual(response.status_code, 400)
def test_out_of_range_lat_returns_400(self):
response = self.client.get(self.url, {"date": "1990-06-15", "lat": "999", "lon": "0"})
self.assertEqual(response.status_code, 400)
def test_invalid_tz_string_returns_400(self):
response = self.client.get(
self.url,
{"date": "1990-06-15", "lat": "51.5", "lon": "-0.1", "tz": "Not/Real"},
)
self.assertEqual(response.status_code, 400)
def test_bad_date_format_returns_400(self):
response = self.client.get(
self.url,
{"date": "baddate", "time": "09:00", "lat": "51.5", "lon": "-0.1", "tz": "UTC"},
)
self.assertEqual(response.status_code, 400)
@patch("apps.epic.views.http_requests")
def test_pyswiss_failure_returns_502(self, mock_requests):
from unittest.mock import MagicMock
tz_r = MagicMock()
tz_r.json.return_value = {"timezone": "UTC"}
tz_r.raise_for_status = MagicMock()
chart_r = MagicMock()
chart_r.raise_for_status.side_effect = Exception("timeout")
mock_requests.get.side_effect = [tz_r, chart_r]
response = self.client.get(self.url, {"date": "1990-06-15", "lat": "51.5", "lon": "-0.1"})
self.assertEqual(response.status_code, 502)
@patch("apps.epic.views.http_requests")
def test_success_returns_chart_distinctions_timezone(self, mock_requests):
from unittest.mock import MagicMock
payload = {
"planets": {"Sun": {"degree": 84.5}},
"houses": {"cusps": [0] * 12},
"elements": {"Earth": 1},
"house_system": "O",
}
tz_r = MagicMock()
tz_r.json.return_value = {"timezone": "Europe/London"}
tz_r.raise_for_status = MagicMock()
ch_r = MagicMock()
ch_r.json.return_value = payload
ch_r.raise_for_status = MagicMock()
mock_requests.get.side_effect = [tz_r, ch_r]
response = self.client.get(self.url, {"date": "1990-06-15", "lat": "51.5", "lon": "-0.1"})
self.assertEqual(response.status_code, 200)
data = response.json()
self.assertIn("distinctions", data)
self.assertIn("Stone", data["elements"])
self.assertNotIn("Earth", data["elements"])
# ── tarot_deal ────────────────────────────────────────────────────────────────
class TarotDealViewTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="dealer@test.io")
self.client.force_login(self.user)
self.room, _ = _make_sig_room(self.user)
def test_non_post_redirects_to_tarot_deck(self):
response = self.client.get(
reverse("epic:tarot_deal", kwargs={"room_id": self.room.id})
)
self.assertRedirects(
response,
reverse("epic:tarot_deck", kwargs={"room_id": self.room.id}),
fetch_redirect_response=False,
)
# ── sky_save (epic) ─────────────────────────────────────────────────────────
class SkySaveViewTest(TestCase):
def setUp(self):
self.user = User.objects.create(email="pc@skysave.io")
self.client.force_login(self.user)
self.room, _ = _make_sig_room(self.user)
self.room.table_status = Room.SKY_SELECT
self.room.save()
self.url = reverse("epic:sky_save", kwargs={"room_id": self.room.id})
def _post(self, payload):
import json as _json
return self.client.post(self.url, data=_json.dumps(payload), content_type="application/json")
def test_get_returns_405(self):
response = self.client.get(self.url)
self.assertEqual(response.status_code, 405)
def test_invalid_json_returns_400(self):
response = self.client.post(self.url, data="not json", content_type="application/json")
self.assertEqual(response.status_code, 400)
def test_save_draft_returns_id_and_not_confirmed(self):
response = self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5,
"birth_lon": -0.1,
"birth_place": "London",
"house_system": "O",
"chart_data": {},
})
self.assertEqual(response.status_code, 200)
data = response.json()
self.assertIn("id", data)
self.assertFalse(data["confirmed"])
def test_confirm_action_locks_character(self):
response = self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5,
"birth_lon": -0.1,
"birth_place": "",
"house_system": "O",
"chart_data": {},
"action": "confirm",
})
self.assertEqual(response.status_code, 200)
self.assertTrue(response.json()["confirmed"])
def test_confirm_records_sky_saved_event_with_top_capacitors(self):
"""When action=confirm, log a SKY_SAVED GameEvent w. the highest-count
capacitor name(s) so the billscroll can render the new prose."""
from apps.drama.models import GameEvent
chart = {
"elements": {
# Earthman uses 6 elements; canonical names map to capacitors:
# Fire→Ardor Stone→Ossum Air→Pneuma Water→Humor Time→Tempo Space→Nexus.
"Fire": 3,
"Stone": 1,
"Air": 2,
"Water": 0,
"Time": 1,
"Space": 1,
}
}
self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5, "birth_lon": -0.1,
"birth_place": "", "house_system": "O",
"chart_data": chart, "action": "confirm",
})
event = GameEvent.objects.get(room=self.room, verb=GameEvent.SKY_SAVED)
self.assertEqual(event.actor, self.user)
self.assertEqual(event.data.get("top_capacitors"), ["Ardor"])
def test_confirm_records_sky_saved_event_with_two_way_tie(self):
from apps.drama.models import GameEvent
chart = {
"elements": {
"Fire": 3, "Stone": 3, # tied at top
"Air": 2, "Water": 0, "Time": 1, "Space": 1,
}
}
self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5, "birth_lon": -0.1,
"birth_place": "", "house_system": "O",
"chart_data": chart, "action": "confirm",
})
event = GameEvent.objects.get(room=self.room, verb=GameEvent.SKY_SAVED)
# Order follows the canonical ELEMENT_ORDER (Fire, Stone, Time, Space, Air, Water)
self.assertEqual(event.data.get("top_capacitors"), ["Ardor", "Ossum"])
def test_save_without_confirm_does_not_record_sky_saved_event(self):
from apps.drama.models import GameEvent
self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5, "birth_lon": -0.1,
"birth_place": "", "house_system": "O",
"chart_data": {"elements": {"Fire": 3}},
# no action=confirm — just a draft save
})
self.assertFalse(
GameEvent.objects.filter(room=self.room, verb=GameEvent.SKY_SAVED).exists()
)
def test_confirm_with_dict_shaped_elements_extracts_count(self):
"""Some chart payloads enrich each element to {count, contributors};
sky_save should read .count rather than treating the dict as a value."""
from apps.drama.models import GameEvent
chart = {
"elements": {
"Fire": {"count": 4, "contributors": ["Sun", "Mars", "Jupiter", "Pluto"]},
"Stone": {"count": 1, "contributors": ["Venus"]},
"Air": {"count": 2, "contributors": ["Mercury", "Uranus"]},
"Water": {"count": 0, "contributors": []},
"Time": {"count": 1, "stellia": ["Saturn"]},
"Space": {"count": 1, "parades": ["Neptune"]},
}
}
self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5, "birth_lon": -0.1,
"birth_place": "", "house_system": "O",
"chart_data": chart, "action": "confirm",
})
event = GameEvent.objects.get(room=self.room, verb=GameEvent.SKY_SAVED)
self.assertEqual(event.data.get("top_capacitors"), ["Ardor"])
def test_confirm_copies_seat_significator_to_character(self):
"""sky_save with action=confirm copies seat.significator onto Character."""
earthman, _ = DeckVariant.objects.get_or_create(
slug="earthman", defaults={"name": "Earthman Deck", "card_count": 108}
)
sig_card = TarotCard.objects.filter(deck_variant=earthman).first()
pc_seat = TableSeat.objects.get(room=self.room, role="PC")
pc_seat.significator = sig_card
pc_seat.save()
self._post({
"birth_dt": "1990-06-15T09:00:00Z",
"birth_lat": 51.5, "birth_lon": -0.1,
"birth_place": "", "house_system": "O",
"chart_data": {}, "action": "confirm",
})
char = Character.objects.get(seat=pc_seat)
self.assertEqual(char.significator, sig_card)
# ── SIG event-retraction branches ────────────────────────────────────────────
# The provenance scrolls use a `data["retracted"] = True` flag to soft-cancel
# prior events when a gamer reverses themselves (WAIT NVM after SAVE SIG, etc).
# These three branches in sig_reserve / sig_ready are the load-bearing ones —
# without them a recanted action stays visible in the billboard scrollback.
class SigEventRetractionTest(TestCase):
"""`data["retracted"] = True` writes on the three reverse-direction paths."""
def setUp(self):
self.room, self.gamers, self.earthman, self.card = _full_sig_setUp(self)
# PC (founder) already logged in; reserve + go ready so subsequent
# actions have prior SIG_READY events to retract.
self.reserve_url = reverse("epic:sig_reserve", kwargs={"room_id": self.room.id})
self.ready_url = reverse("epic:sig_ready", kwargs={"room_id": self.room.id})
def _reserve(self):
return self.client.post(self.reserve_url, data={
"card_id": self.card.id, "action": "reserve",
})
def _ready(self, action="ready"):
return self.client.post(self.ready_url, data={"action": action})
def test_sig_unready_retracts_prior_sig_ready_event(self):
"""sig_ready action=unready flips `data["retracted"]=True` on the most
recent un-retracted SIG_READY event for this actor (views.py L937)."""
self._reserve()
self._ready(action="ready")
prior = self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_READY
).last()
self.assertFalse(prior.data.get("retracted"), "precondition: not yet retracted")
self._ready(action="unready")
prior.refresh_from_db()
self.assertTrue(prior.data.get("retracted"))
def test_sig_ready_retracts_prior_sig_unready_event(self):
"""sig_ready action=ready retracts the most recent un-retracted
SIG_UNREADY event (views.py L907) — the cancellation is now moot."""
self._reserve()
self._ready(action="ready")
self._ready(action="unready")
prior_unready = self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_UNREADY
).last()
self.assertFalse(prior_unready.data.get("retracted"))
self._ready(action="ready")
prior_unready.refresh_from_db()
self.assertTrue(prior_unready.data.get("retracted"))
def test_sig_release_while_ready_retracts_prior_sig_ready_event(self):
"""sig_reserve action=release on a ready reservation acts as implicit
WAIT NVM — retracts the most recent SIG_READY (views.py L823)."""
self._reserve()
self._ready(action="ready")
prior = self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_READY
).last()
self.assertFalse(prior.data.get("retracted"))
self.client.post(self.reserve_url, data={
"card_id": self.card.id, "action": "release",
})
prior.refresh_from_db()
self.assertTrue(prior.data.get("retracted"))
def test_sig_release_while_ready_records_sig_unready_event(self):
"""Same release-while-ready path also records a fresh SIG_UNREADY
(the implicit cancellation event)."""
self._reserve()
self._ready(action="ready")
unready_count_before = self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_UNREADY
).count()
self.client.post(self.reserve_url, data={
"card_id": self.card.id, "action": "release",
})
self.assertEqual(
self.room.events.filter(
actor=self.gamers[0], verb=GameEvent.SIG_UNREADY
).count(),
unready_count_before + 1,
)
# ── SIG_RESERVE invalid card-id branch ───────────────────────────────────────
class SigReserveInvalidCardIdTest(TestCase):
"""sig_reserve POSTed with a card_id that doesn't exist returns 400."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.url = reverse("epic:sig_reserve", kwargs={"room_id": self.room.id})
def test_unknown_card_id_returns_400(self):
"""TarotCard.DoesNotExist branch (views.py L840-841)."""
response = self.client.post(self.url, data={
"card_id": 999999, "action": "reserve",
})
self.assertEqual(response.status_code, 400)
# ── SIG_SELECT gravity-polarity rendering ────────────────────────────────────
class SigSelectGravityContextTest(TestCase):
"""SIG_SELECT room context for a gravity-polarity gamer.
Covers the `user_polarity = 'gravity'` branch (views.py L322) and the
gravity_sig_cards lookup (L357) — both fall through the cracks of the
default founder-as-PC-levity tests."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
# gamers[5] is BC → gravity polarity
self.bc = self.gamers[5]
self.client.force_login(self.bc)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_gravity_gamer_room_context_has_gravity_polarity(self):
response = self.client.get(self.url)
self.assertEqual(response.context["user_polarity"], "gravity")
def test_gravity_gamer_sees_gravity_sig_cards(self):
"""Levity + gravity get the same 16 court cards (filtered by major-arcana
Note unlocks); this test just asserts the gravity branch was taken."""
from apps.epic.models import gravity_sig_cards
response = self.client.get(self.url)
# Same underlying card set; assertion is that the context was populated
# (the gravity branch returned, vs falling into the empty `else`).
self.assertEqual(
list(response.context["sig_cards"]),
list(gravity_sig_cards(self.room, self.bc)),
)
def test_gravity_gamer_sig_card_set_non_empty(self):
response = self.client.get(self.url)
self.assertGreater(len(response.context["sig_cards"]), 0)
# ── SEA_DECK draw view ───────────────────────────────────────────────────────
class SeaDeckViewTest(TestCase):
"""sea_deck — JSON view returning shuffled levity + gravity halves.
Mirrors the FT in test_game_room_select_sea.py:DRAW SEA — that test walks
the full UI; this one isolates the JSON contract + filter semantics."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.room.table_status = Room.SKY_SELECT
self.room.save()
# Use PC seat (founder) — already logged in by _full_sig_setUp
self.url = reverse("epic:sea_deck", kwargs={"room_id": self.room.id})
def test_returns_403_when_not_seated(self):
outsider = User.objects.create(email="outsider@test.io")
self.client.force_login(outsider)
response = self.client.get(self.url)
self.assertEqual(response.status_code, 403)
def test_returns_empty_halves_when_seat_has_no_deck_variant(self):
"""sea_deck early-outs to {levity:[],gravity:[]} if the seat hasn't
committed a deck — guards against null deck_variant FK access."""
TableSeat.objects.filter(room=self.room).update(deck_variant=None)
response = self.client.get(self.url)
self.assertEqual(response.status_code, 200)
data = response.json()
self.assertEqual(data, {"levity": [], "gravity": []})
def test_returns_two_halves(self):
response = self.client.get(self.url)
data = response.json()
self.assertIn("levity", data)
self.assertIn("gravity", data)
def test_card_count_roughly_split_between_halves(self):
"""Total card pool is split in half — within 1 of perfectly even."""
response = self.client.get(self.url)
data = response.json()
self.assertAlmostEqual(len(data["levity"]), len(data["gravity"]), delta=1)
def test_card_dict_contains_expected_keys(self):
response = self.client.get(self.url)
data = response.json()
sample = data["levity"][0]
for key in (
"id", "name", "arcana", "corner_rank", "suit_icon",
"name_group", "name_title", "reversed",
"levity_qualifier", "gravity_qualifier",
):
self.assertIn(key, sample, f"missing key {key!r} in card dict")
def test_reversed_field_is_boolean(self):
response = self.client.get(self.url)
data = response.json()
for card in data["levity"] + data["gravity"]:
self.assertIsInstance(card["reversed"], bool)
def test_excludes_claimed_significators(self):
"""A card already set as a seat.significator must not appear in either
half — it's been claimed for the game and is out of the sea-draw pool."""
sig_card = TarotCard.objects.get(
deck_variant=self.earthman, arcana="MIDDLE", suit="BRANDS", number=11
)
pc_seat = TableSeat.objects.get(room=self.room, role="PC")
pc_seat.significator = sig_card
pc_seat.save()
response = self.client.get(self.url)
data = response.json()
all_ids = {c["id"] for c in data["levity"]} | {c["id"] for c in data["gravity"]}
self.assertNotIn(sig_card.id, all_ids)
class RoomBurgerBtnRenderTest(TestCase):
"""Burger btn + fan of 5 sub-btns renders on room.html unconditionally —
from the gatekeeper state through every table_status. Sub-btns are
pure scaffolding (no handlers in this sprint)."""
def setUp(self):
self.user = User.objects.create(email="founder@test.io")
self.room = Room.objects.create(name="Burger Room", owner=self.user)
self.client.force_login(self.user)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_burger_btn_renders(self):
response = self.client.get(self.url)
self.assertContains(response, 'id="id_burger_btn"')
self.assertContains(response, "fa-burger")
def test_burger_fan_container_renders(self):
response = self.client.get(self.url)
self.assertContains(response, 'id="id_burger_fan"')
def test_five_fan_sub_btns_render_with_correct_ids(self):
response = self.client.get(self.url)
for btn_id in (
"id_sky_btn", "id_earth_btn", "id_sea_btn",
"id_voice_btn", "id_text_btn",
):
self.assertContains(response, f'id="{btn_id}"')
def test_fan_sub_btn_icons_match_spec(self):
response = self.client.get(self.url)
for icon in (
"fa-cloud", "fa-earth-americas", "fa-bridge-water",
"fa-headset", "fa-keyboard",
):
self.assertContains(response, icon)
def test_each_sub_btn_renders_dual_icon_for_inactive_flash_swap(self):
"""Sub-btns carry BOTH the real icon (.burger-fan-icon--on) + a
fa-ban placeholder (.burger-fan-icon--off). CSS keeps the real
icon visible by default; .flash-inactive swaps to fa-ban during
the click-while-inactive pulse. fa-ban itself isn't counted
directly — _table_positions.html also renders fa-ban for
non-starter seats — but the burger-fan-icon classes are unique
to the fan + load-bearing for the CSS swap rule."""
response = self.client.get(self.url)
body = response.content.decode()
self.assertEqual(body.count("burger-fan-icon--on"), 5)
self.assertEqual(body.count("burger-fan-icon--off"), 5)
def test_burger_btn_script_loaded(self):
response = self.client.get(self.url)
self.assertContains(response, "burger-btn.js")
def test_burger_persists_in_table_status_state(self):
"""Burger renders past the gatekeeper too — confirmed for ROLE_SELECT."""
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self.client.get(self.url)
self.assertContains(response, 'id="id_burger_btn"')
class RoomGateViewTest(TestCase):
"""Room renewal gate-view (sprint 2026-05-31) — reachable mid-game (the
gatekeeper redirects to the table once table_status is set; this view
does not). Shows the viewer's own seat/circle + token time-remaining + a
RENEW affordance; the gear-menu NVM returns to the table hex."""
def setUp(self):
self.owner = User.objects.create(email="owner@test.io", username="owner")
self.room = Room.objects.create(
name="Renewal Room", owner=self.owner,
renewal_period=timedelta(days=7),
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.owner
self.slot.status = GateSlot.FILLED
self.slot.filled_at = timezone.now()
self.slot.debited_token_type = Token.FREE
self.slot.save()
self.client.force_login(self.owner)
self.url = reverse("epic:room_gate", args=[self.room.id])
def _lapse(self):
# Backdate the seat into the renewal-grace window (cost lapsed, seat
# still FILLED) so the gate-view renders its renew state.
self.slot.filled_at = timezone.now() - timedelta(days=8)
self.slot.save()
def test_renders_200_even_when_table_status_set(self):
# gatekeeper would 302 to the room; the gate-view must render mid-game.
response = self.client.get(self.url)
self.assertEqual(response.status_code, 200)
def test_cost_current_shows_cont_game_to_hex(self):
response = self.client.get(self.url)
self.assertContains(response, "id_room_cont_game_btn")
self.assertContains(response, "CONT")
self.assertContains(response, reverse("epic:room", args=[self.room.id]))
def test_cost_current_status_shows_deposited_count(self):
# One filled slot (the owner's) → "1 Token(s) Deposited" (literal "(s)").
response = self.client.get(self.url)
self.assertContains(response, "1 Token(s) Deposited")
def test_cost_current_has_no_renew_form(self):
# Rails are static (claimed) while the cost is current — renewing is a
# lapsed-state affordance only.
response = self.client.get(self.url)
self.assertNotContains(
response, reverse("epic:renew_token", args=[self.room.id]))
def test_cost_lapsed_shows_please_deposit_and_renew_rails(self):
self._lapse()
response = self.client.get(self.url)
self.assertContains(response, "Please Deposit Token")
self.assertContains(
response, reverse("epic:renew_token", args=[self.room.id]))
def test_cost_lapsed_hides_cont_game(self):
self._lapse()
response = self.client.get(self.url)
self.assertNotContains(response, "id_room_cont_game_btn")
def test_nvm_returns_to_room_hex(self):
response = self.client.get(self.url)
self.assertContains(
response, f'href="{reverse("epic:room", args=[self.room.id])}"')
def test_page_class_carries_page_room(self):
response = self.client.get(self.url)
self.assertIn("page-room", response.context["page_class"])
def test_no_seat_circle_or_countdown_rendered(self):
# Seat circle + countdown removed (user-spec) — the hex .fa-chair + the
# next-sprint user/seat tooltips carry that info.
response = self.client.get(self.url)
self.assertNotContains(response, "room-gate-seat")
self.assertNotContains(response, "data-cost-until")
self.assertNotContains(response, "id_room_gate_remaining")
class RoomRenewTokenTest(TestCase):
"""The RENEW endpoint — re-deposit a token into the viewer's own FILLED
slot, resetting filled_at=now (via debit_token) so the cost-current
window restarts. Distinct from confirm_token (which needs a RESERVED
slot). 402 when token-depleted."""
def setUp(self):
self.owner = User.objects.create(email="owner@test.io", username="owner")
self.owner.equipped_trinket = None
self.owner.save(update_fields=["equipped_trinket"])
self.owner.tokens.exclude(token_type=Token.FREE).delete() # keep FREE only
self.room = Room.objects.create(
name="Renewal Room", owner=self.owner,
renewal_period=timedelta(days=7),
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.owner
self.slot.status = GateSlot.FILLED
self.slot.filled_at = timezone.now() - timedelta(days=8) # in grace
self.slot.debited_token_type = Token.FREE
self.slot.save()
self.client.force_login(self.owner)
self.url = reverse("epic:renew_token", args=[self.room.id])
def test_renew_resets_filled_at(self):
self.client.post(self.url)
self.slot.refresh_from_db()
self.assertGreater(self.slot.filled_at, timezone.now() - timedelta(minutes=1))
self.assertTrue(self.slot.cost_current)
def test_renew_consumes_free_token(self):
self.client.post(self.url)
self.assertFalse(
self.owner.tokens.filter(token_type=Token.FREE).exists())
self.slot.refresh_from_db()
self.assertEqual(self.slot.debited_token_type, Token.FREE)
def test_renew_records_slot_filled_event(self):
self.client.post(self.url)
self.assertTrue(
self.room.events.filter(
actor=self.owner, verb=GameEvent.SLOT_FILLED).exists())
def test_renew_without_filled_slot_redirects(self):
self.slot.status = GateSlot.EMPTY
self.slot.gamer = None
self.slot.save()
response = self.client.post(self.url)
self.assertEqual(response.status_code, 302)
def test_renew_402_when_token_depleted(self):
self.owner.tokens.all().delete()
response = self.client.post(self.url)
self.assertEqual(response.status_code, 402)
def test_renew_get_redirects(self):
response = self.client.get(self.url)
self.assertEqual(response.status_code, 302)
class RoomNavbarGateViewTest(TestCase):
"""Navbar swaps CONT GAME → GATE VIEW on room pages (mirror my-sea),
routing to the room gate-view. The gameboard listing keeps CONT GAME
(no `page-room` marker)."""
def setUp(self):
self.owner = User.objects.create(email="owner@test.io", username="owner")
self.room = Room.objects.create(
name="Nav Room", owner=self.owner,
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
self.client.force_login(self.owner)
def test_room_page_shows_gate_view_not_cont_game(self):
response = self.client.get(reverse("epic:room", args=[self.room.id]))
self.assertContains(response, "id_navbar_gate_view_btn")
self.assertNotContains(response, 'id="id_cont_game"')
def test_gate_view_btn_links_to_room_gate(self):
response = self.client.get(reverse("epic:room", args=[self.room.id]))
self.assertContains(
response, reverse("epic:room_gate", args=[self.room.id]))
def test_room_gate_page_also_shows_gate_view(self):
response = self.client.get(reverse("epic:room_gate", args=[self.room.id]))
self.assertContains(response, "id_navbar_gate_view_btn")
def test_room_page_carries_page_room_marker(self):
response = self.client.get(reverse("epic:room", args=[self.room.id]))
self.assertIn("page-room", response.context["page_class"])
class RoomCenterSupersessionTest(TestCase):
"""When the viewer's seat token cost lapses (filled_at past the cost-
current window), GATE VIEW supersedes the center-hex phase buttons —
SCAN SIGS, CAST SKY, DRAW SEA, the sig overlay — EXCEPT the gamer's own
ROLE card-stack pick (covered in RoomRoleStackGraceTest)."""
def setUp(self):
self.room, self.gamers, self.earthman, _ = _full_sig_setUp(self)
self.founder = self.gamers[0]
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def _lapse_viewer(self):
slot = self.room.gate_slots.get(slot_number=1) # founder = slot 1
slot.filled_at = timezone.now() - timedelta(days=8) # grace (S=7d)
slot.save()
def test_viewer_cost_current_true_by_default(self):
# _full_sig_setUp leaves filled_at None → never-expires → current.
self.assertTrue(self.client.get(self.url).context["viewer_cost_current"])
def test_cost_current_no_gate_view_btn_in_center(self):
self.assertNotContains(self.client.get(self.url), "id_room_gate_view_btn")
def test_cost_lapsed_supersedes_sig_overlay(self):
self._lapse_viewer()
response = self.client.get(self.url) # _full_sig_setUp room is SIG_SELECT
self.assertContains(response, "id_room_gate_view_btn")
self.assertNotContains(response, "id_sig_deck")
def test_cost_current_shows_sig_overlay(self):
response = self.client.get(self.url)
self.assertContains(response, "id_sig_deck")
self.assertNotContains(response, "id_room_gate_view_btn")
def test_cost_lapsed_supersedes_cast_sky(self):
self.room.table_status = Room.SKY_SELECT
self.room.save()
self._lapse_viewer()
response = self.client.get(self.url)
self.assertContains(response, "id_room_gate_view_btn")
self.assertNotContains(response, "id_pick_sky_btn")
def test_cost_current_shows_cast_sky(self):
self.room.table_status = Room.SKY_SELECT
self.room.save()
response = self.client.get(self.url)
self.assertContains(response, "id_pick_sky_btn")
self.assertNotContains(response, "id_room_gate_view_btn")
def test_cost_lapsed_supersedes_draw_sea(self):
self.room.table_status = Room.SKY_SELECT
self.room.save()
pc = TableSeat.objects.get(room=self.room, gamer=self.founder)
Character.objects.create(seat=pc, confirmed_at=timezone.now())
self._lapse_viewer()
response = self.client.get(self.url)
self.assertContains(response, "id_room_gate_view_btn")
self.assertNotContains(response, "id_pick_sea_btn")
def test_cost_lapsed_supersedes_scan_sigs(self):
self.room.table_status = Room.ROLE_SELECT # roles assigned → SCAN SIGS
self.room.save()
self._lapse_viewer()
response = self.client.get(self.url)
self.assertContains(response, "id_room_gate_view_btn")
self.assertNotContains(response, "id_pick_sigs_btn")
def test_cost_current_shows_scan_sigs(self):
self.room.table_status = Room.ROLE_SELECT
self.room.save()
response = self.client.get(self.url)
self.assertContains(response, "id_pick_sigs_btn")
self.assertNotContains(response, "id_room_gate_view_btn")
class RoomRoleStackGraceTest(TestCase):
"""The gamer's own ROLE card-stack pick survives a lapsed token cost
(deposit-privilege grace) — only SCAN SIGS + later phases get GATE VIEW
(user-spec 2026-05-31)."""
def setUp(self):
self.founder = User.objects.create(email="founder@test.io")
self.room = Room.objects.create(name="Role Room", owner=self.founder)
gamers = [self.founder] + [
User.objects.create(email=f"g{i}@test.io") for i in range(2, 7)
]
for i, gamer in enumerate(gamers, start=1):
slot = self.room.gate_slots.get(slot_number=i)
slot.gamer = gamer
slot.status = GateSlot.FILLED
slot.save()
TableSeat.objects.create(room=self.room, gamer=gamer, slot_number=i)
self.room.gate_status = Room.OPEN
self.room.table_status = Room.ROLE_SELECT
self.room.save()
self.client.force_login(self.founder)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_card_stack_kept_when_cost_lapsed(self):
slot = self.room.gate_slots.get(slot_number=1)
slot.filled_at = timezone.now() - timedelta(days=8)
slot.save()
response = self.client.get(self.url)
# ROLE pick (the gamer's own turn) stays available within grace…
self.assertContains(response, "card-stack")
self.assertContains(response, 'data-state="eligible"')
# …alongside the GATE VIEW supersession of the non-ROLE affordances.
self.assertContains(response, "id_room_gate_view_btn")
class ExpireLapsedSeatsTest(TestCase):
"""Auto-BYE: a seat whose token cost lapsed past the renewal grace
(filled_at + 2*renewal_period) is freed — GateSlot blanked, TableSeat
blanked (row kept for seat-count integrity), SLOT_RETURNED recorded,
room flagged RENEWAL_DUE (gamer needed). Lazy on room/gate-view access.
NULL filled_at never expires (protects fixtures / RESERVED holds)."""
def setUp(self):
self.founder = User.objects.create(email="founder@test.io", username="founder")
self.room = Room.objects.create(
name="Lapse Room", owner=self.founder,
renewal_period=timedelta(days=7),
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
self.slot = self.room.gate_slots.get(slot_number=1)
self.slot.gamer = self.founder
self.slot.status = GateSlot.FILLED
self.slot.debited_token_type = Token.FREE
self.slot.save()
self.seat = TableSeat.objects.create(
room=self.room, gamer=self.founder, slot_number=1, role="PC",
)
self.client.force_login(self.founder)
def _set_filled(self, when):
self.slot.filled_at = when
self.slot.save()
def test_frees_gateslot_past_grace(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now() - timedelta(days=15)) # > 2S = 14d
_expire_lapsed_seats(self.room)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
self.assertIsNone(self.slot.gamer)
self.assertIsNone(self.slot.filled_at)
self.assertIsNone(self.slot.debited_token_type)
def test_blanks_table_seat(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now() - timedelta(days=15))
_expire_lapsed_seats(self.room)
self.seat.refresh_from_db()
self.assertIsNone(self.seat.gamer)
self.assertIsNone(self.seat.role)
def test_noop_within_grace(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now() - timedelta(days=8)) # in grace [7,14)
_expire_lapsed_seats(self.room)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.FILLED)
def test_noop_within_cost_window(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now())
_expire_lapsed_seats(self.room)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.FILLED)
def test_ignores_null_filled_at(self):
from apps.epic.views import _expire_lapsed_seats
self.slot.filled_at = None
self.slot.save()
_expire_lapsed_seats(self.room)
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.FILLED)
def test_sets_renewal_due(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now() - timedelta(days=15))
_expire_lapsed_seats(self.room)
self.room.refresh_from_db()
self.assertEqual(self.room.gate_status, Room.RENEWAL_DUE)
def test_records_slot_returned(self):
from apps.epic.views import _expire_lapsed_seats
self._set_filled(timezone.now() - timedelta(days=15))
_expire_lapsed_seats(self.room)
self.assertTrue(
self.room.events.filter(
actor=self.founder, verb=GameEvent.SLOT_RETURNED).exists())
def test_room_view_runs_expiry_on_access(self):
self._set_filled(timezone.now() - timedelta(days=15))
self.client.get(reverse("epic:room", args=[self.room.id]))
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
def test_room_gate_runs_expiry_on_access(self):
self._set_filled(timezone.now() - timedelta(days=15))
self.client.get(reverse("epic:room_gate", args=[self.room.id]))
self.slot.refresh_from_db()
self.assertEqual(self.slot.status, GateSlot.EMPTY)
def test_gamer_needed_stub_renders(self):
self.room.gate_status = Room.RENEWAL_DUE
self.room.save()
response = self.client.get(reverse("epic:room", args=[self.room.id]))
self.assertContains(response, "id_gamer_needed")
class ExpireLapsedRoomSeatsCommandTest(TestCase):
"""Cron backstop for rooms nobody reopens — `expire_lapsed_room_seats`
runs the same `_expire_lapsed_seats` sweep the lazy view path uses, for
mid-game tables left idle past the grace window."""
def _room_with_filled_slot(self, email, filled_at):
owner = User.objects.create(email=email)
room = Room.objects.create(
name="Cron Room", owner=owner, renewal_period=timedelta(days=7),
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
slot = room.gate_slots.get(slot_number=1)
slot.gamer = owner
slot.status = GateSlot.FILLED
slot.filled_at = filled_at
slot.debited_token_type = Token.FREE
slot.save()
return room, slot
def test_command_frees_lapsed_seat_and_sets_renewal_due(self):
from django.core.management import call_command
room, slot = self._room_with_filled_slot(
"lapsed@test.io", timezone.now() - timedelta(days=15))
call_command("expire_lapsed_room_seats")
slot.refresh_from_db()
room.refresh_from_db()
self.assertEqual(slot.status, GateSlot.EMPTY)
self.assertEqual(room.gate_status, Room.RENEWAL_DUE)
def test_command_noop_within_grace(self):
from django.core.management import call_command
room, slot = self._room_with_filled_slot(
"ingrace@test.io", timezone.now() - timedelta(days=8))
call_command("expire_lapsed_room_seats")
slot.refresh_from_db()
self.assertEqual(slot.status, GateSlot.FILLED)
class RoomScrollOfEventsTest(TestCase):
"""The table-hex aperture gains a 2nd scroll-snap section: scrolling down
past the hex reveals the room's provenance feed (the same GameEvent stream
the Billscroll page shows). Available from Role Select onwards — the gate
phase (no `table_status`) shows no scroll. Mirrors my_sky's wheel<->form
binary scroll-snap toggle; built to DRY-template onto my_sea next.
The shared row list comes from `core/_partials/_scroll.html` (the same
partial the Billscroll page uses), so the feed renders identically here."""
def setUp(self):
self.user = User.objects.create(email="founder@test.io", username="disco")
self.client.force_login(self.user)
self.room = Room.objects.create(
name="Whataburgher", owner=self.user,
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
TableSeat.objects.create(
room=self.room, gamer=self.user, slot_number=1, role="PC",
)
# A welcome line (system-authored) + a deposit line — feed content.
GameEvent.objects.create(room=self.room, verb=GameEvent.ROOM_CREATED)
GameEvent.objects.create(
room=self.room, actor=self.user, verb=GameEvent.SLOT_FILLED,
data={"token_type": "carte", "token_display": "Carte Blanche",
"slot_number": 1, "renewal_days": 7},
)
self.url = reverse("epic:room", kwargs={"room_id": self.room.id})
def test_aperture_wraps_hex_and_scroll_panes(self):
content = self.client.get(self.url).content.decode()
self.assertIn("room-aperture", content)
self.assertIn("room-hex-pane", content)
self.assertIn("room-scroll-pane", content)
def test_aperture_is_scrollable_from_role_select(self):
# `is-scrollable` engages the binary scroll-snap (2 panes present).
content = self.client.get(self.url).content.decode()
self.assertIn("room-aperture is-scrollable", content)
def test_scroll_pane_renders_the_event_feed(self):
content = self.client.get(self.url).content.decode()
self.assertIn("id_drama_scroll", content)
self.assertIn("Welcome to Whataburgher!", content)
self.assertIn("deposits a Carte Blanche for slot 1", content)
def test_scroll_feed_scoped_to_this_room(self):
other = Room.objects.create(
name="Elsewhere", owner=self.user,
gate_status=Room.OPEN, table_status=Room.ROLE_SELECT,
)
GameEvent.objects.create(room=other, verb=GameEvent.ROOM_CREATED)
content = self.client.get(self.url).content.decode()
self.assertNotIn("Welcome to Elsewhere!", content)
def test_no_scroll_pane_in_gate_phase(self):
"""The gate phase (table_status unset) shows no scroll — the feed is
reachable only from Role Select onwards."""
gate_room = Room.objects.create(name="Gatehouse", owner=self.user)
content = self.client.get(
reverse("epic:gatekeeper", args=[gate_room.id])
).content.decode()
self.assertNotIn("room-scroll-pane", content)
self.assertNotIn("is-scrollable", content)