Files
scrabble-game/backend/internal/game/helpers_test.go
T
Ilia Denisov 751e74b14f
Tests · Go / test (push) Successful in 6s
Tests · Integration / integration (push) Successful in 8s
Tests · Go / test (pull_request) Successful in 5s
Tests · Integration / integration (pull_request) Successful in 8s
Stage 3: game domain (lifecycle, journal+cache, hint, word-check, GCG, stats)
internal/game drives the engine over a single match and owns everything the
engine does not: event-sourced persistence (a games row + an append-only decoded
move journal, with the live engine.Game kept warm in a cache and rebuilt by
replay on a miss), the play/pass/exchange/resign transitions with
validate-at-submit scoring, an unlimited score/legality preview, the hint
(per-game allowance + profile wallet), the word-check tool with complaint
capture, per-player game state, history and GCG export (Poslfit dialect), and a
background turn-timeout sweeper that auto-resigns overdue turns honouring each
player's daily away window. Like Stages 1-2 it is a service/store layer with no
HTTP; the gateway surface lands in Stage 6.

Engine: additive decoded domain API (Direction, SubmitPlay/SubmitExchange/
EvaluatePlay/HintView/Hand, MoveRecord.{Dir,MainRow,MainCol}, Registry.Lookup,
ParseVariant) so internal/game never imports scrabble-solver; and a Resign fix
so the resigner keeps their score yet never wins (the other player wins a
two-player game). Timeout reuses Resign.

Persistence: migration 00002 adds games, game_players, game_moves, complaints,
account_stats and extends accounts with away_start/away_end/hint_balance; go-jet
regenerated. account gained SpendHint. Config adds BACKEND_DICT_DIR (required),
BACKEND_DICT_VERSION, BACKEND_GAME_TIMEOUT_SWEEP_INTERVAL, BACKEND_GAME_CACHE_TTL;
main loads the registry at boot (hard dependency) and starts the sweeper.

Tests: engine resign + decoded-API tests; game unit tests (GCG, away-window
boundaries, hint budget, cache, keyed mutex, payload); inttest integration
(lifecycle, replay equivalence, timeout sweep with away grace, resign stats,
hint policy, word-check/complaint, per-game-lock). Docs (PLAN, ARCHITECTURE,
FUNCTIONAL +_ru, TESTING, README) updated.
2026-06-02 17:33:49 +02:00

140 lines
3.5 KiB
Go

package game
import (
"sync"
"testing"
"time"
"github.com/google/uuid"
"scrabble/backend/internal/engine"
)
func TestPayloadPlayRoundTrip(t *testing.T) {
rec := engine.MoveRecord{
Action: engine.ActionPlay, Dir: engine.Vertical, MainRow: 3, MainCol: 4,
Tiles: []engine.TileRecord{{Row: 3, Col: 4, Letter: "q", Blank: true}, {Row: 4, Col: 4, Letter: "i"}},
Words: []string{"qi"},
}
s, err := buildPayload(rec, []string{"q", "i", "?"}, nil).marshal()
if err != nil {
t.Fatalf("marshal: %v", err)
}
p, err := parsePayload(s)
if err != nil {
t.Fatalf("parse: %v", err)
}
if p.direction() != engine.Vertical || p.MainRow != 3 || p.MainCol != 4 {
t.Errorf("dir/anchor = %v/(%d,%d)", p.direction(), p.MainRow, p.MainCol)
}
tiles := p.tileRecords()
if len(tiles) != 2 || tiles[0].Letter != "q" || !tiles[0].Blank || tiles[1].Letter != "i" {
t.Errorf("tiles = %+v", tiles)
}
if len(p.Rack) != 3 || p.Rack[2] != "?" {
t.Errorf("rack = %v", p.Rack)
}
}
func TestPayloadExchangeRoundTrip(t *testing.T) {
rec := engine.MoveRecord{Action: engine.ActionExchange, Count: 2}
s, err := buildPayload(rec, []string{"a", "b", "c"}, []string{"a", "b"}).marshal()
if err != nil {
t.Fatalf("marshal: %v", err)
}
p, err := parsePayload(s)
if err != nil {
t.Fatalf("parse: %v", err)
}
if len(p.Exchanged) != 2 || p.Exchanged[0] != "a" {
t.Errorf("exchanged = %v", p.Exchanged)
}
if len(p.Tiles) != 0 || p.Dir != "" {
t.Errorf("exchange payload carried play fields: %+v", p)
}
}
func TestHintsRemaining(t *testing.T) {
cases := []struct{ allowance, used, wallet, want int }{
{1, 0, 3, 4},
{1, 1, 3, 3},
{1, 2, 3, 3}, // used past allowance clamps to 0
{0, 0, 5, 5},
{2, 1, 0, 1},
}
for _, c := range cases {
if got := hintsRemaining(c.allowance, c.used, c.wallet); got != c.want {
t.Errorf("hintsRemaining(%d,%d,%d) = %d, want %d", c.allowance, c.used, c.wallet, got, c.want)
}
}
}
func TestAllowedTimeout(t *testing.T) {
if !allowedTimeout(24 * time.Hour) {
t.Error("24h must be allowed")
}
if !allowedTimeout(5 * time.Minute) {
t.Error("5m must be allowed")
}
if allowedTimeout(7 * time.Minute) {
t.Error("7m must not be allowed")
}
if allowedTimeout(0) {
t.Error("zero must not be allowed")
}
}
func TestNormalizeWord(t *testing.T) {
if got := normalizeWord(" CaT \n"); got != "cat" {
t.Errorf("normalizeWord = %q, want cat", got)
}
}
func TestGameCacheEviction(t *testing.T) {
cur := time.Unix(1_700_000_000, 0)
cache := newGameCache(time.Hour, func() time.Time { return cur })
id := uuid.New()
cache.put(id, nil)
if _, ok := cache.get(id); !ok {
t.Fatal("game must be resident after put")
}
cur = cur.Add(30 * time.Minute)
cache.get(id) // refresh idle timer
cur = cur.Add(90 * time.Minute)
if n := cache.sweep(); n != 1 {
t.Errorf("sweep evicted %d, want 1", n)
}
if _, ok := cache.get(id); ok {
t.Error("game must be evicted after idle TTL")
}
if cache.size() != 0 {
t.Errorf("cache size = %d, want 0", cache.size())
}
}
func TestKeyedMutexSerializes(t *testing.T) {
km := newKeyedMutex()
id := uuid.New()
var counter int
var wg sync.WaitGroup
for i := 0; i < 200; i++ {
wg.Add(1)
go func() {
defer wg.Done()
unlock := km.lock(id)
counter++ // serialised; -race would flag a missing lock
unlock()
}()
}
wg.Wait()
if counter != 200 {
t.Errorf("counter = %d, want 200", counter)
}
km.mu.Lock()
left := len(km.locks)
km.mu.Unlock()
if left != 0 {
t.Errorf("lock map not cleaned up: %d entries left", left)
}
}