From 2df20c2840e29fb9cc034c69077621360e48e34d Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 11:48:47 -1000
Subject: [PATCH 1/9] refactor: adopt portigo infrastructure patterns
Add config package with build-tag-switched dev/prod environments,
structured logging via zerolog, Taskfile for dev workflow, golangci-lint
config, testutil package, and improved DB setup with proper SQLite
pragmas and cleanup. Rename sqlc output package from gen to repository.
Switch to allowlist .gitignore, Alpine+UPX+scratch Dockerfile, and
CI pipeline with test/lint gates before deploy.
---
.env.example | 15 ++-
.gitea/workflows/deploy.yml | 34 +++++-
.gitignore | 40 +++++--
.golangci.yml | 45 ++++++++
Dockerfile | 31 ++----
Taskfile.yml | 63 +++++++++++
config/config.go | 73 +++++++++++++
config/config_dev.go | 9 ++
config/config_prod.go | 9 ++
config/config_test_helper.go | 19 ++++
db/db.go | 57 ++++++++--
db/persister.go | 28 ++---
db/{gen => repository}/chat.sql.go | 2 +-
db/{gen => repository}/db.go | 2 +-
db/{gen => repository}/games.sql.go | 2 +-
db/{gen => repository}/models.go | 2 +-
db/{gen => repository}/snake_games.sql.go | 2 +-
db/{gen => repository}/users.sql.go | 2 +-
db/sqlc.yaml | 8 +-
go.mod | 27 ++---
go.sum | 62 +++++------
logging/log.go | 41 +++++++
logging/middleware.go | 126 ++++++++++++++++++++++
main.go | 52 +++++----
testutil/db.go | 47 ++++++++
testutil/sessions.go | 31 ++++++
ui/status.go | 8 +-
27 files changed, 694 insertions(+), 143 deletions(-)
create mode 100644 .golangci.yml
create mode 100644 Taskfile.yml
create mode 100644 config/config.go
create mode 100644 config/config_dev.go
create mode 100644 config/config_prod.go
create mode 100644 config/config_test_helper.go
rename db/{gen => repository}/chat.sql.go (98%)
rename db/{gen => repository}/db.go (96%)
rename db/{gen => repository}/games.sql.go (99%)
rename db/{gen => repository}/models.go (98%)
rename db/{gen => repository}/snake_games.sql.go (99%)
rename db/{gen => repository}/users.sql.go (98%)
create mode 100644 logging/log.go
create mode 100644 logging/middleware.go
create mode 100644 testutil/db.go
create mode 100644 testutil/sessions.go
diff --git a/.env.example b/.env.example
index 0643835..7ae4ff9 100644
--- a/.env.example
+++ b/.env.example
@@ -1,5 +1,16 @@
-# Application URL for invite links (defaults to https://games.adriatica.io)
+# Log level (TRACE, DEBUG, INFO, WARN, ERROR). Defaults to INFO.
+# LOG_LEVEL=DEBUG
+
+# SQLite database path. Defaults to data/c4.db.
+# DB_PATH=data/c4.db
+
+# Application URL for invite links. Defaults to https://games.adriatica.io.
# APP_URL=http://localhost:7331
-# Server port (defaults to 7331)
+# Server port. Defaults to 7331.
# PORT=7331
+
+# Goose CLI migration config (only needed for running goose manually)
+GOOSE_DRIVER=sqlite3
+GOOSE_DBSTRING=data/c4.db?_pragma=foreign_keys(1)&_pragma=journal_mode(WAL)
+GOOSE_MIGRATION_DIR=db/migrations
diff --git a/.gitea/workflows/deploy.yml b/.gitea/workflows/deploy.yml
index 1a314c7..f8a7271 100644
--- a/.gitea/workflows/deploy.yml
+++ b/.gitea/workflows/deploy.yml
@@ -1,14 +1,44 @@
-name: Deploy c4
+name: CI / Deploy
on:
push:
branches: [main]
+ pull_request:
env:
DEPLOY_DIR: /home/ryan/c4
jobs:
+ test:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+
+ - uses: actions/setup-go@v5
+ with:
+ go-version: "1.25"
+
+ - name: Run tests
+ run: go test ./...
+
+ lint:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v4
+
+ - uses: actions/setup-go@v5
+ with:
+ go-version: "1.25"
+
+ - name: Install golangci-lint
+ run: go install github.com/golangci/golangci-lint/v2/cmd/golangci-lint@latest
+
+ - name: Run linter
+ run: golangci-lint run
+
deploy:
+ if: github.ref == 'refs/heads/main' && github.event_name == 'push'
+ needs: [test, lint]
runs-on: games
steps:
- uses: actions/checkout@v4
@@ -21,8 +51,6 @@ jobs:
- name: Ensure data directory exists with correct ownership
run: |
mkdir -p $DEPLOY_DIR/data
- # UID 5 / GID 60 = games:games in the container (debian:bookworm-slim)
- sudo chown 5:60 $DEPLOY_DIR/data
- name: Rebuild and restart
run: cd $DEPLOY_DIR && docker compose up -d --build --remove-orphans
diff --git a/.gitignore b/.gitignore
index 58dd18e..9dca485 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,8 +1,34 @@
-c4
-c4.db
-data/
-.env
+# Allowlisting gitignore: ignore everything, then un-ignore what we track.
+# source: https://github.com/github/gitignore/blob/main/community/Golang/Go.AllowList.gitignore
-# Deploy artifacts
-c4-deploy-*.tar.gz
-c4-deploy-*_b64*.txt
+# Ignore everything
+*
+
+# But not these files...
+!.gitignore
+
+!*.go
+!*.sql
+!go.sum
+!go.mod
+!Taskfile.yml
+!sqlc.yaml
+!.golangci.yml
+!.gitea/workflows/*.yml
+
+!.env.example
+!LICENSE
+
+!assets/**/*
+
+# Generated CSS stays out of version control
+assets/css/output.css
+
+# Deploy scripts and configs
+!deploy/*.sh
+!deploy/*.service
+!docker-compose.yml
+!Dockerfile
+
+# ...even if they are in subdirectories
+!*/
diff --git a/.golangci.yml b/.golangci.yml
new file mode 100644
index 0000000..961c8c8
--- /dev/null
+++ b/.golangci.yml
@@ -0,0 +1,45 @@
+version: "2"
+
+linters:
+ default: standard
+ enable:
+ - errcheck
+ - govet
+ - staticcheck
+ - gosec
+ - bodyclose
+ - sqlclosecheck
+ - misspell
+ - errname
+ - copyloopvar
+
+ settings:
+ staticcheck:
+ checks:
+ - all
+ - "-ST1001" # dot imports
+ - "-ST1003" # naming conventions
+ gosec:
+ excludes:
+ - G104 # unhandled errors — redundant with errcheck
+ - G107 # HTTP requests with variable URLs — expected in a web app
+ - G115 # integer overflow conversion
+ - G301 # directory permissions 0750 — 0755 is standard for data dirs
+ - G404 # weak random — acceptable for game IDs and player IDs
+
+formatters:
+ enable:
+ - gofmt
+ - goimports
+
+ settings:
+ goimports:
+ local-prefixes:
+ - github.com/ryanhamamura/c4
+
+issues:
+ exclude-rules:
+ - path: _test\.go
+ linters:
+ - gosec
+ - errcheck
diff --git a/Dockerfile b/Dockerfile
index 08261c3..a22afc9 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,4 +1,6 @@
-FROM golang:1.25.4-bookworm AS build
+FROM docker.io/golang:1.25.4-alpine AS build
+
+RUN apk add --no-cache upx
WORKDIR /src
COPY go.mod go.sum ./
@@ -6,24 +8,11 @@ RUN go mod download
COPY . .
RUN go tool gotailwind -i assets/css/input.css -o assets/css/output.css --minify
-RUN CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -o /c4 .
+RUN --mount=type=cache,target=/root/.cache/go-build \
+ CGO_ENABLED=0 go build -ldflags="-s" -o /bin/c4 .
+RUN upx -9 -k /bin/c4
-FROM debian:bookworm-slim
-
-RUN apt-get update && \
- apt-get install -y --no-install-recommends ca-certificates wget && \
- rm -rf /var/lib/apt/lists/*
-
-COPY --from=build /c4 /usr/local/bin/c4
-
-WORKDIR /app
-RUN mkdir data && chown games:games data
-
-USER games
-
-EXPOSE 8080
-
-HEALTHCHECK --interval=30s --timeout=5s --start-period=5s --retries=3 \
- CMD wget -qO /dev/null http://localhost:8080/
-
-CMD ["c4"]
+FROM scratch
+ENV PORT=8080
+COPY --from=build /bin/c4 /
+ENTRYPOINT ["/c4"]
diff --git a/Taskfile.yml b/Taskfile.yml
new file mode 100644
index 0000000..f5b9d29
--- /dev/null
+++ b/Taskfile.yml
@@ -0,0 +1,63 @@
+version: "3"
+
+tasks:
+ build:styles:
+ desc: Build TailwindCSS styles
+ cmds:
+ - go tool gotailwind -i assets/css/input.css -o assets/css/output.css --minify
+ sources:
+ - "assets/css/input.css"
+ - "**/*.go"
+ generates:
+ - "assets/css/output.css"
+
+ build:
+ desc: Production build to bin/c4
+ cmds:
+ - go build -o bin/c4 .
+ deps:
+ - build:styles
+
+ live:styles:
+ desc: Watch and rebuild TailwindCSS styles
+ cmds:
+ - go tool gotailwind -i assets/css/input.css -o assets/css/output.css -w
+
+ live:server:
+ desc: Run server with hot-reload via air
+ cmds:
+ - |
+ go tool air \
+ -build.cmd "go build -tags=dev -o tmp/bin/c4 ." \
+ -build.bin "tmp/bin/c4" \
+ -build.exclude_dir "data,bin,tmp,deploy" \
+ -build.include_ext "go" \
+ -misc.clean_on_exit "true"
+
+ live:
+ desc: Dev mode with hot-reload
+ deps:
+ - live:styles
+ - live:server
+
+ test:
+ desc: Run the test suite
+ cmds:
+ - go test ./...
+
+ lint:
+ desc: Run golangci-lint
+ cmds:
+ - golangci-lint run
+
+ run:
+ desc: Build and run the server
+ cmds:
+ - ./bin/c4
+ deps:
+ - build
+
+ default:
+ desc: Run the default task (live)
+ cmds:
+ - task: live
diff --git a/config/config.go b/config/config.go
new file mode 100644
index 0000000..6c09cd9
--- /dev/null
+++ b/config/config.go
@@ -0,0 +1,73 @@
+// Package config provides build-tag-switched application configuration.
+// The global Config singleton is initialized at import time via init()
+// and can be overridden in tests with LoadForTest().
+package config
+
+import (
+ "os"
+ "sync"
+
+ "github.com/joho/godotenv"
+ "github.com/rs/zerolog"
+)
+
+type Environment string
+
+const (
+ Dev Environment = "dev"
+ Prod Environment = "prod"
+)
+
+type Config struct {
+ Environment Environment
+ Host string
+ Port string
+ LogLevel zerolog.Level
+ AppURL string
+ DBPath string
+}
+
+var (
+ Global *Config
+ once sync.Once
+)
+
+func init() {
+ once.Do(func() {
+ Global = Load()
+ })
+}
+
+func getEnv(key, fallback string) string {
+ if val, ok := os.LookupEnv(key); ok {
+ return val
+ }
+ return fallback
+}
+
+func loadBase() *Config {
+ godotenv.Load() //nolint:errcheck // .env file is optional
+
+ return &Config{
+ Host: getEnv("HOST", "0.0.0.0"),
+ Port: getEnv("PORT", "7331"),
+ LogLevel: func() zerolog.Level {
+ switch os.Getenv("LOG_LEVEL") {
+ case "TRACE":
+ return zerolog.TraceLevel
+ case "DEBUG":
+ return zerolog.DebugLevel
+ case "INFO":
+ return zerolog.InfoLevel
+ case "WARN":
+ return zerolog.WarnLevel
+ case "ERROR":
+ return zerolog.ErrorLevel
+ default:
+ return zerolog.InfoLevel
+ }
+ }(),
+ AppURL: getEnv("APP_URL", "https://games.adriatica.io"),
+ DBPath: getEnv("DB_PATH", "data/c4.db"),
+ }
+}
diff --git a/config/config_dev.go b/config/config_dev.go
new file mode 100644
index 0000000..53af2dd
--- /dev/null
+++ b/config/config_dev.go
@@ -0,0 +1,9 @@
+//go:build dev
+
+package config
+
+func Load() *Config {
+ cfg := loadBase()
+ cfg.Environment = Dev
+ return cfg
+}
diff --git a/config/config_prod.go b/config/config_prod.go
new file mode 100644
index 0000000..9598fb1
--- /dev/null
+++ b/config/config_prod.go
@@ -0,0 +1,9 @@
+//go:build !dev
+
+package config
+
+func Load() *Config {
+ cfg := loadBase()
+ cfg.Environment = Prod
+ return cfg
+}
diff --git a/config/config_test_helper.go b/config/config_test_helper.go
new file mode 100644
index 0000000..aec2a48
--- /dev/null
+++ b/config/config_test_helper.go
@@ -0,0 +1,19 @@
+package config
+
+import (
+ "github.com/rs/zerolog"
+)
+
+// LoadForTest sets config.Global to safe defaults without reading
+// environment variables or .env files. Call this in TestMain or at the
+// top of tests that import packages which depend on config.Global.
+func LoadForTest() {
+ Global = &Config{
+ Environment: Dev,
+ Host: "127.0.0.1",
+ Port: "0",
+ LogLevel: zerolog.WarnLevel,
+ AppURL: "http://localhost:0",
+ DBPath: ":memory:",
+ }
+}
diff --git a/db/db.go b/db/db.go
index 04bb9bb..3d7031a 100644
--- a/db/db.go
+++ b/db/db.go
@@ -1,33 +1,70 @@
+// Package db handles SQLite database setup, pragma configuration, and
+// goose migrations.
package db
import (
"database/sql"
"embed"
+ "errors"
+ "fmt"
+ "io/fs"
+ "log/slog"
+ "os"
+ "path/filepath"
"github.com/pressly/goose/v3"
_ "modernc.org/sqlite"
)
//go:embed migrations/*.sql
-var migrations embed.FS
+var MigrationFS embed.FS
var DB *sql.DB
-func Init(dbPath string) error {
+func Init(dbPath string) (func(), error) {
+ if err := os.MkdirAll(filepath.Dir(dbPath), 0o755); err != nil {
+ return nil, fmt.Errorf("creating data dir: %w", err)
+ }
+
+ // busy_timeout must be first because the connection needs to block on
+ // busy before WAL mode is set in case it hasn't been set already.
+ pragmas := "?_pragma=busy_timeout(10000)&_pragma=journal_mode(WAL)&_pragma=journal_size_limit(200000000)&_pragma=synchronous(NORMAL)&_pragma=foreign_keys(ON)&_pragma=temp_store(MEMORY)&_pragma=cache_size(-32000)"
var err error
- DB, err = sql.Open("sqlite", dbPath+"?_journal_mode=WAL&_busy_timeout=5000")
+ DB, err = goose.OpenDBWithDriver("sqlite", dbPath+pragmas)
if err != nil {
- return err
+ return nil, fmt.Errorf("opening database: %w", err)
}
- DB.SetMaxOpenConns(1)
- goose.SetBaseFS(migrations)
+ if err := DB.Ping(); err != nil {
+ return nil, errors.Join(fmt.Errorf("pinging database: %w", err), DB.Close())
+ }
+ slog.Info("db connected", "db", dbPath)
+
+ sub, err := fs.Sub(MigrationFS, "migrations")
+ if err != nil {
+ return nil, errors.Join(fmt.Errorf("migrations sub fs: %w", err), DB.Close())
+ }
+ goose.SetBaseFS(sub)
+
if err := goose.SetDialect("sqlite3"); err != nil {
- return err
+ return nil, errors.Join(fmt.Errorf("setting goose dialect: %w", err), DB.Close())
}
- if err := goose.Up(DB, "migrations"); err != nil {
- return err
+ if err := goose.Up(DB, "."); err != nil {
+ return nil, errors.Join(fmt.Errorf("running migrations: %w", err), DB.Close())
}
- return nil
+ if _, err := DB.Exec("PRAGMA optimize"); err != nil {
+ return nil, errors.Join(fmt.Errorf("pragma optimize: %w", err), DB.Close())
+ }
+
+ cleanup := func() {
+ if _, err := DB.Exec("PRAGMA optimize(0x10002)"); err != nil {
+ slog.Error("pragma optimize at shutdown", "error", err)
+ }
+ if err := DB.Close(); err != nil {
+ slog.Error("closing database", "error", err)
+ }
+ }
+
+ return cleanup, nil
}
diff --git a/db/persister.go b/db/persister.go
index 5e68b17..ddab719 100644
--- a/db/persister.go
+++ b/db/persister.go
@@ -5,17 +5,17 @@ import (
"database/sql"
"slices"
- "github.com/ryanhamamura/c4/db/gen"
+ "github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
"github.com/ryanhamamura/c4/snake"
"github.com/ryanhamamura/c4/ui"
)
type GamePersister struct {
- queries *gen.Queries
+ queries *repository.Queries
}
-func NewGamePersister(q *gen.Queries) *GamePersister {
+func NewGamePersister(q *repository.Queries) *GamePersister {
return &GamePersister{queries: q}
}
@@ -24,7 +24,7 @@ func (p *GamePersister) SaveGame(g *game.Game) error {
_, err := p.queries.GetGame(ctx, g.ID)
if err == sql.ErrNoRows {
- _, err = p.queries.CreateGame(ctx, gen.CreateGameParams{
+ _, err = p.queries.CreateGame(ctx, repository.CreateGameParams{
ID: g.ID,
Board: g.BoardToJSON(),
CurrentTurn: int64(g.CurrentTurn),
@@ -51,7 +51,7 @@ func (p *GamePersister) SaveGame(g *game.Game) error {
rematchGameID = sql.NullString{String: *g.RematchGameID, Valid: true}
}
- return p.queries.UpdateGame(ctx, gen.UpdateGameParams{
+ return p.queries.UpdateGame(ctx, repository.UpdateGameParams{
Board: g.BoardToJSON(),
CurrentTurn: int64(g.CurrentTurn),
Status: int64(g.Status),
@@ -100,7 +100,7 @@ func (p *GamePersister) SaveGamePlayer(gameID string, player *game.Player, slot
guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
}
- return p.queries.CreateGamePlayer(ctx, gen.CreateGamePlayerParams{
+ return p.queries.CreateGamePlayer(ctx, repository.CreateGamePlayerParams{
GameID: gameID,
UserID: userID,
GuestPlayerID: guestPlayerID,
@@ -144,10 +144,10 @@ func (p *GamePersister) DeleteGame(id string) error {
// SnakePersister implements snake.Persister
type SnakePersister struct {
- queries *gen.Queries
+ queries *repository.Queries
}
-func NewSnakePersister(q *gen.Queries) *SnakePersister {
+func NewSnakePersister(q *repository.Queries) *SnakePersister {
return &SnakePersister{queries: q}
}
@@ -167,7 +167,7 @@ func (p *SnakePersister) SaveSnakeGame(sg *snake.SnakeGame) error {
_, err := p.queries.GetSnakeGame(ctx, sg.ID)
if err == sql.ErrNoRows {
- _, err = p.queries.CreateSnakeGame(ctx, gen.CreateSnakeGameParams{
+ _, err = p.queries.CreateSnakeGame(ctx, repository.CreateSnakeGameParams{
ID: sg.ID,
Board: boardJSON,
Status: int64(sg.Status),
@@ -192,7 +192,7 @@ func (p *SnakePersister) SaveSnakeGame(sg *snake.SnakeGame) error {
rematchGameID = sql.NullString{String: *sg.RematchGameID, Valid: true}
}
- return p.queries.UpdateSnakeGame(ctx, gen.UpdateSnakeGameParams{
+ return p.queries.UpdateSnakeGame(ctx, repository.UpdateSnakeGameParams{
Board: boardJSON,
Status: int64(sg.Status),
WinnerUserID: winnerUserID,
@@ -247,7 +247,7 @@ func (p *SnakePersister) SaveSnakePlayer(gameID string, player *snake.Player) er
guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
}
- return p.queries.CreateSnakePlayer(ctx, gen.CreateSnakePlayerParams{
+ return p.queries.CreateSnakePlayer(ctx, repository.CreateSnakePlayerParams{
GameID: gameID,
UserID: userID,
GuestPlayerID: guestPlayerID,
@@ -290,15 +290,15 @@ func (p *SnakePersister) DeleteSnakeGame(id string) error {
}
type ChatPersister struct {
- queries *gen.Queries
+ queries *repository.Queries
}
-func NewChatPersister(q *gen.Queries) *ChatPersister {
+func NewChatPersister(q *repository.Queries) *ChatPersister {
return &ChatPersister{queries: q}
}
func (p *ChatPersister) SaveChatMessage(gameID string, msg ui.C4ChatMessage) error {
- return p.queries.CreateChatMessage(context.Background(), gen.CreateChatMessageParams{
+ return p.queries.CreateChatMessage(context.Background(), repository.CreateChatMessageParams{
GameID: gameID,
Nickname: msg.Nickname,
Color: int64(msg.Color),
diff --git a/db/gen/chat.sql.go b/db/repository/chat.sql.go
similarity index 98%
rename from db/gen/chat.sql.go
rename to db/repository/chat.sql.go
index 2d956cd..1694e02 100644
--- a/db/gen/chat.sql.go
+++ b/db/repository/chat.sql.go
@@ -3,7 +3,7 @@
// sqlc v1.30.0
// source: chat.sql
-package gen
+package repository
import (
"context"
diff --git a/db/gen/db.go b/db/repository/db.go
similarity index 96%
rename from db/gen/db.go
rename to db/repository/db.go
index d577e39..998bfd3 100644
--- a/db/gen/db.go
+++ b/db/repository/db.go
@@ -2,7 +2,7 @@
// versions:
// sqlc v1.30.0
-package gen
+package repository
import (
"context"
diff --git a/db/gen/games.sql.go b/db/repository/games.sql.go
similarity index 99%
rename from db/gen/games.sql.go
rename to db/repository/games.sql.go
index 6bb412d..7883526 100644
--- a/db/gen/games.sql.go
+++ b/db/repository/games.sql.go
@@ -3,7 +3,7 @@
// sqlc v1.30.0
// source: games.sql
-package gen
+package repository
import (
"context"
diff --git a/db/gen/models.go b/db/repository/models.go
similarity index 98%
rename from db/gen/models.go
rename to db/repository/models.go
index f539b79..de3f897 100644
--- a/db/gen/models.go
+++ b/db/repository/models.go
@@ -2,7 +2,7 @@
// versions:
// sqlc v1.30.0
-package gen
+package repository
import (
"database/sql"
diff --git a/db/gen/snake_games.sql.go b/db/repository/snake_games.sql.go
similarity index 99%
rename from db/gen/snake_games.sql.go
rename to db/repository/snake_games.sql.go
index ec1d547..1c80c2b 100644
--- a/db/gen/snake_games.sql.go
+++ b/db/repository/snake_games.sql.go
@@ -3,7 +3,7 @@
// sqlc v1.30.0
// source: snake_games.sql
-package gen
+package repository
import (
"context"
diff --git a/db/gen/users.sql.go b/db/repository/users.sql.go
similarity index 98%
rename from db/gen/users.sql.go
rename to db/repository/users.sql.go
index 5adfa6e..4654293 100644
--- a/db/gen/users.sql.go
+++ b/db/repository/users.sql.go
@@ -3,7 +3,7 @@
// sqlc v1.30.0
// source: users.sql
-package gen
+package repository
import (
"context"
diff --git a/db/sqlc.yaml b/db/sqlc.yaml
index 0c1840b..47ad0a3 100644
--- a/db/sqlc.yaml
+++ b/db/sqlc.yaml
@@ -5,5 +5,9 @@ sql:
schema: "migrations"
gen:
go:
- package: "gen"
- out: "gen"
+ package: "repository"
+ out: "repository"
+ emit_db_tags: true
+ emit_json_tags: true
+ emit_result_struct_pointers: true
+ emit_pointers_for_null_types: true
diff --git a/go.mod b/go.mod
index 2c8644a..5b66895 100644
--- a/go.mod
+++ b/go.mod
@@ -3,27 +3,28 @@ module github.com/ryanhamamura/c4
go 1.25.4
require (
+ github.com/alexedwards/scs/sqlite3store v0.0.0-20251002162104-209de6e426de
+ github.com/alexedwards/scs/v2 v2.9.0
github.com/google/uuid v1.6.0
github.com/joho/godotenv v1.5.1
- github.com/pressly/goose/v3 v3.26.0
+ github.com/pressly/goose/v3 v3.27.0
+ github.com/rs/zerolog v1.34.0
github.com/ryanhamamura/via v0.23.0
- golang.org/x/crypto v0.47.0
- modernc.org/sqlite v1.44.0
+ golang.org/x/crypto v0.48.0
+ modernc.org/sqlite v1.46.1
)
require (
github.com/CAFxX/httpcompression v0.0.9 // indirect
- github.com/alexedwards/scs/sqlite3store v0.0.0-20251002162104-209de6e426de // indirect
- github.com/alexedwards/scs/v2 v2.9.0 // indirect
github.com/andybalholm/brotli v1.2.0 // indirect
github.com/antithesishq/antithesis-sdk-go v0.5.0 // indirect
github.com/cenkalti/backoff v2.2.1+incompatible // indirect
github.com/delaneyj/toolbelt v0.9.1 // indirect
github.com/dustin/go-humanize v1.0.1 // indirect
github.com/google/go-tpm v0.9.7 // indirect
- github.com/hookenz/gotailwind/v4 v4.1.18 // indirect
- github.com/klauspost/compress v1.18.2 // indirect
- github.com/mattn/go-colorable v0.1.13 // indirect
+ github.com/hookenz/gotailwind/v4 v4.2.1 // indirect
+ github.com/klauspost/compress v1.18.4 // indirect
+ github.com/mattn/go-colorable v0.1.14 // indirect
github.com/mattn/go-isatty v0.0.20 // indirect
github.com/mfridman/interpolate v0.0.2 // indirect
github.com/minio/highwayhash v1.0.4-0.20251030100505-070ab1a87a76 // indirect
@@ -33,18 +34,18 @@ require (
github.com/nats-io/nkeys v0.4.12 // indirect
github.com/nats-io/nuid v1.0.1 // indirect
github.com/ncruces/go-strftime v1.0.0 // indirect
+ github.com/pkg/errors v0.9.1 // indirect
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
- github.com/rs/zerolog v1.34.0 // indirect
github.com/sethvargo/go-retry v0.3.0 // indirect
github.com/starfederation/datastar-go v1.0.3 // indirect
github.com/valyala/bytebufferpool v1.0.0 // indirect
go.uber.org/multierr v1.11.0 // indirect
- golang.org/x/exp v0.0.0-20251125195548-87e1e737ad39 // indirect
- golang.org/x/sync v0.18.0 // indirect
- golang.org/x/sys v0.40.0 // indirect
+ golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa // indirect
+ golang.org/x/sync v0.19.0 // indirect
+ golang.org/x/sys v0.41.0 // indirect
golang.org/x/time v0.14.0 // indirect
maragu.dev/gomponents v1.2.0 // indirect
- modernc.org/libc v1.67.4 // indirect
+ modernc.org/libc v1.68.0 // indirect
modernc.org/mathutil v1.7.1 // indirect
modernc.org/memory v1.11.0 // indirect
)
diff --git a/go.sum b/go.sum
index 4adb164..e30bc87 100644
--- a/go.sum
+++ b/go.sum
@@ -31,16 +31,17 @@ github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs4luLUK2k=
github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM=
-github.com/hookenz/gotailwind/v4 v4.1.18 h1:1h3XwTVx1dEBm6A0bcosAplNCde+DCmVJG0arLy5fBE=
-github.com/hookenz/gotailwind/v4 v4.1.18/go.mod h1:IfiJtdp8ExV9HV2XUiVjRBvB3QewVXVKWoAGEcpjfNE=
+github.com/hookenz/gotailwind/v4 v4.2.1 h1:FpZLtAAbHH7wMvyGYT+01vTLFITGMGZGMtEbp7dd2dM=
+github.com/hookenz/gotailwind/v4 v4.2.1/go.mod h1:IfiJtdp8ExV9HV2XUiVjRBvB3QewVXVKWoAGEcpjfNE=
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
github.com/klauspost/compress v1.16.7/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE=
-github.com/klauspost/compress v1.18.2 h1:iiPHWW0YrcFgpBYhsA6D1+fqHssJscY/Tm/y2Uqnapk=
-github.com/klauspost/compress v1.18.2/go.mod h1:R0h/fSBs8DE4ENlcrlib3PsXS61voFxhIs2DeRhCvJ4=
+github.com/klauspost/compress v1.18.4 h1:RPhnKRAQ4Fh8zU2FY/6ZFDwTVTxgJ/EMydqSTzE9a2c=
+github.com/klauspost/compress v1.18.4/go.mod h1:R0h/fSBs8DE4ENlcrlib3PsXS61voFxhIs2DeRhCvJ4=
github.com/klauspost/pgzip v1.2.6/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs=
-github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
+github.com/mattn/go-colorable v0.1.14 h1:9A9LHSqF/7dyVVX6g0U9cwm9pG3kP9gSzcuIPHPsaIE=
+github.com/mattn/go-colorable v0.1.14/go.mod h1:6LmQG8QLFO4G5z1gPvYEzlUgJ2wF+stgPZH1UqBm1s8=
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
@@ -65,21 +66,20 @@ github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OS
github.com/ncruces/go-strftime v1.0.0 h1:HMFp8mLCTPp341M/ZnA4qaf7ZlsbTc+miZjCLOFAw7w=
github.com/ncruces/go-strftime v1.0.0/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
github.com/pierrec/lz4/v4 v4.1.18/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4=
-github.com/pierrec/lz4/v4 v4.1.22 h1:cKFw6uJDK+/gfw5BcDL0JL5aBsAFdsIT18eRtLj7VIU=
-github.com/pierrec/lz4/v4 v4.1.22/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4=
+github.com/pierrec/lz4/v4 v4.1.25 h1:kocOqRffaIbU5djlIBr7Wh+cx82C0vtFb0fOurZHqD0=
+github.com/pierrec/lz4/v4 v4.1.25/go.mod h1:EoQMVJgeeEOMsCqCzqFm2O0cJvljX2nGZjcRIPL34O4=
+github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
-github.com/pressly/goose/v3 v3.26.0 h1:KJakav68jdH0WDvoAcj8+n61WqOIaPGgH0bJWS6jpmM=
-github.com/pressly/goose/v3 v3.26.0/go.mod h1:4hC1KrritdCxtuFsqgs1R4AU5bWtTAf+cnWvfhf2DNY=
+github.com/pressly/goose/v3 v3.27.0 h1:/D30gVTuQhu0WsNZYbJi4DMOsx1lNq+6SkLe+Wp59BM=
+github.com/pressly/goose/v3 v3.27.0/go.mod h1:3ZBeCXqzkgIRvrEMDkYh1guvtoJTU5oMMuDdkutoM78=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
github.com/rs/xid v1.6.0/go.mod h1:7XoLgs4eV+QndskICGsho+ADou8ySMSjJKDIan90Nz0=
github.com/rs/zerolog v1.34.0 h1:k43nTLIwcTVQAncfCw4KZ2VY6ukYoZaBPNOE8txlOeY=
github.com/rs/zerolog v1.34.0/go.mod h1:bJsvje4Z08ROH4Nhs5iH600c3IkWhwp44iRc54W6wYQ=
-github.com/ryanhamamura/via v0.21.2 h1:osR6peY/mZSl9SNPeEv6IvzGU0akkQfQzQJgA74+7mk=
-github.com/ryanhamamura/via v0.21.2/go.mod h1:rpJewNVG6tgginZN7Be3qqRuol70+v1sFCKD4UjHsQo=
github.com/ryanhamamura/via v0.23.0 h1:0e7nytisazcWq7uxs6T27GM3FwzosCMenkxJd+78Lko=
github.com/ryanhamamura/via v0.23.0/go.mod h1:rpJewNVG6tgginZN7Be3qqRuol70+v1sFCKD4UjHsQo=
github.com/sethvargo/go-retry v0.3.0 h1:EEt31A35QhrcRZtrYFDTBg91cqZVnFL2navjDrah2SE=
@@ -103,24 +103,24 @@ github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZ
github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
-golang.org/x/crypto v0.47.0 h1:V6e3FRj+n4dbpw86FJ8Fv7XVOql7TEwpHapKoMJ/GO8=
-golang.org/x/crypto v0.47.0/go.mod h1:ff3Y9VzzKbwSSEzWqJsJVBnWmRwRSHt/6Op5n9bQc4A=
-golang.org/x/exp v0.0.0-20251125195548-87e1e737ad39 h1:DHNhtq3sNNzrvduZZIiFyXWOL9IWaDPHqTnLJp+rCBY=
-golang.org/x/exp v0.0.0-20251125195548-87e1e737ad39/go.mod h1:46edojNIoXTNOhySWIWdix628clX9ODXwPsQuG6hsK0=
-golang.org/x/mod v0.30.0 h1:fDEXFVZ/fmCKProc/yAXXUijritrDzahmwwefnjoPFk=
-golang.org/x/mod v0.30.0/go.mod h1:lAsf5O2EvJeSFMiBxXDki7sCgAxEUcZHXoXMKT4GJKc=
-golang.org/x/sync v0.18.0 h1:kr88TuHDroi+UVf+0hZnirlk8o8T+4MrK6mr60WkH/I=
-golang.org/x/sync v0.18.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
+golang.org/x/crypto v0.48.0 h1:/VRzVqiRSggnhY7gNRxPauEQ5Drw9haKdM0jqfcCFts=
+golang.org/x/crypto v0.48.0/go.mod h1:r0kV5h3qnFPlQnBSrULhlsRfryS2pmewsg+XfMgkVos=
+golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa h1:Zt3DZoOFFYkKhDT3v7Lm9FDMEV06GpzjG2jrqW+QTE0=
+golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa/go.mod h1:K79w1Vqn7PoiZn+TkNpx3BUWUQksGO3JcVX6qIjytmA=
+golang.org/x/mod v0.33.0 h1:tHFzIWbBifEmbwtGz65eaWyGiGZatSrT9prnU8DbVL8=
+golang.org/x/mod v0.33.0/go.mod h1:swjeQEj+6r7fODbD2cqrnje9PnziFuw4bmLbBZFrQ5w=
+golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
+golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
-golang.org/x/sys v0.40.0 h1:DBZZqJ2Rkml6QMQsZywtnjnnGvHza6BTfYFWY9kjEWQ=
-golang.org/x/sys v0.40.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
+golang.org/x/sys v0.41.0 h1:Ivj+2Cp/ylzLiEU89QhWblYnOE9zerudt9Ftecq2C6k=
+golang.org/x/sys v0.41.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
golang.org/x/time v0.14.0 h1:MRx4UaLrDotUKUdCIqzPC48t1Y9hANFKIRpNx+Te8PI=
golang.org/x/time v0.14.0/go.mod h1:eL/Oa2bBBK0TkX57Fyni+NgnyQQN4LitPmob2Hjnqw4=
-golang.org/x/tools v0.39.0 h1:ik4ho21kwuQln40uelmciQPp9SipgNDdrafrYA4TmQQ=
-golang.org/x/tools v0.39.0/go.mod h1:JnefbkDPyD8UU2kI5fuf8ZX4/yUeh9W877ZeBONxUqQ=
+golang.org/x/tools v0.42.0 h1:uNgphsn75Tdz5Ji2q36v/nsFSfR/9BRFvqhGBaJGd5k=
+golang.org/x/tools v0.42.0/go.mod h1:Ma6lCIwGZvHK6XtgbswSoWroEkhugApmsXyrUmBhfr0=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
@@ -129,18 +129,18 @@ maragu.dev/gomponents v1.2.0 h1:H7/N5htz1GCnhu0HB1GasluWeU2rJZOYztVEyN61iTc=
maragu.dev/gomponents v1.2.0/go.mod h1:oEDahza2gZoXDoDHhw8jBNgH+3UR5ni7Ur648HORydM=
modernc.org/cc/v4 v4.27.1 h1:9W30zRlYrefrDV2JE2O8VDtJ1yPGownxciz5rrbQZis=
modernc.org/cc/v4 v4.27.1/go.mod h1:uVtb5OGqUKpoLWhqwNQo/8LwvoiEBLvZXIQ/SmO6mL0=
-modernc.org/ccgo/v4 v4.30.1 h1:4r4U1J6Fhj98NKfSjnPUN7Ze2c6MnAdL0hWw6+LrJpc=
-modernc.org/ccgo/v4 v4.30.1/go.mod h1:bIOeI1JL54Utlxn+LwrFyjCx2n2RDiYEaJVSrgdrRfM=
+modernc.org/ccgo/v4 v4.30.2 h1:4yPaaq9dXYXZ2V8s1UgrC3KIj580l2N4ClrLwnbv2so=
+modernc.org/ccgo/v4 v4.30.2/go.mod h1:yZMnhWEdW0qw3EtCndG1+ldRrVGS+bIwyWmAWzS0XEw=
modernc.org/fileutil v1.3.40 h1:ZGMswMNc9JOCrcrakF1HrvmergNLAmxOPjizirpfqBA=
modernc.org/fileutil v1.3.40/go.mod h1:HxmghZSZVAz/LXcMNwZPA/DRrQZEVP9VX0V4LQGQFOc=
modernc.org/gc/v2 v2.6.5 h1:nyqdV8q46KvTpZlsw66kWqwXRHdjIlJOhG6kxiV/9xI=
modernc.org/gc/v2 v2.6.5/go.mod h1:YgIahr1ypgfe7chRuJi2gD7DBQiKSLMPgBQe9oIiito=
-modernc.org/gc/v3 v3.1.1 h1:k8T3gkXWY9sEiytKhcgyiZ2L0DTyCQ/nvX+LoCljoRE=
-modernc.org/gc/v3 v3.1.1/go.mod h1:HFK/6AGESC7Ex+EZJhJ2Gni6cTaYpSMmU/cT9RmlfYY=
+modernc.org/gc/v3 v3.1.2 h1:ZtDCnhonXSZexk/AYsegNRV1lJGgaNZJuKjJSWKyEqo=
+modernc.org/gc/v3 v3.1.2/go.mod h1:HFK/6AGESC7Ex+EZJhJ2Gni6cTaYpSMmU/cT9RmlfYY=
modernc.org/goabi0 v0.2.0 h1:HvEowk7LxcPd0eq6mVOAEMai46V+i7Jrj13t4AzuNks=
modernc.org/goabi0 v0.2.0/go.mod h1:CEFRnnJhKvWT1c1JTI3Avm+tgOWbkOu5oPA8eH8LnMI=
-modernc.org/libc v1.67.4 h1:zZGmCMUVPORtKv95c2ReQN5VDjvkoRm9GWPTEPuvlWg=
-modernc.org/libc v1.67.4/go.mod h1:QvvnnJ5P7aitu0ReNpVIEyesuhmDLQ8kaEoyMjIFZJA=
+modernc.org/libc v1.68.0 h1:PJ5ikFOV5pwpW+VqCK1hKJuEWsonkIJhhIXyuF/91pQ=
+modernc.org/libc v1.68.0/go.mod h1:NnKCYeoYgsEqnY3PgvNgAeaJnso968ygU8Z0DxjoEc0=
modernc.org/mathutil v1.7.1 h1:GCZVGXdaN8gTqB1Mf/usp1Y/hSqgI2vAGGP4jZMCxOU=
modernc.org/mathutil v1.7.1/go.mod h1:4p5IwJITfppl0G4sUEDtCr4DthTaT47/N3aT6MhfgJg=
modernc.org/memory v1.11.0 h1:o4QC8aMQzmcwCK3t3Ux/ZHmwFPzE6hf2Y5LbkRs+hbI=
@@ -149,8 +149,8 @@ modernc.org/opt v0.1.4 h1:2kNGMRiUjrp4LcaPuLY2PzUfqM/w9N23quVwhKt5Qm8=
modernc.org/opt v0.1.4/go.mod h1:03fq9lsNfvkYSfxrfUhZCWPk1lm4cq4N+Bh//bEtgns=
modernc.org/sortutil v1.2.1 h1:+xyoGf15mM3NMlPDnFqrteY07klSFxLElE2PVuWIJ7w=
modernc.org/sortutil v1.2.1/go.mod h1:7ZI3a3REbai7gzCLcotuw9AC4VZVpYMjDzETGsSMqJE=
-modernc.org/sqlite v1.44.0 h1:YjCKJnzZde2mLVy0cMKTSL4PxCmbIguOq9lGp8ZvGOc=
-modernc.org/sqlite v1.44.0/go.mod h1:2Dq41ir5/qri7QJJJKNZcP4UF7TsX/KNeykYgPDtGhE=
+modernc.org/sqlite v1.46.1 h1:eFJ2ShBLIEnUWlLy12raN0Z1plqmFX9Qe3rjQTKt6sU=
+modernc.org/sqlite v1.46.1/go.mod h1:CzbrU2lSB1DKUusvwGz7rqEKIq+NUd8GWuBBZDs9/nA=
modernc.org/strutil v1.2.1 h1:UneZBkQA+DX2Rp35KcM69cSsNES9ly8mQWD71HKlOA0=
modernc.org/strutil v1.2.1/go.mod h1:EHkiggD70koQxjVdSBM3JKM7k6L0FbGE5eymy9i3B9A=
modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y=
diff --git a/logging/log.go b/logging/log.go
new file mode 100644
index 0000000..4cdaed3
--- /dev/null
+++ b/logging/log.go
@@ -0,0 +1,41 @@
+// Package logging configures zerolog and provides HTTP request logging middleware.
+package logging
+
+import (
+ "io"
+ stdlog "log"
+ "os"
+
+ "github.com/ryanhamamura/c4/config"
+
+ "github.com/rs/zerolog"
+ "github.com/rs/zerolog/log"
+ "github.com/rs/zerolog/pkgerrors"
+)
+
+func SetupLogger(env config.Environment, level zerolog.Level) *zerolog.Logger {
+ zerolog.ErrorStackFieldName = "stack_trace"
+ zerolog.ErrorStackMarshaler = pkgerrors.MarshalStack
+
+ zerolog.SetGlobalLevel(level)
+ zerolog.TimeFieldFormat = zerolog.TimeFormatUnix
+
+ var output io.Writer
+ switch env {
+ case config.Dev:
+ output = zerolog.ConsoleWriter{
+ Out: os.Stderr,
+ TimeFormat: "2006/01/02 15:04:05",
+ }
+ case config.Prod:
+ output = os.Stderr
+ }
+
+ logger := zerolog.New(output).With().Timestamp().Stack().Logger()
+ zerolog.DefaultContextLogger = &logger
+ log.Logger = logger
+
+ stdlog.SetFlags(0)
+ stdlog.SetOutput(logger)
+ return &logger
+}
diff --git a/logging/middleware.go b/logging/middleware.go
new file mode 100644
index 0000000..dd6404e
--- /dev/null
+++ b/logging/middleware.go
@@ -0,0 +1,126 @@
+package logging
+
+import (
+ "fmt"
+ "net/http"
+ "time"
+
+ "github.com/ryanhamamura/c4/config"
+
+ "github.com/rs/zerolog"
+ "github.com/rs/zerolog/log"
+)
+
+const (
+ ansiReset = "\033[0m"
+ ansiBrightRed = "\033[31;1m"
+ ansiBrightGreen = "\033[32;1m"
+ ansiBrightYellow = "\033[33;1m"
+ ansiBrightMagenta = "\033[35;1m"
+ ansiBrightCyan = "\033[36;1m"
+ ansiGreen = "\033[32m"
+ ansiYellow = "\033[33m"
+ ansiRed = "\033[31m"
+)
+
+func colorStatus(status int, useColor bool) string {
+ s := fmt.Sprintf("%d", status)
+ if !useColor {
+ return s
+ }
+ switch {
+ case status < 200:
+ return ansiBrightGreen + s + ansiReset
+ case status < 300:
+ return ansiBrightGreen + s + ansiReset
+ case status < 400:
+ return ansiBrightCyan + s + ansiReset
+ case status < 500:
+ return ansiBrightYellow + s + ansiReset
+ default:
+ return ansiBrightRed + s + ansiReset
+ }
+}
+
+func colorMethod(method string, useColor bool) string {
+ if !useColor {
+ return method
+ }
+ return ansiBrightMagenta + method + ansiReset
+}
+
+func colorLatency(d time.Duration, useColor bool) string {
+ s := d.String()
+ if !useColor {
+ return s
+ }
+ switch {
+ case d < 500*time.Millisecond:
+ return ansiGreen + s + ansiReset
+ case d < 5*time.Second:
+ return ansiYellow + s + ansiReset
+ default:
+ return ansiRed + s + ansiReset
+ }
+}
+
+type responseWriter struct {
+ http.ResponseWriter
+ status int
+}
+
+func (rw *responseWriter) WriteHeader(code int) {
+ rw.status = code
+ rw.ResponseWriter.WriteHeader(code)
+}
+
+func RequestLogger(logger *zerolog.Logger, env config.Environment) func(http.Handler) http.Handler {
+ return func(next http.Handler) http.Handler {
+ return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
+ start := time.Now()
+ rw := &responseWriter{ResponseWriter: w}
+
+ next.ServeHTTP(rw, r)
+
+ status := rw.status
+ if status == 0 {
+ status = http.StatusOK
+ }
+ l := log.Ctx(r.Context())
+ if l.GetLevel() == zerolog.Disabled {
+ l = logger
+ }
+
+ var evt *zerolog.Event
+ switch {
+ case status < 400:
+ evt = l.Info()
+ case status < 500:
+ evt = l.Warn()
+ case status < 600:
+ evt = l.Error()
+ default:
+ evt = l.Info()
+ }
+
+ latency := time.Since(start)
+ switch env {
+ case config.Dev:
+ useColor := true
+ evt.Msg(fmt.Sprintf("%s %s %s [%s]",
+ colorStatus(status, useColor),
+ colorMethod(r.Method, useColor),
+ r.URL.Path,
+ colorLatency(latency, useColor),
+ ))
+ default:
+ evt.
+ Int("status", status).
+ Str("method", r.Method).
+ Str("path", r.URL.Path).
+ Dur("latency", latency).
+ Msg("request")
+ }
+ })
+ }
+}
diff --git a/main.go b/main.go
index fa6bce1..e324692 100644
--- a/main.go
+++ b/main.go
@@ -8,20 +8,20 @@ import (
"encoding/hex"
"encoding/json"
"io/fs"
- "log"
- "os"
"sync"
"time"
- "github.com/google/uuid"
- "github.com/joho/godotenv"
-
"github.com/ryanhamamura/c4/auth"
+ "github.com/ryanhamamura/c4/config"
"github.com/ryanhamamura/c4/db"
- "github.com/ryanhamamura/c4/db/gen"
+ "github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
+ "github.com/ryanhamamura/c4/logging"
"github.com/ryanhamamura/c4/snake"
"github.com/ryanhamamura/c4/ui"
+
+ "github.com/google/uuid"
+ "github.com/rs/zerolog/log"
"github.com/ryanhamamura/via"
"github.com/ryanhamamura/via/h"
)
@@ -29,7 +29,7 @@ import (
var (
store = game.NewGameStore()
snakeStore = snake.NewSnakeStore()
- queries *gen.Queries
+ queries *repository.Queries
chatPersister *db.ChatPersister
)
@@ -43,39 +43,35 @@ func DaisyUIPlugin(v *via.V) {
v.AppendToHead(h.Link(h.Rel("stylesheet"), h.Href("/assets/css/output.css?v="+version)))
}
-func port() string {
- if p := os.Getenv("PORT"); p != "" {
- return p
- }
- return "7331"
-}
-
func main() {
- _ = godotenv.Load()
+ cfg := config.Global
+ logger := logging.SetupLogger(cfg.Environment, cfg.LogLevel)
- if err := os.MkdirAll("data", 0o755); err != nil {
- log.Fatal(err)
+ cleanupDB, err := db.Init(cfg.DBPath)
+ if err != nil {
+ log.Fatal().Err(err).Msg("initializing database")
}
- if err := db.Init("data/c4.db"); err != nil {
- log.Fatal(err)
- }
- queries = gen.New(db.DB)
+ defer cleanupDB()
+
+ queries = repository.New(db.DB)
store.SetPersister(db.NewGamePersister(queries))
snakeStore.SetPersister(db.NewSnakePersister(queries))
chatPersister = db.NewChatPersister(queries)
sessionManager, err := via.NewSQLiteSessionManager(db.DB)
if err != nil {
- log.Fatal(err)
+ log.Fatal().Err(err).Msg("creating session manager")
}
+ _ = logger
+
v := via.New()
v.Config(via.Options{
- LogLevel: via.LogLevelDebug,
- DocumentTitle: "Game Lobby",
- ServerAddress: ":" + port(),
- SessionManager: sessionManager,
- Plugins: []via.Plugin{DaisyUIPlugin},
+ LogLevel: via.LogLevelDebug,
+ DocumentTitle: "Game Lobby",
+ ServerAddress: ":" + cfg.Port,
+ SessionManager: sessionManager,
+ Plugins: []via.Plugin{DaisyUIPlugin},
})
subFS, _ := fs.Sub(assets, "assets")
@@ -315,7 +311,7 @@ func main() {
ctx := context.Background()
id := uuid.New().String()
- user, err := queries.CreateUser(ctx, gen.CreateUserParams{
+ user, err := queries.CreateUser(ctx, repository.CreateUserParams{
ID: id,
Username: username.String(),
PasswordHash: hash,
diff --git a/testutil/db.go b/testutil/db.go
new file mode 100644
index 0000000..d2cb2f9
--- /dev/null
+++ b/testutil/db.go
@@ -0,0 +1,47 @@
+// Package testutil provides composable test helpers for spinning up
+// real infrastructure (in-memory SQLite, session managers) in
+// integration tests.
+package testutil
+
+import (
+ "database/sql"
+ "io/fs"
+ "testing"
+
+ "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/db/repository"
+
+ "github.com/pressly/goose/v3"
+ _ "modernc.org/sqlite"
+)
+
+// NewTestDB opens an in-memory SQLite database with the same pragmas as
+// production, runs all goose migrations, and returns the raw connection
+// alongside the sqlc Queries handle. The database is closed automatically
+// when the test finishes.
+func NewTestDB(t *testing.T) (*sql.DB, *repository.Queries) {
+ t.Helper()
+
+ pragmas := "?_pragma=busy_timeout(10000)&_pragma=journal_mode(WAL)&_pragma=journal_size_limit(200000000)&_pragma=synchronous(NORMAL)&_pragma=foreign_keys(ON)&_pragma=temp_store(MEMORY)&_pragma=cache_size(-32000)"
+ database, err := goose.OpenDBWithDriver("sqlite", ":memory:"+pragmas)
+ if err != nil {
+ t.Fatalf("open test database: %v", err)
+ }
+ t.Cleanup(func() { database.Close() }) //nolint:errcheck // test cleanup
+
+ if err := database.Ping(); err != nil {
+ t.Fatalf("ping test database: %v", err)
+ }
+
+ sub, err := fs.Sub(db.MigrationFS, "migrations")
+ if err != nil {
+ t.Fatalf("migrations sub fs: %v", err)
+ }
+ goose.SetBaseFS(sub)
+
+ if err := goose.Up(database, "."); err != nil {
+ t.Fatalf("run migrations: %v", err)
+ }
+
+ return database, repository.New(database)
+}
diff --git a/testutil/sessions.go b/testutil/sessions.go
new file mode 100644
index 0000000..5753f7d
--- /dev/null
+++ b/testutil/sessions.go
@@ -0,0 +1,31 @@
+package testutil
+
+import (
+ "database/sql"
+ "net/http"
+ "testing"
+ "time"
+
+ "github.com/alexedwards/scs/sqlite3store"
+ "github.com/alexedwards/scs/v2"
+)
+
+// NewTestSessionManager creates an SCS session manager backed by the
+// provided SQLite database. The background cleanup goroutine is stopped
+// automatically when the test finishes.
+func NewTestSessionManager(t *testing.T, db *sql.DB) *scs.SessionManager {
+ t.Helper()
+
+ store := sqlite3store.New(db)
+ t.Cleanup(func() { store.StopCleanup() })
+
+ sm := scs.New()
+ sm.Store = store
+ sm.Lifetime = 30 * 24 * time.Hour
+ sm.Cookie.Path = "/"
+ sm.Cookie.HttpOnly = true
+ sm.Cookie.Secure = false
+ sm.Cookie.SameSite = http.SameSiteLaxMode
+
+ return sm
+}
diff --git a/ui/status.go b/ui/status.go
index 37a0e8c..3fb3793 100644
--- a/ui/status.go
+++ b/ui/status.go
@@ -1,8 +1,7 @@
package ui
import (
- "os"
-
+ "github.com/ryanhamamura/c4/config"
"github.com/ryanhamamura/c4/game"
"github.com/ryanhamamura/via/h"
)
@@ -118,10 +117,7 @@ func PlayerInfo(g *game.Game, myColor int) h.H {
}
func getBaseURL() string {
- if url := os.Getenv("APP_URL"); url != "" {
- return url
- }
- return "https://games.adriatica.io"
+ return config.Global.AppURL
}
func InviteLink(gameID string) h.H {
From 8c3b3fc6eae768f9e8546403549b731974bf8225 Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 12:16:25 -1000
Subject: [PATCH 2/9] refactor: replace via framework with chi + templ +
datastar
Migrate from the via meta-framework to direct dependencies:
- chi for routing, templ for HTML templates, datastar for SSE/reactivity
- Feature-sliced architecture (features/{auth,lobby,c4game,snakegame}/)
- Shared layouts and components (features/common/)
- Handler factory pattern (HandleX(deps) http.HandlerFunc)
- Embedded NATS server (nats/), SCS sessions (sessions/), chi router wiring (router/)
- Move ChatMessage domain type from ui package to game package
- Remove old ui/ package (gomponents-based via/h views)
- Remove via dependency from go.mod entirely
---
db/persister.go | 9 +-
features/auth/handlers.go | 133 +++
features/auth/pages/login_templ.go | 89 ++
features/auth/pages/register_templ.go | 89 ++
features/auth/routes.go | 16 +
features/c4game/components/board_templ.go | 199 +++++
features/c4game/components/chat_templ.go | 173 ++++
features/c4game/components/status_templ.go | 352 ++++++++
features/c4game/handlers.go | 368 ++++++++
features/c4game/pages/game_templ.go | 219 +++++
features/c4game/routes.go | 29 +
features/common/components/shared_templ.go | 199 +++++
features/common/layouts/base_templ.go | 69 ++
features/lobby/components/gamelist_templ.go | 239 +++++
features/lobby/components/types.go | 12 +
features/lobby/handlers.go | 168 ++++
features/lobby/pages/lobby_templ.go | 339 +++++++
features/lobby/pages/types.go | 20 +
features/lobby/routes.go | 29 +
features/snakegame/components/board_templ.go | 295 ++++++
features/snakegame/components/chat_templ.go | 173 ++++
features/snakegame/components/status_templ.go | 470 ++++++++++
features/snakegame/handlers.go | 321 +++++++
features/snakegame/pages/game_templ.go | 277 ++++++
features/snakegame/routes.go | 22 +
game/types.go | 8 +
go.mod | 209 ++++-
go.sum | 747 +++++++++++++++-
main.go | 838 ++----------------
nats/nats.go | 69 ++
router/router.go | 76 ++
sessions/sessions.go | 31 +
ui/auth.go | 130 ---
ui/board.go | 69 --
ui/c4chat.go | 64 --
ui/gamelist.go | 110 ---
ui/lobby.go | 153 ----
ui/snakeboard.go | 112 ---
ui/snakechat.go | 63 --
ui/snakelobby.go | 124 ---
ui/snakestatus.go | 161 ----
ui/status.go | 137 ---
42 files changed, 5519 insertions(+), 1891 deletions(-)
create mode 100644 features/auth/handlers.go
create mode 100644 features/auth/pages/login_templ.go
create mode 100644 features/auth/pages/register_templ.go
create mode 100644 features/auth/routes.go
create mode 100644 features/c4game/components/board_templ.go
create mode 100644 features/c4game/components/chat_templ.go
create mode 100644 features/c4game/components/status_templ.go
create mode 100644 features/c4game/handlers.go
create mode 100644 features/c4game/pages/game_templ.go
create mode 100644 features/c4game/routes.go
create mode 100644 features/common/components/shared_templ.go
create mode 100644 features/common/layouts/base_templ.go
create mode 100644 features/lobby/components/gamelist_templ.go
create mode 100644 features/lobby/components/types.go
create mode 100644 features/lobby/handlers.go
create mode 100644 features/lobby/pages/lobby_templ.go
create mode 100644 features/lobby/pages/types.go
create mode 100644 features/lobby/routes.go
create mode 100644 features/snakegame/components/board_templ.go
create mode 100644 features/snakegame/components/chat_templ.go
create mode 100644 features/snakegame/components/status_templ.go
create mode 100644 features/snakegame/handlers.go
create mode 100644 features/snakegame/pages/game_templ.go
create mode 100644 features/snakegame/routes.go
create mode 100644 nats/nats.go
create mode 100644 router/router.go
create mode 100644 sessions/sessions.go
delete mode 100644 ui/auth.go
delete mode 100644 ui/board.go
delete mode 100644 ui/c4chat.go
delete mode 100644 ui/gamelist.go
delete mode 100644 ui/lobby.go
delete mode 100644 ui/snakeboard.go
delete mode 100644 ui/snakechat.go
delete mode 100644 ui/snakelobby.go
delete mode 100644 ui/snakestatus.go
delete mode 100644 ui/status.go
diff --git a/db/persister.go b/db/persister.go
index ddab719..f87760b 100644
--- a/db/persister.go
+++ b/db/persister.go
@@ -8,7 +8,6 @@ import (
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
"github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/c4/ui"
)
type GamePersister struct {
@@ -297,7 +296,7 @@ func NewChatPersister(q *repository.Queries) *ChatPersister {
return &ChatPersister{queries: q}
}
-func (p *ChatPersister) SaveChatMessage(gameID string, msg ui.C4ChatMessage) error {
+func (p *ChatPersister) SaveChatMessage(gameID string, msg game.ChatMessage) error {
return p.queries.CreateChatMessage(context.Background(), repository.CreateChatMessageParams{
GameID: gameID,
Nickname: msg.Nickname,
@@ -307,14 +306,14 @@ func (p *ChatPersister) SaveChatMessage(gameID string, msg ui.C4ChatMessage) err
})
}
-func (p *ChatPersister) LoadChatMessages(gameID string) ([]ui.C4ChatMessage, error) {
+func (p *ChatPersister) LoadChatMessages(gameID string) ([]game.ChatMessage, error) {
rows, err := p.queries.GetChatMessages(context.Background(), gameID)
if err != nil {
return nil, err
}
- msgs := make([]ui.C4ChatMessage, len(rows))
+ msgs := make([]game.ChatMessage, len(rows))
for i, r := range rows {
- msgs[i] = ui.C4ChatMessage{
+ msgs[i] = game.ChatMessage{
Nickname: r.Nickname,
Color: int(r.Color),
Message: r.Message,
diff --git a/features/auth/handlers.go b/features/auth/handlers.go
new file mode 100644
index 0000000..1212e0b
--- /dev/null
+++ b/features/auth/handlers.go
@@ -0,0 +1,133 @@
+package auth
+
+import (
+ "database/sql"
+ "net/http"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/google/uuid"
+ "github.com/ryanhamamura/c4/auth"
+ "github.com/ryanhamamura/c4/db/repository"
+ "github.com/ryanhamamura/c4/features/auth/pages"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+type LoginSignals struct {
+ Username string `json:"username"`
+ Password string `json:"password"`
+}
+
+type RegisterSignals struct {
+ Username string `json:"username"`
+ Password string `json:"password"`
+ Confirm string `json:"confirm"`
+}
+
+func HandleLoginPage() http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ if err := pages.LoginPage().Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ }
+}
+
+func HandleRegisterPage() http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ if err := pages.RegisterPage().Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ }
+}
+
+func HandleLogin(queries *repository.Queries, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ var signals LoginSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ sse := datastar.NewSSE(w, r)
+
+ user, err := queries.GetUserByUsername(r.Context(), signals.Username)
+ if err == sql.ErrNoRows {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "Invalid username or password"}) //nolint:errcheck
+ return
+ }
+ if err != nil {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "An error occurred"}) //nolint:errcheck
+ return
+ }
+ if !auth.CheckPassword(signals.Password, user.PasswordHash) {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "Invalid username or password"}) //nolint:errcheck
+ return
+ }
+
+ sessions.RenewToken(r.Context()) //nolint:errcheck
+ sessions.Put(r.Context(), "user_id", user.ID)
+ sessions.Put(r.Context(), "username", user.Username)
+ sessions.Put(r.Context(), "nickname", user.Username)
+
+ redirectURL := "/"
+ if returnURL := sessions.GetString(r.Context(), "return_url"); returnURL != "" {
+ sessions.Put(r.Context(), "return_url", "")
+ redirectURL = returnURL
+ }
+
+ sse.Redirect(redirectURL) //nolint:errcheck
+ }
+}
+
+func HandleRegister(queries *repository.Queries, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ var signals RegisterSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ sse := datastar.NewSSE(w, r)
+
+ if err := auth.ValidateUsername(signals.Username); err != nil {
+ sse.MarshalAndPatchSignals(map[string]any{"error": err.Error()}) //nolint:errcheck
+ return
+ }
+ if err := auth.ValidatePassword(signals.Password); err != nil {
+ sse.MarshalAndPatchSignals(map[string]any{"error": err.Error()}) //nolint:errcheck
+ return
+ }
+ if signals.Password != signals.Confirm {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "Passwords do not match"}) //nolint:errcheck
+ return
+ }
+
+ hash, err := auth.HashPassword(signals.Password)
+ if err != nil {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "An error occurred"}) //nolint:errcheck
+ return
+ }
+
+ user, err := queries.CreateUser(r.Context(), repository.CreateUserParams{
+ ID: uuid.New().String(),
+ Username: signals.Username,
+ PasswordHash: hash,
+ })
+ if err != nil {
+ sse.MarshalAndPatchSignals(map[string]any{"error": "Username already taken"}) //nolint:errcheck
+ return
+ }
+
+ sessions.RenewToken(r.Context()) //nolint:errcheck
+ sessions.Put(r.Context(), "user_id", user.ID)
+ sessions.Put(r.Context(), "username", user.Username)
+ sessions.Put(r.Context(), "nickname", user.Username)
+
+ redirectURL := "/"
+ if returnURL := sessions.GetString(r.Context(), "return_url"); returnURL != "" {
+ sessions.Put(r.Context(), "return_url", "")
+ redirectURL = returnURL
+ }
+
+ sse.Redirect(redirectURL) //nolint:errcheck
+ }
+}
diff --git a/features/auth/pages/login_templ.go b/features/auth/pages/login_templ.go
new file mode 100644
index 0000000..115ea27
--- /dev/null
+++ b/features/auth/pages/login_templ.go
@@ -0,0 +1,89 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package pages
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "github.com/ryanhamamura/c4/features/common/layouts"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func LoginPage() templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var2 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "Login Sign in to your account
Don't have an account? Register
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Login").Render(templ.WithChildren(ctx, templ_7745c5c3_Var2), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/auth/pages/register_templ.go b/features/auth/pages/register_templ.go
new file mode 100644
index 0000000..89efec3
--- /dev/null
+++ b/features/auth/pages/register_templ.go
@@ -0,0 +1,89 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package pages
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "github.com/ryanhamamura/c4/features/common/layouts"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func RegisterPage() templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var2 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "Register Create a new account
Already have an account? Login
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Register").Render(templ.WithChildren(ctx, templ_7745c5c3_Var2), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/auth/routes.go b/features/auth/routes.go
new file mode 100644
index 0000000..98ad6df
--- /dev/null
+++ b/features/auth/routes.go
@@ -0,0 +1,16 @@
+package auth
+
+import (
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/ryanhamamura/c4/db/repository"
+)
+
+func SetupRoutes(router chi.Router, queries *repository.Queries, sessions *scs.SessionManager) error {
+ router.Get("/login", HandleLoginPage())
+ router.Get("/register", HandleRegisterPage())
+ router.Post("/api/auth/login", HandleLogin(queries, sessions))
+ router.Post("/api/auth/register", HandleRegister(queries, sessions))
+
+ return nil
+}
diff --git a/features/c4game/components/board_templ.go b/features/c4game/components/board_templ.go
new file mode 100644
index 0000000..92f27b5
--- /dev/null
+++ b/features/c4game/components/board_templ.go
@@ -0,0 +1,199 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/ryanhamamura/c4/game"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func Board(g *game.Game, myColor int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for col := 0; col < 7; col++ {
+ templ_7745c5c3_Err = column(g, col, myColor).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func column(g *game.Game, colIdx int, myColor int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var2 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var2 == nil {
+ templ_7745c5c3_Var2 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ if g.Status == game.StatusInProgress && myColor == g.CurrentTurn {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for row := 0; row < 6; row++ {
+ templ_7745c5c3_Err = cell(g, row, colIdx).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for row := 0; row < 6; row++ {
+ templ_7745c5c3_Err = cell(g, row, colIdx).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ return nil
+ })
+}
+
+func cell(g *game.Game, row int, col int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var4 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var4 == nil {
+ templ_7745c5c3_Var4 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ var templ_7745c5c3_Var5 = []any{cellClass(g, row, col)}
+ templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var5...)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func cellClass(g *game.Game, row, col int) string {
+ color := g.Board[row][col]
+ activeTurn := 0
+ if g.Status == game.StatusInProgress {
+ activeTurn = g.CurrentTurn
+ }
+
+ class := "cell"
+ switch color {
+ case 1:
+ class += " red"
+ case 2:
+ class += " yellow"
+ }
+ if g.IsWinningCell(row, col) {
+ class += " winning"
+ }
+ if color != 0 && color == activeTurn {
+ class += " active-turn"
+ }
+ return class
+}
+
+// suppress unused import
+var _ = fmt.Sprintf
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/c4game/components/chat_templ.go b/features/c4game/components/chat_templ.go
new file mode 100644
index 0000000..c6fa087
--- /dev/null
+++ b/features/c4game/components/chat_templ.go
@@ -0,0 +1,173 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+type ChatMessage struct {
+ Nickname string `json:"nickname"`
+ Color int `json:"color"`
+ Message string `json:"message"`
+ Time int64 `json:"time"`
+}
+
+var chatColors = map[int]string{
+ 1: "#4a2a3a",
+ 2: "#2a4545",
+}
+
+func Chat(messages []ChatMessage, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for _, m := range messages {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var3 string
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(m.Nickname)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 27, Col: 18}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, ": ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var4 string
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(m.Message)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 29, Col: 22}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = chatAutoScroll().Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "
Send
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func chatAutoScroll() templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var7 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var7 == nil {
+ templ_7745c5c3_Var7 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func chatColor(color int) string {
+ if c, ok := chatColors[color]; ok {
+ return c
+ }
+ return "#666"
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/c4game/components/status_templ.go b/features/c4game/components/status_templ.go
new file mode 100644
index 0000000..ba58ad9
--- /dev/null
+++ b/features/c4game/components/status_templ.go
@@ -0,0 +1,352 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "github.com/ryanhamamura/c4/config"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func StatusBanner(g *game.Game, myColor int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ var templ_7745c5c3_Var2 = []any{statusClass(g, myColor)}
+ templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var2...)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var4 string
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(statusMessage(g, myColor))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 11, Col: 29}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if g.IsFinished() {
+ if g.RematchGameID != nil {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "
Join Rematch ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "
Play again ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func PlayerInfo(g *game.Game, myColor int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var7 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var7 == nil {
+ templ_7745c5c3_Var7 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for _, info := range playerInfoPairs(g, myColor) {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var8 = []any{"player-chip " + info.ColorClass}
+ templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var8...)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var10 string
+ templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(info.Label)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 38, Col: 22}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func InviteLink(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var11 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var11 == nil {
+ templ_7745c5c3_Var11 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "Share this link with your opponent:
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var12 string
+ templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(config.Global.AppURL + "/game/" + gameID)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 48, Col: 45}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, copyToClipboard(config.Global.AppURL+"/game/"+gameID))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "
Copy Link ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func copyToClipboard(url string) templ.ComponentScript {
+ return templ.ComponentScript{
+ Name: `__templ_copyToClipboard_1463`,
+ Function: `function __templ_copyToClipboard_1463(url){navigator.clipboard.writeText(url)
+}`,
+ Call: templ.SafeScript(`__templ_copyToClipboard_1463`, url),
+ CallInline: templ.SafeScriptInline(`__templ_copyToClipboard_1463`, url),
+ }
+}
+
+func statusClass(g *game.Game, myColor int) string {
+ switch g.Status {
+ case game.StatusWaitingForPlayer:
+ return "alert bg-base-200 text-xl font-bold"
+ case game.StatusInProgress:
+ if g.CurrentTurn == myColor {
+ return "alert alert-success text-xl font-bold"
+ }
+ return "alert bg-base-200 text-xl font-bold"
+ case game.StatusWon:
+ if g.Winner != nil && g.Winner.Color == myColor {
+ return "alert alert-success text-xl font-bold"
+ }
+ return "alert alert-error text-xl font-bold"
+ case game.StatusDraw:
+ return "alert alert-warning text-xl font-bold"
+ }
+ return "alert bg-base-200 text-xl font-bold"
+}
+
+func statusMessage(g *game.Game, myColor int) string {
+ switch g.Status {
+ case game.StatusWaitingForPlayer:
+ return "Waiting for opponent..."
+ case game.StatusInProgress:
+ if g.CurrentTurn == myColor {
+ return "Your turn!"
+ }
+ return opponentName(g, myColor) + "'s turn"
+ case game.StatusWon:
+ if g.Winner != nil && g.Winner.Color == myColor {
+ return "You win!"
+ }
+ if g.Winner != nil {
+ return g.Winner.Nickname + " wins!"
+ }
+ return "Game over"
+ case game.StatusDraw:
+ return "It's a draw!"
+ }
+ return ""
+}
+
+func opponentName(g *game.Game, myColor int) string {
+ for _, p := range g.Players {
+ if p != nil && p.Color != myColor {
+ return p.Nickname
+ }
+ }
+ return "Opponent"
+}
+
+type playerInfoData struct {
+ ColorClass string
+ Label string
+}
+
+func playerInfoPairs(g *game.Game, myColor int) []playerInfoData {
+ var result []playerInfoData
+
+ var myName, oppName string
+ var myClass, oppClass string
+
+ for _, p := range g.Players {
+ if p == nil {
+ continue
+ }
+ colorClass := "yellow"
+ if p.Color == 1 {
+ colorClass = "red"
+ }
+ if p.Color == myColor {
+ myName = p.Nickname
+ myClass = colorClass
+ } else {
+ oppName = p.Nickname
+ oppClass = colorClass
+ }
+ }
+
+ if oppName == "" {
+ oppName = "Waiting..."
+ }
+
+ result = append(result, playerInfoData{ColorClass: myClass, Label: myName + " (You)"})
+ result = append(result, playerInfoData{ColorClass: oppClass, Label: oppName})
+ return result
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/c4game/handlers.go b/features/c4game/handlers.go
new file mode 100644
index 0000000..4df8425
--- /dev/null
+++ b/features/c4game/handlers.go
@@ -0,0 +1,368 @@
+package c4game
+
+import (
+ "encoding/json"
+ "net/http"
+ "strconv"
+ "sync"
+ "time"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/nats-io/nats.go"
+ "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/features/c4game/components"
+ "github.com/ryanhamamura/c4/features/c4game/pages"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ http.Redirect(w, r, "/", http.StatusFound)
+ return
+ }
+
+ playerID := game.PlayerID(sessions.GetString(r.Context(), "player_id"))
+ if playerID == "" {
+ playerID = game.PlayerID(game.GenerateID(8))
+ sessions.Put(r.Context(), "player_id", string(playerID))
+ }
+
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ playerID = game.PlayerID(userID)
+ }
+
+ nickname := sessions.GetString(r.Context(), "nickname")
+
+ // Auto-join if player has a nickname but isn't in the game yet
+ if nickname != "" && gi.GetPlayerColor(playerID) == 0 {
+ player := &game.Player{
+ ID: playerID,
+ Nickname: nickname,
+ }
+ if userID != "" {
+ player.UserID = &userID
+ }
+ gi.Join(&game.PlayerSession{Player: player})
+ }
+
+ myColor := gi.GetPlayerColor(playerID)
+
+ if myColor == 0 {
+ // Player not in game
+ isGuest := r.URL.Query().Get("guest") == "1"
+ if userID == "" && !isGuest {
+ // Show join prompt (login vs guest)
+ if err := pages.JoinPage(gameID).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ return
+ }
+ // Show nickname prompt
+ if err := pages.NicknamePage(gameID).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ return
+ }
+
+ // Player is in the game — render full game page
+ g := gi.GetGame()
+ uiMsgs, _ := chatPersister.LoadChatMessages(gameID)
+ msgs := uiChatToComponents(uiMsgs)
+
+ if err := pages.GamePage(g, myColor, msgs).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ }
+}
+
+func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ playerID := game.PlayerID(sessions.GetString(r.Context(), "player_id"))
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ playerID = game.PlayerID(userID)
+ }
+
+ myColor := gi.GetPlayerColor(playerID)
+
+ sse := datastar.NewSSE(w, r)
+
+ // Load initial chat messages
+ uiMsgs, _ := chatPersister.LoadChatMessages(gameID)
+ var chatMu sync.Mutex
+ chatMessages := uiChatToComponents(uiMsgs)
+
+ // Send initial render of all components
+ sendGameComponents(sse, gi, myColor, chatMessages, &chatMu, gameID)
+
+ // Subscribe to game state updates
+ gameCh := make(chan *nats.Msg, 64)
+ gameSub, err := nc.ChanSubscribe("game."+gameID, gameCh)
+ if err != nil {
+ return
+ }
+ defer gameSub.Unsubscribe()
+
+ // Subscribe to chat messages
+ chatCh := make(chan *nats.Msg, 64)
+ chatSub, err := nc.ChanSubscribe("game.chat."+gameID, chatCh)
+ if err != nil {
+ return
+ }
+ defer chatSub.Unsubscribe()
+
+ ctx := r.Context()
+ for {
+ select {
+ case <-ctx.Done():
+ return
+ case <-gameCh:
+ // Re-read player color in case we just joined
+ myColor = gi.GetPlayerColor(playerID)
+ sendGameComponents(sse, gi, myColor, chatMessages, &chatMu, gameID)
+ case msg := <-chatCh:
+ var uiMsg game.ChatMessage
+ if err := json.Unmarshal(msg.Data, &uiMsg); err != nil {
+ continue
+ }
+ cm := components.ChatMessage{
+ Nickname: uiMsg.Nickname,
+ Color: uiMsg.Color,
+ Message: uiMsg.Message,
+ Time: uiMsg.Time,
+ }
+ chatMu.Lock()
+ chatMessages = append(chatMessages, cm)
+ if len(chatMessages) > 50 {
+ chatMessages = chatMessages[len(chatMessages)-50:]
+ }
+ chatMu.Unlock()
+
+ chatMu.Lock()
+ msgs := make([]components.ChatMessage, len(chatMessages))
+ copy(msgs, chatMessages)
+ chatMu.Unlock()
+
+ if err := sse.PatchElementTempl(components.Chat(msgs, gameID), datastar.WithSelectorID("c4-chat")); err != nil {
+ return
+ }
+ }
+ }
+ }
+}
+
+func HandleDropPiece(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ colStr := r.URL.Query().Get("col")
+ col, err := strconv.Atoi(colStr)
+ if err != nil {
+ http.Error(w, "invalid column", http.StatusBadRequest)
+ return
+ }
+
+ playerID := game.PlayerID(sessions.GetString(r.Context(), "player_id"))
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ playerID = game.PlayerID(userID)
+ }
+
+ myColor := gi.GetPlayerColor(playerID)
+ if myColor == 0 {
+ http.Error(w, "not in game", http.StatusForbidden)
+ return
+ }
+
+ gi.DropPiece(col, myColor)
+
+ // The store's notifyFunc publishes to NATS, which triggers SSE updates.
+ // Return empty SSE response.
+ datastar.NewSSE(w, r)
+ }
+}
+
+func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ type ChatSignals struct {
+ ChatMsg string `json:"chatMsg"`
+ }
+ var signals ChatSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.ChatMsg == "" {
+ datastar.NewSSE(w, r)
+ return
+ }
+
+ playerID := game.PlayerID(sessions.GetString(r.Context(), "player_id"))
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ playerID = game.PlayerID(userID)
+ }
+
+ myColor := gi.GetPlayerColor(playerID)
+ if myColor == 0 {
+ datastar.NewSSE(w, r)
+ return
+ }
+
+ g := gi.GetGame()
+ nick := ""
+ for _, p := range g.Players {
+ if p != nil && p.ID == playerID {
+ nick = p.Nickname
+ break
+ }
+ }
+
+ cm := game.ChatMessage{
+ Nickname: nick,
+ Color: myColor,
+ Message: signals.ChatMsg,
+ Time: time.Now().UnixMilli(),
+ }
+ chatPersister.SaveChatMessage(gameID, cm)
+
+ data, err := json.Marshal(cm)
+ if err != nil {
+ datastar.NewSSE(w, r)
+ return
+ }
+ nc.Publish("game.chat."+gameID, data)
+
+ // Clear the chat input
+ sse := datastar.NewSSE(w, r)
+ sse.MarshalAndPatchSignals(map[string]any{"chatMsg": ""}) //nolint:errcheck
+ }
+}
+
+func HandleSetNickname(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ sse := datastar.NewSSE(w, r)
+ sse.Redirect("/") //nolint:errcheck
+ return
+ }
+
+ type NicknameSignals struct {
+ Nickname string `json:"nickname"`
+ }
+ var signals NicknameSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.Nickname == "" {
+ datastar.NewSSE(w, r)
+ return
+ }
+
+ sessions.Put(r.Context(), "nickname", signals.Nickname)
+
+ playerID := game.PlayerID(sessions.GetString(r.Context(), "player_id"))
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ playerID = game.PlayerID(userID)
+ }
+
+ if gi.GetPlayerColor(playerID) == 0 {
+ player := &game.Player{
+ ID: playerID,
+ Nickname: signals.Nickname,
+ }
+ if userID != "" {
+ player.UserID = &userID
+ }
+ gi.Join(&game.PlayerSession{Player: player})
+ }
+
+ sse := datastar.NewSSE(w, r)
+ sse.Redirect("/game/" + gameID) //nolint:errcheck
+ }
+}
+
+func HandleRematch(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+
+ gi, exists := store.Get(gameID)
+ if !exists {
+ sse := datastar.NewSSE(w, r)
+ sse.Redirect("/") //nolint:errcheck
+ return
+ }
+
+ newGI := gi.CreateRematch(store)
+ sse := datastar.NewSSE(w, r)
+ if newGI != nil {
+ sse.Redirectf("/game/%s", newGI.ID()) //nolint:errcheck
+ }
+ }
+}
+
+// sendGameComponents patches all game-related SSE components.
+func sendGameComponents(sse *datastar.ServerSentEventGenerator, gi *game.GameInstance, myColor int, chatMessages []components.ChatMessage, chatMu *sync.Mutex, gameID string) {
+ g := gi.GetGame()
+
+ sse.PatchElementTempl(components.Board(g, myColor), datastar.WithSelectorID("c4-board")) //nolint:errcheck
+ sse.PatchElementTempl(components.StatusBanner(g, myColor), datastar.WithSelectorID("c4-status")) //nolint:errcheck
+ sse.PatchElementTempl(components.PlayerInfo(g, myColor), datastar.WithSelectorID("c4-players")) //nolint:errcheck
+
+ chatMu.Lock()
+ msgs := make([]components.ChatMessage, len(chatMessages))
+ copy(msgs, chatMessages)
+ chatMu.Unlock()
+
+ sse.PatchElementTempl(components.Chat(msgs, gameID), datastar.WithSelectorID("c4-chat")) //nolint:errcheck
+}
+
+// uiChatToComponents converts ui.C4ChatMessage slice to components.ChatMessage slice.
+func uiChatToComponents(uiMsgs []game.ChatMessage) []components.ChatMessage {
+ msgs := make([]components.ChatMessage, len(uiMsgs))
+ for i, m := range uiMsgs {
+ msgs[i] = components.ChatMessage{
+ Nickname: m.Nickname,
+ Color: m.Color,
+ Message: m.Message,
+ Time: m.Time,
+ }
+ }
+ return msgs
+}
diff --git a/features/c4game/pages/game_templ.go b/features/c4game/pages/game_templ.go
new file mode 100644
index 0000000..3662c23
--- /dev/null
+++ b/features/c4game/pages/game_templ.go
@@ -0,0 +1,219 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package pages
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "github.com/ryanhamamura/c4/features/c4game/components"
+ sharedcomponents "github.com/ryanhamamura/c4/features/common/components"
+ "github.com/ryanhamamura/c4/features/common/layouts"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func GamePage(g *game.Game, myColor int, messages []components.ChatMessage) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var2 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = sharedcomponents.BackToLobby().Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = sharedcomponents.StealthTitle("text-3xl font-bold").Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.PlayerInfo(g, myColor).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.StatusBanner(g, myColor).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.Board(g, myColor).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.Chat(messages, g.ID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if g.Status == game.StatusWaitingForPlayer {
+ templ_7745c5c3_Err = components.InviteLink(g.ID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Connect 4").Render(templ.WithChildren(ctx, templ_7745c5c3_Var2), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func JoinPage(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var4 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var4 == nil {
+ templ_7745c5c3_Var4 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var5 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = sharedcomponents.GameJoinPrompt(
+ "/login?return_url=/game/"+gameID,
+ "/register?return_url=/game/"+gameID,
+ "/game/"+gameID,
+ ).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Connect 4 - Join").Render(templ.WithChildren(ctx, templ_7745c5c3_Var5), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func NicknamePage(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var6 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var6 == nil {
+ templ_7745c5c3_Var6 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var7 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = sharedcomponents.NicknamePrompt("/api/game/"+gameID+"/join").Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Connect 4 - Join").Render(templ.WithChildren(ctx, templ_7745c5c3_Var7), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/c4game/routes.go b/features/c4game/routes.go
new file mode 100644
index 0000000..99783f6
--- /dev/null
+++ b/features/c4game/routes.go
@@ -0,0 +1,29 @@
+package c4game
+
+import (
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/nats-io/nats.go"
+ "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/game"
+)
+
+func SetupRoutes(
+ router chi.Router,
+ store *game.GameStore,
+ nc *nats.Conn,
+ sessions *scs.SessionManager,
+ chatPersister *db.ChatPersister,
+) error {
+ router.Get("/game/{game_id}", HandleGamePage(store, sessions, chatPersister))
+ router.Get("/game/{game_id}/events", HandleGameEvents(store, nc, sessions, chatPersister))
+
+ router.Route("/api/game/{game_id}", func(r chi.Router) {
+ r.Post("/drop", HandleDropPiece(store, sessions))
+ r.Post("/chat", HandleSendChat(store, nc, sessions, chatPersister))
+ r.Post("/join", HandleSetNickname(store, sessions))
+ r.Post("/rematch", HandleRematch(store, sessions))
+ })
+
+ return nil
+}
diff --git a/features/common/components/shared_templ.go b/features/common/components/shared_templ.go
new file mode 100644
index 0000000..9c74d6c
--- /dev/null
+++ b/features/common/components/shared_templ.go
@@ -0,0 +1,199 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import "github.com/starfederation/datastar-go/datastar"
+
+func BackToLobby() templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "← Back ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func StealthTitle(class string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var2 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var2 == nil {
+ templ_7745c5c3_Var2 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ var templ_7745c5c3_Var3 = []any{class}
+ templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var3...)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "● ● ● ● ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func NicknamePrompt(returnPath string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var5 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var5 == nil {
+ templ_7745c5c3_Var5 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "Join Game Enter your nickname to join the game.
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func GameJoinPrompt(loginURL string, registerURL string, gamePath string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var7 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var7 == nil {
+ templ_7745c5c3_Var7 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "Join Game Log in to track your game history, or continue as a guest.
Don't have an account? Register
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/common/layouts/base_templ.go b/features/common/layouts/base_templ.go
new file mode 100644
index 0000000..2f5437f
--- /dev/null
+++ b/features/common/layouts/base_templ.go
@@ -0,0 +1,69 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package layouts
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import "github.com/ryanhamamura/c4/config"
+
+func Base(title string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var2 string
+ templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(title)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/common/layouts/base.templ`, Line: 9, Col: 17}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if config.Global.Environment == config.Dev {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templ_7745c5c3_Var1.Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/lobby/components/gamelist_templ.go b/features/lobby/components/gamelist_templ.go
new file mode 100644
index 0000000..c8539c8
--- /dev/null
+++ b/features/lobby/components/gamelist_templ.go
@@ -0,0 +1,239 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+ "time"
+
+ "github.com/ryanhamamura/c4/game"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func GameList(games []GameListItem) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ if len(games) > 0 {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "Your Games ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for _, g := range games {
+ templ_7745c5c3_Err = gameListEntry(g).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ return nil
+ })
+}
+
+func gameListEntry(g GameListItem) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var2 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var2 == nil {
+ templ_7745c5c3_Var2 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var4 string
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(opponentDisplay(g))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 31, Col: 48}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var5 = []any{statusClass(g)}
+ templ_7745c5c3_Err = templ.RenderCSSItems(ctx, templ_7745c5c3_Buffer, templ_7745c5c3_Var5...)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var7 string
+ templ_7745c5c3_Var7, templ_7745c5c3_Err = templ.JoinStringErrs(statusText(g))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 32, Col: 50}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var7))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var8 string
+ templ_7745c5c3_Var8, templ_7745c5c3_Err = templ.JoinStringErrs(formatTimeAgo(g.LastPlayed))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 35, Col: 66}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var8))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "
× ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func statusText(g GameListItem) string {
+ switch game.GameStatus(g.Status) {
+ case game.StatusWaitingForPlayer:
+ return "Waiting for opponent"
+ case game.StatusInProgress:
+ if g.IsMyTurn {
+ return "Your turn!"
+ }
+ return "Opponent's turn"
+ }
+ return ""
+}
+
+func statusClass(g GameListItem) string {
+ switch game.GameStatus(g.Status) {
+ case game.StatusWaitingForPlayer:
+ return "text-sm opacity-60"
+ case game.StatusInProgress:
+ if g.IsMyTurn {
+ return "text-sm text-success font-bold"
+ }
+ return "text-sm"
+ }
+ return ""
+}
+
+func opponentDisplay(g GameListItem) string {
+ if g.OpponentName == "" {
+ return "Waiting for opponent..."
+ }
+ return "vs " + g.OpponentName
+}
+
+func formatTimeAgo(t time.Time) string {
+ if t.IsZero() {
+ return ""
+ }
+ duration := time.Since(t)
+
+ if duration < time.Minute {
+ return "just now"
+ }
+ if duration < time.Hour {
+ mins := int(duration.Minutes())
+ if mins == 1 {
+ return "1 minute ago"
+ }
+ return fmt.Sprintf("%d minutes ago", mins)
+ }
+ if duration < 24*time.Hour {
+ hours := int(duration.Hours())
+ if hours == 1 {
+ return "1 hour ago"
+ }
+ return fmt.Sprintf("%d hours ago", hours)
+ }
+ days := int(duration.Hours() / 24)
+ if days == 1 {
+ return "yesterday"
+ }
+ return fmt.Sprintf("%d days ago", days)
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/lobby/components/types.go b/features/lobby/components/types.go
new file mode 100644
index 0000000..2608c95
--- /dev/null
+++ b/features/lobby/components/types.go
@@ -0,0 +1,12 @@
+package components
+
+import "time"
+
+// GameListItem represents a connect4 game in the user's active game list.
+type GameListItem struct {
+ ID string
+ Status int
+ OpponentName string
+ IsMyTurn bool
+ LastPlayed time.Time
+}
diff --git a/features/lobby/handlers.go b/features/lobby/handlers.go
new file mode 100644
index 0000000..4c8a86a
--- /dev/null
+++ b/features/lobby/handlers.go
@@ -0,0 +1,168 @@
+package lobby
+
+import (
+ "context"
+ "database/sql"
+ "fmt"
+ "net/http"
+ "strconv"
+
+ "github.com/ryanhamamura/c4/db/repository"
+ lobbycomponents "github.com/ryanhamamura/c4/features/lobby/components"
+ "github.com/ryanhamamura/c4/features/lobby/pages"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/ryanhamamura/c4/snake"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+// HandleLobbyPage renders the main lobby page with active games for logged-in users.
+func HandleLobbyPage(queries *repository.Queries, sessions *scs.SessionManager, snakeStore *snake.SnakeStore) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ userID := sessions.GetString(r.Context(), "user_id")
+ username := sessions.GetString(r.Context(), "username")
+ isLoggedIn := userID != ""
+
+ var userGames []lobbycomponents.GameListItem
+ if isLoggedIn {
+ ctx := context.Background()
+ games, err := queries.GetUserActiveGames(ctx, sql.NullString{String: userID, Valid: true})
+ if err == nil {
+ for _, g := range games {
+ isMyTurn := g.Status == 1 && g.CurrentTurn == g.MyColor
+ userGames = append(userGames, lobbycomponents.GameListItem{
+ ID: g.ID,
+ Status: int(g.Status),
+ OpponentName: g.OpponentNickname.String,
+ IsMyTurn: isMyTurn,
+ LastPlayed: g.UpdatedAt.Time,
+ })
+ }
+ }
+ }
+
+ var activeSnakeGames []pages.SnakeGameListItem
+ for _, g := range snakeStore.ActiveGames() {
+ statusLabel := "Waiting"
+ if g.Status == snake.StatusCountdown {
+ statusLabel = "Starting soon"
+ }
+ activeSnakeGames = append(activeSnakeGames, pages.SnakeGameListItem{
+ ID: g.ID,
+ Width: g.State.Width,
+ Height: g.State.Height,
+ PlayerCount: g.PlayerCount(),
+ StatusLabel: statusLabel,
+ })
+ }
+
+ data := pages.LobbyData{
+ IsLoggedIn: isLoggedIn,
+ Username: username,
+ UserGames: userGames,
+ ActiveSnakeGames: activeSnakeGames,
+ }
+
+ if err := pages.LobbyPage(data).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ }
+}
+
+// HandleCreateGame reads the nickname signal, creates a connect4 game, and redirects via SSE.
+func HandleCreateGame(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ type Signals struct {
+ Nickname string `json:"nickname"`
+ }
+ signals := &Signals{}
+ if err := datastar.ReadSignals(r, signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.Nickname == "" {
+ return
+ }
+
+ sessions.Put(r.Context(), "nickname", signals.Nickname)
+
+ gi := store.Create()
+ sse := datastar.NewSSE(w, r)
+ sse.ExecuteScript(fmt.Sprintf("window.location.href='/game/%s'", gi.ID()))
+ }
+}
+
+// HandleDeleteGame deletes a connect4 game and redirects to the lobby.
+func HandleDeleteGame(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "id")
+ if gameID == "" {
+ http.Error(w, "missing game id", http.StatusBadRequest)
+ return
+ }
+
+ store.Delete(gameID)
+
+ sse := datastar.NewSSE(w, r)
+ sse.ExecuteScript("window.location.href='/'")
+ }
+}
+
+// HandleCreateSnakeGame reads nickname, grid preset, speed, and mode from the request,
+// creates a snake game, and redirects via SSE.
+func HandleCreateSnakeGame(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ type Signals struct {
+ Nickname string `json:"nickname"`
+ SelectedSpeed int `json:"selectedSpeed"`
+ }
+ signals := &Signals{}
+ if err := datastar.ReadSignals(r, signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.Nickname == "" {
+ return
+ }
+
+ sessions.Put(r.Context(), "nickname", signals.Nickname)
+
+ mode := snake.ModeMultiplayer
+ if r.URL.Query().Get("mode") == "solo" {
+ mode = snake.ModeSinglePlayer
+ }
+
+ presetIdx, _ := strconv.Atoi(r.URL.Query().Get("preset"))
+ if presetIdx < 0 || presetIdx >= len(snake.GridPresets) {
+ presetIdx = 0
+ }
+ preset := snake.GridPresets[presetIdx]
+
+ speed := snake.DefaultSpeed
+ if signals.SelectedSpeed >= 0 && signals.SelectedSpeed < len(snake.SpeedPresets) {
+ speed = snake.SpeedPresets[signals.SelectedSpeed].Speed
+ }
+
+ si := snakeStore.Create(preset.Width, preset.Height, mode, speed)
+
+ sse := datastar.NewSSE(w, r)
+ sse.ExecuteScript(fmt.Sprintf("window.location.href='/snake/%s'", si.ID()))
+ }
+}
+
+// HandleLogout clears the session and redirects to the lobby.
+func HandleLogout(sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ if err := sessions.Destroy(r.Context()); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ return
+ }
+
+ sse := datastar.NewSSE(w, r)
+ sse.ExecuteScript("window.location.href='/'")
+ }
+}
diff --git a/features/lobby/pages/lobby_templ.go b/features/lobby/pages/lobby_templ.go
new file mode 100644
index 0000000..bc1aeb2
--- /dev/null
+++ b/features/lobby/pages/lobby_templ.go
@@ -0,0 +1,339 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package pages
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/ryanhamamura/c4/features/common/components"
+ "github.com/ryanhamamura/c4/features/common/layouts"
+ lobbycomponents "github.com/ryanhamamura/c4/features/lobby/components"
+ "github.com/ryanhamamura/c4/snake"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func LobbyPage(data LobbyData) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var2 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if data.IsLoggedIn {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "Logged in as ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var3 string
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(data.Username)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 22, Col: 47}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, " Logout
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.StealthTitle("").Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.StealthTitle("").Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, " ~~~~
Start a new session
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = lobbycomponents.GameList(data.UserGames).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "
Your Nickname Speed ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for i, preset := range snake.SpeedPresets {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var9 string
+ templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(preset.Name)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 116, Col: 21}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "
Play Solo ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for i, preset := range snake.GridPresets {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var11 string
+ templ_7745c5c3_Var11, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s (%d\u00d7%d)", preset.Name, preset.Width, preset.Height))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 131, Col: 82}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var11))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 20, "
Create Multiplayer Game ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for i, preset := range snake.GridPresets {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var13 string
+ templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%s (%d\u00d7%d)", preset.Name, preset.Width, preset.Height))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 146, Col: 82}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if len(data.ActiveSnakeGames) > 0 {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 25, "
Join a Game ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for _, g := range data.ActiveSnakeGames {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var15 string
+ templ_7745c5c3_Var15, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("%d\u00d7%d \u2014 %d/8 players", g.Width, g.Height, g.PlayerCount))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 161, Col: 96}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var15))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var16 string
+ templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(g.StatusLabel)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 162, Col: 57}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Game Lobby").Render(templ.WithChildren(ctx, templ_7745c5c3_Var2), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/lobby/pages/types.go b/features/lobby/pages/types.go
new file mode 100644
index 0000000..a386a6f
--- /dev/null
+++ b/features/lobby/pages/types.go
@@ -0,0 +1,20 @@
+package pages
+
+import "github.com/ryanhamamura/c4/features/lobby/components"
+
+// SnakeGameListItem represents a joinable snake game in the lobby.
+type SnakeGameListItem struct {
+ ID string
+ Width int
+ Height int
+ PlayerCount int
+ StatusLabel string
+}
+
+// LobbyData holds all data needed to render the lobby page.
+type LobbyData struct {
+ IsLoggedIn bool
+ Username string
+ UserGames []components.GameListItem
+ ActiveSnakeGames []SnakeGameListItem
+}
diff --git a/features/lobby/routes.go b/features/lobby/routes.go
new file mode 100644
index 0000000..016eb75
--- /dev/null
+++ b/features/lobby/routes.go
@@ -0,0 +1,29 @@
+package lobby
+
+import (
+ "github.com/ryanhamamura/c4/db/repository"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/ryanhamamura/c4/snake"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+)
+
+func SetupRoutes(
+ router chi.Router,
+ queries *repository.Queries,
+ sessions *scs.SessionManager,
+ store *game.GameStore,
+ snakeStore *snake.SnakeStore,
+) error {
+ router.Get("/", HandleLobbyPage(queries, sessions, snakeStore))
+
+ router.Route("/api/lobby", func(r chi.Router) {
+ r.Post("/create-game", HandleCreateGame(store, sessions))
+ r.Delete("/game/{id}", HandleDeleteGame(store, sessions))
+ r.Post("/create-snake", HandleCreateSnakeGame(snakeStore, sessions))
+ r.Post("/logout", HandleLogout(sessions))
+ })
+
+ return nil
+}
diff --git a/features/snakegame/components/board_templ.go b/features/snakegame/components/board_templ.go
new file mode 100644
index 0000000..cbe1237
--- /dev/null
+++ b/features/snakegame/components/board_templ.go
@@ -0,0 +1,295 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/ryanhamamura/c4/snake"
+)
+
+func cellSizeForGrid(width, height int) int {
+ maxDim := width
+ if height > maxDim {
+ maxDim = height
+ }
+ switch {
+ case maxDim <= 15:
+ return 28
+ case maxDim <= 20:
+ return 24
+ case maxDim <= 30:
+ return 20
+ case maxDim <= 40:
+ return 16
+ default:
+ return 14
+ }
+}
+
+type cellInfo struct {
+ snakeIdx int // -1 = empty, -2 = food
+ isHead bool
+}
+
+func Board(sg *snake.SnakeGame) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if sg.State != nil && (sg.Status == snake.StatusInProgress || sg.Status == snake.StatusFinished) {
+ templ_7745c5c3_Err = boardCells(sg).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func boardCells(sg *snake.SnakeGame) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var3 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var3 == nil {
+ templ_7745c5c3_Var3 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ state := sg.State
+ grid := buildGrid(state)
+ cellSize := cellSizeForGrid(state.Width, state.Height)
+ for y := 0; y < state.Height; y++ {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for x := 0; x < state.Width; x++ {
+ ci := grid[y][x]
+ if ci.snakeIdx == -2 {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else if ci.snakeIdx >= 0 {
+ s := state.Snakes[ci.snakeIdx]
+ bg := snakeColor(ci.snakeIdx)
+ if ci.isHead {
+ if s.Alive {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ } else {
+ if s.Alive {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ return nil
+ })
+}
+
+func buildGrid(state *snake.GameState) [][]cellInfo {
+ grid := make([][]cellInfo, state.Height)
+ for y := 0; y < state.Height; y++ {
+ grid[y] = make([]cellInfo, state.Width)
+ for x := 0; x < state.Width; x++ {
+ grid[y][x] = cellInfo{snakeIdx: -1}
+ }
+ }
+ for fi := range state.Food {
+ f := state.Food[fi]
+ if f.X >= 0 && f.X < state.Width && f.Y >= 0 && f.Y < state.Height {
+ grid[f.Y][f.X] = cellInfo{snakeIdx: -2}
+ }
+ }
+ for si, s := range state.Snakes {
+ if s == nil {
+ continue
+ }
+ for bi, bp := range s.Body {
+ if bp.X >= 0 && bp.X < state.Width && bp.Y >= 0 && bp.Y < state.Height {
+ grid[bp.Y][bp.X] = cellInfo{snakeIdx: si, isHead: bi == 0}
+ }
+ }
+ }
+ return grid
+}
+
+func snakeColor(idx int) string {
+ if idx >= 0 && idx < len(snake.SnakeColors) {
+ return snake.SnakeColors[idx]
+ }
+ return "#666"
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/snakegame/components/chat_templ.go b/features/snakegame/components/chat_templ.go
new file mode 100644
index 0000000..396e9a5
--- /dev/null
+++ b/features/snakegame/components/chat_templ.go
@@ -0,0 +1,173 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/ryanhamamura/c4/snake"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+type ChatMessage struct {
+ Nickname string `json:"nickname"`
+ Slot int `json:"slot"`
+ Message string `json:"message"`
+ Time int64 `json:"time"`
+}
+
+func Chat(messages []ChatMessage, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for _, m := range messages {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var3 string
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(m.Nickname + ": ")
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 23, Col: 25}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var4 string
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(m.Message)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 25, Col: 22}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "
Send
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = chatAutoScroll().Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func chatAutoScroll() templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var7 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var7 == nil {
+ templ_7745c5c3_Var7 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func chatColor(slot int) string {
+ if slot >= 0 && slot < len(snake.SnakeColors) {
+ return snake.SnakeColors[slot]
+ }
+ return "#666"
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/snakegame/components/status_templ.go b/features/snakegame/components/status_templ.go
new file mode 100644
index 0000000..b1733c0
--- /dev/null
+++ b/features/snakegame/components/status_templ.go
@@ -0,0 +1,470 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package components
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+ "math"
+ "time"
+
+ "github.com/ryanhamamura/c4/config"
+ "github.com/ryanhamamura/c4/snake"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func StatusBanner(sg *snake.SnakeGame, mySlot int, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ switch sg.Status {
+ case snake.StatusWaitingForPlayers:
+ if sg.Mode == snake.ModeSinglePlayer {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, "
Ready?
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 3, "
Waiting for players...
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ case snake.StatusCountdown:
+ remaining := time.Until(sg.CountdownEnd)
+ secs := int(math.Ceil(remaining.Seconds()))
+ if secs < 0 {
+ secs = 0
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 4, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var2 string
+ templ_7745c5c3_Var2, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("Starting in %d...", secs))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 29, Col: 45}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var2))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ case snake.StatusInProgress:
+ if sg.State != nil && mySlot >= 0 && mySlot < len(sg.State.Snakes) && sg.State.Snakes[mySlot] != nil && !sg.State.Snakes[mySlot].Alive {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "
You're out!
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else if sg.Mode == snake.ModeSinglePlayer {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var3 string
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("Score: %d", sg.Score))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 36, Col: 42}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 9, "
Go!
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ case snake.StatusFinished:
+ templ_7745c5c3_Err = finishedBanner(sg, mySlot, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 10, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func finishedBanner(sg *snake.SnakeGame, mySlot int, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var4 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var4 == nil {
+ templ_7745c5c3_Var4 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ if sg.Mode == snake.ModeSinglePlayer {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 11, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var5 string
+ templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("Game Over! Score: %d", sg.Score))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 50, Col: 50}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = rematchOrJoin(sg, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 12, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else if sg.Winner != nil {
+ if sg.Winner.Slot == mySlot {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 13, "You win!")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = rematchOrJoin(sg, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 14, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 15, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var6 string
+ templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(sg.Winner.Nickname + " wins!")
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 61, Col: 35}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = rematchOrJoin(sg, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 16, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 17, "It's a draw!")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = rematchOrJoin(sg, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 18, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ return nil
+ })
+}
+
+func rematchOrJoin(sg *snake.SnakeGame, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var7 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var7 == nil {
+ templ_7745c5c3_Var7 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ if sg.RematchGameID != nil {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 19, "Join Rematch ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 21, "Play again ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ return nil
+ })
+}
+
+func PlayerList(sg *snake.SnakeGame, mySlot int) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var10 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var10 == nil {
+ templ_7745c5c3_Var10 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 23, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ for i, p := range sg.Players {
+ if p != nil {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 24, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var12 string
+ templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(p.Nickname)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 96, Col: 18}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 26, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if i == mySlot {
+ var templ_7745c5c3_Var13 string
+ templ_7745c5c3_Var13, templ_7745c5c3_Err = templ.JoinStringErrs(" (You)")
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 98, Col: 17}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var13))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 27, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if sg.Status == snake.StatusInProgress || sg.Status == snake.StatusFinished {
+ if sg.State != nil && i < len(sg.State.Snakes) && sg.State.Snakes[i] != nil {
+ if sg.State.Snakes[i].Alive {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 28, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var14 string
+ templ_7745c5c3_Var14, templ_7745c5c3_Err = templ.JoinStringErrs(fmt.Sprintf("(%d)", len(sg.State.Snakes[i].Body)))
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 105, Col: 60}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var14))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 29, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 30, "(dead) ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 31, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 32, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func InviteLink(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var15 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var15 == nil {
+ templ_7745c5c3_Var15 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ fullURL := config.Global.AppURL + "/snake/" + gameID
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 33, "Share this link to invite players:
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ var templ_7745c5c3_Var16 string
+ templ_7745c5c3_Var16, templ_7745c5c3_Err = templ.JoinStringErrs(fullURL)
+ if templ_7745c5c3_Err != nil {
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 123, Col: 12}
+ }
+ _, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var16))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 34, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, copyToClipboard(fullURL))
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 35, "
Copy Link ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func copyToClipboard(url string) templ.ComponentScript {
+ return templ.ComponentScript{
+ Name: `__templ_copyToClipboard_1463`,
+ Function: `function __templ_copyToClipboard_1463(url){navigator.clipboard.writeText(url)
+}`,
+ Call: templ.SafeScript(`__templ_copyToClipboard_1463`, url),
+ CallInline: templ.SafeScriptInline(`__templ_copyToClipboard_1463`, url),
+ }
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/snakegame/handlers.go b/features/snakegame/handlers.go
new file mode 100644
index 0000000..8fe9675
--- /dev/null
+++ b/features/snakegame/handlers.go
@@ -0,0 +1,321 @@
+package snakegame
+
+import (
+ "encoding/json"
+ "net/http"
+ "strconv"
+ "sync"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/nats-io/nats.go"
+ "github.com/ryanhamamura/c4/features/snakegame/components"
+ "github.com/ryanhamamura/c4/features/snakegame/pages"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/ryanhamamura/c4/snake"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func getPlayerID(sessions *scs.SessionManager, r *http.Request) snake.PlayerID {
+ pid := sessions.GetString(r.Context(), "player_id")
+ if pid == "" {
+ pid = game.GenerateID(8)
+ sessions.Put(r.Context(), "player_id", pid)
+ }
+ userID := sessions.GetString(r.Context(), "user_id")
+ if userID != "" {
+ return snake.PlayerID(userID)
+ }
+ return snake.PlayerID(pid)
+}
+
+func HandleSnakePage(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Redirect(w, r, "/", http.StatusSeeOther)
+ return
+ }
+
+ playerID := getPlayerID(sessions, r)
+ nickname := sessions.GetString(r.Context(), "nickname")
+ userID := sessions.GetString(r.Context(), "user_id")
+
+ // Auto-join if nickname exists and not already in game
+ if nickname != "" && si.GetPlayerSlot(playerID) < 0 {
+ player := &snake.Player{
+ ID: playerID,
+ Nickname: nickname,
+ }
+ if userID != "" {
+ player.UserID = &userID
+ }
+ si.Join(player)
+ }
+
+ mySlot := si.GetPlayerSlot(playerID)
+
+ if mySlot < 0 {
+ // Not in game yet
+ isGuest := r.URL.Query().Get("guest") == "1"
+ if userID == "" && !isGuest {
+ if err := pages.JoinPage(gameID).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ return
+ }
+ if err := pages.NicknamePage(gameID).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ return
+ }
+
+ sg := si.GetGame()
+ if err := pages.GamePage(sg, mySlot, nil, gameID).Render(r.Context(), w); err != nil {
+ http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
+ }
+ }
+}
+
+func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ playerID := getPlayerID(sessions, r)
+ mySlot := si.GetPlayerSlot(playerID)
+
+ sse := datastar.NewSSE(w, r)
+
+ // Send initial render
+ sg := si.GetGame()
+ sse.PatchElementTempl(components.Board(sg)) //nolint:errcheck
+ sse.PatchElementTempl(components.StatusBanner(sg, mySlot, gameID)) //nolint:errcheck
+ sse.PatchElementTempl(components.PlayerList(sg, mySlot)) //nolint:errcheck
+ if sg.Mode == snake.ModeMultiplayer {
+ sse.PatchElementTempl(components.Chat(nil, gameID)) //nolint:errcheck
+ if sg.Status == snake.StatusWaitingForPlayers || sg.Status == snake.StatusCountdown {
+ sse.PatchElementTempl(components.InviteLink(gameID)) //nolint:errcheck
+ }
+ }
+
+ // Subscribe to game updates via NATS
+ gameCh := make(chan *nats.Msg, 64)
+ gameSub, err := nc.ChanSubscribe("snake."+gameID, gameCh)
+ if err != nil {
+ return
+ }
+ defer gameSub.Unsubscribe()
+
+ // Chat subscription (multiplayer only)
+ var chatCh chan *nats.Msg
+ var chatSub *nats.Subscription
+ var chatMessages []components.ChatMessage
+ var chatMu sync.Mutex
+
+ if sg.Mode == snake.ModeMultiplayer {
+ chatCh = make(chan *nats.Msg, 64)
+ chatSub, err = nc.ChanSubscribe("snake.chat."+gameID, chatCh)
+ if err != nil {
+ return
+ }
+ defer chatSub.Unsubscribe()
+ }
+
+ ctx := r.Context()
+ for {
+ select {
+ case <-ctx.Done():
+ return
+
+ case <-gameCh:
+ // Drain backed-up game updates
+ for {
+ select {
+ case <-gameCh:
+ default:
+ goto drained
+ }
+ }
+ drained:
+ si, ok = snakeStore.Get(gameID)
+ if !ok {
+ return
+ }
+ mySlot = si.GetPlayerSlot(playerID)
+ sg = si.GetGame()
+ if err := sse.PatchElementTempl(components.Board(sg)); err != nil {
+ return
+ }
+ if err := sse.PatchElementTempl(components.StatusBanner(sg, mySlot, gameID)); err != nil {
+ return
+ }
+ if err := sse.PatchElementTempl(components.PlayerList(sg, mySlot)); err != nil {
+ return
+ }
+
+ case msg := <-chatCh:
+ if msg == nil {
+ continue
+ }
+ var cm components.ChatMessage
+ if err := json.Unmarshal(msg.Data, &cm); err != nil {
+ continue
+ }
+ chatMu.Lock()
+ chatMessages = append(chatMessages, cm)
+ if len(chatMessages) > 50 {
+ chatMessages = chatMessages[len(chatMessages)-50:]
+ }
+ msgs := make([]components.ChatMessage, len(chatMessages))
+ copy(msgs, chatMessages)
+ chatMu.Unlock()
+
+ if err := sse.PatchElementTempl(components.Chat(msgs, gameID)); err != nil {
+ return
+ }
+ }
+ }
+ }
+}
+
+func HandleSetDirection(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ playerID := getPlayerID(sessions, r)
+ slot := si.GetPlayerSlot(playerID)
+ if slot < 0 {
+ http.Error(w, "not in game", http.StatusForbidden)
+ return
+ }
+
+ dStr := r.URL.Query().Get("d")
+ d, err := strconv.Atoi(dStr)
+ if err != nil || d < 0 || d > 3 {
+ http.Error(w, "invalid direction", http.StatusBadRequest)
+ return
+ }
+
+ si.SetDirection(slot, snake.Direction(d))
+ w.WriteHeader(http.StatusOK)
+ }
+}
+
+type chatSignals struct {
+ ChatMsg string `json:"chatMsg"`
+}
+
+func HandleSendChat(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ var signals chatSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.ChatMsg == "" {
+ return
+ }
+
+ playerID := getPlayerID(sessions, r)
+ slot := si.GetPlayerSlot(playerID)
+ if slot < 0 {
+ http.Error(w, "not in game", http.StatusForbidden)
+ return
+ }
+
+ sg := si.GetGame()
+ cm := components.ChatMessage{
+ Nickname: sg.Players[slot].Nickname,
+ Slot: slot,
+ Message: signals.ChatMsg,
+ }
+ data, err := json.Marshal(cm)
+ if err != nil {
+ return
+ }
+ nc.Publish("snake.chat."+gameID, data) //nolint:errcheck
+
+ sse := datastar.NewSSE(w, r)
+ sse.MarshalAndPatchSignals(map[string]any{"chatMsg": ""}) //nolint:errcheck
+ }
+}
+
+type nicknameSignals struct {
+ Nickname string `json:"nickname"`
+}
+
+func HandleSetNickname(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ var signals nicknameSignals
+ if err := datastar.ReadSignals(r, &signals); err != nil {
+ http.Error(w, err.Error(), http.StatusBadRequest)
+ return
+ }
+
+ if signals.Nickname == "" {
+ return
+ }
+
+ sessions.Put(r.Context(), "nickname", signals.Nickname)
+
+ playerID := getPlayerID(sessions, r)
+ userID := sessions.GetString(r.Context(), "user_id")
+
+ if si.GetPlayerSlot(playerID) < 0 {
+ player := &snake.Player{
+ ID: playerID,
+ Nickname: signals.Nickname,
+ }
+ if userID != "" {
+ player.UserID = &userID
+ }
+ si.Join(player)
+ }
+
+ sse := datastar.NewSSE(w, r)
+ sse.Redirect("/snake/" + gameID) //nolint:errcheck
+ }
+}
+
+func HandleRematch(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
+ return func(w http.ResponseWriter, r *http.Request) {
+ gameID := chi.URLParam(r, "game_id")
+ si, ok := snakeStore.Get(gameID)
+ if !ok {
+ http.Error(w, "game not found", http.StatusNotFound)
+ return
+ }
+
+ newSI := si.CreateRematch()
+ sse := datastar.NewSSE(w, r)
+ if newSI != nil {
+ sse.Redirect("/snake/" + newSI.ID()) //nolint:errcheck
+ }
+ }
+}
diff --git a/features/snakegame/pages/game_templ.go b/features/snakegame/pages/game_templ.go
new file mode 100644
index 0000000..a7023cb
--- /dev/null
+++ b/features/snakegame/pages/game_templ.go
@@ -0,0 +1,277 @@
+// Code generated by templ - DO NOT EDIT.
+
+// templ: version: v0.3.1001
+package pages
+
+//lint:file-ignore SA4006 This context is only used if a nested component is present.
+
+import "github.com/a-h/templ"
+import templruntime "github.com/a-h/templ/runtime"
+
+import (
+ "fmt"
+
+ "github.com/ryanhamamura/c4/features/common/components"
+ "github.com/ryanhamamura/c4/features/common/layouts"
+ snakecomponents "github.com/ryanhamamura/c4/features/snakegame/components"
+ "github.com/ryanhamamura/c4/snake"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+// keydownScript builds the inline JS for a single data-on:keydown handler
+// that dispatches WASD/arrow keys to direction POST endpoints.
+func keydownScript(gameID string) string {
+ return fmt.Sprintf(
+ "const k=evt.key;"+
+ "if(k==='w'||k==='ArrowUp'){evt.preventDefault();%s}"+
+ "else if(k==='s'||k==='ArrowDown'){evt.preventDefault();%s}"+
+ "else if(k==='a'||k==='ArrowLeft'){evt.preventDefault();%s}"+
+ "else if(k==='d'||k==='ArrowRight'){evt.preventDefault();%s}",
+ datastar.PostSSE("/api/snake/%s/dir?d=0", gameID),
+ datastar.PostSSE("/api/snake/%s/dir?d=1", gameID),
+ datastar.PostSSE("/api/snake/%s/dir?d=2", gameID),
+ datastar.PostSSE("/api/snake/%s/dir?d=3", gameID),
+ )
+}
+
+func GamePage(sg *snake.SnakeGame, mySlot int, messages []snakecomponents.ChatMessage, gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var1 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var1 == nil {
+ templ_7745c5c3_Var1 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var2 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = components.BackToLobby().Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "~~~~ ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = snakecomponents.PlayerList(sg, mySlot).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = snakecomponents.StatusBanner(sg, mySlot, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ if sg.Status == snake.StatusInProgress || sg.Status == snake.StatusFinished {
+ if sg.Mode == snake.ModeMultiplayer {
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 6, "")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = snakecomponents.Board(sg).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = snakecomponents.Chat(messages, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "
")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ } else {
+ templ_7745c5c3_Err = snakecomponents.Board(sg).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ } else if sg.Mode == snake.ModeMultiplayer {
+ templ_7745c5c3_Err = snakecomponents.Chat(messages, gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ if sg.Mode == snake.ModeMultiplayer && (sg.Status == snake.StatusWaitingForPlayers || sg.Status == snake.StatusCountdown) {
+ templ_7745c5c3_Err = snakecomponents.InviteLink(gameID).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ }
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, " ")
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Snake").Render(templ.WithChildren(ctx, templ_7745c5c3_Var2), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func JoinPage(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var6 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var6 == nil {
+ templ_7745c5c3_Var6 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var7 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = components.GameJoinPrompt(
+ fmt.Sprintf("/login?return=/snake/%s", gameID),
+ fmt.Sprintf("/register?return=/snake/%s", gameID),
+ fmt.Sprintf("/snake/%s", gameID),
+ ).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Snake - Join").Render(templ.WithChildren(ctx, templ_7745c5c3_Var7), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+func NicknamePage(gameID string) templ.Component {
+ return templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ if templ_7745c5c3_CtxErr := ctx.Err(); templ_7745c5c3_CtxErr != nil {
+ return templ_7745c5c3_CtxErr
+ }
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Var8 := templ.GetChildren(ctx)
+ if templ_7745c5c3_Var8 == nil {
+ templ_7745c5c3_Var8 = templ.NopComponent
+ }
+ ctx = templ.ClearChildren(ctx)
+ templ_7745c5c3_Var9 := templruntime.GeneratedTemplate(func(templ_7745c5c3_Input templruntime.GeneratedComponentInput) (templ_7745c5c3_Err error) {
+ templ_7745c5c3_W, ctx := templ_7745c5c3_Input.Writer, templ_7745c5c3_Input.Context
+ templ_7745c5c3_Buffer, templ_7745c5c3_IsBuffer := templruntime.GetBuffer(templ_7745c5c3_W)
+ if !templ_7745c5c3_IsBuffer {
+ defer func() {
+ templ_7745c5c3_BufErr := templruntime.ReleaseBuffer(templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err == nil {
+ templ_7745c5c3_Err = templ_7745c5c3_BufErr
+ }
+ }()
+ }
+ ctx = templ.InitializeContext(ctx)
+ templ_7745c5c3_Err = components.NicknamePrompt(fmt.Sprintf("/api/snake/%s/join", gameID)).Render(ctx, templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+ templ_7745c5c3_Err = layouts.Base("Snake - Join").Render(templ.WithChildren(ctx, templ_7745c5c3_Var9), templ_7745c5c3_Buffer)
+ if templ_7745c5c3_Err != nil {
+ return templ_7745c5c3_Err
+ }
+ return nil
+ })
+}
+
+var _ = templruntime.GeneratedTemplate
diff --git a/features/snakegame/routes.go b/features/snakegame/routes.go
new file mode 100644
index 0000000..c4c8334
--- /dev/null
+++ b/features/snakegame/routes.go
@@ -0,0 +1,22 @@
+package snakegame
+
+import (
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/nats-io/nats.go"
+ "github.com/ryanhamamura/c4/snake"
+)
+
+func SetupRoutes(router chi.Router, snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) error {
+ router.Get("/snake/{game_id}", HandleSnakePage(snakeStore, sessions))
+ router.Get("/snake/{game_id}/events", HandleSnakeEvents(snakeStore, nc, sessions))
+
+ router.Route("/api/snake/{game_id}", func(r chi.Router) {
+ r.Post("/dir", HandleSetDirection(snakeStore, sessions))
+ r.Post("/chat", HandleSendChat(snakeStore, nc, sessions))
+ r.Post("/join", HandleSetNickname(snakeStore, sessions))
+ r.Post("/rematch", HandleRematch(snakeStore, sessions))
+ })
+
+ return nil
+}
diff --git a/game/types.go b/game/types.go
index 73cd33e..71f0ae8 100644
--- a/game/types.go
+++ b/game/types.go
@@ -67,3 +67,11 @@ func (g *Game) WinningCellsFromJSON(data string) error {
}
return json.Unmarshal([]byte(data), &g.WinningCells)
}
+
+// ChatMessage is the domain type for persisted C4 chat messages.
+type ChatMessage struct {
+ Nickname string `json:"nickname"`
+ Color int `json:"color"` // 1=Red, 2=Yellow
+ Message string `json:"message"`
+ Time int64 `json:"time"`
+}
diff --git a/go.mod b/go.mod
index 5b66895..d7a24c5 100644
--- a/go.mod
+++ b/go.mod
@@ -3,51 +3,244 @@ module github.com/ryanhamamura/c4
go 1.25.4
require (
+ github.com/a-h/templ v0.3.1001
github.com/alexedwards/scs/sqlite3store v0.0.0-20251002162104-209de6e426de
github.com/alexedwards/scs/v2 v2.9.0
+ github.com/delaneyj/toolbelt v0.9.1
+ github.com/go-chi/chi/v5 v5.2.5
github.com/google/uuid v1.6.0
github.com/joho/godotenv v1.5.1
+ github.com/nats-io/nats-server/v2 v2.12.2
+ github.com/nats-io/nats.go v1.48.0
github.com/pressly/goose/v3 v3.27.0
github.com/rs/zerolog v1.34.0
- github.com/ryanhamamura/via v0.23.0
+ github.com/starfederation/datastar-go v1.1.0
golang.org/x/crypto v0.48.0
+ golang.org/x/sync v0.19.0
modernc.org/sqlite v1.46.1
)
require (
+ cel.dev/expr v0.25.1 // indirect
+ charm.land/bubbles/v2 v2.0.0-rc.1 // indirect
+ charm.land/bubbletea/v2 v2.0.0-rc.2 // indirect
+ charm.land/lipgloss/v2 v2.0.0-beta.3.0.20251106192539-4b304240aab7 // indirect
+ cloud.google.com/go v0.123.0 // indirect
+ cloud.google.com/go/auth v0.17.0 // indirect
+ cloud.google.com/go/auth/oauth2adapt v0.2.8 // indirect
+ cloud.google.com/go/compute/metadata v0.9.0 // indirect
+ cloud.google.com/go/iam v1.5.3 // indirect
+ cloud.google.com/go/monitoring v1.24.2 // indirect
+ cloud.google.com/go/storage v1.58.0 // indirect
+ dario.cat/mergo v1.0.2 // indirect
+ filippo.io/edwards25519 v1.2.0 // indirect
github.com/CAFxX/httpcompression v0.0.9 // indirect
+ github.com/ClickHouse/ch-go v0.71.0 // indirect
+ github.com/ClickHouse/clickhouse-go/v2 v2.43.0 // indirect
+ github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.30.0 // indirect
+ github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.54.0 // indirect
+ github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.54.0 // indirect
+ github.com/Ladicle/tabwriter v1.0.0 // indirect
+ github.com/Masterminds/semver/v3 v3.4.0 // indirect
+ github.com/a-h/parse v0.0.0-20250122154542-74294addb73e // indirect
+ github.com/air-verse/air v1.64.5 // indirect
+ github.com/alecthomas/chroma/v2 v2.23.0 // indirect
github.com/andybalholm/brotli v1.2.0 // indirect
github.com/antithesishq/antithesis-sdk-go v0.5.0 // indirect
+ github.com/antlr4-go/antlr/v4 v4.13.1 // indirect
+ github.com/atotto/clipboard v0.1.4 // indirect
+ github.com/aws/aws-sdk-go-v2 v1.41.0 // indirect
+ github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.7.4 // indirect
+ github.com/aws/aws-sdk-go-v2/config v1.32.6 // indirect
+ github.com/aws/aws-sdk-go-v2/credentials v1.19.6 // indirect
+ github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.18.16 // indirect
+ github.com/aws/aws-sdk-go-v2/internal/configsources v1.4.16 // indirect
+ github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.7.16 // indirect
+ github.com/aws/aws-sdk-go-v2/internal/ini v1.8.4 // indirect
+ github.com/aws/aws-sdk-go-v2/internal/v4a v1.4.16 // indirect
+ github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.13.4 // indirect
+ github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.9.7 // indirect
+ github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.13.16 // indirect
+ github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.19.16 // indirect
+ github.com/aws/aws-sdk-go-v2/service/s3 v1.95.0 // indirect
+ github.com/aws/aws-sdk-go-v2/service/signin v1.0.4 // indirect
+ github.com/aws/aws-sdk-go-v2/service/sso v1.30.8 // indirect
+ github.com/aws/aws-sdk-go-v2/service/ssooidc v1.35.12 // indirect
+ github.com/aws/aws-sdk-go-v2/service/sts v1.41.5 // indirect
+ github.com/aws/smithy-go v1.24.0 // indirect
+ github.com/bep/godartsass/v2 v2.5.0 // indirect
+ github.com/bep/golibsass v1.2.0 // indirect
+ github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d // indirect
github.com/cenkalti/backoff v2.2.1+incompatible // indirect
- github.com/delaneyj/toolbelt v0.9.1 // indirect
+ github.com/cenkalti/backoff/v4 v4.3.0 // indirect
+ github.com/cespare/xxhash/v2 v2.3.0 // indirect
+ github.com/chainguard-dev/git-urls v1.0.2 // indirect
+ github.com/charmbracelet/colorprofile v0.3.3 // indirect
+ github.com/charmbracelet/ultraviolet v0.0.0-20251116181749-377898bcce38 // indirect
+ github.com/charmbracelet/x/ansi v0.11.1 // indirect
+ github.com/charmbracelet/x/term v0.2.2 // indirect
+ github.com/charmbracelet/x/termios v0.1.1 // indirect
+ github.com/charmbracelet/x/windows v0.2.2 // indirect
+ github.com/cli/browser v1.3.0 // indirect
+ github.com/clipperhouse/displaywidth v0.5.0 // indirect
+ github.com/clipperhouse/stringish v0.1.1 // indirect
+ github.com/clipperhouse/uax29/v2 v2.3.0 // indirect
+ github.com/cncf/xds/go v0.0.0-20251210132809-ee656c7534f5 // indirect
+ github.com/coder/websocket v1.8.14 // indirect
+ github.com/cubicdaiya/gonp v1.0.4 // indirect
+ github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
+ github.com/dlclark/regexp2 v1.11.5 // indirect
+ github.com/dominikbraun/graph v0.23.0 // indirect
github.com/dustin/go-humanize v1.0.1 // indirect
+ github.com/elastic/go-sysinfo v1.15.4 // indirect
+ github.com/elastic/go-windows v1.0.2 // indirect
+ github.com/elliotchance/orderedmap/v3 v3.1.0 // indirect
+ github.com/envoyproxy/go-control-plane/envoy v1.36.0 // indirect
+ github.com/envoyproxy/protoc-gen-validate v1.3.0 // indirect
+ github.com/fatih/color v1.18.0 // indirect
+ github.com/fatih/structtag v1.2.0 // indirect
+ github.com/felixge/httpsnoop v1.0.4 // indirect
+ github.com/fsnotify/fsnotify v1.9.0 // indirect
+ github.com/go-faster/city v1.0.1 // indirect
+ github.com/go-faster/errors v0.7.1 // indirect
+ github.com/go-jose/go-jose/v4 v4.1.3 // indirect
+ github.com/go-logr/logr v1.4.3 // indirect
+ github.com/go-logr/stdr v1.2.2 // indirect
+ github.com/go-sql-driver/mysql v1.9.3 // indirect
+ github.com/go-task/slim-sprig/v3 v3.0.0 // indirect
+ github.com/go-task/task/v3 v3.48.0 // indirect
+ github.com/go-task/template v0.2.0 // indirect
+ github.com/gobwas/glob v0.2.3 // indirect
+ github.com/gohugoio/hugo v0.149.1 // indirect
+ github.com/golang-jwt/jwt/v4 v4.5.2 // indirect
+ github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 // indirect
+ github.com/golang-sql/sqlexp v0.1.0 // indirect
+ github.com/google/cel-go v0.26.1 // indirect
github.com/google/go-tpm v0.9.7 // indirect
+ github.com/google/s2a-go v0.1.9 // indirect
+ github.com/googleapis/enterprise-certificate-proxy v0.3.7 // indirect
+ github.com/googleapis/gax-go/v2 v2.15.0 // indirect
+ github.com/hashicorp/aws-sdk-go-base/v2 v2.0.0-beta.70 // indirect
+ github.com/hashicorp/go-cleanhttp v0.5.2 // indirect
+ github.com/hashicorp/go-getter v1.8.4 // indirect
+ github.com/hashicorp/go-version v1.8.0 // indirect
github.com/hookenz/gotailwind/v4 v4.2.1 // indirect
+ github.com/inconshreveable/mousetrap v1.1.0 // indirect
+ github.com/jackc/pgpassfile v1.0.0 // indirect
+ github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 // indirect
+ github.com/jackc/pgx/v5 v5.8.0 // indirect
+ github.com/jackc/puddle/v2 v2.2.2 // indirect
+ github.com/jinzhu/inflection v1.0.0 // indirect
+ github.com/jonboulle/clockwork v0.5.0 // indirect
github.com/klauspost/compress v1.18.4 // indirect
+ github.com/klauspost/cpuid/v2 v2.3.0 // indirect
+ github.com/klauspost/pgzip v1.2.6 // indirect
+ github.com/lucasb-eyer/go-colorful v1.3.0 // indirect
github.com/mattn/go-colorable v0.1.14 // indirect
github.com/mattn/go-isatty v0.0.20 // indirect
+ github.com/mattn/go-runewidth v0.0.19 // indirect
+ github.com/mattn/go-sqlite3 v1.14.32 // indirect
github.com/mfridman/interpolate v0.0.2 // indirect
+ github.com/mfridman/xflag v0.1.0 // indirect
+ github.com/microsoft/go-mssqldb v1.9.6 // indirect
github.com/minio/highwayhash v1.0.4-0.20251030100505-070ab1a87a76 // indirect
+ github.com/mitchellh/go-homedir v1.1.0 // indirect
+ github.com/mitchellh/hashstructure/v2 v2.0.2 // indirect
+ github.com/muesli/cancelreader v0.2.2 // indirect
+ github.com/natefinch/atomic v1.0.1 // indirect
github.com/nats-io/jwt/v2 v2.8.0 // indirect
- github.com/nats-io/nats-server/v2 v2.12.2 // indirect
- github.com/nats-io/nats.go v1.48.0 // indirect
github.com/nats-io/nkeys v0.4.12 // indirect
github.com/nats-io/nuid v1.0.1 // indirect
github.com/ncruces/go-strftime v1.0.0 // indirect
+ github.com/paulmach/orb v0.12.0 // indirect
+ github.com/pelletier/go-toml v1.9.5 // indirect
+ github.com/pelletier/go-toml/v2 v2.2.4 // indirect
+ github.com/pganalyze/pg_query_go/v6 v6.1.0 // indirect
+ github.com/pierrec/lz4/v4 v4.1.25 // indirect
+ github.com/pingcap/errors v0.11.5-0.20240311024730-e056997136bb // indirect
+ github.com/pingcap/failpoint v0.0.0-20240528011301-b51a646c7c86 // indirect
+ github.com/pingcap/log v1.1.0 // indirect
+ github.com/pingcap/tidb/pkg/parser v0.0.0-20250324122243-d51e00e5bbf0 // indirect
github.com/pkg/errors v0.9.1 // indirect
+ github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10 // indirect
+ github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
+ github.com/prometheus/procfs v0.19.2 // indirect
+ github.com/puzpuzpuz/xsync/v4 v4.3.0 // indirect
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
+ github.com/rivo/uniseg v0.4.7 // indirect
+ github.com/riza-io/grpc-go v0.2.0 // indirect
+ github.com/sajari/fuzzy v1.0.0 // indirect
+ github.com/segmentio/asm v1.2.1 // indirect
github.com/sethvargo/go-retry v0.3.0 // indirect
- github.com/starfederation/datastar-go v1.0.3 // indirect
+ github.com/shopspring/decimal v1.4.0 // indirect
+ github.com/spf13/afero v1.14.0 // indirect
+ github.com/spf13/cast v1.9.2 // indirect
+ github.com/spf13/cobra v1.9.1 // indirect
+ github.com/spf13/pflag v1.0.10 // indirect
+ github.com/spiffe/go-spiffe/v2 v2.6.0 // indirect
+ github.com/sqlc-dev/sqlc v1.30.0 // indirect
+ github.com/stoewer/go-strcase v1.2.0 // indirect
+ github.com/stretchr/objx v0.5.2 // indirect
+ github.com/stretchr/testify v1.11.1 // indirect
+ github.com/tdewolff/parse/v2 v2.8.3 // indirect
+ github.com/tetratelabs/wazero v1.9.0 // indirect
+ github.com/tursodatabase/libsql-client-go v0.0.0-20251219100830-236aa1ff8acc // indirect
+ github.com/u-root/u-root v0.15.1-0.20251208185023-2f8c7e763cf8 // indirect
+ github.com/u-root/uio v0.0.0-20240224005618-d2acac8f3701 // indirect
+ github.com/ulikunitz/xz v0.5.15 // indirect
github.com/valyala/bytebufferpool v1.0.0 // indirect
+ github.com/vertica/vertica-sql-go v1.3.5 // indirect
+ github.com/wasilibs/go-pgquery v0.0.0-20250409022910-10ac41983c07 // indirect
+ github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52 // indirect
+ github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
+ github.com/ydb-platform/ydb-go-genproto v0.0.0-20260128080146-c4ed16b24b37 // indirect
+ github.com/ydb-platform/ydb-go-sdk/v3 v3.127.0 // indirect
+ github.com/zeebo/xxh3 v1.0.2 // indirect
+ github.com/ziutek/mymysql v1.5.4 // indirect
+ go.opentelemetry.io/auto/sdk v1.2.1 // indirect
+ go.opentelemetry.io/contrib/detectors/gcp v1.39.0 // indirect
+ go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.63.0 // indirect
+ go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.65.0 // indirect
+ go.opentelemetry.io/otel v1.40.0 // indirect
+ go.opentelemetry.io/otel/metric v1.40.0 // indirect
+ go.opentelemetry.io/otel/sdk v1.40.0 // indirect
+ go.opentelemetry.io/otel/sdk/metric v1.40.0 // indirect
+ go.opentelemetry.io/otel/trace v1.40.0 // indirect
+ go.uber.org/atomic v1.11.0 // indirect
go.uber.org/multierr v1.11.0 // indirect
+ go.uber.org/zap v1.27.1 // indirect
+ go.yaml.in/yaml/v3 v3.0.4 // indirect
+ go.yaml.in/yaml/v4 v4.0.0-rc.3 // indirect
golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa // indirect
- golang.org/x/sync v0.19.0 // indirect
+ golang.org/x/mod v0.33.0 // indirect
+ golang.org/x/net v0.50.0 // indirect
+ golang.org/x/oauth2 v0.34.0 // indirect
golang.org/x/sys v0.41.0 // indirect
+ golang.org/x/term v0.40.0 // indirect
+ golang.org/x/text v0.34.0 // indirect
golang.org/x/time v0.14.0 // indirect
- maragu.dev/gomponents v1.2.0 // indirect
+ golang.org/x/tools v0.42.0 // indirect
+ google.golang.org/api v0.256.0 // indirect
+ google.golang.org/genproto v0.0.0-20250922171735-9219d122eba9 // indirect
+ google.golang.org/genproto/googleapis/api v0.0.0-20251202230838-ff82c1b0f217 // indirect
+ google.golang.org/genproto/googleapis/rpc v0.0.0-20260217215200-42d3e9bedb6d // indirect
+ google.golang.org/grpc v1.79.1 // indirect
+ google.golang.org/protobuf v1.36.11 // indirect
+ gopkg.in/natefinch/lumberjack.v2 v2.2.1 // indirect
+ gopkg.in/yaml.v3 v3.0.1 // indirect
+ howett.net/plist v1.0.1 // indirect
modernc.org/libc v1.68.0 // indirect
modernc.org/mathutil v1.7.1 // indirect
modernc.org/memory v1.11.0 // indirect
+ mvdan.cc/sh/moreinterp v0.0.0-20260120230322-19def062a997 // indirect
+ mvdan.cc/sh/v3 v3.12.1-0.20260124232039-e74afc18e65b // indirect
)
-tool github.com/hookenz/gotailwind/v4
+tool (
+ github.com/a-h/templ/cmd/templ
+ github.com/air-verse/air
+ github.com/go-task/task/v3/cmd/task
+ github.com/hookenz/gotailwind/v4
+ github.com/pressly/goose/v3/cmd/goose
+ github.com/sqlc-dev/sqlc/cmd/sqlc
+)
diff --git a/go.sum b/go.sum
index e30bc87..df2ccbd 100644
--- a/go.sum
+++ b/go.sum
@@ -1,5 +1,82 @@
+cel.dev/expr v0.25.1 h1:1KrZg61W6TWSxuNZ37Xy49ps13NUovb66QLprthtwi4=
+cel.dev/expr v0.25.1/go.mod h1:hrXvqGP6G6gyx8UAHSHJ5RGk//1Oj5nXQ2NI02Nrsg4=
+charm.land/bubbles/v2 v2.0.0-rc.1 h1:EiIFVAc3Zi/yY86td+79mPhHR7AqZ1OxF+6ztpOCRaM=
+charm.land/bubbles/v2 v2.0.0-rc.1/go.mod h1:5AbN6cEd/47gkEf8TgiQ2O3RZ5QxMS14l9W+7F9fPC4=
+charm.land/bubbletea/v2 v2.0.0-rc.2 h1:TdTbUOFzbufDJmSz/3gomL6q+fR6HwfY+P13hXQzD7k=
+charm.land/bubbletea/v2 v2.0.0-rc.2/go.mod h1:IXFmnCnMLTWw/KQ9rEatSYqbAPAYi8kA3Yqwa1SFnLk=
+charm.land/lipgloss/v2 v2.0.0-beta.3.0.20251106192539-4b304240aab7 h1:059k1h5vvZ4ASinki9nmBguxu9Rq0UDDSa6q8LOUphk=
+charm.land/lipgloss/v2 v2.0.0-beta.3.0.20251106192539-4b304240aab7/go.mod h1:1qZyvvVCenJO2M1ac2mX0yyiIZJoZmDM4DG4s0udJkU=
+cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+cloud.google.com/go v0.123.0 h1:2NAUJwPR47q+E35uaJeYoNhuNEM9kM8SjgRgdeOJUSE=
+cloud.google.com/go v0.123.0/go.mod h1:xBoMV08QcqUGuPW65Qfm1o9Y4zKZBpGS+7bImXLTAZU=
+cloud.google.com/go/auth v0.17.0 h1:74yCm7hCj2rUyyAocqnFzsAYXgJhrG26XCFimrc/Kz4=
+cloud.google.com/go/auth v0.17.0/go.mod h1:6wv/t5/6rOPAX4fJiRjKkJCvswLwdet7G8+UGXt7nCQ=
+cloud.google.com/go/auth/oauth2adapt v0.2.8 h1:keo8NaayQZ6wimpNSmW5OPc283g65QNIiLpZnkHRbnc=
+cloud.google.com/go/auth/oauth2adapt v0.2.8/go.mod h1:XQ9y31RkqZCcwJWNSx2Xvric3RrU88hAYYbjDWYDL+c=
+cloud.google.com/go/compute/metadata v0.9.0 h1:pDUj4QMoPejqq20dK0Pg2N4yG9zIkYGdBtwLoEkH9Zs=
+cloud.google.com/go/compute/metadata v0.9.0/go.mod h1:E0bWwX5wTnLPedCKqk3pJmVgCBSM6qQI1yTBdEb3C10=
+cloud.google.com/go/iam v1.5.3 h1:+vMINPiDF2ognBJ97ABAYYwRgsaqxPbQDlMnbHMjolc=
+cloud.google.com/go/iam v1.5.3/go.mod h1:MR3v9oLkZCTlaqljW6Eb2d3HGDGK5/bDv93jhfISFvU=
+cloud.google.com/go/logging v1.13.0 h1:7j0HgAp0B94o1YRDqiqm26w4q1rDMH7XNRU34lJXHYc=
+cloud.google.com/go/logging v1.13.0/go.mod h1:36CoKh6KA/M0PbhPKMq6/qety2DCAErbhXT62TuXALA=
+cloud.google.com/go/longrunning v0.7.0 h1:FV0+SYF1RIj59gyoWDRi45GiYUMM3K1qO51qoboQT1E=
+cloud.google.com/go/longrunning v0.7.0/go.mod h1:ySn2yXmjbK9Ba0zsQqunhDkYi0+9rlXIwnoAf+h+TPY=
+cloud.google.com/go/monitoring v1.24.2 h1:5OTsoJ1dXYIiMiuL+sYscLc9BumrL3CarVLL7dd7lHM=
+cloud.google.com/go/monitoring v1.24.2/go.mod h1:x7yzPWcgDRnPEv3sI+jJGBkwl5qINf+6qY4eq0I9B4U=
+cloud.google.com/go/storage v1.58.0 h1:PflFXlmFJjG/nBeR9B7pKddLQWaFaRWx4uUi/LyNxxo=
+cloud.google.com/go/storage v1.58.0/go.mod h1:cMWbtM+anpC74gn6qjLh+exqYcfmB9Hqe5z6adx+CLI=
+cloud.google.com/go/trace v1.11.6 h1:2O2zjPzqPYAHrn3OKl029qlqG6W8ZdYaOWRyr8NgMT4=
+cloud.google.com/go/trace v1.11.6/go.mod h1:GA855OeDEBiBMzcckLPE2kDunIpC72N+Pq8WFieFjnI=
+dario.cat/mergo v1.0.2 h1:85+piFYR1tMbRrLcDwR18y4UKJ3aH1Tbzi24VRW1TK8=
+dario.cat/mergo v1.0.2/go.mod h1:E/hbnu0NxMFBjpMIE34DRGLWqDy0g5FuKDhCb31ngxA=
+filippo.io/edwards25519 v1.2.0 h1:crnVqOiS4jqYleHd9vaKZ+HKtHfllngJIiOpNpoJsjo=
+filippo.io/edwards25519 v1.2.0/go.mod h1:xzAOLCNug/yB62zG1bQ8uziwrIqIuxhctzJT18Q77mc=
+github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.1 h1:Wc1ml6QlJs2BHQ/9Bqu1jiyggbsSjramq2oUmp5WeIo=
+github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.1/go.mod h1:Ot/6aikWnKWi4l9QB7qVSwa8iMphQNqkWALMoNT3rzM=
+github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1 h1:B+blDbyVIG3WaikNxPnhPiJ1MThR03b3vKGtER95TP4=
+github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1/go.mod h1:JdM5psgjfBf5fo2uWOZhflPWyDBZ/O/CNAH9CtsuZE4=
+github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1 h1:FPKJS1T+clwv+OLGt13a8UjqeRuh0O4SJ3lUriThc+4=
+github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.1/go.mod h1:j2chePtV91HrC22tGoRX3sGY42uF13WzmmV80/OdVAA=
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1 h1:Wgf5rZba3YZqeTNJPtvqZoBu1sBN/L4sry+u2U3Y75w=
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/azkeys v1.3.1/go.mod h1:xxCBG/f/4Vbmh2XQJBsOmNdxWUY5j/s27jujKPbQf14=
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1 h1:bFWuoEKg+gImo7pvkiQEFAc8ocibADgXeiLAxWhWmkI=
+github.com/Azure/azure-sdk-for-go/sdk/security/keyvault/internal v1.1.1/go.mod h1:Vih/3yc6yac2JzU4hzpaDupBJP0Flaia9rXXrU8xyww=
+github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2 h1:oygO0locgZJe7PpYPXT5A29ZkwJaPqcva7BVeemZOZs=
+github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
+github.com/BurntSushi/locker v0.0.0-20171006230638-a6e239ea1c69 h1:+tu3HOoMXB7RXEINRVIpxJCT+KdYiI7LAEAUrOw3dIU=
+github.com/BurntSushi/locker v0.0.0-20171006230638-a6e239ea1c69/go.mod h1:L1AbZdiDllfyYH5l5OkAaZtk7VkWe89bPJFmnDBNHxg=
+github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
github.com/CAFxX/httpcompression v0.0.9 h1:0ue2X8dOLEpxTm8tt+OdHcgA+gbDge0OqFQWGKSqgrg=
github.com/CAFxX/httpcompression v0.0.9/go.mod h1:XX8oPZA+4IDcfZ0A71Hz0mZsv/YJOgYygkFhizVPilM=
+github.com/ClickHouse/ch-go v0.71.0 h1:bUdZ/EZj/LcVHsMqaRUP2holqygrPWQKeMjc6nZoyRM=
+github.com/ClickHouse/ch-go v0.71.0/go.mod h1:NwbNc+7jaqfY58dmdDUbG4Jl22vThgx1cYjBw0vtgXw=
+github.com/ClickHouse/clickhouse-go/v2 v2.43.0 h1:fUR05TrF1GyvLDa/mAQjkx7KbgwdLRffs2n9O3WobtE=
+github.com/ClickHouse/clickhouse-go/v2 v2.43.0/go.mod h1:o6jf7JM/zveWC/PP277BLxjHy5KjnGX/jfljhM4s34g=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.30.0 h1:sBEjpZlNHzK1voKq9695PJSX2o5NEXl7/OL3coiIY0c=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.30.0/go.mod h1:P4WPRUkOhJC13W//jWpyfJNDAIpvRbAUIYLX/4jtlE0=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.54.0 h1:lhhYARPUu3LmHysQ/igznQphfzynnqI3D75oUyw1HXk=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.54.0/go.mod h1:l9rva3ApbBpEJxSNYnwT9N4CDLrWgtq3u8736C5hyJw=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/cloudmock v0.54.0 h1:xfK3bbi6F2RDtaZFtUdKO3osOBIhNb+xTs8lFW6yx9o=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/cloudmock v0.54.0/go.mod h1:vB2GH9GAYYJTO3mEn8oYwzEdhlayZIdQz6zdzgUIRvA=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.54.0 h1:s0WlVbf9qpvkh1c/uDAPElam0WrL7fHRIidgZJ7UqZI=
+github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.54.0/go.mod h1:Mf6O40IAyB9zR/1J8nGDDPirZQQPbYJni8Yisy7NTMc=
+github.com/Ladicle/tabwriter v1.0.0 h1:DZQqPvMumBDwVNElso13afjYLNp0Z7pHqHnu0r4t9Dg=
+github.com/Ladicle/tabwriter v1.0.0/go.mod h1:c4MdCjxQyTbGuQO/gvqJ+IA/89UEwrsD6hUCW98dyp4=
+github.com/Masterminds/semver/v3 v3.4.0 h1:Zog+i5UMtVoCU8oKka5P7i9q9HgrJeGzI9SA1Xbatp0=
+github.com/Masterminds/semver/v3 v3.4.0/go.mod h1:4V+yj/TJE1HU9XfppCwVMZq3I84lprf4nC11bSS5beM=
+github.com/a-h/parse v0.0.0-20250122154542-74294addb73e h1:HjVbSQHy+dnlS6C3XajZ69NYAb5jbGNfHanvm1+iYlo=
+github.com/a-h/parse v0.0.0-20250122154542-74294addb73e/go.mod h1:3mnrkvGpurZ4ZrTDbYU84xhwXW2TjTKShSwjRi2ihfQ=
+github.com/a-h/templ v0.3.1001 h1:yHDTgexACdJttyiyamcTHXr2QkIeVF1MukLy44EAhMY=
+github.com/a-h/templ v0.3.1001/go.mod h1:oCZcnKRf5jjsGpf2yELzQfodLphd2mwecwG4Crk5HBo=
+github.com/air-verse/air v1.64.5 h1:+gs/NgTzYYe+gGPyfHy3XxpJReQWC1pIsiKIg0LgNt4=
+github.com/air-verse/air v1.64.5/go.mod h1:OaJZSfZqf7wyjS2oP/CcEVyIt0JmZuPh5x1gdtklmmY=
+github.com/alecthomas/assert/v2 v2.11.0 h1:2Q9r3ki8+JYXvGsDyBXwH3LcJ+WK5D0gc5E8vS6K3D0=
+github.com/alecthomas/assert/v2 v2.11.0/go.mod h1:Bze95FyfUr7x34QZrjL+XP+0qgp/zg8yS+TtBj1WA3k=
+github.com/alecthomas/chroma/v2 v2.23.0 h1:u/Orux1J0eLuZDeQ44froV8smumheieI0EofhbyKhhk=
+github.com/alecthomas/chroma/v2 v2.23.0/go.mod h1:NqVhfBR0lte5Ouh3DcthuUCTUpDC9cxBOfyMbMQPs3o=
+github.com/alecthomas/repr v0.5.2 h1:SU73FTI9D1P5UNtvseffFSGmdNci/O6RsqzeXJtP0Qs=
+github.com/alecthomas/repr v0.5.2/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4=
github.com/alexedwards/scs/sqlite3store v0.0.0-20251002162104-209de6e426de h1:c72K9HLu6K442et0j3BUL/9HEYaUJouLkkVANdmqTOo=
github.com/alexedwards/scs/sqlite3store v0.0.0-20251002162104-209de6e426de/go.mod h1:Iyk7S76cxGaiEX/mSYmTZzYehp4KfyylcLaV3OnToss=
github.com/alexedwards/scs/v2 v2.9.0 h1:xa05mVpwTBm1iLeTMNFfAWpKUm4fXAW7CeAViqBVS90=
@@ -7,38 +84,366 @@ github.com/alexedwards/scs/v2 v2.9.0/go.mod h1:ToaROZxyKukJKT/xLcVQAChi5k6+Pn1Gv
github.com/andybalholm/brotli v1.0.5/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ=
github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY=
+github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY=
github.com/antithesishq/antithesis-sdk-go v0.5.0 h1:cudCFF83pDDANcXFzkQPUHHedfnnIbUO3JMr9fqwFJs=
github.com/antithesishq/antithesis-sdk-go v0.5.0/go.mod h1:IUpT2DPAKh6i/YhSbt6Gl3v2yvUZjmKncl7U91fup7E=
+github.com/antlr4-go/antlr/v4 v4.13.1 h1:SqQKkuVZ+zWkMMNkjy5FZe5mr5WURWnlpmOuzYWrPrQ=
+github.com/antlr4-go/antlr/v4 v4.13.1/go.mod h1:GKmUxMtwp6ZgGwZSva4eWPC5mS6vUAmOABFgjdkM7Nw=
+github.com/armon/go-radix v1.0.1-0.20221118154546-54df44f2176c h1:651/eoCRnQ7YtSjAnSzRucrJz+3iGEFt+ysraELS81M=
+github.com/armon/go-radix v1.0.1-0.20221118154546-54df44f2176c/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
+github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4=
+github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI=
+github.com/aws/aws-sdk-go-v2 v1.41.0 h1:tNvqh1s+v0vFYdA1xq0aOJH+Y5cRyZ5upu6roPgPKd4=
+github.com/aws/aws-sdk-go-v2 v1.41.0/go.mod h1:MayyLB8y+buD9hZqkCW3kX1AKq07Y5pXxtgB+rRFhz0=
+github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.7.4 h1:489krEF9xIGkOaaX3CE/Be2uWjiXrkCH6gUX+bZA/BU=
+github.com/aws/aws-sdk-go-v2/aws/protocol/eventstream v1.7.4/go.mod h1:IOAPF6oT9KCsceNTvvYMNHy0+kMF8akOjeDvPENWxp4=
+github.com/aws/aws-sdk-go-v2/config v1.32.6 h1:hFLBGUKjmLAekvi1evLi5hVvFQtSo3GYwi+Bx4lpJf8=
+github.com/aws/aws-sdk-go-v2/config v1.32.6/go.mod h1:lcUL/gcd8WyjCrMnxez5OXkO3/rwcNmvfno62tnXNcI=
+github.com/aws/aws-sdk-go-v2/credentials v1.19.6 h1:F9vWao2TwjV2MyiyVS+duza0NIRtAslgLUM0vTA1ZaE=
+github.com/aws/aws-sdk-go-v2/credentials v1.19.6/go.mod h1:SgHzKjEVsdQr6Opor0ihgWtkWdfRAIwxYzSJ8O85VHY=
+github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.18.16 h1:80+uETIWS1BqjnN9uJ0dBUaETh+P1XwFy5vwHwK5r9k=
+github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.18.16/go.mod h1:wOOsYuxYuB/7FlnVtzeBYRcjSRtQpAW0hCP7tIULMwo=
+github.com/aws/aws-sdk-go-v2/internal/configsources v1.4.16 h1:rgGwPzb82iBYSvHMHXc8h9mRoOUBZIGFgKb9qniaZZc=
+github.com/aws/aws-sdk-go-v2/internal/configsources v1.4.16/go.mod h1:L/UxsGeKpGoIj6DxfhOWHWQ/kGKcd4I1VncE4++IyKA=
+github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.7.16 h1:1jtGzuV7c82xnqOVfx2F0xmJcOw5374L7N6juGW6x6U=
+github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.7.16/go.mod h1:M2E5OQf+XLe+SZGmmpaI2yy+J326aFf6/+54PoxSANc=
+github.com/aws/aws-sdk-go-v2/internal/ini v1.8.4 h1:WKuaxf++XKWlHWu9ECbMlha8WOEGm0OUEZqm4K/Gcfk=
+github.com/aws/aws-sdk-go-v2/internal/ini v1.8.4/go.mod h1:ZWy7j6v1vWGmPReu0iSGvRiise4YI5SkR3OHKTZ6Wuc=
+github.com/aws/aws-sdk-go-v2/internal/v4a v1.4.16 h1:CjMzUs78RDDv4ROu3JnJn/Ig1r6ZD7/T2DXLLRpejic=
+github.com/aws/aws-sdk-go-v2/internal/v4a v1.4.16/go.mod h1:uVW4OLBqbJXSHJYA9svT9BluSvvwbzLQ2Crf6UPzR3c=
+github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.13.4 h1:0ryTNEdJbzUCEWkVXEXoqlXV72J5keC1GvILMOuD00E=
+github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.13.4/go.mod h1:HQ4qwNZh32C3CBeO6iJLQlgtMzqeG17ziAA/3KDJFow=
+github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.9.7 h1:DIBqIrJ7hv+e4CmIk2z3pyKT+3B6qVMgRsawHiR3qso=
+github.com/aws/aws-sdk-go-v2/service/internal/checksum v1.9.7/go.mod h1:vLm00xmBke75UmpNvOcZQ/Q30ZFjbczeLFqGx5urmGo=
+github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.13.16 h1:oHjJHeUy0ImIV0bsrX0X91GkV5nJAyv1l1CC9lnO0TI=
+github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.13.16/go.mod h1:iRSNGgOYmiYwSCXxXaKb9HfOEj40+oTKn8pTxMlYkRM=
+github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.19.16 h1:NSbvS17MlI2lurYgXnCOLvCFX38sBW4eiVER7+kkgsU=
+github.com/aws/aws-sdk-go-v2/service/internal/s3shared v1.19.16/go.mod h1:SwT8Tmqd4sA6G1qaGdzWCJN99bUmPGHfRwwq3G5Qb+A=
+github.com/aws/aws-sdk-go-v2/service/s3 v1.95.0 h1:MIWra+MSq53CFaXXAywB2qg9YvVZifkk6vEGl/1Qor0=
+github.com/aws/aws-sdk-go-v2/service/s3 v1.95.0/go.mod h1:79S2BdqCJpScXZA2y+cpZuocWsjGjJINyXnOsf5DTz8=
+github.com/aws/aws-sdk-go-v2/service/signin v1.0.4 h1:HpI7aMmJ+mm1wkSHIA2t5EaFFv5EFYXePW30p1EIrbQ=
+github.com/aws/aws-sdk-go-v2/service/signin v1.0.4/go.mod h1:C5RdGMYGlfM0gYq/tifqgn4EbyX99V15P2V3R+VHbQU=
+github.com/aws/aws-sdk-go-v2/service/sso v1.30.8 h1:aM/Q24rIlS3bRAhTyFurowU8A0SMyGDtEOY/l/s/1Uw=
+github.com/aws/aws-sdk-go-v2/service/sso v1.30.8/go.mod h1:+fWt2UHSb4kS7Pu8y+BMBvJF0EWx+4H0hzNwtDNRTrg=
+github.com/aws/aws-sdk-go-v2/service/ssooidc v1.35.12 h1:AHDr0DaHIAo8c9t1emrzAlVDFp+iMMKnPdYy6XO4MCE=
+github.com/aws/aws-sdk-go-v2/service/ssooidc v1.35.12/go.mod h1:GQ73XawFFiWxyWXMHWfhiomvP3tXtdNar/fi8z18sx0=
+github.com/aws/aws-sdk-go-v2/service/sts v1.41.5 h1:SciGFVNZ4mHdm7gpD1dgZYnCuVdX1s+lFTg4+4DOy70=
+github.com/aws/aws-sdk-go-v2/service/sts v1.41.5/go.mod h1:iW40X4QBmUxdP+fZNOpfmkdMZqsovezbAeO+Ubiv2pk=
+github.com/aws/smithy-go v1.24.0 h1:LpilSUItNPFr1eY85RYgTIg5eIEPtvFbskaFcmmIUnk=
+github.com/aws/smithy-go v1.24.0/go.mod h1:LEj2LM3rBRQJxPZTB4KuzZkaZYnZPnvgIhb4pu07mx0=
+github.com/aymanbagabas/go-udiff v0.3.1 h1:LV+qyBQ2pqe0u42ZsUEtPiCaUoqgA9gYRDs3vj1nolY=
+github.com/aymanbagabas/go-udiff v0.3.1/go.mod h1:G0fsKmG+P6ylD0r6N/KgQD/nWzgfnl8ZBcNLgcbrw8E=
+github.com/aymerick/douceur v0.2.0 h1:Mv+mAeH1Q+n9Fr+oyamOlAkUNPWPlA8PPGR0QAaYuPk=
+github.com/aymerick/douceur v0.2.0/go.mod h1:wlT5vV2O3h55X9m7iVYN0TBM0NH/MmbLnd30/FjWUq4=
+github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA=
+github.com/bep/clocks v0.5.0 h1:hhvKVGLPQWRVsBP/UB7ErrHYIO42gINVbvqxvYTPVps=
+github.com/bep/clocks v0.5.0/go.mod h1:SUq3q+OOq41y2lRQqH5fsOoxN8GbxSiT6jvoVVLCVhU=
+github.com/bep/debounce v1.2.1 h1:v67fRdBA9UQu2NhLFXrSg0Brw7CexQekrBwDMM8bzeY=
+github.com/bep/debounce v1.2.1/go.mod h1:H8yggRPQKLUhUoqrJC1bO2xNya7vanpDl7xR3ISbCJ0=
+github.com/bep/gitmap v1.9.0 h1:2pyb1ex+cdwF6c4tsrhEgEKfyNfxE34d5K+s2sa9byc=
+github.com/bep/gitmap v1.9.0/go.mod h1:Juq6e1qqCRvc1W7nzgadPGI9IGV13ZncEebg5atj4Vo=
+github.com/bep/goat v0.5.0 h1:S8jLXHCVy/EHIoCY+btKkmcxcXFd34a0Q63/0D4TKeA=
+github.com/bep/goat v0.5.0/go.mod h1:Md9x7gRxiWKs85yHlVTvHQw9rg86Bm+Y4SuYE8CTH7c=
+github.com/bep/godartsass/v2 v2.5.0 h1:tKRvwVdyjCIr48qgtLa4gHEdtRkPF8H1OeEhJAEv7xg=
+github.com/bep/godartsass/v2 v2.5.0/go.mod h1:rjsi1YSXAl/UbsGL85RLDEjRKdIKUlMQHr6ChUNYOFU=
+github.com/bep/golibsass v1.2.0 h1:nyZUkKP/0psr8nT6GR2cnmt99xS93Ji82ZD9AgOK6VI=
+github.com/bep/golibsass v1.2.0/go.mod h1:DL87K8Un/+pWUS75ggYv41bliGiolxzDKWJAq3eJ1MA=
+github.com/bep/goportabletext v0.1.0 h1:8dqym2So1cEqVZiBa4ZnMM1R9l/DnC1h4ONg4J5kujw=
+github.com/bep/goportabletext v0.1.0/go.mod h1:6lzSTsSue75bbcyvVc0zqd1CdApuT+xkZQ6Re5DzZFg=
+github.com/bep/gowebp v0.4.0 h1:QihuVnvIKbRoeBNQkN0JPMM8ClLmD6V2jMftTFwSK3Q=
+github.com/bep/gowebp v0.4.0/go.mod h1:95gtYkAA8iIn1t3HkAPurRCVGV/6NhgaHJ1urz0iIwc=
+github.com/bep/helpers v0.6.0 h1:qtqMCK8XPFNM9hp5Ztu9piPjxNNkk8PIyUVjg6v8Bsw=
+github.com/bep/helpers v0.6.0/go.mod h1:IOZlgx5PM/R/2wgyCatfsgg5qQ6rNZJNDpWGXqDR044=
+github.com/bep/imagemeta v0.12.0 h1:ARf+igs5B7pf079LrqRnwzQ/wEB8Q9v4NSDRZO1/F5k=
+github.com/bep/imagemeta v0.12.0/go.mod h1:23AF6O+4fUi9avjiydpKLStUNtJr5hJB4rarG18JpN8=
+github.com/bep/lazycache v0.8.0 h1:lE5frnRjxaOFbkPZ1YL6nijzOPPz6zeXasJq8WpG4L8=
+github.com/bep/lazycache v0.8.0/go.mod h1:BQ5WZepss7Ko91CGdWz8GQZi/fFnCcyWupv8gyTeKwk=
+github.com/bep/logg v0.4.0 h1:luAo5mO4ZkhA5M1iDVDqDqnBBnlHjmtZF6VAyTp+nCQ=
+github.com/bep/logg v0.4.0/go.mod h1:Ccp9yP3wbR1mm++Kpxet91hAZBEQgmWgFgnXX3GkIV0=
+github.com/bep/overlayfs v0.10.0 h1:wS3eQ6bRsLX+4AAmwGjvoFSAQoeheamxofFiJ2SthSE=
+github.com/bep/overlayfs v0.10.0/go.mod h1:ouu4nu6fFJaL0sPzNICzxYsBeWwrjiTdFZdK4lI3tro=
+github.com/bep/tmc v0.5.1 h1:CsQnSC6MsomH64gw0cT5f+EwQDcvZz4AazKunFwTpuI=
+github.com/bep/tmc v0.5.1/go.mod h1:tGYHN8fS85aJPhDLgXETVKp+PR382OvFi2+q2GkGsq0=
+github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d h1:xDfNPAt8lFiC1UJrqV3uuy861HCTo708pDMbjHHdCas=
+github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d/go.mod h1:6QX/PXZ00z/TKoufEY6K/a0k6AhaJrQKdFe6OfVXsa4=
github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4=
github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM=
+github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK343L8=
+github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
+github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
+github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
+github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
+github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
+github.com/chainguard-dev/git-urls v1.0.2 h1:pSpT7ifrpc5X55n4aTTm7FFUE+ZQHKiqpiwNkJrVcKQ=
+github.com/chainguard-dev/git-urls v1.0.2/go.mod h1:rbGgj10OS7UgZlbzdUQIQpT0k/D4+An04HJY7Ol+Y/o=
+github.com/charmbracelet/colorprofile v0.3.3 h1:DjJzJtLP6/NZ8p7Cgjno0CKGr7wwRJGxWUwh2IyhfAI=
+github.com/charmbracelet/colorprofile v0.3.3/go.mod h1:nB1FugsAbzq284eJcjfah2nhdSLppN2NqvfotkfRYP4=
+github.com/charmbracelet/ultraviolet v0.0.0-20251116181749-377898bcce38 h1:7Rs87fbKJoIIxsQS8YKJYGYa0tlsDwwb0twQjV1KB+g=
+github.com/charmbracelet/ultraviolet v0.0.0-20251116181749-377898bcce38/go.mod h1:6lfcr3MNP+kZR25sF1nQwJFuQnNYBlFy3PGX5rvslXc=
+github.com/charmbracelet/x/ansi v0.11.1 h1:iXAC8SyMQDJgtcz9Jnw+HU8WMEctHzoTAETIeA3JXMk=
+github.com/charmbracelet/x/ansi v0.11.1/go.mod h1:M49wjzpIujwPceJ+t5w3qh2i87+HRtHohgb5iTyepL0=
+github.com/charmbracelet/x/exp/golden v0.0.0-20250806222409-83e3a29d542f h1:pk6gmGpCE7F3FcjaOEKYriCvpmIN4+6OS/RD0vm4uIA=
+github.com/charmbracelet/x/exp/golden v0.0.0-20250806222409-83e3a29d542f/go.mod h1:IfZAMTHB6XkZSeXUqriemErjAWCCzT0LwjKFYCZyw0I=
+github.com/charmbracelet/x/term v0.2.2 h1:xVRT/S2ZcKdhhOuSP4t5cLi5o+JxklsoEObBSgfgZRk=
+github.com/charmbracelet/x/term v0.2.2/go.mod h1:kF8CY5RddLWrsgVwpw4kAa6TESp6EB5y3uxGLeCqzAI=
+github.com/charmbracelet/x/termios v0.1.1 h1:o3Q2bT8eqzGnGPOYheoYS8eEleT5ZVNYNy8JawjaNZY=
+github.com/charmbracelet/x/termios v0.1.1/go.mod h1:rB7fnv1TgOPOyyKRJ9o+AsTU/vK5WHJ2ivHeut/Pcwo=
+github.com/charmbracelet/x/windows v0.2.2 h1:IofanmuvaxnKHuV04sC0eBy/smG6kIKrWG2/jYn2GuM=
+github.com/charmbracelet/x/windows v0.2.2/go.mod h1:/8XtdKZzedat74NQFn0NGlGL4soHB0YQZrETF96h75k=
+github.com/clbanning/mxj/v2 v2.7.0 h1:WA/La7UGCanFe5NpHF0Q3DNtnCsVoxbPKuyBNHWRyME=
+github.com/clbanning/mxj/v2 v2.7.0/go.mod h1:hNiWqW14h+kc+MdF9C6/YoRfjEJoR3ou6tn/Qo+ve2s=
+github.com/cli/browser v1.3.0 h1:LejqCrpWr+1pRqmEPDGnTZOjsMe7sehifLynZJuqJpo=
+github.com/cli/browser v1.3.0/go.mod h1:HH8s+fOAxjhQoBUAsKuPCbqUuxZDhQ2/aD+SzsEfBTk=
+github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
+github.com/clipperhouse/displaywidth v0.5.0 h1:AIG5vQaSL2EKqzt0M9JMnvNxOCRTKUc4vUnLWGgP89I=
+github.com/clipperhouse/displaywidth v0.5.0/go.mod h1:R+kHuzaYWFkTm7xoMmK1lFydbci4X2CicfbGstSGg0o=
+github.com/clipperhouse/stringish v0.1.1 h1:+NSqMOr3GR6k1FdRhhnXrLfztGzuG+VuFDfatpWHKCs=
+github.com/clipperhouse/stringish v0.1.1/go.mod h1:v/WhFtE1q0ovMta2+m+UbpZ+2/HEXNWYXQgCt4hdOzA=
+github.com/clipperhouse/uax29/v2 v2.3.0 h1:SNdx9DVUqMoBuBoW3iLOj4FQv3dN5mDtuqwuhIGpJy4=
+github.com/clipperhouse/uax29/v2 v2.3.0/go.mod h1:Wn1g7MK6OoeDT0vL+Q0SQLDz/KpfsVRgg6W7ihQeh4g=
+github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
+github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
+github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI=
+github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
+github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
+github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
+github.com/cncf/xds/go v0.0.0-20251210132809-ee656c7534f5 h1:6xNmx7iTtyBRev0+D/Tv1FZd4SCg8axKApyNyRsAt/w=
+github.com/cncf/xds/go v0.0.0-20251210132809-ee656c7534f5/go.mod h1:KdCmV+x/BuvyMxRnYBlmVaq4OLiKW6iRQfvC62cvdkI=
+github.com/coder/websocket v1.8.14 h1:9L0p0iKiNOibykf283eHkKUHHrpG7f65OE3BhhO7v9g=
+github.com/coder/websocket v1.8.14/go.mod h1:NX3SzP+inril6yawo5CQXx8+fk145lPDC6pumgx0mVg=
github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
+github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6NIQQ7OS05n1F4g=
+github.com/creack/pty v1.1.24 h1:bJrF4RRfyJnbTJqzRLHzcGaZK1NeM5kTC9jGgovnR1s=
+github.com/creack/pty v1.1.24/go.mod h1:08sCNb52WyoAwi2QDyzUCTgcvVFhUzewun7wtTfvcwE=
+github.com/cubicdaiya/gonp v1.0.4 h1:ky2uIAJh81WiLcGKBVD5R7KsM/36W6IqqTy6Bo6rGws=
+github.com/cubicdaiya/gonp v1.0.4/go.mod h1:iWGuP/7+JVTn02OWhRemVbMmG1DOUnmrGTYYACpOI0I=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/delaneyj/toolbelt v0.9.1 h1:QJComn2qoaQ4azl5uRkGpdHSO9e+JtoxDTXCiQHvH8o=
github.com/delaneyj/toolbelt v0.9.1/go.mod h1:eNXpPuThjTD4tpRNCBl4JEz9jdg9LpyzNuyG+stnIbs=
+github.com/disintegration/gift v1.2.1 h1:Y005a1X4Z7Uc+0gLpSAsKhWi4qLtsdEcMIbbdvdZ6pc=
+github.com/disintegration/gift v1.2.1/go.mod h1:Jh2i7f7Q2BM7Ezno3PhfezbR1xpUg9dUg3/RlKGr4HI=
+github.com/dlclark/regexp2 v1.11.5 h1:Q/sSnsKerHeCkc/jSTNq1oCm7KiVgUMZRDUoRu0JQZQ=
+github.com/dlclark/regexp2 v1.11.5/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
+github.com/dominikbraun/graph v0.23.0 h1:TdZB4pPqCLFxYhdyMFb1TBdFxp8XLcJfTTBQucVPgCo=
+github.com/dominikbraun/graph v0.23.0/go.mod h1:yOjYyogZLY1LSG9E33JWZJiq5k83Qy2C6POAuiViluc=
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
+github.com/elastic/go-sysinfo v1.8.1/go.mod h1:JfllUnzoQV/JRYymbH3dO1yggI3mV2oTKSXsDHM+uIM=
+github.com/elastic/go-sysinfo v1.15.4 h1:A3zQcunCxik14MgXu39cXFXcIw2sFXZ0zL886eyiv1Q=
+github.com/elastic/go-sysinfo v1.15.4/go.mod h1:ZBVXmqS368dOn/jvijV/zHLfakWTYHBZPk3G244lHrU=
+github.com/elastic/go-windows v1.0.0/go.mod h1:TsU0Nrp7/y3+VwE82FoZF8gC/XFg/Elz6CcloAxnPgU=
+github.com/elastic/go-windows v1.0.2 h1:yoLLsAsV5cfg9FLhZ9EXZ2n2sQFKeDYrHenkcivY4vI=
+github.com/elastic/go-windows v1.0.2/go.mod h1:bGcDpBzXgYSqM0Gx3DM4+UxFj300SZLixie9u9ixLM8=
+github.com/elliotchance/orderedmap/v3 v3.1.0 h1:j4DJ5ObEmMBt/lcwIecKcoRxIQUEnw0L804lXYDt/pg=
+github.com/elliotchance/orderedmap/v3 v3.1.0/go.mod h1:G+Hc2RwaZvJMcS4JpGCOyViCnGeKf0bTYCGTO4uhjSo=
+github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
+github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
+github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
+github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
+github.com/envoyproxy/go-control-plane v0.10.2-0.20220325020618-49ff273808a1/go.mod h1:KJwIaB5Mv44NWtYuAOFCVOjcI94vtpEz2JU/D2v6IjE=
+github.com/envoyproxy/go-control-plane v0.14.0 h1:hbG2kr4RuFj222B6+7T83thSPqLjwBIfQawTkC++2HA=
+github.com/envoyproxy/go-control-plane v0.14.0/go.mod h1:NcS5X47pLl/hfqxU70yPwL9ZMkUlwlKxtAohpi2wBEU=
+github.com/envoyproxy/go-control-plane/envoy v1.36.0 h1:yg/JjO5E7ubRyKX3m07GF3reDNEnfOboJ0QySbH736g=
+github.com/envoyproxy/go-control-plane/envoy v1.36.0/go.mod h1:ty89S1YCCVruQAm9OtKeEkQLTb+Lkz0k8v9W0Oxsv98=
+github.com/envoyproxy/go-control-plane/ratelimit v0.1.0 h1:/G9QYbddjL25KvtKTv3an9lx6VBE2cnb8wp1vEGNYGI=
+github.com/envoyproxy/go-control-plane/ratelimit v0.1.0/go.mod h1:Wk+tMFAFbCXaJPzVVHnPgRKdUdwW/KdbRt94AzgRee4=
+github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
+github.com/envoyproxy/protoc-gen-validate v1.3.0 h1:TvGH1wof4H33rezVKWSpqKz5NXWg5VPuZ0uONDT6eb4=
+github.com/envoyproxy/protoc-gen-validate v1.3.0/go.mod h1:HvYl7zwPa5mffgyeTUHA9zHIH36nmrm7oCbo4YKoSWA=
+github.com/evanw/esbuild v0.25.9 h1:aU7GVC4lxJGC1AyaPwySWjSIaNLAdVEEuq3chD0Khxs=
+github.com/evanw/esbuild v0.25.9/go.mod h1:D2vIQZqV/vIf/VRHtViaUtViZmG7o+kKmlBfVQuRi48=
+github.com/fatih/color v1.18.0 h1:S8gINlzdQ840/4pfAwic/ZE0djQEH3wM94VfqLTZcOM=
+github.com/fatih/color v1.18.0/go.mod h1:4FelSpRwEGDpQ12mAdzqdOukCy4u8WUtOY6lkT/6HfU=
+github.com/fatih/structtag v1.2.0 h1:/OdNE99OxoI/PqaW/SuSK9uxxT3f/tcSZgon/ssNSx4=
+github.com/fatih/structtag v1.2.0/go.mod h1:mBJUNpUnHmRKrKlQQlmCrh5PuhftFbNv8Ys4/aAZl94=
+github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg=
+github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
+github.com/frankban/quicktest v1.7.2/go.mod h1:jaStnuzAqU1AJdCO0l53JDCJrVDKcS03DbaAcR7Ks/o=
+github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
+github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
+github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k=
+github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
+github.com/getkin/kin-openapi v0.133.0 h1:pJdmNohVIJ97r4AUFtEXRXwESr8b0bD721u/Tz6k8PQ=
+github.com/getkin/kin-openapi v0.133.0/go.mod h1:boAciF6cXk5FhPqe/NQeBTeenbjqU4LhWBf09ILVvWE=
+github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk=
+github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
+github.com/go-chi/chi/v5 v5.2.5 h1:Eg4myHZBjyvJmAFjFvWgrqDTXFyOzjj7YIm3L3mu6Ug=
+github.com/go-chi/chi/v5 v5.2.5/go.mod h1:X7Gx4mteadT3eDOMTsXzmI4/rwUpOwBHLpAfupzFJP0=
+github.com/go-faster/city v1.0.1 h1:4WAxSZ3V2Ws4QRDrscLEDcibJY8uf41H6AhXDrNDcGw=
+github.com/go-faster/city v1.0.1/go.mod h1:jKcUJId49qdW3L1qKHH/3wPeUstCVpVSXTM6vO3VcTw=
+github.com/go-faster/errors v0.7.1 h1:MkJTnDoEdi9pDabt1dpWf7AA8/BaSYZqibYyhZ20AYg=
+github.com/go-faster/errors v0.7.1/go.mod h1:5ySTjWFiphBs07IKuiL69nxdfd5+fzh1u7FPGZP2quo=
+github.com/go-jose/go-jose/v4 v4.1.3 h1:CVLmWDhDVRa6Mi/IgCgaopNosCaHz7zrMeF9MlZRkrs=
+github.com/go-jose/go-jose/v4 v4.1.3/go.mod h1:x4oUasVrzR7071A4TnHLGSPpNOm2a21K9Kf04k1rs08=
+github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
+github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI=
+github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
+github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
+github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
+github.com/go-openapi/jsonpointer v0.21.0 h1:YgdVicSA9vH5RiHs9TZW5oyafXZFc6+2Vc1rr/O9oNQ=
+github.com/go-openapi/jsonpointer v0.21.0/go.mod h1:IUyH9l/+uyhIYQ/PXVA41Rexl+kOkAPDdXEYns6fzUY=
+github.com/go-openapi/swag v0.23.0 h1:vsEVJDUo2hPJ2tu0/Xc+4noaxyEffXNIs3cOULZ+GrE=
+github.com/go-openapi/swag v0.23.0/go.mod h1:esZ8ITTYEsH1V2trKHjAN8Ai7xHb8RV+YSZ577vPjgQ=
+github.com/go-quicktest/qt v1.101.0 h1:O1K29Txy5P2OK0dGo59b7b0LR6wKfIhttaAhHUyn7eI=
+github.com/go-quicktest/qt v1.101.0/go.mod h1:14Bz/f7NwaXPtdYEgzsx46kqSxVwTbzVZsDC26tQJow=
+github.com/go-sql-driver/mysql v1.9.3 h1:U/N249h2WzJ3Ukj8SowVFjdtZKfu9vlLZxjPXV1aweo=
+github.com/go-sql-driver/mysql v1.9.3/go.mod h1:qn46aNg1333BRMNU69Lq93t8du/dwxI64Gl8i5p1WMU=
+github.com/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI=
+github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8=
+github.com/go-task/task/v3 v3.48.0 h1:HEim5OOpgmob5ONfq7ji3QHUyJdcwqL5ctOT5CPWCzA=
+github.com/go-task/task/v3 v3.48.0/go.mod h1:ChDoJV0k919miEJJu1yJ846tg+4Ivv9ZE/1YwQXvIRY=
+github.com/go-task/template v0.2.0 h1:xW7ek0o65FUSTbKcSNeg2Vyf/I7wYXFgLUznptvviBE=
+github.com/go-task/template v0.2.0/go.mod h1:dbdoUb6qKnHQi1y6o+IdIrs0J4o/SEhSTA6bbzZmdtc=
+github.com/gobuffalo/flect v1.0.3 h1:xeWBM2nui+qnVvNM4S3foBhCAL2XgPU+a7FdpelbTq4=
+github.com/gobuffalo/flect v1.0.3/go.mod h1:A5msMlrHtLqh9umBSnvabjsMrCcCpAyzglnDvkbYKHs=
+github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y=
+github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8=
github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
+github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
+github.com/gohugoio/go-i18n/v2 v2.1.3-0.20230805085216-e63c13218d0e h1:QArsSubW7eDh8APMXkByjQWvuljwPGAGQpJEFn0F0wY=
+github.com/gohugoio/go-i18n/v2 v2.1.3-0.20230805085216-e63c13218d0e/go.mod h1:3Ltoo9Banwq0gOtcOwxuHG6omk+AwsQPADyw2vQYOJQ=
+github.com/gohugoio/hashstructure v0.5.0 h1:G2fjSBU36RdwEJBWJ+919ERvOVqAg9tfcYp47K9swqg=
+github.com/gohugoio/hashstructure v0.5.0/go.mod h1:Ser0TniXuu/eauYmrwM4o64EBvySxNzITEOLlm4igec=
+github.com/gohugoio/httpcache v0.7.0 h1:ukPnn04Rgvx48JIinZvZetBfHaWE7I01JR2Q2RrQ3Vs=
+github.com/gohugoio/httpcache v0.7.0/go.mod h1:fMlPrdY/vVJhAriLZnrF5QpN3BNAcoBClgAyQd+lGFI=
+github.com/gohugoio/hugo v0.149.1 h1:uWOc8Ve4h4e48FyYhBquRoHCJviyxA5yGrFJLT48yio=
+github.com/gohugoio/hugo v0.149.1/go.mod h1:HS6BP6e8FGxungP4CHC3zeLDvhBLnTJIjHJZWTZjs7o=
+github.com/gohugoio/hugo-goldmark-extensions/extras v0.5.0 h1:dco+7YiOryRoPOMXwwaf+kktZSCtlFtreNdiJbETvYE=
+github.com/gohugoio/hugo-goldmark-extensions/extras v0.5.0/go.mod h1:CRrxQTKeM3imw+UoS4EHKyrqB7Zp6sAJiqHit+aMGTE=
+github.com/gohugoio/hugo-goldmark-extensions/passthrough v0.3.1 h1:nUzXfRTszLliZuN0JTKeunXTRaiFX6ksaWP0puLLYAY=
+github.com/gohugoio/hugo-goldmark-extensions/passthrough v0.3.1/go.mod h1:Wy8ThAA8p2/w1DY05vEzq6EIeI2mzDjvHsu7ULBVwog=
+github.com/gohugoio/locales v0.14.0 h1:Q0gpsZwfv7ATHMbcTNepFd59H7GoykzWJIxi113XGDc=
+github.com/gohugoio/locales v0.14.0/go.mod h1:ip8cCAv/cnmVLzzXtiTpPwgJ4xhKZranqNqtoIu0b/4=
+github.com/gohugoio/localescompressed v1.0.1 h1:KTYMi8fCWYLswFyJAeOtuk/EkXR/KPTHHNN9OS+RTxo=
+github.com/gohugoio/localescompressed v1.0.1/go.mod h1:jBF6q8D7a0vaEmcWPNcAjUZLJaIVNiwvM3WlmTvooB0=
+github.com/golang-jwt/jwt/v4 v4.5.2 h1:YtQM7lnr8iZ+j5q71MGKkNw9Mn7AjHM68uc9g5fXeUI=
+github.com/golang-jwt/jwt/v4 v4.5.2/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
+github.com/golang-jwt/jwt/v5 v5.2.3 h1:kkGXqQOBSDDWRhWNXTFpqGSCMyh/PLnqUvMGJPDJDs0=
+github.com/golang-jwt/jwt/v5 v5.2.3/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
+github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 h1:au07oEsX2xN0ktxqI+Sida1w446QrXBRJ0nee3SNZlA=
+github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
+github.com/golang-sql/sqlexp v0.1.0 h1:ZCD6MBpcuOVfGVqsEmY5/4FtYiKz6tSyUv9LPEDei6A=
+github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI=
+github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
+github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
+github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
+github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
+github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
+github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
+github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
+github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
+github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
+github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
+github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
+github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
+github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
+github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
+github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
+github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
github.com/google/brotli/go/cbrotli v0.0.0-20230829110029-ed738e842d2f h1:jopqB+UTSdJGEJT8tEqYyE29zN91fi2827oLET8tl7k=
github.com/google/brotli/go/cbrotli v0.0.0-20230829110029-ed738e842d2f/go.mod h1:nOPhAkwVliJdNTkj3gXpljmWhjc4wCaVqbMJcPKWP4s=
+github.com/google/cel-go v0.26.1 h1:iPbVVEdkhTX++hpe3lzSk7D3G3QSYqLGoHOcEio+UXQ=
+github.com/google/cel-go v0.26.1/go.mod h1:A9O8OU9rdvrK5MQyrqfIxo1a0u4g3sF8KB6PUIaryMM=
+github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
+github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
+github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
+github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
+github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
github.com/google/go-tpm v0.9.7 h1:u89J4tUUeDTlH8xxC3CTW7OHZjbjKoHdQ9W7gCUhtxA=
github.com/google/go-tpm v0.9.7/go.mod h1:h9jEsEECg7gtLis0upRBQU+GhYVH6jMjrFxI8u6bVUY=
+github.com/google/martian/v3 v3.3.3 h1:DIhPTQrbPkgs2yJYdXU/eNACCG5DVQjySNRNlflZ9Fc=
+github.com/google/martian/v3 v3.3.3/go.mod h1:iEPrYcgCF7jA9OtScMFQyAlZZ4YXTKEtJ1E6RWzmBA0=
github.com/google/pprof v0.0.0-20250317173921-a4b03ec1a45e h1:ijClszYn+mADRFY17kjQEVQ1XRhq2/JR1M3sGqeJoxs=
github.com/google/pprof v0.0.0-20250317173921-a4b03ec1a45e/go.mod h1:boTsfXsheKC2y+lKOCMpSfarhxDeIzfZG1jqGcPl3cA=
+github.com/google/s2a-go v0.1.9 h1:LGD7gtMgezd8a/Xak7mEWL0PjoTQFvpRudN895yqKW0=
+github.com/google/s2a-go v0.1.9/go.mod h1:YA0Ei2ZQL3acow2O62kdp9UlnvMmU7kA6Eutn0dXayM=
+github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
+github.com/googleapis/enterprise-certificate-proxy v0.3.7 h1:zrn2Ee/nWmHulBx5sAVrGgAa0f2/R35S4DJwfFaUPFQ=
+github.com/googleapis/enterprise-certificate-proxy v0.3.7/go.mod h1:MkHOF77EYAE7qfSuSS9PU6g4Nt4e11cnsDUowfwewLA=
+github.com/googleapis/gax-go/v2 v2.15.0 h1:SyjDc1mGgZU5LncH8gimWo9lW1DtIfPibOG81vgd/bo=
+github.com/googleapis/gax-go/v2 v2.15.0/go.mod h1:zVVkkxAQHa1RQpg9z2AUCMnKhi0Qld9rcmyfL1OZhoc=
+github.com/gorilla/css v1.0.1 h1:ntNaBIghp6JmvWnxbZKANoLyuXTPZ4cAMlo6RyhlbO8=
+github.com/gorilla/css v1.0.1/go.mod h1:BvnYkspnSzMmwRK+b8/xgNPLiIuNZr6vbZBTPQ2A3b0=
+github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
+github.com/hairyhenderson/go-codeowners v0.7.0 h1:s0W4wF8bdsBEjTWzwzSlsatSthWtTAF2xLgo4a4RwAo=
+github.com/hairyhenderson/go-codeowners v0.7.0/go.mod h1:wUlNgQ3QjqC4z8DnM5nnCYVq/icpqXJyJOukKx5U8/Q=
+github.com/hashicorp/aws-sdk-go-base/v2 v2.0.0-beta.70 h1:0HADrxxqaQkGycO1JoUUA+B4FnIkuo8d2bz/hSaTFFQ=
+github.com/hashicorp/aws-sdk-go-base/v2 v2.0.0-beta.70/go.mod h1:fm2FdDCzJdtbXF7WKAMvBb5NEPouXPHFbGNYs9ShFns=
+github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ=
+github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48=
+github.com/hashicorp/go-getter v1.8.4 h1:hGEd2xsuVKgwkMtPVufq73fAmZU/x65PPcqH3cb0D9A=
+github.com/hashicorp/go-getter v1.8.4/go.mod h1:x27pPGSg9kzoB147QXI8d/nDvp2IgYGcwuRjpaXE9Yg=
+github.com/hashicorp/go-version v1.8.0 h1:KAkNb1HAiZd1ukkxDFGmokVZe1Xy9HG6NUp+bPle2i4=
+github.com/hashicorp/go-version v1.8.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA=
github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs4luLUK2k=
github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM=
+github.com/hexops/gotextdiff v1.0.3 h1:gitA9+qJrrTCsiCl7+kh75nPqQt1cx4ZkudSTLoUqJM=
+github.com/hexops/gotextdiff v1.0.3/go.mod h1:pSWU5MAI3yDq+fZBTazCSJysOMbxWL1BSow5/V2vxeg=
github.com/hookenz/gotailwind/v4 v4.2.1 h1:FpZLtAAbHH7wMvyGYT+01vTLFITGMGZGMtEbp7dd2dM=
github.com/hookenz/gotailwind/v4 v4.2.1/go.mod h1:IfiJtdp8ExV9HV2XUiVjRBvB3QewVXVKWoAGEcpjfNE=
+github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
+github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
+github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
+github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
+github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 h1:iCEnooe7UlwOQYpKFhBabPMi4aNAfoODPEFNiAnClxo=
+github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM=
+github.com/jackc/pgx/v5 v5.8.0 h1:TYPDoleBBme0xGSAX3/+NujXXtpZn9HBONkQC7IEZSo=
+github.com/jackc/pgx/v5 v5.8.0/go.mod h1:QVeDInX2m9VyzvNeiCJVjCkNFqzsNb43204HshNSZKw=
+github.com/jackc/puddle/v2 v2.2.2 h1:PR8nw+E/1w0GLuRFSmiioY6UooMp6KJv0/61nB7icHo=
+github.com/jackc/puddle/v2 v2.2.2/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4=
+github.com/jdkato/prose v1.2.1 h1:Fp3UnJmLVISmlc57BgKUzdjr0lOtjqTZicL3PaYy6cU=
+github.com/jdkato/prose v1.2.1/go.mod h1:AiRHgVagnEx2JbQRQowVBKjG0bcs/vtkGCH1dYAL1rA=
+github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI=
+github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E=
+github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc=
+github.com/joeshaw/multierror v0.0.0-20140124173710-69b34d4ec901/go.mod h1:Z86h9688Y0wesXCyonoVr47MasHilkuLMqGhRZ4Hpak=
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
+github.com/jonboulle/clockwork v0.5.0 h1:Hyh9A8u51kptdkR+cqRpT1EebBwTn1oK9YfGYbdFz6I=
+github.com/jonboulle/clockwork v0.5.0/go.mod h1:3mZlmanh0g2NDKO5TWZVJAfofYk64M7XN3SzBPjZF60=
+github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
+github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
+github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
+github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
+github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
github.com/klauspost/compress v1.16.7/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE=
github.com/klauspost/compress v1.18.4 h1:RPhnKRAQ4Fh8zU2FY/6ZFDwTVTxgJ/EMydqSTzE9a2c=
github.com/klauspost/compress v1.18.4/go.mod h1:R0h/fSBs8DE4ENlcrlib3PsXS61voFxhIs2DeRhCvJ4=
+github.com/klauspost/cpuid/v2 v2.3.0 h1:S4CRMLnYUhGeDFDqkGriYKdfoFlDnMtqTiI/sFzhA9Y=
+github.com/klauspost/cpuid/v2 v2.3.0/go.mod h1:hqwkgyIinND0mEev00jJYCxPNVRVXFQeu1XKlok6oO0=
+github.com/klauspost/pgzip v1.2.6 h1:8RXeL5crjEUFnR2/Sn6GJNWtSQ3Dk8pq4CL3jvdDyjU=
github.com/klauspost/pgzip v1.2.6/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs=
+github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
+github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
+github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
+github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
+github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
+github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
+github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
+github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc=
+github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw=
+github.com/kyokomi/emoji/v2 v2.2.13 h1:GhTfQa67venUUvmleTNFnb+bi7S3aocF7ZCXU9fSO7U=
+github.com/kyokomi/emoji/v2 v2.2.13/go.mod h1:JUcn42DTdsXJo1SWanHh4HKDEyPaR5CqkmoirZZP9qE=
+github.com/lucasb-eyer/go-colorful v1.3.0 h1:2/yBRLdWBZKrf7gB40FoiKfAWYQ0lqNcbuQwVHXptag=
+github.com/lucasb-eyer/go-colorful v1.3.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
+github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0=
+github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc=
+github.com/makeworld-the-better-one/dither/v2 v2.4.0 h1:Az/dYXiTcwcRSe59Hzw4RI1rSnAZns+1msaCXetrMFE=
+github.com/makeworld-the-better-one/dither/v2 v2.4.0/go.mod h1:VBtN8DXO7SNtyGmLiGA7IsFeKrBkQPze1/iAeM95arc=
+github.com/marekm4/color-extractor v1.2.1 h1:3Zb2tQsn6bITZ8MBVhc33Qn1k5/SEuZ18mrXGUqIwn0=
+github.com/marekm4/color-extractor v1.2.1/go.mod h1:90VjmiHI6M8ez9eYUaXLdcKnS+BAOp7w+NpwBdkJmpA=
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
github.com/mattn/go-colorable v0.1.14 h1:9A9LHSqF/7dyVVX6g0U9cwm9pG3kP9gSzcuIPHPsaIE=
github.com/mattn/go-colorable v0.1.14/go.mod h1:6LmQG8QLFO4G5z1gPvYEzlUgJ2wF+stgPZH1UqBm1s8=
@@ -46,13 +451,36 @@ github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
+github.com/mattn/go-runewidth v0.0.19 h1:v++JhqYnZuu5jSKrk9RbgF5v4CGUjqRfBm05byFGLdw=
+github.com/mattn/go-runewidth v0.0.19/go.mod h1:XBkDxAl56ILZc9knddidhrOlY5R/pDhgLpndooCuJAs=
github.com/mattn/go-sqlite3 v1.14.6/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU=
github.com/mattn/go-sqlite3 v1.14.32 h1:JD12Ag3oLy1zQA+BNn74xRgaBbdhbNIDYvQUEuuErjs=
github.com/mattn/go-sqlite3 v1.14.32/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y=
github.com/mfridman/interpolate v0.0.2 h1:pnuTK7MQIxxFz1Gr+rjSIx9u7qVjf5VOoM/u6BbAxPY=
github.com/mfridman/interpolate v0.0.2/go.mod h1:p+7uk6oE07mpE/Ik1b8EckO0O4ZXiGAfshKBWLUM9Xg=
+github.com/mfridman/xflag v0.1.0 h1:TWZrZwG1QklFX5S4j1vxfF1sZbZeZSGofMwPMLAF29M=
+github.com/mfridman/xflag v0.1.0/go.mod h1:/483ywM5ZO5SuMVjrIGquYNE5CzLrj5Ux/LxWWnjRaE=
+github.com/microcosm-cc/bluemonday v1.0.27 h1:MpEUotklkwCSLeH+Qdx1VJgNqLlpY2KXwXFM08ygZfk=
+github.com/microcosm-cc/bluemonday v1.0.27/go.mod h1:jFi9vgW+H7c3V0lb6nR74Ib/DIB5OBs92Dimizgw2cA=
+github.com/microsoft/go-mssqldb v1.9.6 h1:1MNQg5UiSsokiPz3++K2KPx4moKrwIqly1wv+RyCKTw=
+github.com/microsoft/go-mssqldb v1.9.6/go.mod h1:yYMPDufyoF2vVuVCUGtZARr06DKFIhMrluTcgWlXpr4=
github.com/minio/highwayhash v1.0.4-0.20251030100505-070ab1a87a76 h1:KGuD/pM2JpL9FAYvBrnBBeENKZNh6eNtjqytV6TYjnk=
github.com/minio/highwayhash v1.0.4-0.20251030100505-070ab1a87a76/go.mod h1:GGYsuwP/fPD6Y9hMiXuapVvlIUEhFhMTh0rxU3ik1LQ=
+github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
+github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
+github.com/mitchellh/hashstructure/v2 v2.0.2 h1:vGKWl0YJqUNxE8d+h8f6NJLcCJrgbhC4NcD46KavDd4=
+github.com/mitchellh/hashstructure/v2 v2.0.2/go.mod h1:MG3aRVU/N29oo/V/IhBX8GR/zz4kQkprJgF2EVszyDE=
+github.com/mitchellh/mapstructure v1.5.1-0.20231216201459-8508981c8b6c h1:cqn374mizHuIWj+OSJCajGr/phAmuMug9qIX3l9CflE=
+github.com/mitchellh/mapstructure v1.5.1-0.20231216201459-8508981c8b6c/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
+github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826 h1:RWengNIwukTxcDr9M+97sNutRR1RKhG96O6jWumTTnw=
+github.com/mohae/deepcopy v0.0.0-20170929034955-c48cc78d4826/go.mod h1:TaXosZuwdSHYgviHp1DAtfrULt5eUgsSMsZf+YrPgl8=
+github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc=
+github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA=
+github.com/muesli/cancelreader v0.2.2/go.mod h1:3XuTXfFS2VjM+HTLZY9Ak0l6eUKfijIfMUZ4EgX0QYo=
+github.com/muesli/smartcrop v0.3.0 h1:JTlSkmxWg/oQ1TcLDoypuirdE8Y/jzNirQeLkxpA6Oc=
+github.com/muesli/smartcrop v0.3.0/go.mod h1:i2fCI/UorTfgEpPPLWiFBv4pye+YAG78RwcQLUkocpI=
+github.com/natefinch/atomic v1.0.1 h1:ZPYKxkqQOx3KZ+RsbnP/YsgvxWQPGxjC0oBt2AhwV0A=
+github.com/natefinch/atomic v1.0.1/go.mod h1:N/D/ELrljoqDyT3rZrsUmtsuzvHkeB/wWjHV22AZRbM=
github.com/nats-io/jwt/v2 v2.8.0 h1:K7uzyz50+yGZDO5o772eRE7atlcSEENpL7P+b74JV1g=
github.com/nats-io/jwt/v2 v2.8.0/go.mod h1:me11pOkwObtcBNR8AiMrUbtVOUGkqYjMQZ6jnSdVUIA=
github.com/nats-io/nats-server/v2 v2.12.2 h1:4TEQd0Y4zvcW0IsVxjlXnRso1hBkQl3TS0BI+SxgPhE=
@@ -65,68 +493,369 @@ github.com/nats-io/nuid v1.0.1 h1:5iA8DT8V7q8WK2EScv2padNa/rTESc1KdnPw4TC2paw=
github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c=
github.com/ncruces/go-strftime v1.0.0 h1:HMFp8mLCTPp341M/ZnA4qaf7ZlsbTc+miZjCLOFAw7w=
github.com/ncruces/go-strftime v1.0.0/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls=
+github.com/niklasfasching/go-org v1.9.1 h1:/3s4uTPOF06pImGa2Yvlp24yKXZoTYM+nsIlMzfpg/0=
+github.com/niklasfasching/go-org v1.9.1/go.mod h1:ZAGFFkWvUQcpazmi/8nHqwvARpr1xpb+Es67oUGX/48=
+github.com/oasdiff/yaml v0.0.0-20250309154309-f31be36b4037 h1:G7ERwszslrBzRxj//JalHPu/3yz+De2J+4aLtSRlHiY=
+github.com/oasdiff/yaml v0.0.0-20250309154309-f31be36b4037/go.mod h1:2bpvgLBZEtENV5scfDFEtB/5+1M4hkQhDQrccEJ/qGw=
+github.com/oasdiff/yaml3 v0.0.0-20250309153720-d2182401db90 h1:bQx3WeLcUWy+RletIKwUIt4x3t8n2SxavmoclizMb8c=
+github.com/oasdiff/yaml3 v0.0.0-20250309153720-d2182401db90/go.mod h1:y5+oSEHCPT/DGrS++Wc/479ERge0zTFxaF8PbGKcg2o=
+github.com/olekukonko/errors v1.1.0 h1:RNuGIh15QdDenh+hNvKrJkmxxjV4hcS50Db478Ou5sM=
+github.com/olekukonko/errors v1.1.0/go.mod h1:ppzxA5jBKcO1vIpCXQ9ZqgDh8iwODz6OXIGKU8r5m4Y=
+github.com/olekukonko/ll v0.0.9 h1:Y+1YqDfVkqMWuEQMclsF9HUR5+a82+dxJuL1HHSRpxI=
+github.com/olekukonko/ll v0.0.9/go.mod h1:En+sEW0JNETl26+K8eZ6/W4UQ7CYSrrgg/EdIYT2H8g=
+github.com/olekukonko/tablewriter v1.0.9 h1:XGwRsYLC2bY7bNd93Dk51bcPZksWZmLYuaTHR0FqfL8=
+github.com/olekukonko/tablewriter v1.0.9/go.mod h1:5c+EBPeSqvXnLLgkm9isDdzR3wjfBkHR9Nhfp3NWrzo=
+github.com/paulmach/orb v0.12.0 h1:z+zOwjmG3MyEEqzv92UN49Lg1JFYx0L9GpGKNVDKk1s=
+github.com/paulmach/orb v0.12.0/go.mod h1:5mULz1xQfs3bmQm63QEJA6lNGujuRafwA5S/EnuLaLU=
+github.com/paulmach/protoscan v0.2.1/go.mod h1:SpcSwydNLrxUGSDvXvO0P7g7AuhJ7lcKfDlhJCDw2gY=
+github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58 h1:onHthvaw9LFnH4t2DcNVpwGmV9E1BkGknEliJkfwQj0=
+github.com/pbnjay/memory v0.0.0-20210728143218-7b4eea64cf58/go.mod h1:DXv8WO4yhMYhSNPKjeNKa5WY9YCIEBRbNzFFPJbWO6Y=
+github.com/pelletier/go-toml v1.9.5 h1:4yBQzkHv+7BHq2PQUZF3Mx0IYxG7LsP222s7Agd3ve8=
+github.com/pelletier/go-toml v1.9.5/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c=
+github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4=
+github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY=
+github.com/perimeterx/marshmallow v1.1.5 h1:a2LALqQ1BlHM8PZblsDdidgv1mWi1DgC2UmX50IvK2s=
+github.com/perimeterx/marshmallow v1.1.5/go.mod h1:dsXbUu8CRzfYP5a87xpp0xq9S3u0Vchtcl8we9tYaXw=
+github.com/pganalyze/pg_query_go/v6 v6.1.0 h1:jG5ZLhcVgL1FAw4C/0VNQaVmX1SUJx71wBGdtTtBvls=
+github.com/pganalyze/pg_query_go/v6 v6.1.0/go.mod h1:nvTHIuoud6e1SfrUaFwHqT0i4b5Nr+1rPWVds3B5+50=
github.com/pierrec/lz4/v4 v4.1.18/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4=
github.com/pierrec/lz4/v4 v4.1.25 h1:kocOqRffaIbU5djlIBr7Wh+cx82C0vtFb0fOurZHqD0=
github.com/pierrec/lz4/v4 v4.1.25/go.mod h1:EoQMVJgeeEOMsCqCzqFm2O0cJvljX2nGZjcRIPL34O4=
+github.com/pingcap/errors v0.11.0/go.mod h1:Oi8TUi2kEtXXLMJk9l1cGmz20kV3TaQ0usTwv5KuLY8=
+github.com/pingcap/errors v0.11.5-0.20240311024730-e056997136bb h1:3pSi4EDG6hg0orE1ndHkXvX6Qdq2cZn8gAPir8ymKZk=
+github.com/pingcap/errors v0.11.5-0.20240311024730-e056997136bb/go.mod h1:X2r9ueLEUZgtx2cIogM0v4Zj5uvvzhuuiu7Pn8HzMPg=
+github.com/pingcap/failpoint v0.0.0-20240528011301-b51a646c7c86 h1:tdMsjOqUR7YXHoBitzdebTvOjs/swniBTOLy5XiMtuE=
+github.com/pingcap/failpoint v0.0.0-20240528011301-b51a646c7c86/go.mod h1:exzhVYca3WRtd6gclGNErRWb1qEgff3LYta0LvRmON4=
+github.com/pingcap/log v1.1.0 h1:ELiPxACz7vdo1qAvvaWJg1NrYFoY6gqAh/+Uo6aXdD8=
+github.com/pingcap/log v1.1.0/go.mod h1:DWQW5jICDR7UJh4HtxXSM20Churx4CQL0fwL/SoOSA4=
+github.com/pingcap/tidb/pkg/parser v0.0.0-20250324122243-d51e00e5bbf0 h1:W3rpAI3bubR6VWOcwxDIG0Gz9G5rl5b3SL116T0vBt0=
+github.com/pingcap/tidb/pkg/parser v0.0.0-20250324122243-d51e00e5bbf0/go.mod h1:+8feuexTKcXHZF/dkDfvCwEyBAmgb4paFc3/WeYV2eE=
+github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ=
+github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU=
+github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
+github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10 h1:GFCKgmp0tecUJ0sJuv4pzYCqS9+RGSn52M3FUwPs+uo=
+github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10/go.mod h1:t/avpk3KcrXxUnYOhZhMXJlSEyie6gQbtLq5NM3loB8=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/pressly/goose/v3 v3.27.0 h1:/D30gVTuQhu0WsNZYbJi4DMOsx1lNq+6SkLe+Wp59BM=
github.com/pressly/goose/v3 v3.27.0/go.mod h1:3ZBeCXqzkgIRvrEMDkYh1guvtoJTU5oMMuDdkutoM78=
+github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
+github.com/prometheus/procfs v0.0.0-20190425082905-87a4384529e0/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
+github.com/prometheus/procfs v0.19.2 h1:zUMhqEW66Ex7OXIiDkll3tl9a1ZdilUOd/F6ZXw4Vws=
+github.com/prometheus/procfs v0.19.2/go.mod h1:M0aotyiemPhBCM0z5w87kL22CxfcH05ZpYlu+b4J7mw=
+github.com/puzpuzpuz/xsync/v4 v4.3.0 h1:w/bWkEJdYuRNYhHn5eXnIT8LzDM1O629X1I9MJSkD7Q=
+github.com/puzpuzpuz/xsync/v4 v4.3.0/go.mod h1:VJDmTCJMBt8igNxnkQd86r+8KUeN1quSfNKu5bLYFQo=
+github.com/rekby/fixenv v0.6.1 h1:jUFiSPpajT4WY2cYuc++7Y1zWrnCxnovGCIX72PZniM=
+github.com/rekby/fixenv v0.6.1/go.mod h1:/b5LRc06BYJtslRtHKxsPWFT/ySpHV+rWvzTg+XWk4c=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
+github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ=
+github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
+github.com/riza-io/grpc-go v0.2.0 h1:2HxQKFVE7VuYstcJ8zqpN84VnAoJ4dCL6YFhJewNcHQ=
+github.com/riza-io/grpc-go v0.2.0/go.mod h1:2bDvR9KkKC3KhtlSHfR3dAXjUMT86kg4UfWFyVGWqi8=
+github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ=
+github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ=
+github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
github.com/rs/xid v1.6.0/go.mod h1:7XoLgs4eV+QndskICGsho+ADou8ySMSjJKDIan90Nz0=
github.com/rs/zerolog v1.34.0 h1:k43nTLIwcTVQAncfCw4KZ2VY6ukYoZaBPNOE8txlOeY=
github.com/rs/zerolog v1.34.0/go.mod h1:bJsvje4Z08ROH4Nhs5iH600c3IkWhwp44iRc54W6wYQ=
-github.com/ryanhamamura/via v0.23.0 h1:0e7nytisazcWq7uxs6T27GM3FwzosCMenkxJd+78Lko=
-github.com/ryanhamamura/via v0.23.0/go.mod h1:rpJewNVG6tgginZN7Be3qqRuol70+v1sFCKD4UjHsQo=
+github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
+github.com/sajari/fuzzy v1.0.0 h1:+FmwVvJErsd0d0hAPlj4CxqxUtQY/fOoY0DwX4ykpRY=
+github.com/sajari/fuzzy v1.0.0/go.mod h1:OjYR6KxoWOe9+dOlXeiCJd4dIbED4Oo8wpS89o0pwOo=
+github.com/sebdah/goldie/v2 v2.8.0 h1:dZb9wR8q5++oplmEiJT+U/5KyotVD+HNGCAc5gNr8rc=
+github.com/sebdah/goldie/v2 v2.8.0/go.mod h1:oZ9fp0+se1eapSRjfYbsV/0Hqhbuu3bJVvKI/NNtssI=
+github.com/segmentio/asm v1.2.1 h1:DTNbBqs57ioxAD4PrArqftgypG4/qNpXoJx8TVXxPR0=
+github.com/segmentio/asm v1.2.1/go.mod h1:BqMnlJP91P8d+4ibuonYZw9mfnzI9HfxselHZr5aAcs=
+github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3 h1:n661drycOFuPLCN3Uc8sB6B/s6Z4t2xvBgU1htSHuq8=
+github.com/sergi/go-diff v1.3.2-0.20230802210424-5b0b94c5c0d3/go.mod h1:A0bzQcvG0E7Rwjx0REVgAGH58e96+X0MeOfepqsbeW4=
github.com/sethvargo/go-retry v0.3.0 h1:EEt31A35QhrcRZtrYFDTBg91cqZVnFL2navjDrah2SE=
github.com/sethvargo/go-retry v0.3.0/go.mod h1:mNX17F0C/HguQMyMyJxcnU471gOZGxCLyYaFyAZraas=
-github.com/starfederation/datastar-go v1.0.3 h1:DnzgsJ6tDHDM6y5Nxsk0AGW/m8SyKch2vQg3P1xGTcU=
-github.com/starfederation/datastar-go v1.0.3/go.mod h1:stm83LQkhZkwa5GzzdPEN6dLuu8FVwxIv0w1DYkbD3w=
+github.com/shopspring/decimal v1.4.0 h1:bxl37RwXBklmTi0C79JfXCEBD1cqqHt0bbgBAGFp81k=
+github.com/shopspring/decimal v1.4.0/go.mod h1:gawqmDU56v4yIKSwfBSFip1HdCCXN8/+DMd9qYNcwME=
+github.com/spf13/afero v1.14.0 h1:9tH6MapGnn/j0eb0yIXiLjERO8RB6xIVZRDCX7PtqWA=
+github.com/spf13/afero v1.14.0/go.mod h1:acJQ8t0ohCGuMN3O+Pv0V0hgMxNYDlvdk+VTfyZmbYo=
+github.com/spf13/cast v1.9.2 h1:SsGfm7M8QOFtEzumm7UZrZdLLquNdzFYfIbEXntcFbE=
+github.com/spf13/cast v1.9.2/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo=
+github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo=
+github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0=
+github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
+github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk=
+github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
+github.com/spiffe/go-spiffe/v2 v2.6.0 h1:l+DolpxNWYgruGQVV0xsfeya3CsC7m8iBzDnMpsbLuo=
+github.com/spiffe/go-spiffe/v2 v2.6.0/go.mod h1:gm2SeUoMZEtpnzPNs2Csc0D/gX33k1xIx7lEzqblHEs=
+github.com/sqlc-dev/sqlc v1.30.0 h1:H4HrNwPc0hntxGWzAbhlfplPRN4bQpXFx+CaEMcKz6c=
+github.com/sqlc-dev/sqlc v1.30.0/go.mod h1:QnEN+npugyhUg1A+1kkYM3jc2OMOFsNlZ1eh8mdhad0=
+github.com/starfederation/datastar-go v1.1.0 h1:UVOYpbNfKPfrEq3MBOa1FRPO/YsxxcIduUxUTJiEQbQ=
+github.com/starfederation/datastar-go v1.1.0/go.mod h1:stm83LQkhZkwa5GzzdPEN6dLuu8FVwxIv0w1DYkbD3w=
+github.com/stoewer/go-strcase v1.2.0 h1:Z2iHWqGXH00XYgqDmNgQbIBxf3wrNq0F3feEy0ainaU=
+github.com/stoewer/go-strcase v1.2.0/go.mod h1:IBiWB2sKIp3wVVQ3Y035++gc+knqhUQag1KpM8ahLw8=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
+github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY=
+github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
+github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
+github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
+github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
+github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
+github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
+github.com/tdewolff/minify/v2 v2.24.2 h1:vnY3nTulEAbCAAlxTxPPDkzG24rsq31SOzp63yT+7mo=
+github.com/tdewolff/minify/v2 v2.24.2/go.mod h1:1JrCtoZXaDbqioQZfk3Jdmr0GPJKiU7c1Apmb+7tCeE=
+github.com/tdewolff/parse/v2 v2.8.3 h1:5VbvtJ83cfb289A1HzRA9sf02iT8YyUwN84ezjkdY1I=
+github.com/tdewolff/parse/v2 v2.8.3/go.mod h1:Hwlni2tiVNKyzR1o6nUs4FOF07URA+JLBLd6dlIXYqo=
+github.com/tdewolff/test v1.0.11 h1:FdLbwQVHxqG16SlkGveC0JVyrJN62COWTRyUFzfbtBE=
+github.com/tdewolff/test v1.0.11/go.mod h1:XPuWBzvdUzhCuxWO1ojpXsyzsA5bFoS3tO/Q3kFuTG8=
+github.com/tetratelabs/wazero v1.9.0 h1:IcZ56OuxrtaEz8UYNRHBrUa9bYeX9oVY93KspZZBf/I=
+github.com/tetratelabs/wazero v1.9.0/go.mod h1:TSbcXCfFP0L2FGkRPxHphadXPjo1T6W+CseNNY7EkjM=
+github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
+github.com/tursodatabase/libsql-client-go v0.0.0-20251219100830-236aa1ff8acc h1:lzi/5fg2EfinRlh3v//YyIhnc4tY7BTqazQGwb1ar+0=
+github.com/tursodatabase/libsql-client-go v0.0.0-20251219100830-236aa1ff8acc/go.mod h1:08inkKyguB6CGGssc/JzhmQWwBgFQBgjlYFjxjRh7nU=
+github.com/u-root/u-root v0.15.1-0.20251208185023-2f8c7e763cf8 h1:cq+DjLAjz3ZPwh0+G571O/jMH0c0DzReDPLjQGL2/BA=
+github.com/u-root/u-root v0.15.1-0.20251208185023-2f8c7e763cf8/go.mod h1:JNauIV2zopCBv/6o+umxcT3bKe8YUqYJaTZQYSYpKss=
+github.com/u-root/uio v0.0.0-20240224005618-d2acac8f3701 h1:pyC9PaHYZFgEKFdlp3G8RaCKgVpHZnecvArXvPXcFkM=
+github.com/u-root/uio v0.0.0-20240224005618-d2acac8f3701/go.mod h1:P3a5rG4X7tI17Nn3aOIAYr5HbIMukwXG0urG0WuL8OA=
github.com/ulikunitz/xz v0.5.11/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
+github.com/ulikunitz/xz v0.5.15 h1:9DNdB5s+SgV3bQ2ApL10xRc35ck0DuIX/isZvIk+ubY=
+github.com/ulikunitz/xz v0.5.15/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
github.com/valyala/gozstd v1.20.1 h1:xPnnnvjmaDDitMFfDxmQ4vpx0+3CdTg2o3lALvXTU/g=
github.com/valyala/gozstd v1.20.1/go.mod h1:y5Ew47GLlP37EkTB+B4s7r6A5rdaeB7ftbl9zoYiIPQ=
+github.com/vertica/vertica-sql-go v1.3.5 h1:IrfH2WIgzZ45yDHyjVFrXU2LuKNIjF5Nwi90a6cfgUI=
+github.com/vertica/vertica-sql-go v1.3.5/go.mod h1:jnn2GFuv+O2Jcjktb7zyc4Utlbu9YVqpHH/lx63+1M4=
+github.com/wasilibs/go-pgquery v0.0.0-20250409022910-10ac41983c07 h1:mJdDDPblDfPe7z7go8Dvv1AJQDI3eQ/5xith3q2mFlo=
+github.com/wasilibs/go-pgquery v0.0.0-20250409022910-10ac41983c07/go.mod h1:Ak17IJ037caFp4jpCw/iQQ7/W74Sqpb1YuKJU6HTKfM=
+github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52 h1:OvLBa8SqJnZ6P+mjlzc2K7PM22rRUPE1x32G9DTPrC4=
+github.com/wasilibs/wazero-helpers v0.0.0-20240620070341-3dff1577cd52/go.mod h1:jMeV4Vpbi8osrE/pKUxRZkVaA0EX7NZN0A9/oRzgpgY=
+github.com/woodsbury/decimal128 v1.3.0 h1:8pffMNWIlC0O5vbyHWFZAt5yWvWcrHA+3ovIIjVWss0=
+github.com/woodsbury/decimal128 v1.3.0/go.mod h1:C5UTmyTjW3JftjUFzOVhC20BEQa2a4ZKOB5I6Zjb+ds=
+github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI=
+github.com/xdg-go/scram v1.1.1/go.mod h1:RaEWvsqvNKKvBPvcKeFjrG2cJqOkHTiyTpzz23ni57g=
+github.com/xdg-go/stringprep v1.0.3/go.mod h1:W3f5j4i+9rC0kuIEJL0ky1VpHXQU3ocBgklLGvcBnW8=
+github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e h1:JVG44RsyaB9T2KIHavMF/ppJZNG9ZpyihvCd0w101no=
+github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e/go.mod h1:RbqR21r5mrJuqunuUZ/Dhy/avygyECGrLceyNeo4LiM=
github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZqKjWU=
github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E=
+github.com/ydb-platform/ydb-go-genproto v0.0.0-20260128080146-c4ed16b24b37 h1:kUXMT/fM/DpDT66WQgRUf3I8VOAWjypkMf52W5PChwA=
+github.com/ydb-platform/ydb-go-genproto v0.0.0-20260128080146-c4ed16b24b37/go.mod h1:Er+FePu1dNUieD+XTMDduGpQuCPssK5Q4BjF+IIXJ3I=
+github.com/ydb-platform/ydb-go-sdk/v3 v3.127.0 h1:OfHS9ZkZgCy6y/CJ9N8123DXrgaY2BPxWsQiQ8e3wC8=
+github.com/ydb-platform/ydb-go-sdk/v3 v3.127.0/go.mod h1:stS1mQYjbJvwwYaYzKyFY9eMiuVXWWXQA6T+SpOLg9c=
+github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA=
+github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
+github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
+github.com/yuin/goldmark v1.7.13 h1:GPddIs617DnBLFFVJFgpo1aBfe/4xcvMc3SB5t/D0pA=
+github.com/yuin/goldmark v1.7.13/go.mod h1:ip/1k0VRfGynBgxOz0yCqHrbZXhcjxyuS66Brc7iBKg=
+github.com/yuin/goldmark-emoji v1.0.6 h1:QWfF2FYaXwL74tfGOW5izeiZepUDroDJfWubQI9HTHs=
+github.com/yuin/goldmark-emoji v1.0.6/go.mod h1:ukxJDKFpdFb5x0a5HqbdlcKtebh086iJpI31LTKmWuA=
+github.com/zeebo/assert v1.3.0 h1:g7C04CbJuIDKNPFHmsk4hwZDO5O+kntRxzaUoNXj+IQ=
+github.com/zeebo/assert v1.3.0/go.mod h1:Pq9JiuJQpG8JLJdtkwrJESF0Foym2/D9XMU5ciN/wJ0=
+github.com/zeebo/xxh3 v1.0.2 h1:xZmwmqxHZA8AI603jOQ0tMqmBr9lPeFwGg6d+xy9DC0=
+github.com/zeebo/xxh3 v1.0.2/go.mod h1:5NWz9Sef7zIDm2JHfFlcQvNekmcEl9ekUZQQKCYaDcA=
+github.com/ziutek/mymysql v1.5.4 h1:GB0qdRGsTwQSBVYuVShFBKaXSnSnYYC2d9knnE1LHFs=
+github.com/ziutek/mymysql v1.5.4/go.mod h1:LMSpPZ6DbqWFxNCHW77HeMg9I646SAhApZ/wKdgO/C0=
+go.mongodb.org/mongo-driver v1.11.4/go.mod h1:PTSz5yu21bkT/wXpkS7WR5f0ddqw5quethTUn9WM+2g=
+go.opentelemetry.io/auto/sdk v1.2.1 h1:jXsnJ4Lmnqd11kwkBV2LgLoFMZKizbCi5fNZ/ipaZ64=
+go.opentelemetry.io/auto/sdk v1.2.1/go.mod h1:KRTj+aOaElaLi+wW1kO/DZRXwkF4C5xPbEe3ZiIhN7Y=
+go.opentelemetry.io/contrib/detectors/gcp v1.39.0 h1:kWRNZMsfBHZ+uHjiH4y7Etn2FK26LAGkNFw7RHv1DhE=
+go.opentelemetry.io/contrib/detectors/gcp v1.39.0/go.mod h1:t/OGqzHBa5v6RHZwrDBJ2OirWc+4q/w2fTbLZwAKjTk=
+go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.63.0 h1:YH4g8lQroajqUwWbq/tr2QX1JFmEXaDLgG+ew9bLMWo=
+go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.63.0/go.mod h1:fvPi2qXDqFs8M4B4fmJhE92TyQs9Ydjlg3RvfUp+NbQ=
+go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.65.0 h1:7iP2uCb7sGddAr30RRS6xjKy7AZ2JtTOPA3oolgVSw8=
+go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.65.0/go.mod h1:c7hN3ddxs/z6q9xwvfLPk+UHlWRQyaeR1LdgfL/66l0=
+go.opentelemetry.io/otel v1.40.0 h1:oA5YeOcpRTXq6NN7frwmwFR0Cn3RhTVZvXsP4duvCms=
+go.opentelemetry.io/otel v1.40.0/go.mod h1:IMb+uXZUKkMXdPddhwAHm6UfOwJyh4ct1ybIlV14J0g=
+go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.38.0 h1:wm/Q0GAAykXv83wzcKzGGqAnnfLFyFe7RslekZuv+VI=
+go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.38.0/go.mod h1:ra3Pa40+oKjvYh+ZD3EdxFZZB0xdMfuileHAm4nNN7w=
+go.opentelemetry.io/otel/metric v1.40.0 h1:rcZe317KPftE2rstWIBitCdVp89A2HqjkxR3c11+p9g=
+go.opentelemetry.io/otel/metric v1.40.0/go.mod h1:ib/crwQH7N3r5kfiBZQbwrTge743UDc7DTFVZrrXnqc=
+go.opentelemetry.io/otel/sdk v1.40.0 h1:KHW/jUzgo6wsPh9At46+h4upjtccTmuZCFAc9OJ71f8=
+go.opentelemetry.io/otel/sdk v1.40.0/go.mod h1:Ph7EFdYvxq72Y8Li9q8KebuYUr2KoeyHx0DRMKrYBUE=
+go.opentelemetry.io/otel/sdk/metric v1.40.0 h1:mtmdVqgQkeRxHgRv4qhyJduP3fYJRMX4AtAlbuWdCYw=
+go.opentelemetry.io/otel/sdk/metric v1.40.0/go.mod h1:4Z2bGMf0KSK3uRjlczMOeMhKU2rhUqdWNoKcYrtcBPg=
+go.opentelemetry.io/otel/trace v1.40.0 h1:WA4etStDttCSYuhwvEa8OP8I5EWu24lkOzp+ZYblVjw=
+go.opentelemetry.io/otel/trace v1.40.0/go.mod h1:zeAhriXecNGP/s2SEG3+Y8X9ujcJOTqQ5RgdEJcawiA=
+go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI=
+go.uber.org/atomic v1.6.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
+go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
+go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
+go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
+go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
+go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A=
+go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
+go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
+go.uber.org/mock v0.4.0 h1:VcM4ZOtdbR4f6VXfiOpwpVJDL6lCReaZ6mw31wqh7KU=
+go.uber.org/mock v0.4.0/go.mod h1:a6FSlNadKUHUa9IP5Vyt1zh4fC7uAwxMutEAscFbkZc=
+go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
+go.uber.org/multierr v1.7.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak=
go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
+go.uber.org/zap v1.19.0/go.mod h1:xg/QME4nWcxGxrpdeYfq7UvYrLh66cuVKdrbD1XF/NI=
+go.uber.org/zap v1.27.1 h1:08RqriUEv8+ArZRYSTXy1LeBScaMpVSTBhCeaZYfMYc=
+go.uber.org/zap v1.27.1/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E=
+go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc=
+go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg=
+go.yaml.in/yaml/v4 v4.0.0-rc.3 h1:3h1fjsh1CTAPjW7q/EMe+C8shx5d8ctzZTrLcs/j8Go=
+go.yaml.in/yaml/v4 v4.0.0-rc.3/go.mod h1:aZqd9kCMsGL7AuUv/m/PvWLdg5sjJsZ4oHDEnfPPfY0=
+golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
+golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
+golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
+golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.48.0 h1:/VRzVqiRSggnhY7gNRxPauEQ5Drw9haKdM0jqfcCFts=
golang.org/x/crypto v0.48.0/go.mod h1:r0kV5h3qnFPlQnBSrULhlsRfryS2pmewsg+XfMgkVos=
+golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa h1:Zt3DZoOFFYkKhDT3v7Lm9FDMEV06GpzjG2jrqW+QTE0=
golang.org/x/exp v0.0.0-20260218203240-3dfff04db8fa/go.mod h1:K79w1Vqn7PoiZn+TkNpx3BUWUQksGO3JcVX6qIjytmA=
+golang.org/x/image v0.30.0 h1:jD5RhkmVAnjqaCUXfbGBrn3lpxbknfN9w2UhHHU+5B4=
+golang.org/x/image v0.30.0/go.mod h1:SAEUTxCCMWSrJcCy/4HwavEsfZZJlYxeHLc6tTiAe/c=
+golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
+golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
+golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
+golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
+golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
+golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
golang.org/x/mod v0.33.0 h1:tHFzIWbBifEmbwtGz65eaWyGiGZatSrT9prnU8DbVL8=
golang.org/x/mod v0.33.0/go.mod h1:swjeQEj+6r7fODbD2cqrnje9PnziFuw4bmLbBZFrQ5w=
+golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
+golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
+golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
+golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
+golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
+golang.org/x/net v0.50.0 h1:ucWh9eiCGyDR3vtzso0WMQinm2Dnt8cFMuQa9K33J60=
+golang.org/x/net v0.50.0/go.mod h1:UgoSli3F/pBgdJBHCTc+tp3gmrU4XswgGRgtnwWTfyM=
+golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
+golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
+golang.org/x/oauth2 v0.34.0 h1:hqK/t4AKgbqWkdkcAeI8XLmbK+4m4G5YeQRrmiotGlw=
+golang.org/x/oauth2 v0.34.0/go.mod h1:lzm5WQJQwKZ3nwavOZ3IS5Aulzxi68dUSgRHujetwEA=
+golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
+golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.21.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
golang.org/x/sys v0.41.0 h1:Ivj+2Cp/ylzLiEU89QhWblYnOE9zerudt9Ftecq2C6k=
golang.org/x/sys v0.41.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
+golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
+golang.org/x/term v0.40.0 h1:36e4zGLqU4yhjlmxEaagx2KuYbJq3EwY8K943ZsHcvg=
+golang.org/x/term v0.40.0/go.mod h1:w2P8uVp06p2iyKKuvXIm7N/y0UCRt3UfJTfZ7oOpglM=
+golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
+golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
+golang.org/x/text v0.34.0 h1:oL/Qq0Kdaqxa1KbNeMKwQq0reLCCaFtqu2eNuSeNHbk=
+golang.org/x/text v0.34.0/go.mod h1:homfLqTYRFyVYemLBFl5GgL/DWEiH5wcsQ5gSh1yziA=
golang.org/x/time v0.14.0 h1:MRx4UaLrDotUKUdCIqzPC48t1Y9hANFKIRpNx+Te8PI=
golang.org/x/time v0.14.0/go.mod h1:eL/Oa2bBBK0TkX57Fyni+NgnyQQN4LitPmob2Hjnqw4=
+golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
+golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
+golang.org/x/tools v0.0.0-20191029041327-9cc4af7d6b2c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
+golang.org/x/tools v0.0.0-20191108193012-7d206e10da11/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
+golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
+golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
+golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.42.0 h1:uNgphsn75Tdz5Ji2q36v/nsFSfR/9BRFvqhGBaJGd5k=
golang.org/x/tools v0.42.0/go.mod h1:Ma6lCIwGZvHK6XtgbswSoWroEkhugApmsXyrUmBhfr0=
+golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
+golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
+golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
+golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
+gonum.org/v1/gonum v0.16.0 h1:5+ul4Swaf3ESvrOnidPp4GZbzf0mxVQpDCYUQE7OJfk=
+gonum.org/v1/gonum v0.16.0/go.mod h1:fef3am4MQ93R2HHpKnLk4/Tbh/s0+wqD5nfa6Pnwy4E=
+google.golang.org/api v0.256.0 h1:u6Khm8+F9sxbCTYNoBHg6/Hwv0N/i+V94MvkOSor6oI=
+google.golang.org/api v0.256.0/go.mod h1:KIgPhksXADEKJlnEoRa9qAII4rXcy40vfI8HRqcU964=
+google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
+google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
+google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
+google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
+google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
+google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
+google.golang.org/genproto v0.0.0-20250922171735-9219d122eba9 h1:LvZVVaPE0JSqL+ZWb6ErZfnEOKIqqFWUJE2D0fObSmc=
+google.golang.org/genproto v0.0.0-20250922171735-9219d122eba9/go.mod h1:QFOrLhdAe2PsTp3vQY4quuLKTi9j3XG3r6JPPaw7MSc=
+google.golang.org/genproto/googleapis/api v0.0.0-20251202230838-ff82c1b0f217 h1:fCvbg86sFXwdrl5LgVcTEvNC+2txB5mgROGmRL5mrls=
+google.golang.org/genproto/googleapis/api v0.0.0-20251202230838-ff82c1b0f217/go.mod h1:+rXWjjaukWZun3mLfjmVnQi18E1AsFbDN9QdJ5YXLto=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20260217215200-42d3e9bedb6d h1:t/LOSXPJ9R0B6fnZNyALBRfZBH0Uy0gT+uR+SJ6syqQ=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20260217215200-42d3e9bedb6d/go.mod h1:4Hqkh8ycfw05ld/3BWL7rJOSfebL2Q+DVDeRgYgxUU8=
+google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
+google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
+google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
+google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
+google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0=
+google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
+google.golang.org/grpc v1.47.0/go.mod h1:vN9eftEi1UMyUsIF80+uQXhHjbXYbm0uXoFCACuMGWk=
+google.golang.org/grpc v1.79.1 h1:zGhSi45ODB9/p3VAawt9a+O/MULLl9dpizzNNpq7flY=
+google.golang.org/grpc v1.79.1/go.mod h1:KmT0Kjez+0dde/v2j9vzwoAScgEPx/Bw1CYChhHLrHQ=
+google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
+google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
+google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
+google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
+google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
+google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
+google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
+google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
+google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
+google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
+google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
+google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
+google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
+google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
+google.golang.org/protobuf v1.36.11 h1:fV6ZwhNocDyBLK0dj+fg8ektcVegBBuEolpbTQyBNVE=
+google.golang.org/protobuf v1.36.11/go.mod h1:HTf+CrKn2C3g5S8VImy6tdcUvCska2kB7j23XfzDpco=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
+gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
+gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k=
+gopkg.in/natefinch/lumberjack.v2 v2.2.1 h1:bBRl1b0OH9s/DuPhuXpNl+VtCaJXFZ5/uEFST95x9zc=
+gopkg.in/natefinch/lumberjack.v2 v2.2.1/go.mod h1:YD8tP3GAjkrDg1eZH7EGmyESg/lsYskCTPBJVb9jqSc=
+gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0/go.mod h1:WDnlLJ4WF5VGsH/HVa3CI79GS0ol3YnhVnKP89i0kNg=
+gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
+gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
+gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
-maragu.dev/gomponents v1.2.0 h1:H7/N5htz1GCnhu0HB1GasluWeU2rJZOYztVEyN61iTc=
-maragu.dev/gomponents v1.2.0/go.mod h1:oEDahza2gZoXDoDHhw8jBNgH+3UR5ni7Ur648HORydM=
+honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
+honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
+howett.net/plist v0.0.0-20181124034731-591f970eefbb/go.mod h1:vMygbs4qMhSZSc4lCUl2OEE+rDiIIJAIdR4m7MiMcm0=
+howett.net/plist v1.0.1 h1:37GdZ8tP09Q35o9ych3ehygcsL+HqKSwzctveSlarvM=
+howett.net/plist v1.0.1/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g=
modernc.org/cc/v4 v4.27.1 h1:9W30zRlYrefrDV2JE2O8VDtJ1yPGownxciz5rrbQZis=
modernc.org/cc/v4 v4.27.1/go.mod h1:uVtb5OGqUKpoLWhqwNQo/8LwvoiEBLvZXIQ/SmO6mL0=
modernc.org/ccgo/v4 v4.30.2 h1:4yPaaq9dXYXZ2V8s1UgrC3KIj580l2N4ClrLwnbv2so=
@@ -155,3 +884,9 @@ modernc.org/strutil v1.2.1 h1:UneZBkQA+DX2Rp35KcM69cSsNES9ly8mQWD71HKlOA0=
modernc.org/strutil v1.2.1/go.mod h1:EHkiggD70koQxjVdSBM3JKM7k6L0FbGE5eymy9i3B9A=
modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y=
modernc.org/token v1.1.0/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM=
+mvdan.cc/sh/moreinterp v0.0.0-20260120230322-19def062a997 h1:3bbJwtPFh98dJ6lxRdR3eLHTH1CmR3BcU6TriIMiXjE=
+mvdan.cc/sh/moreinterp v0.0.0-20260120230322-19def062a997/go.mod h1:Qy/zdaMDxq9sT72Gi43K3gsV+TtTohyDO3f1cyBVwuo=
+mvdan.cc/sh/v3 v3.12.1-0.20260124232039-e74afc18e65b h1:PUPnLxbDzRO9kg/03l7TZk7+ywTv7FxmOhDHOtOdOtk=
+mvdan.cc/sh/v3 v3.12.1-0.20260124232039-e74afc18e65b/go.mod h1:mencVHx2sy9XZG5wJbCA9nRUOE3zvMtoRXOmXMxH7sc=
+rsc.io/qr v0.2.0 h1:6vBLea5/NRMVTz8V66gipeLycZMl/+UlFmk8DvqQ6WY=
+rsc.io/qr v0.2.0/go.mod h1:IF+uZjkb9fqyeF/4tlBoynqmQxUoPfWEKh921coOuXs=
diff --git a/main.go b/main.go
index e324692..bb437a1 100644
--- a/main.go
+++ b/main.go
@@ -2,787 +2,127 @@ package main
import (
"context"
- "crypto/md5"
- "database/sql"
"embed"
- "encoding/hex"
- "encoding/json"
- "io/fs"
- "sync"
+ "fmt"
+ "log/slog"
+ "net"
+ "net/http"
+ "os/signal"
+ "syscall"
"time"
- "github.com/ryanhamamura/c4/auth"
"github.com/ryanhamamura/c4/config"
"github.com/ryanhamamura/c4/db"
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
"github.com/ryanhamamura/c4/logging"
+ appnats "github.com/ryanhamamura/c4/nats"
+ "github.com/ryanhamamura/c4/router"
+ "github.com/ryanhamamura/c4/sessions"
"github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/c4/ui"
- "github.com/google/uuid"
+ "github.com/go-chi/chi/v5"
+ "github.com/go-chi/chi/v5/middleware"
"github.com/rs/zerolog/log"
- "github.com/ryanhamamura/via"
- "github.com/ryanhamamura/via/h"
-)
-
-var (
- store = game.NewGameStore()
- snakeStore = snake.NewSnakeStore()
- queries *repository.Queries
- chatPersister *db.ChatPersister
+ "golang.org/x/sync/errgroup"
)
//go:embed assets
var assets embed.FS
-func DaisyUIPlugin(v *via.V) {
- css, _ := fs.ReadFile(assets, "assets/css/output.css")
- sum := md5.Sum(css)
- version := hex.EncodeToString(sum[:4])
- v.AppendToHead(h.Link(h.Rel("stylesheet"), h.Href("/assets/css/output.css?v="+version)))
+func main() {
+ ctx, cancel := signal.NotifyContext(context.Background(), syscall.SIGINT, syscall.SIGTERM)
+ defer cancel()
+
+ cfg := config.Global
+ logging.SetupLogger(cfg.Environment, cfg.LogLevel)
+
+ if err := run(ctx); err != nil && err != http.ErrServerClosed {
+ log.Fatal().Err(err).Msg("server error")
+ }
}
-func main() {
+func run(ctx context.Context) error {
cfg := config.Global
- logger := logging.SetupLogger(cfg.Environment, cfg.LogLevel)
+ addr := fmt.Sprintf("%s:%s", cfg.Host, cfg.Port)
+ slog.Info("server starting", "addr", addr)
+ defer slog.Info("server shutdown complete")
+ eg, egctx := errgroup.WithContext(ctx)
+
+ // Database
cleanupDB, err := db.Init(cfg.DBPath)
if err != nil {
- log.Fatal().Err(err).Msg("initializing database")
+ return fmt.Errorf("initializing database: %w", err)
}
defer cleanupDB()
- queries = repository.New(db.DB)
- store.SetPersister(db.NewGamePersister(queries))
- snakeStore.SetPersister(db.NewSnakePersister(queries))
- chatPersister = db.NewChatPersister(queries)
+ queries := repository.New(db.DB)
- sessionManager, err := via.NewSQLiteSessionManager(db.DB)
+ // Sessions
+ sessionManager, cleanupSessions := sessions.SetupSessionManager(db.DB)
+ defer cleanupSessions()
+
+ // NATS
+ nc, cleanupNATS, err := appnats.SetupNATS(egctx)
if err != nil {
- log.Fatal().Err(err).Msg("creating session manager")
+ return fmt.Errorf("setting up NATS: %w", err)
+ }
+ defer cleanupNATS()
+
+ // Game stores
+ store := game.NewGameStore()
+ store.SetPersister(db.NewGamePersister(queries))
+ store.SetNotifyFunc(func(gameID string) {
+ nc.Publish("game."+gameID, nil) //nolint:errcheck // best-effort notification
+ })
+
+ snakeStore := snake.NewSnakeStore()
+ snakeStore.SetPersister(db.NewSnakePersister(queries))
+ snakeStore.SetNotifyFunc(func(gameID string) {
+ nc.Publish("snake."+gameID, nil) //nolint:errcheck // best-effort notification
+ })
+
+ chatPersister := db.NewChatPersister(queries)
+
+ // Router
+ logger := log.Logger
+ r := chi.NewMux()
+ r.Use(
+ logging.RequestLogger(&logger, cfg.Environment),
+ middleware.Recoverer,
+ sessionManager.LoadAndSave,
+ )
+
+ if err := router.SetupRoutes(r, queries, sessionManager, nc, store, snakeStore, chatPersister, assets); err != nil {
+ return fmt.Errorf("setting up routes: %w", err)
}
- _ = logger
+ // HTTP server
+ srv := &http.Server{
+ Addr: addr,
+ Handler: r,
+ ReadHeaderTimeout: 10 * time.Second,
+ BaseContext: func(l net.Listener) context.Context {
+ return egctx
+ },
+ }
- v := via.New()
- v.Config(via.Options{
- LogLevel: via.LogLevelDebug,
- DocumentTitle: "Game Lobby",
- ServerAddress: ":" + cfg.Port,
- SessionManager: sessionManager,
- Plugins: []via.Plugin{DaisyUIPlugin},
+ eg.Go(func() error {
+ err := srv.ListenAndServe()
+ if err != nil && err != http.ErrServerClosed {
+ return fmt.Errorf("server error: %w", err)
+ }
+ return nil
})
- subFS, _ := fs.Sub(assets, "assets")
- v.StaticFS("/assets/", subFS)
-
- store.SetNotifyFunc(func(gameID string) {
- v.PubSub().Publish("game."+gameID, nil)
- })
- snakeStore.SetNotifyFunc(func(gameID string) {
- v.PubSub().Publish("snake."+gameID, nil)
+ eg.Go(func() error {
+ <-egctx.Done()
+ shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
+ defer cancel()
+ slog.Debug("shutting down server...")
+ return srv.Shutdown(shutdownCtx)
})
- // Home page - tabbed lobby
- v.Page("/", func(c *via.Context) {
- userID := c.Session().GetString("user_id")
- username := c.Session().GetString("username")
- isLoggedIn := userID != ""
-
- var userGames []ui.GameListItem
- if isLoggedIn {
- ctx := context.Background()
- games, err := queries.GetUserActiveGames(ctx, sql.NullString{String: userID, Valid: true})
- if err == nil {
- for _, g := range games {
- isMyTurn := g.Status == 1 && g.CurrentTurn == g.MyColor
- userGames = append(userGames, ui.GameListItem{
- ID: g.ID,
- Status: int(g.Status),
- OpponentName: g.OpponentNickname.String,
- IsMyTurn: isMyTurn,
- LastPlayed: g.UpdatedAt.Time,
- })
- }
- }
- }
-
- nickname := c.Signal("")
- if isLoggedIn {
- nickname = c.Signal(username)
- }
- activeTab := c.Signal("connect4")
-
- logout := c.Action(func() {
- c.Session().Clear()
- c.Redirect("/")
- })
-
- createGame := c.Action(func() {
- name := nickname.String()
- if name == "" {
- return
- }
- c.Session().Set("nickname", name)
-
- gi := store.Create()
- c.Redirectf("/game/%s", gi.ID())
- })
-
- deleteGame := func(id string) h.H {
- return c.Action(func() {
- for _, g := range userGames {
- if g.ID == id {
- store.Delete(id)
- break
- }
- }
- c.Redirect("/")
- }).OnClick()
- }
-
- tabClickConnect4 := c.Action(func() {
- activeTab.SetValue("connect4")
- c.Sync()
- })
-
- tabClickSnake := c.Action(func() {
- activeTab.SetValue("snake")
- c.Sync()
- })
-
- snakeNickname := c.Signal("")
- if isLoggedIn {
- snakeNickname = c.Signal(username)
- }
-
- // Speed selection signal (index into SpeedPresets, default to Normal which is index 1)
- selectedSpeedIndex := c.Signal(1)
-
- // Speed selector actions
- var speedSelectClicks []h.H
- for i := range snake.SpeedPresets {
- idx := i
- speedSelectClicks = append(speedSelectClicks, c.Action(func() {
- selectedSpeedIndex.SetValue(idx)
- c.Sync()
- }).OnClick())
- }
-
- // Snake create game actions — one per preset for solo and multiplayer
- var snakeSoloClicks []h.H
- var snakeMultiClicks []h.H
- for _, preset := range snake.GridPresets {
- w, ht := preset.Width, preset.Height
- snakeSoloClicks = append(snakeSoloClicks, c.Action(func() {
- name := snakeNickname.String()
- if name == "" {
- return
- }
- c.Session().Set("nickname", name)
- speedIdx := selectedSpeedIndex.Int()
- speed := snake.DefaultSpeed
- if speedIdx >= 0 && speedIdx < len(snake.SpeedPresets) {
- speed = snake.SpeedPresets[speedIdx].Speed
- }
- si := snakeStore.Create(w, ht, snake.ModeSinglePlayer, speed)
- c.Redirectf("/snake/%s", si.ID())
- }).OnClick())
- snakeMultiClicks = append(snakeMultiClicks, c.Action(func() {
- name := snakeNickname.String()
- if name == "" {
- return
- }
- c.Session().Set("nickname", name)
- speedIdx := selectedSpeedIndex.Int()
- speed := snake.DefaultSpeed
- if speedIdx >= 0 && speedIdx < len(snake.SpeedPresets) {
- speed = snake.SpeedPresets[speedIdx].Speed
- }
- si := snakeStore.Create(w, ht, snake.ModeMultiplayer, speed)
- c.Redirectf("/snake/%s", si.ID())
- }).OnClick())
- }
-
- c.View(func() h.H {
- return ui.LobbyView(ui.LobbyProps{
- NicknameBind: nickname.Bind(),
- CreateGameKeyDown: createGame.OnKeyDown("Enter"),
- CreateGameClick: createGame.OnClick(),
- IsLoggedIn: isLoggedIn,
- Username: username,
- LogoutClick: logout.OnClick(),
- UserGames: userGames,
- DeleteGameClick: deleteGame,
- ActiveTab: activeTab.String(),
- TabClickConnect4: tabClickConnect4.OnClick(),
- TabClickSnake: tabClickSnake.OnClick(),
- SnakeNicknameBind: snakeNickname.Bind(),
- SnakeSoloClicks: snakeSoloClicks,
- SnakeMultiClicks: snakeMultiClicks,
- ActiveSnakeGames: snakeStore.ActiveGames(),
- SelectedSpeedIndex: selectedSpeedIndex.Int(),
- SpeedSelectClicks: speedSelectClicks,
- })
- })
- })
-
- // Login page
- v.Page("/login", func(c *via.Context) {
- username := c.Signal("")
- password := c.Signal("")
- errorMsg := c.Signal("")
-
- login := c.Action(func() {
- ctx := context.Background()
- user, err := queries.GetUserByUsername(ctx, username.String())
- if err == sql.ErrNoRows {
- errorMsg.SetValue("Invalid username or password")
- c.Sync()
- return
- }
- if err != nil {
- errorMsg.SetValue("An error occurred")
- c.Sync()
- return
- }
- if !auth.CheckPassword(password.String(), user.PasswordHash) {
- errorMsg.SetValue("Invalid username or password")
- c.Sync()
- return
- }
-
- c.Session().RenewToken()
- c.Session().Set("user_id", user.ID)
- c.Session().Set("username", user.Username)
- c.Session().Set("nickname", user.Username)
-
- returnURL := c.Session().GetString("return_url")
- if returnURL != "" {
- c.Session().Set("return_url", "")
- c.Redirect(returnURL)
- } else {
- c.Redirect("/")
- }
- })
-
- c.View(func() h.H {
- return ui.LoginView(
- username.Bind(),
- password.Bind(),
- login.OnKeyDown("Enter"),
- login.OnClick(),
- errorMsg.String(),
- )
- })
- })
-
- // Register page
- v.Page("/register", func(c *via.Context) {
- username := c.Signal("")
- password := c.Signal("")
- confirm := c.Signal("")
- errorMsg := c.Signal("")
-
- register := c.Action(func() {
- if err := auth.ValidateUsername(username.String()); err != nil {
- errorMsg.SetValue(err.Error())
- c.Sync()
- return
- }
- if err := auth.ValidatePassword(password.String()); err != nil {
- errorMsg.SetValue(err.Error())
- c.Sync()
- return
- }
- if password.String() != confirm.String() {
- errorMsg.SetValue("Passwords do not match")
- c.Sync()
- return
- }
-
- hash, err := auth.HashPassword(password.String())
- if err != nil {
- errorMsg.SetValue("An error occurred")
- c.Sync()
- return
- }
-
- ctx := context.Background()
- id := uuid.New().String()
- user, err := queries.CreateUser(ctx, repository.CreateUserParams{
- ID: id,
- Username: username.String(),
- PasswordHash: hash,
- })
- if err != nil {
- errorMsg.SetValue("Username already taken")
- c.Sync()
- return
- }
-
- c.Session().RenewToken()
- c.Session().Set("user_id", user.ID)
- c.Session().Set("username", user.Username)
- c.Session().Set("nickname", user.Username)
-
- returnURL := c.Session().GetString("return_url")
- if returnURL != "" {
- c.Session().Set("return_url", "")
- c.Redirect(returnURL)
- } else {
- c.Redirect("/")
- }
- })
-
- c.View(func() h.H {
- return ui.RegisterView(
- username.Bind(),
- password.Bind(),
- confirm.Bind(),
- register.OnKeyDown("Enter"),
- register.OnClick(),
- errorMsg.String(),
- )
- })
- })
-
- // Connect 4 game page
- v.Page("/game/{game_id}", func(c *via.Context) {
- gameID := c.GetPathParam("game_id")
- sessionNickname := c.Session().GetString("nickname")
- sessionUserID := c.Session().GetString("user_id")
-
- nickname := c.Signal(sessionNickname)
- colSignal := c.Signal(0)
- showGuestPrompt := c.Signal(false)
- chatMsg := c.Signal("")
- chatMessages, _ := chatPersister.LoadChatMessages(gameID)
- var chatMu sync.Mutex
-
- goToLogin := c.Action(func() {
- c.Session().Set("return_url", "/game/"+gameID)
- c.Redirect("/login")
- })
-
- goToRegister := c.Action(func() {
- c.Session().Set("return_url", "/game/"+gameID)
- c.Redirect("/register")
- })
-
- continueAsGuest := c.Action(func() {
- showGuestPrompt.SetValue(true)
- c.Sync()
- })
-
- var gi *game.GameInstance
- var gameExists bool
-
- if gameID != "" {
- gi, gameExists = store.Get(gameID)
- }
-
- playerID := game.PlayerID(c.Session().GetString("player_id"))
- if playerID == "" {
- playerID = game.PlayerID(game.GenerateID(8))
- c.Session().Set("player_id", string(playerID))
- }
-
- if sessionUserID != "" {
- playerID = game.PlayerID(sessionUserID)
- }
-
- setNickname := c.Action(func() {
- if gi == nil {
- return
- }
- name := nickname.String()
- if name == "" {
- return
- }
- c.Session().Set("nickname", name)
-
- if gi.GetPlayerColor(playerID) == 0 {
- player := &game.Player{
- ID: playerID,
- Nickname: name,
- }
- if sessionUserID != "" {
- player.UserID = &sessionUserID
- }
- gi.Join(&game.PlayerSession{
- Player: player,
- })
- }
- c.Sync()
- })
-
- dropPiece := c.Action(func() {
- if gi == nil {
- return
- }
- myColor := gi.GetPlayerColor(playerID)
- if myColor == 0 {
- return
- }
- col := colSignal.Int()
- gi.DropPiece(col, myColor)
- c.Sync()
- })
-
- createRematch := c.Action(func() {
- if gi == nil {
- return
- }
- newGI := gi.CreateRematch(store)
- if newGI != nil {
- c.Redirectf("/game/%s", newGI.ID())
- }
- })
-
- sendChat := c.Action(func() {
- msg := chatMsg.String()
- if msg == "" || gi == nil {
- return
- }
- color := gi.GetPlayerColor(playerID)
- if color == 0 {
- return
- }
- g := gi.GetGame()
- nick := ""
- for _, p := range g.Players {
- if p != nil && p.ID == playerID {
- nick = p.Nickname
- break
- }
- }
- cm := ui.C4ChatMessage{
- Nickname: nick,
- Color: color,
- Message: msg,
- Time: time.Now().UnixMilli(),
- }
- chatPersister.SaveChatMessage(gameID, cm)
- data, err := json.Marshal(cm)
- if err != nil {
- return
- }
- c.Publish("game.chat."+gameID, data)
- chatMsg.SetValue("")
- })
-
- if gameExists {
- c.Subscribe("game."+gameID, func(data []byte) { c.Sync() })
-
- c.Subscribe("game.chat."+gameID, func(data []byte) {
- var cm ui.C4ChatMessage
- if err := json.Unmarshal(data, &cm); err != nil {
- return
- }
- chatMu.Lock()
- chatMessages = append(chatMessages, cm)
- if len(chatMessages) > 50 {
- chatMessages = chatMessages[len(chatMessages)-50:]
- }
- chatMu.Unlock()
- c.Sync()
- })
- }
-
- if gameExists && sessionNickname != "" && gi.GetPlayerColor(playerID) == 0 {
- player := &game.Player{
- ID: playerID,
- Nickname: sessionNickname,
- }
- if sessionUserID != "" {
- player.UserID = &sessionUserID
- }
- gi.Join(&game.PlayerSession{
- Player: player,
- })
- }
-
- c.View(func() h.H {
- if !gameExists {
- c.Redirect("/")
- return h.Div()
- }
-
- myColor := gi.GetPlayerColor(playerID)
-
- if myColor == 0 {
- if sessionUserID == "" && !showGuestPrompt.Bool() {
- return ui.GameJoinPrompt(
- goToLogin.OnClick(),
- continueAsGuest.OnClick(),
- goToRegister.OnClick(),
- )
- }
- return ui.NicknamePrompt(
- nickname.Bind(),
- setNickname.OnKeyDown("Enter"),
- setNickname.OnClick(),
- )
- }
-
- g := gi.GetGame()
-
- columnClick := func(col int) h.H {
- return dropPiece.OnClick(via.WithSignalInt(colSignal, col))
- }
-
- chatMu.Lock()
- msgs := make([]ui.C4ChatMessage, len(chatMessages))
- copy(msgs, chatMessages)
- chatMu.Unlock()
- chat := ui.C4Chat(msgs, chatMsg.Bind(), sendChat.OnClick(), sendChat.OnKeyDown("Enter"))
-
- var content []h.H
- content = append(content,
- ui.BackToLobby(),
- ui.StealthTitle("text-3xl font-bold"),
- ui.PlayerInfo(g, myColor),
- ui.StatusBanner(g, myColor, createRematch.OnClick()),
- h.Div(h.Class("c4-game-area"), ui.BoardComponent(g, columnClick, myColor), chat),
- )
-
- if g.Status == game.StatusWaitingForPlayer {
- content = append(content, ui.InviteLink(g.ID))
- }
-
- mainAttrs := []h.H{h.Class("flex flex-col items-center gap-4 p-4")}
- mainAttrs = append(mainAttrs, content...)
- return h.Main(mainAttrs...)
- })
- })
-
- // Snake game page
- v.Page("/snake/{game_id}", func(c *via.Context) {
- gameID := c.GetPathParam("game_id")
- sessionNickname := c.Session().GetString("nickname")
- sessionUserID := c.Session().GetString("user_id")
-
- nickname := c.Signal(sessionNickname)
- showGuestPrompt := c.Signal(false)
-
- goToLogin := c.Action(func() {
- c.Session().Set("return_url", "/snake/"+gameID)
- c.Redirect("/login")
- })
-
- goToRegister := c.Action(func() {
- c.Session().Set("return_url", "/snake/"+gameID)
- c.Redirect("/register")
- })
-
- continueAsGuest := c.Action(func() {
- showGuestPrompt.SetValue(true)
- c.Sync()
- })
-
- var si *snake.SnakeGameInstance
- var gameExists bool
-
- if gameID != "" {
- si, gameExists = snakeStore.Get(gameID)
- }
-
- playerID := snake.PlayerID(c.Session().GetString("player_id"))
- if playerID == "" {
- pid := game.GenerateID(8)
- playerID = snake.PlayerID(pid)
- c.Session().Set("player_id", pid)
- }
- if sessionUserID != "" {
- playerID = snake.PlayerID(sessionUserID)
- }
-
- setNickname := c.Action(func() {
- if si == nil {
- return
- }
- name := nickname.String()
- if name == "" {
- return
- }
- c.Session().Set("nickname", name)
-
- if si.GetPlayerSlot(playerID) < 0 {
- player := &snake.Player{
- ID: playerID,
- Nickname: name,
- }
- if sessionUserID != "" {
- player.UserID = &sessionUserID
- }
- si.Join(player)
- }
- c.Sync()
- })
-
- // Direction input: single action with a direction signal
- dirSignal := c.Signal(-1)
- handleDir := c.Action(func() {
- if si == nil {
- return
- }
- slot := si.GetPlayerSlot(playerID)
- if slot < 0 {
- return
- }
- dir := snake.Direction(dirSignal.Int())
- si.SetDirection(slot, dir)
- })
-
- createRematch := c.Action(func() {
- if si == nil {
- return
- }
- newSI := si.CreateRematch()
- if newSI != nil {
- c.Redirectf("/snake/%s", newSI.ID())
- }
- })
-
- chatMsg := c.Signal("")
- var chatMessages []ui.ChatMessage
- var chatMu sync.Mutex
-
- sendChat := c.Action(func() {
- msg := chatMsg.String()
- if msg == "" || si == nil {
- return
- }
- slot := si.GetPlayerSlot(playerID)
- if slot < 0 {
- return
- }
- cm := ui.ChatMessage{
- Nickname: si.GetGame().Players[slot].Nickname,
- Slot: slot,
- Message: msg,
- Time: time.Now().UnixMilli(),
- }
- data, err := json.Marshal(cm)
- if err != nil {
- return
- }
- c.Publish("snake.chat."+gameID, data)
- chatMsg.SetValue("")
- })
-
- if gameExists {
- c.Subscribe("snake."+gameID, func(data []byte) { c.Sync() })
-
- if si.GetGame().Mode == snake.ModeMultiplayer {
- c.Subscribe("snake.chat."+gameID, func(data []byte) {
- var cm ui.ChatMessage
- if err := json.Unmarshal(data, &cm); err != nil {
- return
- }
- chatMu.Lock()
- chatMessages = append(chatMessages, cm)
- if len(chatMessages) > 50 {
- chatMessages = chatMessages[len(chatMessages)-50:]
- }
- chatMu.Unlock()
- c.Sync()
- })
- }
- }
-
- // Auto-join if nickname exists
- if gameExists && sessionNickname != "" && si.GetPlayerSlot(playerID) < 0 {
- player := &snake.Player{
- ID: playerID,
- Nickname: sessionNickname,
- }
- if sessionUserID != "" {
- player.UserID = &sessionUserID
- }
- si.Join(player)
- }
-
- c.View(func() h.H {
- if !gameExists {
- c.Redirect("/")
- return h.Div()
- }
-
- mySlot := si.GetPlayerSlot(playerID)
-
- if mySlot < 0 {
- if sessionUserID == "" && !showGuestPrompt.Bool() {
- return ui.GameJoinPrompt(
- goToLogin.OnClick(),
- continueAsGuest.OnClick(),
- goToRegister.OnClick(),
- )
- }
- return ui.NicknamePrompt(
- nickname.Bind(),
- setNickname.OnKeyDown("Enter"),
- setNickname.OnClick(),
- )
- }
-
- sg := si.GetGame()
-
- var content []h.H
- content = append(content,
- ui.BackToLobby(),
- h.H1(h.Class("text-3xl font-bold"), h.Text("~~~~")),
- ui.SnakePlayerList(sg, mySlot),
- ui.SnakeStatusBanner(sg, mySlot, createRematch.OnClick()),
- )
-
- if sg.Status == snake.StatusInProgress || sg.Status == snake.StatusFinished {
- board := ui.SnakeBoard(sg)
-
- if sg.Mode == snake.ModeMultiplayer {
- chatMu.Lock()
- msgs := make([]ui.ChatMessage, len(chatMessages))
- copy(msgs, chatMessages)
- chatMu.Unlock()
- chat := ui.SnakeChat(msgs, chatMsg.Bind(), sendChat.OnClick(), sendChat.OnKeyDown("Enter"))
- content = append(content, h.Div(h.Class("snake-game-area"), board, chat))
- } else {
- content = append(content, board)
- }
- } else if sg.Mode == snake.ModeMultiplayer {
- // Show chat even before game starts (waiting/countdown)
- chatMu.Lock()
- msgs := make([]ui.ChatMessage, len(chatMessages))
- copy(msgs, chatMessages)
- chatMu.Unlock()
- content = append(content, ui.SnakeChat(msgs, chatMsg.Bind(), sendChat.OnClick(), sendChat.OnKeyDown("Enter")))
- }
-
- // Only show invite link for multiplayer games
- if sg.Mode == snake.ModeMultiplayer && (sg.Status == snake.StatusWaitingForPlayers || sg.Status == snake.StatusCountdown) {
- content = append(content, ui.SnakeInviteLink(sg.ID))
- }
-
- wrapperAttrs := []h.H{
- h.Class("snake-wrapper flex flex-col items-center gap-4 p-4"),
- via.OnKeyDownMap(
- via.KeyBind("w", handleDir, via.WithSignalInt(dirSignal, int(snake.DirUp)), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("a", handleDir, via.WithSignalInt(dirSignal, int(snake.DirLeft)), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("s", handleDir, via.WithSignalInt(dirSignal, int(snake.DirDown)), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("d", handleDir, via.WithSignalInt(dirSignal, int(snake.DirRight)), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("ArrowUp", handleDir, via.WithSignalInt(dirSignal, int(snake.DirUp)), via.WithPreventDefault(), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("ArrowLeft", handleDir, via.WithSignalInt(dirSignal, int(snake.DirLeft)), via.WithPreventDefault(), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("ArrowDown", handleDir, via.WithSignalInt(dirSignal, int(snake.DirDown)), via.WithPreventDefault(), via.WithThrottle(100*time.Millisecond)),
- via.KeyBind("ArrowRight", handleDir, via.WithSignalInt(dirSignal, int(snake.DirRight)), via.WithPreventDefault(), via.WithThrottle(100*time.Millisecond)),
- ),
- }
-
- wrapperAttrs = append(wrapperAttrs, content...)
- return h.Main(wrapperAttrs...)
- })
- })
-
- v.Start()
+ return eg.Wait()
}
diff --git a/nats/nats.go b/nats/nats.go
new file mode 100644
index 0000000..9eeabc9
--- /dev/null
+++ b/nats/nats.go
@@ -0,0 +1,69 @@
+// Package nats sets up an embedded NATS server for real-time pub/sub
+// messaging between game clients.
+package nats
+
+import (
+ "context"
+ "fmt"
+ "log/slog"
+ "net"
+ "os"
+ "strconv"
+
+ "github.com/delaneyj/toolbelt"
+ "github.com/delaneyj/toolbelt/embeddednats"
+ natsserver "github.com/nats-io/nats-server/v2/server"
+ "github.com/nats-io/nats.go"
+)
+
+func SetupNATS(ctx context.Context) (*nats.Conn, func(), error) {
+ natsPort, err := getFreeNatsPort()
+ if err != nil {
+ return nil, nil, fmt.Errorf("obtaining NATS port: %w", err)
+ }
+
+ ns, err := embeddednats.New(ctx, embeddednats.WithNATSServerOptions(&natsserver.Options{
+ NoSigs: true,
+ Port: natsPort,
+ }))
+ if err != nil {
+ return nil, nil, fmt.Errorf("creating embedded nats server: %w", err)
+ }
+
+ ns.WaitForServer()
+ slog.Info("NATS started", "port", natsPort)
+
+ nc, err := ns.Client()
+ if err != nil {
+ return nil, nil, fmt.Errorf("creating nats client: %w", err)
+ }
+
+ cleanup := func() {
+ nc.Close()
+ ns.Close()
+ }
+
+ return nc, cleanup, nil
+}
+
+func getFreeNatsPort() (int, error) {
+ if p, ok := os.LookupEnv("NATS_PORT"); ok {
+ natsPort, err := strconv.Atoi(p)
+ if err != nil {
+ return 0, fmt.Errorf("parsing NATS_PORT: %w", err)
+ }
+ if isPortFree(natsPort) {
+ return natsPort, nil
+ }
+ }
+ return toolbelt.FreePort()
+}
+
+func isPortFree(port int) bool {
+ ln, err := net.Listen("tcp", fmt.Sprintf(":%d", port))
+ if err != nil {
+ return false
+ }
+ ln.Close() //nolint:errcheck // checking port availability
+ return true
+}
diff --git a/router/router.go b/router/router.go
new file mode 100644
index 0000000..a42a09d
--- /dev/null
+++ b/router/router.go
@@ -0,0 +1,76 @@
+// Package router wires feature routes and middleware into the central chi mux.
+package router
+
+import (
+ "embed"
+ "io/fs"
+ "net/http"
+ "sync"
+
+ "github.com/ryanhamamura/c4/config"
+ "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/db/repository"
+ "github.com/ryanhamamura/c4/features/auth"
+ "github.com/ryanhamamura/c4/features/c4game"
+ "github.com/ryanhamamura/c4/features/lobby"
+ "github.com/ryanhamamura/c4/features/snakegame"
+ "github.com/ryanhamamura/c4/game"
+ "github.com/ryanhamamura/c4/snake"
+
+ "github.com/alexedwards/scs/v2"
+ "github.com/go-chi/chi/v5"
+ "github.com/nats-io/nats.go"
+ "github.com/starfederation/datastar-go/datastar"
+)
+
+func SetupRoutes(
+ router chi.Router,
+ queries *repository.Queries,
+ sessions *scs.SessionManager,
+ nc *nats.Conn,
+ store *game.GameStore,
+ snakeStore *snake.SnakeStore,
+ chatPersister *db.ChatPersister,
+ assets embed.FS,
+) error {
+ // Static assets
+ subFS, _ := fs.Sub(assets, "assets")
+ router.Handle("/assets/*", http.StripPrefix("/assets/", http.FileServerFS(subFS)))
+
+ // Hot-reload for development
+ if config.Global.Environment == config.Dev {
+ setupReload(router)
+ }
+
+ auth.SetupRoutes(router, queries, sessions)
+ lobby.SetupRoutes(router, queries, sessions, store, snakeStore)
+ c4game.SetupRoutes(router, store, nc, sessions, chatPersister)
+ snakegame.SetupRoutes(router, snakeStore, nc, sessions)
+
+ return nil
+}
+
+func setupReload(router chi.Router) {
+ reloadChan := make(chan struct{}, 1)
+ var hotReloadOnce sync.Once
+
+ router.Get("/reload", func(w http.ResponseWriter, r *http.Request) {
+ sse := datastar.NewSSE(w, r)
+ reload := func() { sse.ExecuteScript("window.location.reload()") } //nolint:errcheck // dev-only
+ hotReloadOnce.Do(reload)
+ select {
+ case <-reloadChan:
+ reload()
+ case <-r.Context().Done():
+ }
+ })
+
+ router.Get("/hotreload", func(w http.ResponseWriter, r *http.Request) {
+ select {
+ case reloadChan <- struct{}{}:
+ default:
+ }
+ w.WriteHeader(http.StatusOK)
+ w.Write([]byte("OK")) //nolint:errcheck // dev-only
+ })
+}
diff --git a/sessions/sessions.go b/sessions/sessions.go
new file mode 100644
index 0000000..489a942
--- /dev/null
+++ b/sessions/sessions.go
@@ -0,0 +1,31 @@
+// Package sessions configures the SCS session manager backed by SQLite.
+package sessions
+
+import (
+ "database/sql"
+ "log/slog"
+ "net/http"
+ "time"
+
+ "github.com/alexedwards/scs/sqlite3store"
+ "github.com/alexedwards/scs/v2"
+)
+
+// SetupSessionManager creates a configured session manager backed by SQLite.
+// Returns the manager and a cleanup function the caller should defer.
+func SetupSessionManager(db *sql.DB) (*scs.SessionManager, func()) {
+ store := sqlite3store.New(db)
+ cleanup := func() { store.StopCleanup() }
+
+ sessionManager := scs.New()
+ sessionManager.Store = store
+ sessionManager.Lifetime = 30 * 24 * time.Hour
+ sessionManager.Cookie.Path = "/"
+ sessionManager.Cookie.HttpOnly = true
+ sessionManager.Cookie.Secure = false
+ sessionManager.Cookie.SameSite = http.SameSiteLaxMode
+
+ slog.Info("session manager configured")
+
+ return sessionManager, cleanup
+}
diff --git a/ui/auth.go b/ui/auth.go
deleted file mode 100644
index a3bfa24..0000000
--- a/ui/auth.go
+++ /dev/null
@@ -1,130 +0,0 @@
-package ui
-
-import (
- "github.com/ryanhamamura/via/h"
-)
-
-func LoginView(usernameBind, passwordBind, loginKeyDown, loginClick h.H, errorMsg string) h.H {
- var errorEl h.H
- if errorMsg != "" {
- errorEl = h.P(h.Class("alert alert-error mb-4"), h.Text(errorMsg))
- }
-
- return h.Main(h.Class("max-w-sm mx-auto mt-8 text-center"),
- h.H1(h.Class("text-3xl font-bold"), h.Text("Login")),
- h.P(h.Class("mb-4"), h.Text("Sign in to your account")),
- errorEl,
- h.Form(
- h.FieldSet(h.Class("fieldset"),
- h.Label(h.Class("label"), h.Text("Username"), h.Attr("for", "username")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("username"),
- h.Type("text"),
- h.Placeholder("Enter your username"),
- usernameBind,
- h.Attr("required"),
- h.Attr("autofocus"),
- ),
- h.Label(h.Class("label"), h.Text("Password"), h.Attr("for", "password")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("password"),
- h.Type("password"),
- h.Placeholder("Enter your password"),
- passwordBind,
- h.Attr("required"),
- loginKeyDown,
- ),
- ),
- h.Button(
- h.Class("btn btn-primary w-full"),
- h.Type("button"),
- h.Text("Login"),
- loginClick,
- ),
- ),
- h.P(
- h.Text("Don't have an account? "),
- h.A(h.Class("link"), h.Href("/register"), h.Text("Register")),
- ),
- )
-}
-
-func RegisterView(usernameBind, passwordBind, confirmBind, registerKeyDown, registerClick h.H, errorMsg string) h.H {
- var errorEl h.H
- if errorMsg != "" {
- errorEl = h.P(h.Class("alert alert-error mb-4"), h.Text(errorMsg))
- }
-
- return h.Main(h.Class("max-w-sm mx-auto mt-8 text-center"),
- h.H1(h.Class("text-3xl font-bold"), h.Text("Register")),
- h.P(h.Class("mb-4"), h.Text("Create a new account")),
- errorEl,
- h.Form(
- h.FieldSet(h.Class("fieldset"),
- h.Label(h.Class("label"), h.Text("Username"), h.Attr("for", "username")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("username"),
- h.Type("text"),
- h.Placeholder("Choose a username"),
- usernameBind,
- h.Attr("required"),
- h.Attr("autofocus"),
- ),
- h.Label(h.Class("label"), h.Text("Password"), h.Attr("for", "password")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("password"),
- h.Type("password"),
- h.Placeholder("Choose a password (min 8 chars)"),
- passwordBind,
- h.Attr("required"),
- ),
- h.Label(h.Class("label"), h.Text("Confirm Password"), h.Attr("for", "confirm")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("confirm"),
- h.Type("password"),
- h.Placeholder("Confirm your password"),
- confirmBind,
- h.Attr("required"),
- registerKeyDown,
- ),
- ),
- h.Button(
- h.Class("btn btn-primary w-full"),
- h.Type("button"),
- h.Text("Register"),
- registerClick,
- ),
- ),
- h.P(
- h.Text("Already have an account? "),
- h.A(h.Class("link"), h.Href("/login"), h.Text("Login")),
- ),
- )
-}
-
-func AuthHeader(username string, logoutClick h.H) h.H {
- return h.Div(h.Class("flex justify-center items-center gap-4 mb-4 p-2 bg-base-200 rounded-lg"),
- h.Span(h.Text("Logged in as "), h.Strong(h.Text(username))),
- h.Button(
- h.Type("button"),
- h.Class("btn btn-ghost btn-sm"),
- h.Text("Logout"),
- logoutClick,
- ),
- )
-}
-
-func GuestBanner() h.H {
- return h.Div(h.Class("alert text-sm mb-4"),
- h.Text("Playing as guest. "),
- h.A(h.Class("link"), h.Href("/login"), h.Text("Login")),
- h.Text(" or "),
- h.A(h.Class("link"), h.Href("/register"), h.Text("Register")),
- h.Text(" to save your games."),
- )
-}
diff --git a/ui/board.go b/ui/board.go
deleted file mode 100644
index c99b018..0000000
--- a/ui/board.go
+++ /dev/null
@@ -1,69 +0,0 @@
-package ui
-
-import (
- "github.com/ryanhamamura/c4/game"
- "github.com/ryanhamamura/via/h"
-)
-
-// ColumnClickFn returns an h.H onClick attribute for a given column index
-type ColumnClickFn func(col int) h.H
-
-func BoardComponent(g *game.Game, columnClick ColumnClickFn, myColor int) h.H {
- var cols []h.H
-
- activeTurn := 0
- if g.Status == game.StatusInProgress {
- activeTurn = g.CurrentTurn
- }
-
- for col := 0; col < 7; col++ {
- var cells []h.H
- for row := 0; row < 6; row++ {
- cellColor := g.Board[row][col]
- isWinning := g.IsWinningCell(row, col)
- isActiveTurn := cellColor != 0 && cellColor == activeTurn
- cells = append(cells, Cell(cellColor, isWinning, isActiveTurn))
- }
-
- // Column is clickable only if it's player's turn and game is in progress
- canClick := g.Status == game.StatusInProgress && g.CurrentTurn == myColor
- cols = append(cols, Column(col, cells, columnClick, canClick))
- }
-
- boardAttrs := []h.H{h.Class("board")}
- boardAttrs = append(boardAttrs, cols...)
- return h.Div(boardAttrs...)
-}
-
-func Column(colIdx int, cells []h.H, columnClick ColumnClickFn, canClick bool) h.H {
- class := "column"
- if canClick {
- class += " clickable"
- }
-
- attrs := []h.H{h.Class(class)}
-
- if canClick && columnClick != nil {
- attrs = append(attrs, columnClick(colIdx))
- }
-
- attrs = append(attrs, cells...)
- return h.Div(attrs...)
-}
-
-func Cell(color int, isWinning, isActiveTurn bool) h.H {
- class := "cell"
- switch color {
- case 1:
- class += " red"
- case 2:
- class += " yellow"
- }
- if isWinning {
- class += " winning"
- }
- if isActiveTurn {
- class += " active-turn"
- }
- return h.Div(h.Class(class))
-}
diff --git a/ui/c4chat.go b/ui/c4chat.go
deleted file mode 100644
index a796085..0000000
--- a/ui/c4chat.go
+++ /dev/null
@@ -1,64 +0,0 @@
-package ui
-
-import (
- "fmt"
-
- "github.com/ryanhamamura/via/h"
-)
-
-type C4ChatMessage struct {
- Nickname string `json:"nickname"`
- Color int `json:"color"` // 1=Red, 2=Yellow
- Message string `json:"message"`
- Time int64 `json:"time"`
-}
-
-var c4ChatColors = map[int]string{
- 1: "#4a2a3a",
- 2: "#2a4545",
-}
-
-func C4Chat(messages []C4ChatMessage, msgBind, sendClick, sendKeyDown h.H) h.H {
- var msgEls []h.H
- for _, m := range messages {
- color := "#666"
- if c, ok := c4ChatColors[m.Color]; ok {
- color = c
- }
- msgEls = append(msgEls, h.Div(h.Class("c4-chat-msg"),
- h.Span(
- h.Attr("style", fmt.Sprintf("color:%s;font-weight:bold;", color)),
- h.Text(m.Nickname+": "),
- ),
- h.Span(h.Text(m.Message)),
- ))
- }
-
- autoScroll := h.Script(h.Text(`
-(function(){
- var el = document.querySelector('.c4-chat-history');
- if (!el) return;
- el.scrollTop = el.scrollHeight;
- new MutationObserver(function(){ el.scrollTop = el.scrollHeight; })
- .observe(el, {childList:true, subtree:true});
-})();
-`))
-
- historyAttrs := []h.H{h.Class("c4-chat-history")}
- historyAttrs = append(historyAttrs, msgEls...)
- historyAttrs = append(historyAttrs, autoScroll)
-
- return h.Div(h.Class("c4-chat"),
- h.Div(historyAttrs...),
- h.Div(h.Class("c4-chat-input"), h.DataIgnoreMorph(),
- h.Input(
- h.Type("text"),
- h.Attr("placeholder", "Chat..."),
- h.Attr("autocomplete", "off"),
- msgBind,
- sendKeyDown,
- ),
- h.Button(h.Type("button"), h.Text("Send"), sendClick),
- ),
- )
-}
diff --git a/ui/gamelist.go b/ui/gamelist.go
deleted file mode 100644
index 0ed4254..0000000
--- a/ui/gamelist.go
+++ /dev/null
@@ -1,110 +0,0 @@
-package ui
-
-import (
- "fmt"
- "time"
-
- "github.com/ryanhamamura/c4/game"
- "github.com/ryanhamamura/via/h"
-)
-
-type GameListItem struct {
- ID string
- Status int
- OpponentName string
- IsMyTurn bool
- LastPlayed time.Time
-}
-
-func GameList(games []GameListItem, deleteClick func(id string) h.H) h.H {
- if len(games) == 0 {
- return nil
- }
-
- var items []h.H
- for _, g := range games {
- items = append(items, gameListEntry(g, deleteClick))
- }
-
- listItems := []h.H{h.Class("flex flex-col gap-2")}
- listItems = append(listItems, items...)
-
- return h.Div(h.Class("mt-8 text-left"),
- h.H3(h.Class("mb-4 text-center text-lg font-bold"), h.Text("Your Games")),
- h.Div(listItems...),
- )
-}
-
-func gameListEntry(g GameListItem, deleteClick func(id string) h.H) h.H {
- statusText, statusClass := getStatusDisplay(g)
-
- return h.Div(h.Class("flex items-center gap-2 p-2 bg-base-200 rounded-lg transition-colors hover:bg-base-300"),
- h.A(
- h.Href("/game/"+g.ID),
- h.Class("flex-1 flex justify-between items-center px-2 py-1 no-underline text-base-content"),
- h.Div(h.Class("flex flex-col gap-1"),
- h.Span(h.Class("font-bold"), h.Text(getOpponentDisplay(g))),
- h.Span(h.Class(statusClass), h.Text(statusText)),
- ),
- h.Div(
- h.Span(h.Class("text-xs opacity-60"), h.Text(formatTimeAgo(g.LastPlayed))),
- ),
- ),
- h.Button(
- h.Type("button"),
- h.Class("btn btn-ghost btn-sm btn-square hover:btn-error"),
- h.Text("\u00d7"),
- deleteClick(g.ID),
- ),
- )
-}
-
-func getStatusDisplay(g GameListItem) (string, string) {
- switch game.GameStatus(g.Status) {
- case game.StatusWaitingForPlayer:
- return "Waiting for opponent", "text-sm opacity-60"
- case game.StatusInProgress:
- if g.IsMyTurn {
- return "Your turn!", "text-sm text-success font-bold"
- }
- return "Opponent's turn", "text-sm"
- }
- return "", ""
-}
-
-func getOpponentDisplay(g GameListItem) string {
- if g.OpponentName == "" {
- return "Waiting for opponent..."
- }
- return "vs " + g.OpponentName
-}
-
-func formatTimeAgo(t time.Time) string {
- if t.IsZero() {
- return ""
- }
- duration := time.Since(t)
-
- if duration < time.Minute {
- return "just now"
- }
- if duration < time.Hour {
- mins := int(duration.Minutes())
- if mins == 1 {
- return "1 minute ago"
- }
- return fmt.Sprintf("%d minutes ago", mins)
- }
- if duration < 24*time.Hour {
- hours := int(duration.Hours())
- if hours == 1 {
- return "1 hour ago"
- }
- return fmt.Sprintf("%d hours ago", hours)
- }
- days := int(duration.Hours() / 24)
- if days == 1 {
- return "yesterday"
- }
- return fmt.Sprintf("%d days ago", days)
-}
diff --git a/ui/lobby.go b/ui/lobby.go
deleted file mode 100644
index 6421670..0000000
--- a/ui/lobby.go
+++ /dev/null
@@ -1,153 +0,0 @@
-package ui
-
-import (
- "github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/via/h"
-)
-
-type LobbyProps struct {
- NicknameBind h.H
- CreateGameKeyDown h.H
- CreateGameClick h.H
- IsLoggedIn bool
- Username string
- LogoutClick h.H
- UserGames []GameListItem
- DeleteGameClick func(id string) h.H
- ActiveTab string
- TabClickConnect4 h.H
- TabClickSnake h.H
- SnakeNicknameBind h.H
- SnakeSoloClicks []h.H
- SnakeMultiClicks []h.H
- ActiveSnakeGames []*snake.SnakeGame
- SelectedSpeedIndex int
- SpeedSelectClicks []h.H
-}
-
-func BackToLobby() h.H {
- return h.A(h.Class("link text-sm opacity-70"), h.Href("/"), h.Text("← Back"))
-}
-
-func StealthTitle(class string) h.H {
- return h.Span(h.Class(class),
- h.Span(h.Style("color:#4a2a3a"), h.Text("●")),
- h.Span(h.Style("color:#2a4545"), h.Text("●")),
- h.Span(h.Style("color:#4a2a3a"), h.Text("●")),
- h.Span(h.Style("color:#2a4545"), h.Text("●")),
- )
-}
-
-func LobbyView(p LobbyProps) h.H {
- var authSection h.H
- if p.IsLoggedIn {
- authSection = AuthHeader(p.Username, p.LogoutClick)
- } else {
- authSection = GuestBanner()
- }
-
- connect4Class := "tab"
- snakeClass := "tab"
- if p.ActiveTab == "snake" {
- snakeClass += " tab-active"
- } else {
- connect4Class += " tab-active"
- }
-
- var tabContent h.H
- if p.ActiveTab == "snake" {
- tabContent = SnakeLobbyTab(p.SnakeNicknameBind, p.SnakeSoloClicks, p.SnakeMultiClicks, p.ActiveSnakeGames, p.SelectedSpeedIndex, p.SpeedSelectClicks)
- } else {
- tabContent = connect4LobbyContent(p)
- }
-
- return h.Main(h.Class("max-w-md mx-auto mt-8 text-center"),
- authSection,
- h.H1(h.Class("text-3xl font-bold mb-4"), StealthTitle("")),
- h.Div(h.Class("tabs tabs-box mb-6 justify-center"),
- h.Button(h.Class(connect4Class), h.Type("button"), StealthTitle(""), p.TabClickConnect4),
- h.Button(h.Class(snakeClass), h.Type("button"), h.Text("~~~~"), p.TabClickSnake),
- ),
- tabContent,
- )
-}
-
-func connect4LobbyContent(p LobbyProps) h.H {
- return h.Div(
- h.P(h.Class("mb-4"), h.Text("Start a new session")),
- h.Form(
- h.FieldSet(h.Class("fieldset"),
- h.Label(h.Class("label"), h.Text("Your Nickname"), h.Attr("for", "nickname")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("nickname"),
- h.Type("text"),
- h.Placeholder("Enter your nickname"),
- p.NicknameBind,
- h.Attr("required"),
- p.CreateGameKeyDown,
- ),
- ),
- h.Button(
- h.Class("btn btn-primary w-full"),
- h.Type("button"),
- h.Text("Create Game"),
- p.CreateGameClick,
- ),
- ),
- GameList(p.UserGames, p.DeleteGameClick),
- )
-}
-
-func NicknamePrompt(nicknameBind, setNicknameKeyDown, setNicknameClick h.H) h.H {
- return h.Main(h.Class("max-w-sm mx-auto mt-8 text-center"),
- h.H1(h.Class("text-3xl font-bold"), h.Text("Join Game")),
- h.P(h.Class("mb-4"), h.Text("Enter your nickname to join the game.")),
- h.Form(
- h.FieldSet(h.Class("fieldset"),
- h.Label(h.Class("label"), h.Text("Your Nickname"), h.Attr("for", "nickname")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("nickname"),
- h.Type("text"),
- h.Placeholder("Enter your nickname"),
- nicknameBind,
- h.Attr("required"),
- h.Attr("autofocus"),
- setNicknameKeyDown,
- ),
- ),
- h.Button(
- h.Class("btn btn-primary w-full"),
- h.Type("button"),
- h.Text("Join"),
- setNicknameClick,
- ),
- ),
- )
-}
-
-func GameJoinPrompt(loginClick, guestClick, registerClick h.H) h.H {
- return h.Main(h.Class("max-w-sm mx-auto mt-8 text-center"),
- h.H1(h.Class("text-3xl font-bold"), h.Text("Join Game")),
- h.P(h.Class("mb-4"), h.Text("Log in to track your game history, or continue as a guest.")),
- h.Div(h.Class("flex flex-col gap-2 my-4"),
- h.Button(
- h.Class("btn btn-primary w-full"),
- h.Type("button"),
- h.Text("Login"),
- loginClick,
- ),
- h.Button(
- h.Class("btn btn-secondary w-full"),
- h.Type("button"),
- h.Text("Continue as Guest"),
- guestClick,
- ),
- ),
- h.P(h.Class("text-sm opacity-60"),
- h.Text("Don't have an account? "),
- h.A(h.Class("link"), h.Href("#"), h.Text("Register"), registerClick),
- ),
- )
-}
diff --git a/ui/snakeboard.go b/ui/snakeboard.go
deleted file mode 100644
index 15097b8..0000000
--- a/ui/snakeboard.go
+++ /dev/null
@@ -1,112 +0,0 @@
-package ui
-
-import (
- "fmt"
-
- "github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/via/h"
-)
-
-func SnakeBoard(sg *snake.SnakeGame) h.H {
- state := sg.State
- if state == nil || sg.Status != snake.StatusInProgress && sg.Status != snake.StatusFinished {
- return nil
- }
-
- // Build a lookup grid for rendering
- type cellInfo struct {
- snakeIdx int // -1 = empty, -2 = food
- isHead bool
- }
- grid := make([][]cellInfo, state.Height)
- for y := 0; y < state.Height; y++ {
- grid[y] = make([]cellInfo, state.Width)
- for x := 0; x < state.Width; x++ {
- grid[y][x] = cellInfo{snakeIdx: -1}
- }
- }
-
- for fi := range state.Food {
- f := state.Food[fi]
- if f.X >= 0 && f.X < state.Width && f.Y >= 0 && f.Y < state.Height {
- grid[f.Y][f.X] = cellInfo{snakeIdx: -2}
- }
- }
-
- for si, s := range state.Snakes {
- if s == nil {
- continue
- }
- for bi, bp := range s.Body {
- if bp.X >= 0 && bp.X < state.Width && bp.Y >= 0 && bp.Y < state.Height {
- grid[bp.Y][bp.X] = cellInfo{snakeIdx: si, isHead: bi == 0}
- }
- }
- }
-
- // Cell size scales with grid dimensions
- cellSize := cellSizeForGrid(state.Width, state.Height)
-
- var rows []h.H
- for y := 0; y < state.Height; y++ {
- var cells []h.H
- for x := 0; x < state.Width; x++ {
- ci := grid[y][x]
- class := "snake-cell"
- style := fmt.Sprintf("width:%dpx;height:%dpx;", cellSize, cellSize)
-
- switch {
- case ci.snakeIdx == -2:
- class += " snake-food"
- case ci.snakeIdx >= 0:
- s := state.Snakes[ci.snakeIdx]
- colorIdx := ci.snakeIdx
- bg := ""
- if colorIdx < len(snake.SnakeColors) {
- bg = snake.SnakeColors[colorIdx]
- style += fmt.Sprintf("background:%s;", bg)
- }
- if !s.Alive {
- class += " snake-dead"
- }
- if ci.isHead {
- class += " snake-head"
- if bg != "" {
- style += fmt.Sprintf("box-shadow:0 0 8px %s;", bg)
- }
- }
- }
-
- cells = append(cells, h.Div(h.Class(class), h.Attr("style", style)))
- }
- rowAttrs := append([]h.H{h.Class("snake-row")}, cells...)
- rows = append(rows, h.Div(rowAttrs...))
- }
-
- boardStyle := fmt.Sprintf("grid-template-columns:repeat(%d,1fr);", state.Width)
- attrs := []h.H{
- h.Class("snake-board"),
- h.Attr("style", boardStyle),
- }
- attrs = append(attrs, rows...)
- return h.Div(attrs...)
-}
-
-func cellSizeForGrid(width, height int) int {
- maxDim := width
- if height > maxDim {
- maxDim = height
- }
- switch {
- case maxDim <= 15:
- return 28
- case maxDim <= 20:
- return 24
- case maxDim <= 30:
- return 20
- case maxDim <= 40:
- return 16
- default:
- return 14
- }
-}
diff --git a/ui/snakechat.go b/ui/snakechat.go
deleted file mode 100644
index 66bc663..0000000
--- a/ui/snakechat.go
+++ /dev/null
@@ -1,63 +0,0 @@
-package ui
-
-import (
- "fmt"
-
- "github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/via/h"
-)
-
-type ChatMessage struct {
- Nickname string `json:"nickname"`
- Slot int `json:"slot"`
- Message string `json:"message"`
- Time int64 `json:"time"`
-}
-
-func SnakeChat(messages []ChatMessage, msgBind, sendClick, sendKeyDown h.H) h.H {
- var msgEls []h.H
- for _, m := range messages {
- color := "#666"
- if m.Slot >= 0 && m.Slot < len(snake.SnakeColors) {
- color = snake.SnakeColors[m.Slot]
- }
- msgEls = append(msgEls, h.Div(h.Class("snake-chat-msg"),
- h.Span(
- h.Attr("style", fmt.Sprintf("color:%s;font-weight:bold;", color)),
- h.Text(m.Nickname+": "),
- ),
- h.Span(h.Text(m.Message)),
- ))
- }
-
- // Auto-scroll chat history to bottom on new messages
- autoScroll := h.Script(h.Text(`
-(function(){
- var el = document.querySelector('.snake-chat-history');
- if (!el) return;
- el.scrollTop = el.scrollHeight;
- new MutationObserver(function(){ el.scrollTop = el.scrollHeight; })
- .observe(el, {childList:true, subtree:true});
-})();
-`))
-
- historyAttrs := []h.H{h.Class("snake-chat-history")}
- historyAttrs = append(historyAttrs, msgEls...)
- historyAttrs = append(historyAttrs, autoScroll)
-
- return h.Div(h.Class("snake-chat"),
- h.Div(historyAttrs...),
- h.Div(h.Class("snake-chat-input"),
- h.Input(
- h.Type("text"),
- h.Attr("placeholder", "Chat..."),
- h.Attr("autocomplete", "off"),
- // Prevent key events from bubbling to the game's window-level handler
- h.Attr("onkeydown", "event.stopPropagation()"),
- msgBind,
- sendKeyDown,
- ),
- h.Button(h.Type("button"), h.Text("Send"), sendClick),
- ),
- )
-}
diff --git a/ui/snakelobby.go b/ui/snakelobby.go
deleted file mode 100644
index e8b12e5..0000000
--- a/ui/snakelobby.go
+++ /dev/null
@@ -1,124 +0,0 @@
-package ui
-
-import (
- "fmt"
-
- "github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/via/h"
-)
-
-func SnakeLobbyTab(nicknameBind h.H, soloClicks, multiClicks []h.H, activeGames []*snake.SnakeGame, selectedSpeedIndex int, speedSelectClicks []h.H) h.H {
- // Solo play buttons
- var soloButtons []h.H
- for i, preset := range snake.GridPresets {
- var click h.H
- if i < len(soloClicks) {
- click = soloClicks[i]
- }
- soloButtons = append(soloButtons,
- h.Button(
- h.Class("btn btn-secondary"),
- h.Type("button"),
- h.Text(fmt.Sprintf("%s (%d×%d)", preset.Name, preset.Width, preset.Height)),
- click,
- ),
- )
- }
-
- // Multiplayer buttons
- var multiButtons []h.H
- for i, preset := range snake.GridPresets {
- var click h.H
- if i < len(multiClicks) {
- click = multiClicks[i]
- }
- multiButtons = append(multiButtons,
- h.Button(
- h.Class("btn btn-primary"),
- h.Type("button"),
- h.Text(fmt.Sprintf("%s (%d×%d)", preset.Name, preset.Width, preset.Height)),
- click,
- ),
- )
- }
-
- nicknameField := h.Div(h.Class("mb-4"),
- h.FieldSet(h.Class("fieldset"),
- h.Label(h.Class("label"), h.Text("Your Nickname"), h.Attr("for", "snake-nickname")),
- h.Input(
- h.Class("input input-bordered w-full"),
- h.ID("snake-nickname"),
- h.Type("text"),
- h.Placeholder("Enter your nickname"),
- nicknameBind,
- h.Attr("required"),
- ),
- ),
- )
-
- // Speed selector
- var speedButtons []h.H
- for i, preset := range snake.SpeedPresets {
- btnClass := "btn btn-sm"
- if i == selectedSpeedIndex {
- btnClass += " btn-active"
- }
- var click h.H
- if i < len(speedSelectClicks) {
- click = speedSelectClicks[i]
- }
- speedButtons = append(speedButtons, h.Button(
- h.Class(btnClass),
- h.Type("button"),
- h.Text(preset.Name),
- click,
- ))
- }
- speedSelector := h.Div(h.Class("mb-4"),
- h.Label(h.Class("label"), h.Text("Speed")),
- h.Div(append([]h.H{h.Class("btn-group")}, speedButtons...)...),
- )
-
- soloSection := h.Div(h.Class("mb-6"),
- h.H3(h.Class("text-lg font-bold mb-2"), h.Text("Play Solo")),
- h.Div(append([]h.H{h.Class("flex gap-2 justify-center")}, soloButtons...)...),
- )
-
- multiSection := h.Div(h.Class("mb-6"),
- h.H3(h.Class("text-lg font-bold mb-2"), h.Text("Create Multiplayer Game")),
- h.Div(append([]h.H{h.Class("flex gap-2 justify-center")}, multiButtons...)...),
- )
-
- var gameListEl h.H
- if len(activeGames) > 0 {
- var items []h.H
- for _, g := range activeGames {
- playerCount := g.PlayerCount()
- sizeLabel := fmt.Sprintf("%d×%d", g.State.Width, g.State.Height)
- statusLabel := "Waiting"
- if g.Status == snake.StatusCountdown {
- statusLabel = "Starting soon"
- }
- items = append(items, h.A(
- h.Href("/snake/"+g.ID),
- h.Class("flex justify-between items-center p-3 bg-base-200 rounded-lg hover:bg-base-300 no-underline text-base-content"),
- h.Span(h.Text(fmt.Sprintf("%s — %d/8 players", sizeLabel, playerCount))),
- h.Span(h.Class("text-sm opacity-60"), h.Text(statusLabel)),
- ))
- }
- listAttrs := []h.H{h.Class("flex flex-col gap-2")}
- listAttrs = append(listAttrs, items...)
- gameListEl = h.Div(h.Class("mt-6"),
- h.H3(h.Class("text-lg font-bold mb-2 text-center"), h.Text("Join a Game")),
- h.Div(listAttrs...),
- )
- }
-
- return h.Div(
- nicknameField,
- speedSelector,
- soloSection,
- multiSection,
- gameListEl,
- )
-}
diff --git a/ui/snakestatus.go b/ui/snakestatus.go
deleted file mode 100644
index e50b989..0000000
--- a/ui/snakestatus.go
+++ /dev/null
@@ -1,161 +0,0 @@
-package ui
-
-import (
- "fmt"
- "math"
- "time"
-
- "github.com/ryanhamamura/c4/snake"
- "github.com/ryanhamamura/via/h"
-)
-
-func SnakeStatusBanner(sg *snake.SnakeGame, mySlot int, rematchClick h.H) h.H {
- switch sg.Status {
- case snake.StatusWaitingForPlayers:
- if sg.Mode == snake.ModeSinglePlayer {
- return h.Div(h.Class("alert bg-base-200 text-xl font-bold"),
- h.Text("Ready?"),
- )
- }
- return h.Div(h.Class("alert bg-base-200 text-xl font-bold"),
- h.Text("Waiting for players..."),
- )
-
- case snake.StatusCountdown:
- remaining := time.Until(sg.CountdownEnd)
- secs := int(math.Ceil(remaining.Seconds()))
- if secs < 0 {
- secs = 0
- }
- return h.Div(h.Class("alert alert-info text-xl font-bold"),
- h.Text(fmt.Sprintf("Starting in %d...", secs)),
- )
-
- case snake.StatusInProgress:
- if sg.State != nil && mySlot >= 0 && mySlot < len(sg.State.Snakes) {
- s := sg.State.Snakes[mySlot]
- if s != nil && !s.Alive {
- return h.Div(h.Class("alert alert-error text-xl font-bold"),
- h.Text("You're out!"),
- )
- }
- }
- // Show score during single player gameplay
- if sg.Mode == snake.ModeSinglePlayer {
- return h.Div(h.Class("alert alert-success text-xl font-bold"),
- h.Text(fmt.Sprintf("Score: %d", sg.Score)),
- )
- }
- return h.Div(h.Class("alert alert-success text-xl font-bold"),
- h.Text("Go!"),
- )
-
- case snake.StatusFinished:
- var msg string
- var class string
-
- if sg.Mode == snake.ModeSinglePlayer {
- msg = fmt.Sprintf("Game Over! Score: %d", sg.Score)
- class = "alert alert-info text-xl font-bold"
- } else if sg.Winner != nil {
- if sg.Winner.Slot == mySlot {
- msg = "You win!"
- class = "alert alert-success text-xl font-bold"
- } else {
- msg = sg.Winner.Nickname + " wins!"
- class = "alert alert-error text-xl font-bold"
- }
- } else {
- msg = "It's a draw!"
- class = "alert alert-warning text-xl font-bold"
- }
-
- content := []h.H{h.Class(class), h.Text(msg)}
-
- if sg.RematchGameID != nil {
- content = append(content,
- h.A(
- h.Class("btn btn-sm bg-white text-gray-800 border-none ml-4"),
- h.Href("/snake/"+*sg.RematchGameID),
- h.Text("Join Rematch"),
- ),
- )
- } else if rematchClick != nil {
- content = append(content,
- h.Button(
- h.Class("btn btn-sm bg-white text-gray-800 border-none ml-4"),
- h.Type("button"),
- h.Text("Play again"),
- rematchClick,
- ),
- )
- }
-
- return h.Div(content...)
- }
-
- return nil
-}
-
-func SnakePlayerList(sg *snake.SnakeGame, mySlot int) h.H {
- var items []h.H
-
- for i, p := range sg.Players {
- if p == nil {
- continue
- }
-
- colorHex := "#666"
- if i < len(snake.SnakeColors) {
- colorHex = snake.SnakeColors[i]
- }
-
- name := p.Nickname
- if i == mySlot {
- name += " (You)"
- }
-
- var statusEl h.H
- if sg.Status == snake.StatusInProgress || sg.Status == snake.StatusFinished {
- if sg.State != nil && i < len(sg.State.Snakes) {
- s := sg.State.Snakes[i]
- if s != nil {
- if s.Alive {
- length := len(s.Body)
- statusEl = h.Span(h.Class("text-sm opacity-60"), h.Text(fmt.Sprintf(" (%d)", length)))
- } else {
- statusEl = h.Span(h.Class("text-sm opacity-40"), h.Text(" (dead)"))
- }
- }
- }
- }
-
- chipStyle := fmt.Sprintf("width:16px;height:16px;border-radius:50%%;background:%s;display:inline-block;", colorHex)
-
- items = append(items, h.Div(h.Class("flex items-center gap-2"),
- h.Span(h.Attr("style", chipStyle)),
- h.Span(h.Text(name)),
- statusEl,
- ))
- }
-
- listAttrs := []h.H{h.Class("flex flex-wrap gap-4 mb-2")}
- listAttrs = append(listAttrs, items...)
- return h.Div(listAttrs...)
-}
-
-func SnakeInviteLink(gameID string) h.H {
- fullURL := getBaseURL() + "/snake/" + gameID
- return h.Div(h.Class("mt-4 text-center"),
- h.P(h.Text("Share this link to invite players:")),
- h.Div(h.Class("bg-base-200 p-4 rounded-lg font-mono break-all my-2"),
- h.Text(fullURL),
- ),
- h.Button(
- h.Class("btn btn-sm mt-2"),
- h.Type("button"),
- h.Text("Copy Link"),
- h.Attr("onclick", "navigator.clipboard.writeText('"+fullURL+"')"),
- ),
- )
-}
diff --git a/ui/status.go b/ui/status.go
deleted file mode 100644
index 3fb3793..0000000
--- a/ui/status.go
+++ /dev/null
@@ -1,137 +0,0 @@
-package ui
-
-import (
- "github.com/ryanhamamura/c4/config"
- "github.com/ryanhamamura/c4/game"
- "github.com/ryanhamamura/via/h"
-)
-
-func StatusBanner(g *game.Game, myColor int, playAgainClick h.H) h.H {
- var message string
- var class string
-
- switch g.Status {
- case game.StatusWaitingForPlayer:
- message = "Waiting for opponent..."
- class = "alert bg-base-200 text-xl font-bold"
- case game.StatusInProgress:
- if g.CurrentTurn == myColor {
- message = "Your turn!"
- class = "alert alert-success text-xl font-bold"
- } else {
- opponentName := getOpponentName(g, myColor)
- message = opponentName + "'s turn"
- class = "alert bg-base-200 text-xl font-bold"
- }
- case game.StatusWon:
- if g.Winner != nil && g.Winner.Color == myColor {
- message = "You win!"
- class = "alert alert-success text-xl font-bold"
- } else if g.Winner != nil {
- message = g.Winner.Nickname + " wins!"
- class = "alert alert-error text-xl font-bold"
- }
- case game.StatusDraw:
- message = "It's a draw!"
- class = "alert alert-warning text-xl font-bold"
- }
-
- content := []h.H{
- h.Class(class),
- h.Text(message),
- }
-
- // Show rematch options for finished games
- if g.IsFinished() {
- if g.RematchGameID != nil {
- content = append(content,
- h.A(
- h.Class("btn btn-sm bg-white text-gray-800 border-none ml-4"),
- h.Href("/game/"+*g.RematchGameID),
- h.Text("Join Rematch"),
- ),
- )
- } else if playAgainClick != nil {
- content = append(content,
- h.Button(
- h.Class("btn btn-sm bg-white text-gray-800 border-none ml-4"),
- h.Type("button"),
- h.Text("Play again"),
- playAgainClick,
- ),
- )
- }
- }
-
- return h.Div(content...)
-}
-
-func getOpponentName(g *game.Game, myColor int) string {
- for _, p := range g.Players {
- if p != nil && p.Color != myColor {
- return p.Nickname
- }
- }
- return "Opponent"
-}
-
-func PlayerInfo(g *game.Game, myColor int) h.H {
- var myName, opponentName string
- var myColorClass, opponentColorClass string
-
- for _, p := range g.Players {
- if p == nil {
- continue
- }
- if p.Color == myColor {
- myName = p.Nickname
- if p.Color == 1 {
- myColorClass = "red"
- } else {
- myColorClass = "yellow"
- }
- } else {
- opponentName = p.Nickname
- if p.Color == 1 {
- opponentColorClass = "red"
- } else {
- opponentColorClass = "yellow"
- }
- }
- }
-
- if opponentName == "" {
- opponentName = "Waiting..."
- }
-
- return h.Div(h.Class("flex gap-8 mb-2"),
- h.Div(h.Class("flex items-center gap-2"),
- h.Span(h.Class("player-chip "+myColorClass)),
- h.Span(h.Text(myName+" (You)")),
- ),
- h.Div(h.Class("flex items-center gap-2"),
- h.Span(h.Class("player-chip "+opponentColorClass)),
- h.Span(h.Text(opponentName)),
- ),
- )
-}
-
-func getBaseURL() string {
- return config.Global.AppURL
-}
-
-func InviteLink(gameID string) h.H {
- fullURL := getBaseURL() + "/game/" + gameID
- return h.Div(h.Class("mt-4 text-center"),
- h.P(h.Text("Share this link with your opponent:")),
- h.Div(h.Class("bg-base-200 p-4 rounded-lg font-mono break-all my-2"),
- h.Text(fullURL),
- ),
- h.Button(
- h.Class("btn btn-sm mt-2"),
- h.Type("button"),
- h.Text("Copy Link"),
- h.Attr("onclick", "navigator.clipboard.writeText('"+fullURL+"')"),
- ),
- )
-}
From 2aa026b1d5ea9f2653abd5753cec9eaa4d076651 Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 12:30:33 -1000
Subject: [PATCH 3/9] refactor: remove persister abstraction layer
Inline persistence logic directly into game stores and handlers:
- game/persist.go: DB mapping methods on GameStore and GameInstance
- snake/persist.go: DB mapping methods on SnakeStore and SnakeGameInstance
- Chat persistence inlined into c4game handlers
- Delete db/persister.go (GamePersister, SnakePersister, ChatPersister)
- Stores now take *repository.Queries directly instead of Persister interface
---
db/persister.go | 326 ------------------------------------
features/c4game/handlers.go | 58 +++++--
features/c4game/routes.go | 10 +-
game/persist.go | 157 +++++++++++++++++
game/store.go | 76 ++++-----
main.go | 10 +-
router/router.go | 4 +-
snake/loop.go | 16 +-
snake/persist.go | 186 ++++++++++++++++++++
snake/store.go | 80 ++++-----
10 files changed, 475 insertions(+), 448 deletions(-)
delete mode 100644 db/persister.go
create mode 100644 game/persist.go
create mode 100644 snake/persist.go
diff --git a/db/persister.go b/db/persister.go
deleted file mode 100644
index f87760b..0000000
--- a/db/persister.go
+++ /dev/null
@@ -1,326 +0,0 @@
-package db
-
-import (
- "context"
- "database/sql"
- "slices"
-
- "github.com/ryanhamamura/c4/db/repository"
- "github.com/ryanhamamura/c4/game"
- "github.com/ryanhamamura/c4/snake"
-)
-
-type GamePersister struct {
- queries *repository.Queries
-}
-
-func NewGamePersister(q *repository.Queries) *GamePersister {
- return &GamePersister{queries: q}
-}
-
-func (p *GamePersister) SaveGame(g *game.Game) error {
- ctx := context.Background()
-
- _, err := p.queries.GetGame(ctx, g.ID)
- if err == sql.ErrNoRows {
- _, err = p.queries.CreateGame(ctx, repository.CreateGameParams{
- ID: g.ID,
- Board: g.BoardToJSON(),
- CurrentTurn: int64(g.CurrentTurn),
- Status: int64(g.Status),
- })
- return err
- }
- if err != nil {
- return err
- }
-
- var winnerUserID sql.NullString
- if g.Winner != nil && g.Winner.UserID != nil {
- winnerUserID = sql.NullString{String: *g.Winner.UserID, Valid: true}
- }
-
- winningCells := sql.NullString{}
- if wc := g.WinningCellsToJSON(); wc != "" {
- winningCells = sql.NullString{String: wc, Valid: true}
- }
-
- rematchGameID := sql.NullString{}
- if g.RematchGameID != nil {
- rematchGameID = sql.NullString{String: *g.RematchGameID, Valid: true}
- }
-
- return p.queries.UpdateGame(ctx, repository.UpdateGameParams{
- Board: g.BoardToJSON(),
- CurrentTurn: int64(g.CurrentTurn),
- Status: int64(g.Status),
- WinnerUserID: winnerUserID,
- WinningCells: winningCells,
- RematchGameID: rematchGameID,
- ID: g.ID,
- })
-}
-
-func (p *GamePersister) LoadGame(id string) (*game.Game, error) {
- ctx := context.Background()
- row, err := p.queries.GetGame(ctx, id)
- if err != nil {
- return nil, err
- }
-
- g := &game.Game{
- ID: row.ID,
- CurrentTurn: int(row.CurrentTurn),
- Status: game.GameStatus(row.Status),
- }
-
- if err := g.BoardFromJSON(row.Board); err != nil {
- return nil, err
- }
-
- if row.WinningCells.Valid {
- g.WinningCellsFromJSON(row.WinningCells.String)
- }
-
- if row.RematchGameID.Valid {
- g.RematchGameID = &row.RematchGameID.String
- }
-
- return g, nil
-}
-
-func (p *GamePersister) SaveGamePlayer(gameID string, player *game.Player, slot int) error {
- ctx := context.Background()
-
- var userID, guestPlayerID sql.NullString
- if player.UserID != nil {
- userID = sql.NullString{String: *player.UserID, Valid: true}
- } else {
- guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
- }
-
- return p.queries.CreateGamePlayer(ctx, repository.CreateGamePlayerParams{
- GameID: gameID,
- UserID: userID,
- GuestPlayerID: guestPlayerID,
- Nickname: player.Nickname,
- Color: int64(player.Color),
- Slot: int64(slot),
- })
-}
-
-func (p *GamePersister) LoadGamePlayers(gameID string) ([]*game.Player, error) {
- ctx := context.Background()
- rows, err := p.queries.GetGamePlayers(ctx, gameID)
- if err != nil {
- return nil, err
- }
-
- players := make([]*game.Player, 0, len(rows))
- for _, row := range rows {
- player := &game.Player{
- Nickname: row.Nickname,
- Color: int(row.Color),
- }
-
- if row.UserID.Valid {
- player.UserID = &row.UserID.String
- player.ID = game.PlayerID(row.UserID.String)
- } else if row.GuestPlayerID.Valid {
- player.ID = game.PlayerID(row.GuestPlayerID.String)
- }
-
- players = append(players, player)
- }
-
- return players, nil
-}
-
-func (p *GamePersister) DeleteGame(id string) error {
- ctx := context.Background()
- return p.queries.DeleteGame(ctx, id)
-}
-
-// SnakePersister implements snake.Persister
-type SnakePersister struct {
- queries *repository.Queries
-}
-
-func NewSnakePersister(q *repository.Queries) *SnakePersister {
- return &SnakePersister{queries: q}
-}
-
-func (p *SnakePersister) SaveSnakeGame(sg *snake.SnakeGame) error {
- ctx := context.Background()
-
- boardJSON := "{}"
- if sg.State != nil {
- boardJSON = sg.State.ToJSON()
- }
-
- var gridWidth, gridHeight sql.NullInt64
- if sg.State != nil {
- gridWidth = sql.NullInt64{Int64: int64(sg.State.Width), Valid: true}
- gridHeight = sql.NullInt64{Int64: int64(sg.State.Height), Valid: true}
- }
-
- _, err := p.queries.GetSnakeGame(ctx, sg.ID)
- if err == sql.ErrNoRows {
- _, err = p.queries.CreateSnakeGame(ctx, repository.CreateSnakeGameParams{
- ID: sg.ID,
- Board: boardJSON,
- Status: int64(sg.Status),
- GridWidth: gridWidth,
- GridHeight: gridHeight,
- GameMode: int64(sg.Mode),
- SnakeSpeed: int64(sg.Speed),
- })
- return err
- }
- if err != nil {
- return err
- }
-
- var winnerUserID sql.NullString
- if sg.Winner != nil && sg.Winner.UserID != nil {
- winnerUserID = sql.NullString{String: *sg.Winner.UserID, Valid: true}
- }
-
- rematchGameID := sql.NullString{}
- if sg.RematchGameID != nil {
- rematchGameID = sql.NullString{String: *sg.RematchGameID, Valid: true}
- }
-
- return p.queries.UpdateSnakeGame(ctx, repository.UpdateSnakeGameParams{
- Board: boardJSON,
- Status: int64(sg.Status),
- WinnerUserID: winnerUserID,
- RematchGameID: rematchGameID,
- Score: int64(sg.Score),
- ID: sg.ID,
- })
-}
-
-func (p *SnakePersister) LoadSnakeGame(id string) (*snake.SnakeGame, error) {
- ctx := context.Background()
- row, err := p.queries.GetSnakeGame(ctx, id)
- if err != nil {
- return nil, err
- }
-
- state, err := snake.GameStateFromJSON(row.Board)
- if err != nil {
- state = &snake.GameState{}
- }
- if row.GridWidth.Valid {
- state.Width = int(row.GridWidth.Int64)
- }
- if row.GridHeight.Valid {
- state.Height = int(row.GridHeight.Int64)
- }
-
- sg := &snake.SnakeGame{
- ID: row.ID,
- State: state,
- Players: make([]*snake.Player, 8),
- Status: snake.Status(row.Status),
- Mode: snake.GameMode(row.GameMode),
- Score: int(row.Score),
- Speed: int(row.SnakeSpeed),
- }
-
- if row.RematchGameID.Valid {
- sg.RematchGameID = &row.RematchGameID.String
- }
-
- return sg, nil
-}
-
-func (p *SnakePersister) SaveSnakePlayer(gameID string, player *snake.Player) error {
- ctx := context.Background()
-
- var userID, guestPlayerID sql.NullString
- if player.UserID != nil {
- userID = sql.NullString{String: *player.UserID, Valid: true}
- } else {
- guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
- }
-
- return p.queries.CreateSnakePlayer(ctx, repository.CreateSnakePlayerParams{
- GameID: gameID,
- UserID: userID,
- GuestPlayerID: guestPlayerID,
- Nickname: player.Nickname,
- Color: int64(player.Slot + 1),
- Slot: int64(player.Slot),
- })
-}
-
-func (p *SnakePersister) LoadSnakePlayers(gameID string) ([]*snake.Player, error) {
- ctx := context.Background()
- rows, err := p.queries.GetSnakePlayers(ctx, gameID)
- if err != nil {
- return nil, err
- }
-
- players := make([]*snake.Player, 0, len(rows))
- for _, row := range rows {
- player := &snake.Player{
- Nickname: row.Nickname,
- Slot: int(row.Slot),
- }
-
- if row.UserID.Valid {
- player.UserID = &row.UserID.String
- player.ID = snake.PlayerID(row.UserID.String)
- } else if row.GuestPlayerID.Valid {
- player.ID = snake.PlayerID(row.GuestPlayerID.String)
- }
-
- players = append(players, player)
- }
-
- return players, nil
-}
-
-func (p *SnakePersister) DeleteSnakeGame(id string) error {
- ctx := context.Background()
- return p.queries.DeleteSnakeGame(ctx, id)
-}
-
-type ChatPersister struct {
- queries *repository.Queries
-}
-
-func NewChatPersister(q *repository.Queries) *ChatPersister {
- return &ChatPersister{queries: q}
-}
-
-func (p *ChatPersister) SaveChatMessage(gameID string, msg game.ChatMessage) error {
- return p.queries.CreateChatMessage(context.Background(), repository.CreateChatMessageParams{
- GameID: gameID,
- Nickname: msg.Nickname,
- Color: int64(msg.Color),
- Message: msg.Message,
- CreatedAt: msg.Time,
- })
-}
-
-func (p *ChatPersister) LoadChatMessages(gameID string) ([]game.ChatMessage, error) {
- rows, err := p.queries.GetChatMessages(context.Background(), gameID)
- if err != nil {
- return nil, err
- }
- msgs := make([]game.ChatMessage, len(rows))
- for i, r := range rows {
- msgs[i] = game.ChatMessage{
- Nickname: r.Nickname,
- Color: int(r.Color),
- Message: r.Message,
- Time: r.CreatedAt,
- }
- }
- // Query returns newest-first; reverse to oldest-first for display
- slices.Reverse(msgs)
- return msgs, nil
-}
diff --git a/features/c4game/handlers.go b/features/c4game/handlers.go
index 4df8425..e0b107b 100644
--- a/features/c4game/handlers.go
+++ b/features/c4game/handlers.go
@@ -1,8 +1,10 @@
package c4game
import (
+ "context"
"encoding/json"
"net/http"
+ "slices"
"strconv"
"sync"
"time"
@@ -10,14 +12,14 @@ import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
- "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/features/c4game/components"
"github.com/ryanhamamura/c4/features/c4game/pages"
"github.com/ryanhamamura/c4/game"
"github.com/starfederation/datastar-go/datastar"
)
-func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
gameID := chi.URLParam(r, "game_id")
@@ -73,8 +75,8 @@ func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, chatPer
// Player is in the game — render full game page
g := gi.GetGame()
- uiMsgs, _ := chatPersister.LoadChatMessages(gameID)
- msgs := uiChatToComponents(uiMsgs)
+ chatMsgs := loadChatMessages(queries, gameID)
+ msgs := chatToComponents(chatMsgs)
if err := pages.GamePage(g, myColor, msgs).Render(r.Context(), w); err != nil {
http.Error(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
@@ -82,7 +84,7 @@ func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, chatPer
}
}
-func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
gameID := chi.URLParam(r, "game_id")
@@ -103,9 +105,9 @@ func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.Sessio
sse := datastar.NewSSE(w, r)
// Load initial chat messages
- uiMsgs, _ := chatPersister.LoadChatMessages(gameID)
+ chatMsgs := loadChatMessages(queries, gameID)
var chatMu sync.Mutex
- chatMessages := uiChatToComponents(uiMsgs)
+ chatMessages := chatToComponents(chatMsgs)
// Send initial render of all components
sendGameComponents(sse, gi, myColor, chatMessages, &chatMu, gameID)
@@ -203,7 +205,7 @@ func HandleDropPiece(store *game.GameStore, sessions *scs.SessionManager) http.H
}
}
-func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, chatPersister *db.ChatPersister) http.HandlerFunc {
+func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
gameID := chi.URLParam(r, "game_id")
@@ -254,7 +256,7 @@ func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionM
Message: signals.ChatMsg,
Time: time.Now().UnixMilli(),
}
- chatPersister.SaveChatMessage(gameID, cm)
+ saveChatMessage(queries, gameID, cm)
data, err := json.Marshal(cm)
if err != nil {
@@ -353,10 +355,40 @@ func sendGameComponents(sse *datastar.ServerSentEventGenerator, gi *game.GameIns
sse.PatchElementTempl(components.Chat(msgs, gameID), datastar.WithSelectorID("c4-chat")) //nolint:errcheck
}
-// uiChatToComponents converts ui.C4ChatMessage slice to components.ChatMessage slice.
-func uiChatToComponents(uiMsgs []game.ChatMessage) []components.ChatMessage {
- msgs := make([]components.ChatMessage, len(uiMsgs))
- for i, m := range uiMsgs {
+// Chat persistence helpers — inlined from the former ChatPersister.
+
+func saveChatMessage(queries *repository.Queries, gameID string, msg game.ChatMessage) {
+ queries.CreateChatMessage(context.Background(), repository.CreateChatMessageParams{ //nolint:errcheck
+ GameID: gameID,
+ Nickname: msg.Nickname,
+ Color: int64(msg.Color),
+ Message: msg.Message,
+ CreatedAt: msg.Time,
+ })
+}
+
+func loadChatMessages(queries *repository.Queries, gameID string) []game.ChatMessage {
+ rows, err := queries.GetChatMessages(context.Background(), gameID)
+ if err != nil {
+ return nil
+ }
+ msgs := make([]game.ChatMessage, len(rows))
+ for i, r := range rows {
+ msgs[i] = game.ChatMessage{
+ Nickname: r.Nickname,
+ Color: int(r.Color),
+ Message: r.Message,
+ Time: r.CreatedAt,
+ }
+ }
+ // DB returns newest-first; reverse for display
+ slices.Reverse(msgs)
+ return msgs
+}
+
+func chatToComponents(chatMsgs []game.ChatMessage) []components.ChatMessage {
+ msgs := make([]components.ChatMessage, len(chatMsgs))
+ for i, m := range chatMsgs {
msgs[i] = components.ChatMessage{
Nickname: m.Nickname,
Color: m.Color,
diff --git a/features/c4game/routes.go b/features/c4game/routes.go
index 99783f6..917e5a0 100644
--- a/features/c4game/routes.go
+++ b/features/c4game/routes.go
@@ -4,7 +4,7 @@ import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
- "github.com/ryanhamamura/c4/db"
+ "github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
)
@@ -13,14 +13,14 @@ func SetupRoutes(
store *game.GameStore,
nc *nats.Conn,
sessions *scs.SessionManager,
- chatPersister *db.ChatPersister,
+ queries *repository.Queries,
) error {
- router.Get("/game/{game_id}", HandleGamePage(store, sessions, chatPersister))
- router.Get("/game/{game_id}/events", HandleGameEvents(store, nc, sessions, chatPersister))
+ router.Get("/game/{game_id}", HandleGamePage(store, sessions, queries))
+ router.Get("/game/{game_id}/events", HandleGameEvents(store, nc, sessions, queries))
router.Route("/api/game/{game_id}", func(r chi.Router) {
r.Post("/drop", HandleDropPiece(store, sessions))
- r.Post("/chat", HandleSendChat(store, nc, sessions, chatPersister))
+ r.Post("/chat", HandleSendChat(store, nc, sessions, queries))
r.Post("/join", HandleSetNickname(store, sessions))
r.Post("/rematch", HandleRematch(store, sessions))
})
diff --git a/game/persist.go b/game/persist.go
new file mode 100644
index 0000000..2322adc
--- /dev/null
+++ b/game/persist.go
@@ -0,0 +1,157 @@
+package game
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/ryanhamamura/c4/db/repository"
+)
+
+// Persistence methods on GameStore (used during Get to hydrate from DB).
+
+func (gs *GameStore) saveGame(g *Game) error {
+ ctx := context.Background()
+
+ _, err := gs.queries.GetGame(ctx, g.ID)
+ if err == sql.ErrNoRows {
+ _, err = gs.queries.CreateGame(ctx, repository.CreateGameParams{
+ ID: g.ID,
+ Board: g.BoardToJSON(),
+ CurrentTurn: int64(g.CurrentTurn),
+ Status: int64(g.Status),
+ })
+ return err
+ }
+ if err != nil {
+ return err
+ }
+
+ return gs.queries.UpdateGame(ctx, updateGameParams(g))
+}
+
+func (gs *GameStore) loadGame(id string) (*Game, error) {
+ row, err := gs.queries.GetGame(context.Background(), id)
+ if err != nil {
+ return nil, err
+ }
+ return gameFromRow(row)
+}
+
+func (gs *GameStore) loadGamePlayers(id string) ([]*Player, error) {
+ rows, err := gs.queries.GetGamePlayers(context.Background(), id)
+ if err != nil {
+ return nil, err
+ }
+ return playersFromRows(rows), nil
+}
+
+// Persistence methods on GameInstance (used during gameplay mutations).
+
+func (gi *GameInstance) saveGame(g *Game) error {
+ ctx := context.Background()
+
+ _, err := gi.queries.GetGame(ctx, g.ID)
+ if err == sql.ErrNoRows {
+ _, err = gi.queries.CreateGame(ctx, repository.CreateGameParams{
+ ID: g.ID,
+ Board: g.BoardToJSON(),
+ CurrentTurn: int64(g.CurrentTurn),
+ Status: int64(g.Status),
+ })
+ return err
+ }
+ if err != nil {
+ return err
+ }
+
+ return gi.queries.UpdateGame(ctx, updateGameParams(g))
+}
+
+func (gi *GameInstance) saveGamePlayer(gameID string, player *Player, slot int) error {
+ var userID, guestPlayerID sql.NullString
+ if player.UserID != nil {
+ userID = sql.NullString{String: *player.UserID, Valid: true}
+ } else {
+ guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
+ }
+
+ return gi.queries.CreateGamePlayer(context.Background(), repository.CreateGamePlayerParams{
+ GameID: gameID,
+ UserID: userID,
+ GuestPlayerID: guestPlayerID,
+ Nickname: player.Nickname,
+ Color: int64(player.Color),
+ Slot: int64(slot),
+ })
+}
+
+// Shared helpers for domain ↔ DB mapping.
+
+func updateGameParams(g *Game) repository.UpdateGameParams {
+ var winnerUserID sql.NullString
+ if g.Winner != nil && g.Winner.UserID != nil {
+ winnerUserID = sql.NullString{String: *g.Winner.UserID, Valid: true}
+ }
+
+ var winningCells sql.NullString
+ if wc := g.WinningCellsToJSON(); wc != "" {
+ winningCells = sql.NullString{String: wc, Valid: true}
+ }
+
+ var rematchGameID sql.NullString
+ if g.RematchGameID != nil {
+ rematchGameID = sql.NullString{String: *g.RematchGameID, Valid: true}
+ }
+
+ return repository.UpdateGameParams{
+ Board: g.BoardToJSON(),
+ CurrentTurn: int64(g.CurrentTurn),
+ Status: int64(g.Status),
+ WinnerUserID: winnerUserID,
+ WinningCells: winningCells,
+ RematchGameID: rematchGameID,
+ ID: g.ID,
+ }
+}
+
+func gameFromRow(row repository.Game) (*Game, error) {
+ g := &Game{
+ ID: row.ID,
+ CurrentTurn: int(row.CurrentTurn),
+ Status: GameStatus(row.Status),
+ }
+
+ if err := g.BoardFromJSON(row.Board); err != nil {
+ return nil, err
+ }
+
+ if row.WinningCells.Valid {
+ g.WinningCellsFromJSON(row.WinningCells.String)
+ }
+
+ if row.RematchGameID.Valid {
+ g.RematchGameID = &row.RematchGameID.String
+ }
+
+ return g, nil
+}
+
+func playersFromRows(rows []repository.GamePlayer) []*Player {
+ players := make([]*Player, 0, len(rows))
+ for _, row := range rows {
+ player := &Player{
+ Nickname: row.Nickname,
+ Color: int(row.Color),
+ }
+
+ if row.UserID.Valid {
+ player.UserID = &row.UserID.String
+ player.ID = PlayerID(row.UserID.String)
+ } else if row.GuestPlayerID.Valid {
+ player.ID = PlayerID(row.GuestPlayerID.String)
+ }
+
+ players = append(players, player)
+ }
+ return players
+}
diff --git a/game/store.go b/game/store.go
index 9e7aae8..e1c9c40 100644
--- a/game/store.go
+++ b/game/store.go
@@ -1,40 +1,32 @@
package game
import (
+ "context"
"crypto/rand"
"encoding/hex"
"sync"
+
+ "github.com/ryanhamamura/c4/db/repository"
)
type PlayerSession struct {
Player *Player
}
-type Persister interface {
- SaveGame(g *Game) error
- LoadGame(id string) (*Game, error)
- SaveGamePlayer(gameID string, player *Player, slot int) error
- LoadGamePlayers(gameID string) ([]*Player, error)
- DeleteGame(id string) error
-}
-
type GameStore struct {
- games map[string]*GameInstance
- gamesMu sync.RWMutex
- persister Persister
+ games map[string]*GameInstance
+ gamesMu sync.RWMutex
+ queries *repository.Queries
notifyFunc func(gameID string)
}
-func NewGameStore() *GameStore {
+func NewGameStore(queries *repository.Queries) *GameStore {
return &GameStore{
- games: make(map[string]*GameInstance),
+ games: make(map[string]*GameInstance),
+ queries: queries,
}
}
-func (gs *GameStore) SetPersister(p Persister) {
- gs.persister = p
-}
-
func (gs *GameStore) SetNotifyFunc(f func(gameID string)) {
gs.notifyFunc = f
}
@@ -50,14 +42,14 @@ func (gs *GameStore) makeNotify(gameID string) func() {
func (gs *GameStore) Create() *GameInstance {
id := GenerateID(4)
gi := NewGameInstance(id)
- gi.persister = gs.persister
+ gi.queries = gs.queries
gi.notify = gs.makeNotify(id)
gs.gamesMu.Lock()
gs.games[id] = gi
gs.gamesMu.Unlock()
- if gs.persister != nil {
- gs.persister.SaveGame(gi.game)
+ if gs.queries != nil {
+ gs.saveGame(gi.game)
}
return gi
@@ -72,28 +64,28 @@ func (gs *GameStore) Get(id string) (*GameInstance, bool) {
return gi, true
}
- if gs.persister == nil {
+ if gs.queries == nil {
return nil, false
}
- game, err := gs.persister.LoadGame(id)
- if err != nil || game == nil {
+ g, err := gs.loadGame(id)
+ if err != nil || g == nil {
return nil, false
}
- players, _ := gs.persister.LoadGamePlayers(id)
+ players, _ := gs.loadGamePlayers(id)
for _, p := range players {
if p.Color == 1 {
- game.Players[0] = p
+ g.Players[0] = p
} else if p.Color == 2 {
- game.Players[1] = p
+ g.Players[1] = p
}
}
gi = &GameInstance{
- game: game,
- persister: gs.persister,
- notify: gs.makeNotify(id),
+ game: g,
+ queries: gs.queries,
+ notify: gs.makeNotify(id),
}
gs.gamesMu.Lock()
@@ -108,8 +100,8 @@ func (gs *GameStore) Delete(id string) error {
delete(gs.games, id)
gs.gamesMu.Unlock()
- if gs.persister != nil {
- return gs.persister.DeleteGame(id)
+ if gs.queries != nil {
+ return gs.queries.DeleteGame(context.Background(), id)
}
return nil
}
@@ -121,10 +113,10 @@ func GenerateID(size int) string {
}
type GameInstance struct {
- game *Game
- gameMu sync.RWMutex
- notify func()
- persister Persister
+ game *Game
+ gameMu sync.RWMutex
+ notify func()
+ queries *repository.Queries
}
func NewGameInstance(id string) *GameInstance {
@@ -158,9 +150,9 @@ func (gi *GameInstance) Join(ps *PlayerSession) bool {
return false
}
- if gi.persister != nil {
- gi.persister.SaveGamePlayer(gi.game.ID, ps.Player, slot)
- gi.persister.SaveGame(gi.game)
+ if gi.queries != nil {
+ gi.saveGamePlayer(gi.game.ID, ps.Player, slot)
+ gi.saveGame(gi.game)
}
gi.notify()
@@ -196,8 +188,8 @@ func (gi *GameInstance) CreateRematch(gs *GameStore) *GameInstance {
newID := newGI.ID()
gi.game.RematchGameID = &newID
- if gi.persister != nil {
- if err := gi.persister.SaveGame(gi.game); err != nil {
+ if gi.queries != nil {
+ if err := gi.saveGame(gi.game); err != nil {
gs.Delete(newID)
gi.game.RematchGameID = nil
return nil
@@ -230,8 +222,8 @@ func (gi *GameInstance) DropPiece(col int, playerColor int) bool {
gi.game.SwitchTurn()
}
- if gi.persister != nil {
- gi.persister.SaveGame(gi.game)
+ if gi.queries != nil {
+ gi.saveGame(gi.game)
}
gi.notify()
diff --git a/main.go b/main.go
index bb437a1..5cae7e6 100644
--- a/main.go
+++ b/main.go
@@ -71,20 +71,16 @@ func run(ctx context.Context) error {
defer cleanupNATS()
// Game stores
- store := game.NewGameStore()
- store.SetPersister(db.NewGamePersister(queries))
+ store := game.NewGameStore(queries)
store.SetNotifyFunc(func(gameID string) {
nc.Publish("game."+gameID, nil) //nolint:errcheck // best-effort notification
})
- snakeStore := snake.NewSnakeStore()
- snakeStore.SetPersister(db.NewSnakePersister(queries))
+ snakeStore := snake.NewSnakeStore(queries)
snakeStore.SetNotifyFunc(func(gameID string) {
nc.Publish("snake."+gameID, nil) //nolint:errcheck // best-effort notification
})
- chatPersister := db.NewChatPersister(queries)
-
// Router
logger := log.Logger
r := chi.NewMux()
@@ -94,7 +90,7 @@ func run(ctx context.Context) error {
sessionManager.LoadAndSave,
)
- if err := router.SetupRoutes(r, queries, sessionManager, nc, store, snakeStore, chatPersister, assets); err != nil {
+ if err := router.SetupRoutes(r, queries, sessionManager, nc, store, snakeStore, assets); err != nil {
return fmt.Errorf("setting up routes: %w", err)
}
diff --git a/router/router.go b/router/router.go
index a42a09d..c6198fc 100644
--- a/router/router.go
+++ b/router/router.go
@@ -8,7 +8,6 @@ import (
"sync"
"github.com/ryanhamamura/c4/config"
- "github.com/ryanhamamura/c4/db"
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/features/auth"
"github.com/ryanhamamura/c4/features/c4game"
@@ -30,7 +29,6 @@ func SetupRoutes(
nc *nats.Conn,
store *game.GameStore,
snakeStore *snake.SnakeStore,
- chatPersister *db.ChatPersister,
assets embed.FS,
) error {
// Static assets
@@ -44,7 +42,7 @@ func SetupRoutes(
auth.SetupRoutes(router, queries, sessions)
lobby.SetupRoutes(router, queries, sessions, store, snakeStore)
- c4game.SetupRoutes(router, store, nc, sessions, chatPersister)
+ c4game.SetupRoutes(router, store, nc, sessions, queries)
snakegame.SetupRoutes(router, snakeStore, nc, sessions)
return nil
diff --git a/snake/loop.go b/snake/loop.go
index 9d46c09..fb1a839 100644
--- a/snake/loop.go
+++ b/snake/loop.go
@@ -61,16 +61,16 @@ func (si *SnakeGameInstance) countdownPhase() {
si.initGame()
si.game.Status = StatusInProgress
- if si.persister != nil {
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakeGame(si.game)
}
si.gameMu.Unlock()
si.notify()
return
}
- if si.persister != nil {
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakeGame(si.game)
}
si.gameMu.Unlock()
si.notify()
@@ -123,8 +123,8 @@ func (si *SnakeGameInstance) gamePhase() {
// Inactivity timeout
if time.Since(lastInput) > inactivityLimit {
si.game.Status = StatusFinished
- if si.persister != nil {
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakeGame(si.game)
}
si.gameMu.Unlock()
si.notify()
@@ -195,8 +195,8 @@ func (si *SnakeGameInstance) gamePhase() {
si.game.Status = StatusFinished
}
- if si.persister != nil {
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakeGame(si.game)
}
si.gameMu.Unlock()
diff --git a/snake/persist.go b/snake/persist.go
new file mode 100644
index 0000000..6ef51d0
--- /dev/null
+++ b/snake/persist.go
@@ -0,0 +1,186 @@
+package snake
+
+import (
+ "context"
+ "database/sql"
+
+ "github.com/ryanhamamura/c4/db/repository"
+)
+
+// Persistence methods on SnakeStore (used during Get to hydrate from DB).
+
+func (ss *SnakeStore) saveSnakeGame(sg *SnakeGame) error {
+ ctx := context.Background()
+
+ boardJSON := "{}"
+ if sg.State != nil {
+ boardJSON = sg.State.ToJSON()
+ }
+
+ var gridWidth, gridHeight sql.NullInt64
+ if sg.State != nil {
+ gridWidth = sql.NullInt64{Int64: int64(sg.State.Width), Valid: true}
+ gridHeight = sql.NullInt64{Int64: int64(sg.State.Height), Valid: true}
+ }
+
+ _, err := ss.queries.GetSnakeGame(ctx, sg.ID)
+ if err == sql.ErrNoRows {
+ _, err = ss.queries.CreateSnakeGame(ctx, repository.CreateSnakeGameParams{
+ ID: sg.ID,
+ Board: boardJSON,
+ Status: int64(sg.Status),
+ GridWidth: gridWidth,
+ GridHeight: gridHeight,
+ GameMode: int64(sg.Mode),
+ SnakeSpeed: int64(sg.Speed),
+ })
+ return err
+ }
+ if err != nil {
+ return err
+ }
+
+ return ss.queries.UpdateSnakeGame(ctx, updateSnakeGameParams(sg, boardJSON))
+}
+
+func (ss *SnakeStore) loadSnakeGame(id string) (*SnakeGame, error) {
+ row, err := ss.queries.GetSnakeGame(context.Background(), id)
+ if err != nil {
+ return nil, err
+ }
+ return snakeGameFromRow(row)
+}
+
+func (ss *SnakeStore) loadSnakePlayers(id string) ([]*Player, error) {
+ rows, err := ss.queries.GetSnakePlayers(context.Background(), id)
+ if err != nil {
+ return nil, err
+ }
+ return snakePlayersFromRows(rows), nil
+}
+
+// Persistence methods on SnakeGameInstance (used during gameplay mutations).
+
+func (si *SnakeGameInstance) saveSnakeGame(sg *SnakeGame) error {
+ ctx := context.Background()
+
+ boardJSON := "{}"
+ if sg.State != nil {
+ boardJSON = sg.State.ToJSON()
+ }
+
+ var gridWidth, gridHeight sql.NullInt64
+ if sg.State != nil {
+ gridWidth = sql.NullInt64{Int64: int64(sg.State.Width), Valid: true}
+ gridHeight = sql.NullInt64{Int64: int64(sg.State.Height), Valid: true}
+ }
+
+ _, err := si.queries.GetSnakeGame(ctx, sg.ID)
+ if err == sql.ErrNoRows {
+ _, err = si.queries.CreateSnakeGame(ctx, repository.CreateSnakeGameParams{
+ ID: sg.ID,
+ Board: boardJSON,
+ Status: int64(sg.Status),
+ GridWidth: gridWidth,
+ GridHeight: gridHeight,
+ GameMode: int64(sg.Mode),
+ SnakeSpeed: int64(sg.Speed),
+ })
+ return err
+ }
+ if err != nil {
+ return err
+ }
+
+ return si.queries.UpdateSnakeGame(ctx, updateSnakeGameParams(sg, boardJSON))
+}
+
+func (si *SnakeGameInstance) saveSnakePlayer(gameID string, player *Player) error {
+ var userID, guestPlayerID sql.NullString
+ if player.UserID != nil {
+ userID = sql.NullString{String: *player.UserID, Valid: true}
+ } else {
+ guestPlayerID = sql.NullString{String: string(player.ID), Valid: true}
+ }
+
+ return si.queries.CreateSnakePlayer(context.Background(), repository.CreateSnakePlayerParams{
+ GameID: gameID,
+ UserID: userID,
+ GuestPlayerID: guestPlayerID,
+ Nickname: player.Nickname,
+ Color: int64(player.Slot + 1),
+ Slot: int64(player.Slot),
+ })
+}
+
+// Shared helpers for domain ↔ DB mapping.
+
+func updateSnakeGameParams(sg *SnakeGame, boardJSON string) repository.UpdateSnakeGameParams {
+ var winnerUserID sql.NullString
+ if sg.Winner != nil && sg.Winner.UserID != nil {
+ winnerUserID = sql.NullString{String: *sg.Winner.UserID, Valid: true}
+ }
+
+ var rematchGameID sql.NullString
+ if sg.RematchGameID != nil {
+ rematchGameID = sql.NullString{String: *sg.RematchGameID, Valid: true}
+ }
+
+ return repository.UpdateSnakeGameParams{
+ Board: boardJSON,
+ Status: int64(sg.Status),
+ WinnerUserID: winnerUserID,
+ RematchGameID: rematchGameID,
+ Score: int64(sg.Score),
+ ID: sg.ID,
+ }
+}
+
+func snakeGameFromRow(row repository.Game) (*SnakeGame, error) {
+ state, err := GameStateFromJSON(row.Board)
+ if err != nil {
+ state = &GameState{}
+ }
+ if row.GridWidth.Valid {
+ state.Width = int(row.GridWidth.Int64)
+ }
+ if row.GridHeight.Valid {
+ state.Height = int(row.GridHeight.Int64)
+ }
+
+ sg := &SnakeGame{
+ ID: row.ID,
+ State: state,
+ Players: make([]*Player, 8),
+ Status: Status(row.Status),
+ Mode: GameMode(row.GameMode),
+ Score: int(row.Score),
+ Speed: int(row.SnakeSpeed),
+ }
+
+ if row.RematchGameID.Valid {
+ sg.RematchGameID = &row.RematchGameID.String
+ }
+
+ return sg, nil
+}
+
+func snakePlayersFromRows(rows []repository.GamePlayer) []*Player {
+ players := make([]*Player, 0, len(rows))
+ for _, row := range rows {
+ player := &Player{
+ Nickname: row.Nickname,
+ Slot: int(row.Slot),
+ }
+
+ if row.UserID.Valid {
+ player.UserID = &row.UserID.String
+ player.ID = PlayerID(row.UserID.String)
+ } else if row.GuestPlayerID.Valid {
+ player.ID = PlayerID(row.GuestPlayerID.String)
+ }
+
+ players = append(players, player)
+ }
+ return players
+}
diff --git a/snake/store.go b/snake/store.go
index fac2574..ef1749e 100644
--- a/snake/store.go
+++ b/snake/store.go
@@ -1,36 +1,28 @@
package snake
import (
+ "context"
"crypto/rand"
"encoding/hex"
"sync"
+
+ "github.com/ryanhamamura/c4/db/repository"
)
-type Persister interface {
- SaveSnakeGame(sg *SnakeGame) error
- LoadSnakeGame(id string) (*SnakeGame, error)
- SaveSnakePlayer(gameID string, player *Player) error
- LoadSnakePlayers(gameID string) ([]*Player, error)
- DeleteSnakeGame(id string) error
-}
-
type SnakeStore struct {
- games map[string]*SnakeGameInstance
- gamesMu sync.RWMutex
- persister Persister
+ games map[string]*SnakeGameInstance
+ gamesMu sync.RWMutex
+ queries *repository.Queries
notifyFunc func(gameID string)
}
-func NewSnakeStore() *SnakeStore {
+func NewSnakeStore(queries *repository.Queries) *SnakeStore {
return &SnakeStore{
- games: make(map[string]*SnakeGameInstance),
+ games: make(map[string]*SnakeGameInstance),
+ queries: queries,
}
}
-func (ss *SnakeStore) SetPersister(p Persister) {
- ss.persister = p
-}
-
func (ss *SnakeStore) SetNotifyFunc(f func(gameID string)) {
ss.notifyFunc = f
}
@@ -60,18 +52,18 @@ func (ss *SnakeStore) Create(width, height int, mode GameMode, speed int) *Snake
Speed: speed,
}
si := &SnakeGameInstance{
- game: sg,
- notify: ss.makeNotify(id),
- persister: ss.persister,
- store: ss,
+ game: sg,
+ notify: ss.makeNotify(id),
+ queries: ss.queries,
+ store: ss,
}
ss.gamesMu.Lock()
ss.games[id] = si
ss.gamesMu.Unlock()
- if ss.persister != nil {
- ss.persister.SaveSnakeGame(sg)
+ if ss.queries != nil {
+ ss.saveSnakeGame(sg)
}
return si
@@ -86,16 +78,16 @@ func (ss *SnakeStore) Get(id string) (*SnakeGameInstance, bool) {
return si, true
}
- if ss.persister == nil {
+ if ss.queries == nil {
return nil, false
}
- sg, err := ss.persister.LoadSnakeGame(id)
+ sg, err := ss.loadSnakeGame(id)
if err != nil || sg == nil {
return nil, false
}
- players, _ := ss.persister.LoadSnakePlayers(id)
+ players, _ := ss.loadSnakePlayers(id)
if sg.Players == nil {
sg.Players = make([]*Player, 8)
}
@@ -106,10 +98,10 @@ func (ss *SnakeStore) Get(id string) (*SnakeGameInstance, bool) {
}
si = &SnakeGameInstance{
- game: sg,
- notify: ss.makeNotify(id),
- persister: ss.persister,
- store: ss,
+ game: sg,
+ notify: ss.makeNotify(id),
+ queries: ss.queries,
+ store: ss,
}
ss.gamesMu.Lock()
@@ -129,8 +121,8 @@ func (ss *SnakeStore) Delete(id string) error {
si.Stop()
}
- if ss.persister != nil {
- return ss.persister.DeleteSnakeGame(id)
+ if ss.queries != nil {
+ return ss.queries.DeleteSnakeGame(context.Background(), id)
}
return nil
}
@@ -158,14 +150,14 @@ func (ss *SnakeStore) ActiveGames() []*SnakeGame {
}
type SnakeGameInstance struct {
- game *SnakeGame
- gameMu sync.RWMutex
+ game *SnakeGame
+ gameMu sync.RWMutex
pendingDirQueue [8][]Direction // queued directions per slot (max 3)
- notify func()
- persister Persister
- store *SnakeStore
- stopCh chan struct{}
- loopOnce sync.Once
+ notify func()
+ queries *repository.Queries
+ store *SnakeStore
+ stopCh chan struct{}
+ loopOnce sync.Once
}
func (si *SnakeGameInstance) ID() string {
@@ -214,9 +206,9 @@ func (si *SnakeGameInstance) Join(player *Player) bool {
player.Slot = slot
si.game.Players[slot] = player
- if si.persister != nil {
- si.persister.SaveSnakePlayer(si.game.ID, player)
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakePlayer(si.game.ID, player)
+ si.saveSnakeGame(si.game)
}
si.notify()
@@ -301,8 +293,8 @@ func (si *SnakeGameInstance) CreateRematch() *SnakeGameInstance {
}
si.game.RematchGameID = &newID
- if si.persister != nil {
- si.persister.SaveSnakeGame(si.game)
+ if si.queries != nil {
+ si.saveSnakeGame(si.game)
}
si.gameMu.Unlock()
From afd8a3e9d0350121f05f05a2a0b615fd1036130c Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 12:38:21 -1000
Subject: [PATCH 4/9] fix: resolve all linting errors and add SSE compression
- Add brotli compression (level 5) to long-lived SSE event streams
(HandleGameEvents, HandleSnakeEvents) to reduce wire payload
- Fix all errcheck violations with nolint annotations for best-effort calls
- Fix goimports: separate stdlib, third-party, and local import groups
- Fix staticcheck: add package comments, use tagged switch
- Zero lint issues remaining
---
auth/auth.go | 1 +
features/auth/handlers.go | 3 ++-
features/auth/routes.go | 2 ++
features/c4game/handlers.go | 13 ++++++++-----
features/c4game/routes.go | 2 ++
features/lobby/handlers.go | 10 +++++-----
features/lobby/routes.go | 1 +
features/snakegame/handlers.go | 11 +++++++----
features/snakegame/routes.go | 2 ++
game/logic.go | 1 +
game/persist.go | 2 +-
game/store.go | 17 +++++++++--------
nats/nats.go | 2 +-
router/router.go | 16 ++++++++++++----
snake/logic.go | 1 +
snake/loop.go | 8 ++++----
snake/store.go | 10 +++++-----
snake/types.go | 6 +++---
18 files changed, 67 insertions(+), 41 deletions(-)
diff --git a/auth/auth.go b/auth/auth.go
index 17b03d2..16a2d45 100644
--- a/auth/auth.go
+++ b/auth/auth.go
@@ -1,3 +1,4 @@
+// Package auth provides password hashing and verification using bcrypt.
package auth
import (
diff --git a/features/auth/handlers.go b/features/auth/handlers.go
index 1212e0b..e39e59e 100644
--- a/features/auth/handlers.go
+++ b/features/auth/handlers.go
@@ -6,10 +6,11 @@ import (
"github.com/alexedwards/scs/v2"
"github.com/google/uuid"
+ "github.com/starfederation/datastar-go/datastar"
+
"github.com/ryanhamamura/c4/auth"
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/features/auth/pages"
- "github.com/starfederation/datastar-go/datastar"
)
type LoginSignals struct {
diff --git a/features/auth/routes.go b/features/auth/routes.go
index 98ad6df..f028547 100644
--- a/features/auth/routes.go
+++ b/features/auth/routes.go
@@ -1,8 +1,10 @@
+// Package auth handles user authentication routes and handlers.
package auth
import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
+
"github.com/ryanhamamura/c4/db/repository"
)
diff --git a/features/c4game/handlers.go b/features/c4game/handlers.go
index e0b107b..2e90886 100644
--- a/features/c4game/handlers.go
+++ b/features/c4game/handlers.go
@@ -12,11 +12,12 @@ import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
+ "github.com/starfederation/datastar-go/datastar"
+
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/features/c4game/components"
"github.com/ryanhamamura/c4/features/c4game/pages"
"github.com/ryanhamamura/c4/game"
- "github.com/starfederation/datastar-go/datastar"
)
func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
@@ -102,7 +103,9 @@ func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.Sessio
myColor := gi.GetPlayerColor(playerID)
- sse := datastar.NewSSE(w, r)
+ sse := datastar.NewSSE(w, r, datastar.WithCompression(
+ datastar.WithBrotli(datastar.WithBrotliLevel(5)),
+ ))
// Load initial chat messages
chatMsgs := loadChatMessages(queries, gameID)
@@ -118,7 +121,7 @@ func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.Sessio
if err != nil {
return
}
- defer gameSub.Unsubscribe()
+ defer gameSub.Unsubscribe() //nolint:errcheck
// Subscribe to chat messages
chatCh := make(chan *nats.Msg, 64)
@@ -126,7 +129,7 @@ func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.Sessio
if err != nil {
return
}
- defer chatSub.Unsubscribe()
+ defer chatSub.Unsubscribe() //nolint:errcheck
ctx := r.Context()
for {
@@ -263,7 +266,7 @@ func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionM
datastar.NewSSE(w, r)
return
}
- nc.Publish("game.chat."+gameID, data)
+ nc.Publish("game.chat."+gameID, data) //nolint:errcheck
// Clear the chat input
sse := datastar.NewSSE(w, r)
diff --git a/features/c4game/routes.go b/features/c4game/routes.go
index 917e5a0..1bba4b2 100644
--- a/features/c4game/routes.go
+++ b/features/c4game/routes.go
@@ -1,9 +1,11 @@
+// Package c4game handles Connect 4 game routes, SSE event streaming, and chat.
package c4game
import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
+
"github.com/ryanhamamura/c4/db/repository"
"github.com/ryanhamamura/c4/game"
)
diff --git a/features/lobby/handlers.go b/features/lobby/handlers.go
index 4c8a86a..a200703 100644
--- a/features/lobby/handlers.go
+++ b/features/lobby/handlers.go
@@ -91,7 +91,7 @@ func HandleCreateGame(store *game.GameStore, sessions *scs.SessionManager) http.
gi := store.Create()
sse := datastar.NewSSE(w, r)
- sse.ExecuteScript(fmt.Sprintf("window.location.href='/game/%s'", gi.ID()))
+ sse.ExecuteScript(fmt.Sprintf("window.location.href='/game/%s'", gi.ID())) //nolint:errcheck
}
}
@@ -104,10 +104,10 @@ func HandleDeleteGame(store *game.GameStore, sessions *scs.SessionManager) http.
return
}
- store.Delete(gameID)
+ store.Delete(gameID) //nolint:errcheck
sse := datastar.NewSSE(w, r)
- sse.ExecuteScript("window.location.href='/'")
+ sse.ExecuteScript("window.location.href='/'") //nolint:errcheck
}
}
@@ -150,7 +150,7 @@ func HandleCreateSnakeGame(snakeStore *snake.SnakeStore, sessions *scs.SessionMa
si := snakeStore.Create(preset.Width, preset.Height, mode, speed)
sse := datastar.NewSSE(w, r)
- sse.ExecuteScript(fmt.Sprintf("window.location.href='/snake/%s'", si.ID()))
+ sse.ExecuteScript(fmt.Sprintf("window.location.href='/snake/%s'", si.ID())) //nolint:errcheck
}
}
@@ -163,6 +163,6 @@ func HandleLogout(sessions *scs.SessionManager) http.HandlerFunc {
}
sse := datastar.NewSSE(w, r)
- sse.ExecuteScript("window.location.href='/'")
+ sse.ExecuteScript("window.location.href='/'") //nolint:errcheck
}
}
diff --git a/features/lobby/routes.go b/features/lobby/routes.go
index 016eb75..7f05748 100644
--- a/features/lobby/routes.go
+++ b/features/lobby/routes.go
@@ -1,3 +1,4 @@
+// Package lobby handles the game lobby page, game creation, and navigation.
package lobby
import (
diff --git a/features/snakegame/handlers.go b/features/snakegame/handlers.go
index 8fe9675..d97453c 100644
--- a/features/snakegame/handlers.go
+++ b/features/snakegame/handlers.go
@@ -9,11 +9,12 @@ import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
+ "github.com/starfederation/datastar-go/datastar"
+
"github.com/ryanhamamura/c4/features/snakegame/components"
"github.com/ryanhamamura/c4/features/snakegame/pages"
"github.com/ryanhamamura/c4/game"
"github.com/ryanhamamura/c4/snake"
- "github.com/starfederation/datastar-go/datastar"
)
func getPlayerID(sessions *scs.SessionManager, r *http.Request) snake.PlayerID {
@@ -90,7 +91,9 @@ func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *sc
playerID := getPlayerID(sessions, r)
mySlot := si.GetPlayerSlot(playerID)
- sse := datastar.NewSSE(w, r)
+ sse := datastar.NewSSE(w, r, datastar.WithCompression(
+ datastar.WithBrotli(datastar.WithBrotliLevel(5)),
+ ))
// Send initial render
sg := si.GetGame()
@@ -110,7 +113,7 @@ func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *sc
if err != nil {
return
}
- defer gameSub.Unsubscribe()
+ defer gameSub.Unsubscribe() //nolint:errcheck
// Chat subscription (multiplayer only)
var chatCh chan *nats.Msg
@@ -124,7 +127,7 @@ func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *sc
if err != nil {
return
}
- defer chatSub.Unsubscribe()
+ defer chatSub.Unsubscribe() //nolint:errcheck
}
ctx := r.Context()
diff --git a/features/snakegame/routes.go b/features/snakegame/routes.go
index c4c8334..d1ccc23 100644
--- a/features/snakegame/routes.go
+++ b/features/snakegame/routes.go
@@ -1,9 +1,11 @@
+// Package snakegame handles snake game routes, SSE event streaming, and chat.
package snakegame
import (
"github.com/alexedwards/scs/v2"
"github.com/go-chi/chi/v5"
"github.com/nats-io/nats.go"
+
"github.com/ryanhamamura/c4/snake"
)
diff --git a/game/logic.go b/game/logic.go
index e8429cd..7a4d167 100644
--- a/game/logic.go
+++ b/game/logic.go
@@ -1,3 +1,4 @@
+// Package game implements Connect 4 game logic, state management, and persistence.
package game
// DropPiece attempts to drop a piece in the given column.
diff --git a/game/persist.go b/game/persist.go
index 2322adc..fb05716 100644
--- a/game/persist.go
+++ b/game/persist.go
@@ -126,7 +126,7 @@ func gameFromRow(row repository.Game) (*Game, error) {
}
if row.WinningCells.Valid {
- g.WinningCellsFromJSON(row.WinningCells.String)
+ _ = g.WinningCellsFromJSON(row.WinningCells.String)
}
if row.RematchGameID.Valid {
diff --git a/game/store.go b/game/store.go
index e1c9c40..e56e750 100644
--- a/game/store.go
+++ b/game/store.go
@@ -49,7 +49,7 @@ func (gs *GameStore) Create() *GameInstance {
gs.gamesMu.Unlock()
if gs.queries != nil {
- gs.saveGame(gi.game)
+ gs.saveGame(gi.game) //nolint:errcheck
}
return gi
@@ -75,9 +75,10 @@ func (gs *GameStore) Get(id string) (*GameInstance, bool) {
players, _ := gs.loadGamePlayers(id)
for _, p := range players {
- if p.Color == 1 {
+ switch p.Color {
+ case 1:
g.Players[0] = p
- } else if p.Color == 2 {
+ case 2:
g.Players[1] = p
}
}
@@ -108,7 +109,7 @@ func (gs *GameStore) Delete(id string) error {
func GenerateID(size int) string {
b := make([]byte, size)
- rand.Read(b)
+ _, _ = rand.Read(b)
return hex.EncodeToString(b)
}
@@ -151,8 +152,8 @@ func (gi *GameInstance) Join(ps *PlayerSession) bool {
}
if gi.queries != nil {
- gi.saveGamePlayer(gi.game.ID, ps.Player, slot)
- gi.saveGame(gi.game)
+ gi.saveGamePlayer(gi.game.ID, ps.Player, slot) //nolint:errcheck
+ gi.saveGame(gi.game) //nolint:errcheck
}
gi.notify()
@@ -190,7 +191,7 @@ func (gi *GameInstance) CreateRematch(gs *GameStore) *GameInstance {
if gi.queries != nil {
if err := gi.saveGame(gi.game); err != nil {
- gs.Delete(newID)
+ gs.Delete(newID) //nolint:errcheck
gi.game.RematchGameID = nil
return nil
}
@@ -223,7 +224,7 @@ func (gi *GameInstance) DropPiece(col int, playerColor int) bool {
}
if gi.queries != nil {
- gi.saveGame(gi.game)
+ gi.saveGame(gi.game) //nolint:errcheck
}
gi.notify()
diff --git a/nats/nats.go b/nats/nats.go
index 9eeabc9..86bee3d 100644
--- a/nats/nats.go
+++ b/nats/nats.go
@@ -40,7 +40,7 @@ func SetupNATS(ctx context.Context) (*nats.Conn, func(), error) {
cleanup := func() {
nc.Close()
- ns.Close()
+ ns.Close() //nolint:errcheck
}
return nc, cleanup, nil
diff --git a/router/router.go b/router/router.go
index c6198fc..f4bded2 100644
--- a/router/router.go
+++ b/router/router.go
@@ -40,10 +40,18 @@ func SetupRoutes(
setupReload(router)
}
- auth.SetupRoutes(router, queries, sessions)
- lobby.SetupRoutes(router, queries, sessions, store, snakeStore)
- c4game.SetupRoutes(router, store, nc, sessions, queries)
- snakegame.SetupRoutes(router, snakeStore, nc, sessions)
+ if err := auth.SetupRoutes(router, queries, sessions); err != nil {
+ return err
+ }
+ if err := lobby.SetupRoutes(router, queries, sessions, store, snakeStore); err != nil {
+ return err
+ }
+ if err := c4game.SetupRoutes(router, store, nc, sessions, queries); err != nil {
+ return err
+ }
+ if err := snakegame.SetupRoutes(router, snakeStore, nc, sessions); err != nil {
+ return err
+ }
return nil
}
diff --git a/snake/logic.go b/snake/logic.go
index c48c6ef..a26a5a6 100644
--- a/snake/logic.go
+++ b/snake/logic.go
@@ -1,3 +1,4 @@
+// Package snake implements snake game logic, state management, and persistence.
package snake
import "math/rand"
diff --git a/snake/loop.go b/snake/loop.go
index fb1a839..cf44f84 100644
--- a/snake/loop.go
+++ b/snake/loop.go
@@ -62,7 +62,7 @@ func (si *SnakeGameInstance) countdownPhase() {
si.game.Status = StatusInProgress
if si.queries != nil {
- si.saveSnakeGame(si.game)
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.gameMu.Unlock()
si.notify()
@@ -70,7 +70,7 @@ func (si *SnakeGameInstance) countdownPhase() {
}
if si.queries != nil {
- si.saveSnakeGame(si.game)
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.gameMu.Unlock()
si.notify()
@@ -124,7 +124,7 @@ func (si *SnakeGameInstance) gamePhase() {
if time.Since(lastInput) > inactivityLimit {
si.game.Status = StatusFinished
if si.queries != nil {
- si.saveSnakeGame(si.game)
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.gameMu.Unlock()
si.notify()
@@ -196,7 +196,7 @@ func (si *SnakeGameInstance) gamePhase() {
}
if si.queries != nil {
- si.saveSnakeGame(si.game)
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.gameMu.Unlock()
diff --git a/snake/store.go b/snake/store.go
index ef1749e..670b9da 100644
--- a/snake/store.go
+++ b/snake/store.go
@@ -63,7 +63,7 @@ func (ss *SnakeStore) Create(width, height int, mode GameMode, speed int) *Snake
ss.gamesMu.Unlock()
if ss.queries != nil {
- ss.saveSnakeGame(sg)
+ ss.saveSnakeGame(sg) //nolint:errcheck
}
return si
@@ -207,8 +207,8 @@ func (si *SnakeGameInstance) Join(player *Player) bool {
si.game.Players[slot] = player
if si.queries != nil {
- si.saveSnakePlayer(si.game.ID, player)
- si.saveSnakeGame(si.game)
+ si.saveSnakePlayer(si.game.ID, player) //nolint:errcheck
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.notify()
@@ -294,7 +294,7 @@ func (si *SnakeGameInstance) CreateRematch() *SnakeGameInstance {
si.game.RematchGameID = &newID
if si.queries != nil {
- si.saveSnakeGame(si.game)
+ si.saveSnakeGame(si.game) //nolint:errcheck
}
si.gameMu.Unlock()
@@ -304,6 +304,6 @@ func (si *SnakeGameInstance) CreateRematch() *SnakeGameInstance {
func generateID(size int) string {
b := make([]byte, size)
- rand.Read(b)
+ _, _ = rand.Read(b)
return hex.EncodeToString(b)
}
diff --git a/snake/types.go b/snake/types.go
index 5bb21a9..8272765 100644
--- a/snake/types.go
+++ b/snake/types.go
@@ -100,7 +100,7 @@ type SnakeGame struct {
Speed int // cells per second
}
-// Speed presets
+// SpeedPreset defines a named speed option for the snake game.
type SpeedPreset struct {
Name string
Speed int
@@ -129,7 +129,7 @@ func (sg *SnakeGame) PlayerCount() int {
return count
}
-// Grid presets
+// GridPreset defines a named grid size option for the snake game.
type GridPreset struct {
Name string
Width int
@@ -163,7 +163,7 @@ func (sg *SnakeGame) snapshot() *SnakeGame {
return &cp
}
-// Snake colors (hex values for CSS)
+// SnakeColors are hex color values for CSS, indexed by player slot.
var SnakeColors = []string{
"#00b894", // 1: Green
"#e17055", // 2: Orange
From 67d4dba37f4334fd5c7ffe9d462eb1a1a2ac48c4 Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 12:40:02 -1000
Subject: [PATCH 5/9] fix: suppress gosec G117 on auth form signal structs
---
features/auth/handlers.go | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/features/auth/handlers.go b/features/auth/handlers.go
index e39e59e..dc34675 100644
--- a/features/auth/handlers.go
+++ b/features/auth/handlers.go
@@ -15,12 +15,12 @@ import (
type LoginSignals struct {
Username string `json:"username"`
- Password string `json:"password"`
+ Password string `json:"password"` //nolint:gosec // form input, not stored
}
type RegisterSignals struct {
Username string `json:"username"`
- Password string `json:"password"`
+ Password string `json:"password"` //nolint:gosec // form input, not stored
Confirm string `json:"confirm"`
}
From fcc6b70e846139d33484b7cf227409b673714fb6 Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 12:42:10 -1000
Subject: [PATCH 6/9] fix: warn when .env file is missing instead of silently
ignoring
---
config/config.go | 5 ++++-
1 file changed, 4 insertions(+), 1 deletion(-)
diff --git a/config/config.go b/config/config.go
index 6c09cd9..31774f2 100644
--- a/config/config.go
+++ b/config/config.go
@@ -4,6 +4,7 @@
package config
import (
+ "log/slog"
"os"
"sync"
@@ -46,7 +47,9 @@ func getEnv(key, fallback string) string {
}
func loadBase() *Config {
- godotenv.Load() //nolint:errcheck // .env file is optional
+ if err := godotenv.Load(); err != nil {
+ slog.Warn("no .env file found, using environment variables and defaults")
+ }
return &Config{
Host: getEnv("HOST", "0.0.0.0"),
From 5120eef776f7cfac210cd2f581274d65f531e98e Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 13:19:03 -1000
Subject: [PATCH 7/9] refactor: streamline routes to RESTful naming conventions
Remove /api/ prefix and consolidate route groups:
- /api/lobby/* -> /games, /snake, /logout (top-level)
- /game/{game_id} + /api/game/{game_id}/* -> /games/{id}/*
- /snake/{game_id} + /api/snake/{game_id}/* -> /snake/{id}/*
- /api/auth/* -> /auth/*
- Standardize snake join page to use return_url= (was return=)
---
features/auth/pages/login_templ.go | 8 ++++----
features/auth/pages/register_templ.go | 8 ++++----
features/auth/routes.go | 4 ++--
features/c4game/components/board_templ.go | 4 ++--
features/c4game/components/chat_templ.go | 8 ++++----
features/c4game/components/status_templ.go | 16 +++++++--------
features/c4game/handlers.go | 16 +++++++--------
features/c4game/pages/game_templ.go | 12 +++++------
features/c4game/routes.go | 7 +++----
features/lobby/components/gamelist_templ.go | 8 ++++----
features/lobby/handlers.go | 2 +-
features/lobby/pages/lobby_templ.go | 20 +++++++++----------
features/lobby/routes.go | 10 ++++------
features/snakegame/components/chat_templ.go | 8 ++++----
features/snakegame/components/status_templ.go | 4 ++--
features/snakegame/handlers.go | 12 +++++------
features/snakegame/pages/game_templ.go | 14 ++++++-------
features/snakegame/routes.go | 7 +++----
18 files changed, 82 insertions(+), 86 deletions(-)
diff --git a/features/auth/pages/login_templ.go b/features/auth/pages/login_templ.go
index 115ea27..3640508 100644
--- a/features/auth/pages/login_templ.go
+++ b/features/auth/pages/login_templ.go
@@ -51,9 +51,9 @@ func LoginPage() templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var3 string
- templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/auth/login"))
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/auth/login"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/login.templ`, Line: 34, Col: 69}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/login.templ`, Line: 34, Col: 65}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
if templ_7745c5c3_Err != nil {
@@ -64,9 +64,9 @@ func LoginPage() templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var4 string
- templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/auth/login"))
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/auth/login"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/login.templ`, Line: 40, Col: 56}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/login.templ`, Line: 40, Col: 52}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
if templ_7745c5c3_Err != nil {
diff --git a/features/auth/pages/register_templ.go b/features/auth/pages/register_templ.go
index 89efec3..61c8f21 100644
--- a/features/auth/pages/register_templ.go
+++ b/features/auth/pages/register_templ.go
@@ -51,9 +51,9 @@ func RegisterPage() templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var3 string
- templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/auth/register"))
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/auth/register"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/register.templ`, Line: 43, Col: 72}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/register.templ`, Line: 43, Col: 68}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
if templ_7745c5c3_Err != nil {
@@ -64,9 +64,9 @@ func RegisterPage() templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var4 string
- templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/auth/register"))
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/auth/register"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/register.templ`, Line: 49, Col: 59}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/auth/pages/register.templ`, Line: 49, Col: 55}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
if templ_7745c5c3_Err != nil {
diff --git a/features/auth/routes.go b/features/auth/routes.go
index f028547..ac50938 100644
--- a/features/auth/routes.go
+++ b/features/auth/routes.go
@@ -11,8 +11,8 @@ import (
func SetupRoutes(router chi.Router, queries *repository.Queries, sessions *scs.SessionManager) error {
router.Get("/login", HandleLoginPage())
router.Get("/register", HandleRegisterPage())
- router.Post("/api/auth/login", HandleLogin(queries, sessions))
- router.Post("/api/auth/register", HandleRegister(queries, sessions))
+ router.Post("/auth/login", HandleLogin(queries, sessions))
+ router.Post("/auth/register", HandleRegister(queries, sessions))
return nil
}
diff --git a/features/c4game/components/board_templ.go b/features/c4game/components/board_templ.go
index 92f27b5..7c638be 100644
--- a/features/c4game/components/board_templ.go
+++ b/features/c4game/components/board_templ.go
@@ -81,9 +81,9 @@ func column(g *game.Game, colIdx int, myColor int) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var3 string
- templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/game/%s/drop?col=%d", g.ID, colIdx))
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games/%s/drop?col=%d", g.ID, colIdx))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/board.templ`, Line: 22, Col: 77}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/board.templ`, Line: 22, Col: 74}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
if templ_7745c5c3_Err != nil {
diff --git a/features/c4game/components/chat_templ.go b/features/c4game/components/chat_templ.go
index c6fa087..b70da12 100644
--- a/features/c4game/components/chat_templ.go
+++ b/features/c4game/components/chat_templ.go
@@ -105,9 +105,9 @@ func Chat(messages []ChatMessage, gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var5 string
- templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/game/%s/chat", gameID))
+ templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games/%s/chat", gameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 40, Col: 73}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 40, Col: 70}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5))
if templ_7745c5c3_Err != nil {
@@ -118,9 +118,9 @@ func Chat(messages []ChatMessage, gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var6 string
- templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/game/%s/chat", gameID))
+ templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games/%s/chat", gameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 44, Col: 65}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/chat.templ`, Line: 44, Col: 62}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
if templ_7745c5c3_Err != nil {
diff --git a/features/c4game/components/status_templ.go b/features/c4game/components/status_templ.go
index ba58ad9..c87c799 100644
--- a/features/c4game/components/status_templ.go
+++ b/features/c4game/components/status_templ.go
@@ -77,9 +77,9 @@ func StatusBanner(g *game.Game, myColor int) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var5 templ.SafeURL
- templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinURLErrs(templ.SafeURL("/game/" + *g.RematchGameID))
+ templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinURLErrs(templ.SafeURL("/games/" + *g.RematchGameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 16, Col: 54}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 16, Col: 55}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5))
if templ_7745c5c3_Err != nil {
@@ -95,9 +95,9 @@ func StatusBanner(g *game.Game, myColor int) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var6 string
- templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/game/%s/rematch", g.ID))
+ templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games/%s/rematch", g.ID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 24, Col: 67}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 24, Col: 64}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
if templ_7745c5c3_Err != nil {
@@ -217,9 +217,9 @@ func InviteLink(gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var12 string
- templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(config.Global.AppURL + "/game/" + gameID)
+ templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(config.Global.AppURL + "/games/" + gameID)
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 48, Col: 45}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/components/status.templ`, Line: 48, Col: 46}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
if templ_7745c5c3_Err != nil {
@@ -229,7 +229,7 @@ func InviteLink(gameID string) templ.Component {
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
- templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, copyToClipboard(config.Global.AppURL+"/game/"+gameID))
+ templ_7745c5c3_Err = templ.RenderScriptItems(ctx, templ_7745c5c3_Buffer, copyToClipboard(config.Global.AppURL+"/games/"+gameID))
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
@@ -237,7 +237,7 @@ func InviteLink(gameID string) templ.Component {
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
- var templ_7745c5c3_Var13 templ.ComponentScript = copyToClipboard(config.Global.AppURL + "/game/" + gameID)
+ var templ_7745c5c3_Var13 templ.ComponentScript = copyToClipboard(config.Global.AppURL + "/games/" + gameID)
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ_7745c5c3_Var13.Call)
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
diff --git a/features/c4game/handlers.go b/features/c4game/handlers.go
index 2e90886..6d24ed2 100644
--- a/features/c4game/handlers.go
+++ b/features/c4game/handlers.go
@@ -22,7 +22,7 @@ import (
func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -87,7 +87,7 @@ func HandleGamePage(store *game.GameStore, sessions *scs.SessionManager, queries
func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -173,7 +173,7 @@ func HandleGameEvents(store *game.GameStore, nc *nats.Conn, sessions *scs.Sessio
func HandleDropPiece(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -210,7 +210,7 @@ func HandleDropPiece(store *game.GameStore, sessions *scs.SessionManager) http.H
func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionManager, queries *repository.Queries) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -276,7 +276,7 @@ func HandleSendChat(store *game.GameStore, nc *nats.Conn, sessions *scs.SessionM
func HandleSetNickname(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -319,13 +319,13 @@ func HandleSetNickname(store *game.GameStore, sessions *scs.SessionManager) http
}
sse := datastar.NewSSE(w, r)
- sse.Redirect("/game/" + gameID) //nolint:errcheck
+ sse.Redirect("/games/" + gameID) //nolint:errcheck
}
}
func HandleRematch(store *game.GameStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
gi, exists := store.Get(gameID)
if !exists {
@@ -337,7 +337,7 @@ func HandleRematch(store *game.GameStore, sessions *scs.SessionManager) http.Han
newGI := gi.CreateRematch(store)
sse := datastar.NewSSE(w, r)
if newGI != nil {
- sse.Redirectf("/game/%s", newGI.ID()) //nolint:errcheck
+ sse.Redirectf("/games/%s", newGI.ID()) //nolint:errcheck
}
}
}
diff --git a/features/c4game/pages/game_templ.go b/features/c4game/pages/game_templ.go
index 3662c23..903e9a3 100644
--- a/features/c4game/pages/game_templ.go
+++ b/features/c4game/pages/game_templ.go
@@ -54,9 +54,9 @@ func GamePage(g *game.Game, myColor int, messages []components.ChatMessage) temp
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var3 string
- templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.GetSSE("/game/%s/events", g.ID))
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.GetSSE("/games/%s/events", g.ID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/pages/game.templ`, Line: 16, Col: 55}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/c4game/pages/game.templ`, Line: 16, Col: 56}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
if templ_7745c5c3_Err != nil {
@@ -152,9 +152,9 @@ func JoinPage(gameID string) templ.Component {
}
ctx = templ.InitializeContext(ctx)
templ_7745c5c3_Err = sharedcomponents.GameJoinPrompt(
- "/login?return_url=/game/"+gameID,
- "/register?return_url=/game/"+gameID,
- "/game/"+gameID,
+ "/login?return_url=/games/"+gameID,
+ "/register?return_url=/games/"+gameID,
+ "/games/"+gameID,
).Render(ctx, templ_7745c5c3_Buffer)
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
@@ -202,7 +202,7 @@ func NicknamePage(gameID string) templ.Component {
}()
}
ctx = templ.InitializeContext(ctx)
- templ_7745c5c3_Err = sharedcomponents.NicknamePrompt("/api/game/"+gameID+"/join").Render(ctx, templ_7745c5c3_Buffer)
+ templ_7745c5c3_Err = sharedcomponents.NicknamePrompt("/games/"+gameID+"/join").Render(ctx, templ_7745c5c3_Buffer)
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
diff --git a/features/c4game/routes.go b/features/c4game/routes.go
index 1bba4b2..4cc4641 100644
--- a/features/c4game/routes.go
+++ b/features/c4game/routes.go
@@ -17,10 +17,9 @@ func SetupRoutes(
sessions *scs.SessionManager,
queries *repository.Queries,
) error {
- router.Get("/game/{game_id}", HandleGamePage(store, sessions, queries))
- router.Get("/game/{game_id}/events", HandleGameEvents(store, nc, sessions, queries))
-
- router.Route("/api/game/{game_id}", func(r chi.Router) {
+ router.Route("/games/{id}", func(r chi.Router) {
+ r.Get("/", HandleGamePage(store, sessions, queries))
+ r.Get("/events", HandleGameEvents(store, nc, sessions, queries))
r.Post("/drop", HandleDropPiece(store, sessions))
r.Post("/chat", HandleSendChat(store, nc, sessions, queries))
r.Post("/join", HandleSetNickname(store, sessions))
diff --git a/features/lobby/components/gamelist_templ.go b/features/lobby/components/gamelist_templ.go
index c8539c8..4157630 100644
--- a/features/lobby/components/gamelist_templ.go
+++ b/features/lobby/components/gamelist_templ.go
@@ -83,9 +83,9 @@ func gameListEntry(g GameListItem) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var3 templ.SafeURL
- templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinURLErrs(templ.SafeURL("/game/" + g.ID))
+ templ_7745c5c3_Var3, templ_7745c5c3_Err = templ.JoinURLErrs(templ.SafeURL("/games/" + g.ID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 27, Col: 40}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 27, Col: 41}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var3))
if templ_7745c5c3_Err != nil {
@@ -157,9 +157,9 @@ func gameListEntry(g GameListItem) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var9 string
- templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.DeleteSSE("/api/lobby/game/%s", g.ID))
+ templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.DeleteSSE("/games/%s", g.ID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 41, Col: 65}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/components/gamelist.templ`, Line: 41, Col: 56}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9))
if templ_7745c5c3_Err != nil {
diff --git a/features/lobby/handlers.go b/features/lobby/handlers.go
index a200703..8144543 100644
--- a/features/lobby/handlers.go
+++ b/features/lobby/handlers.go
@@ -91,7 +91,7 @@ func HandleCreateGame(store *game.GameStore, sessions *scs.SessionManager) http.
gi := store.Create()
sse := datastar.NewSSE(w, r)
- sse.ExecuteScript(fmt.Sprintf("window.location.href='/game/%s'", gi.ID())) //nolint:errcheck
+ sse.ExecuteScript(fmt.Sprintf("window.location.href='/games/%s'", gi.ID())) //nolint:errcheck
}
}
diff --git a/features/lobby/pages/lobby_templ.go b/features/lobby/pages/lobby_templ.go
index bc1aeb2..ffae496 100644
--- a/features/lobby/pages/lobby_templ.go
+++ b/features/lobby/pages/lobby_templ.go
@@ -74,9 +74,9 @@ func LobbyPage(data LobbyData) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var4 string
- templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/lobby/logout"))
+ templ_7745c5c3_Var4, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/logout"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 26, Col: 59}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 26, Col: 49}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var4))
if templ_7745c5c3_Err != nil {
@@ -113,9 +113,9 @@ func LobbyPage(data LobbyData) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var5 string
- templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/lobby/create-game"))
+ templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 76, Col: 73}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 76, Col: 57}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5))
if templ_7745c5c3_Err != nil {
@@ -126,9 +126,9 @@ func LobbyPage(data LobbyData) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var6 string
- templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/lobby/create-game"))
+ templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/games"))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 82, Col: 64}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 82, Col: 48}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
if templ_7745c5c3_Err != nil {
@@ -201,9 +201,9 @@ func LobbyPage(data LobbyData) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var10 string
- templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/lobby/create-snake?mode=solo&preset=%d", i))
+ templ_7745c5c3_Var10, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/snake?mode=solo&preset=%d", i))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 129, Col: 90}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 129, Col: 73}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var10))
if templ_7745c5c3_Err != nil {
@@ -237,9 +237,9 @@ func LobbyPage(data LobbyData) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var12 string
- templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/lobby/create-snake?mode=multi&preset=%d", i))
+ templ_7745c5c3_Var12, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/snake?mode=multi&preset=%d", i))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 144, Col: 91}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/lobby/pages/lobby.templ`, Line: 144, Col: 74}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var12))
if templ_7745c5c3_Err != nil {
diff --git a/features/lobby/routes.go b/features/lobby/routes.go
index 7f05748..5dde496 100644
--- a/features/lobby/routes.go
+++ b/features/lobby/routes.go
@@ -19,12 +19,10 @@ func SetupRoutes(
) error {
router.Get("/", HandleLobbyPage(queries, sessions, snakeStore))
- router.Route("/api/lobby", func(r chi.Router) {
- r.Post("/create-game", HandleCreateGame(store, sessions))
- r.Delete("/game/{id}", HandleDeleteGame(store, sessions))
- r.Post("/create-snake", HandleCreateSnakeGame(snakeStore, sessions))
- r.Post("/logout", HandleLogout(sessions))
- })
+ router.Post("/games", HandleCreateGame(store, sessions))
+ router.Delete("/games/{id}", HandleDeleteGame(store, sessions))
+ router.Post("/snake", HandleCreateSnakeGame(snakeStore, sessions))
+ router.Post("/logout", HandleLogout(sessions))
return nil
}
diff --git a/features/snakegame/components/chat_templ.go b/features/snakegame/components/chat_templ.go
index 396e9a5..db1946e 100644
--- a/features/snakegame/components/chat_templ.go
+++ b/features/snakegame/components/chat_templ.go
@@ -97,9 +97,9 @@ func Chat(messages []ChatMessage, gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var5 string
- templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/snake/%s/chat", gameID))
+ templ_7745c5c3_Var5, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/snake/%s/chat", gameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 36, Col: 78}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 36, Col: 74}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var5))
if templ_7745c5c3_Err != nil {
@@ -110,9 +110,9 @@ func Chat(messages []ChatMessage, gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var6 string
- templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/snake/%s/chat", gameID))
+ templ_7745c5c3_Var6, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/snake/%s/chat", gameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 40, Col: 66}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/chat.templ`, Line: 40, Col: 62}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var6))
if templ_7745c5c3_Err != nil {
diff --git a/features/snakegame/components/status_templ.go b/features/snakegame/components/status_templ.go
index b1733c0..57416eb 100644
--- a/features/snakegame/components/status_templ.go
+++ b/features/snakegame/components/status_templ.go
@@ -266,9 +266,9 @@ func rematchOrJoin(sg *snake.SnakeGame, gameID string) templ.Component {
return templ_7745c5c3_Err
}
var templ_7745c5c3_Var9 string
- templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/api/snake/%s/rematch", gameID))
+ templ_7745c5c3_Var9, templ_7745c5c3_Err = templ.JoinStringErrs(datastar.PostSSE("/snake/%s/rematch", gameID))
if templ_7745c5c3_Err != nil {
- return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 82, Col: 68}
+ return templ.Error{Err: templ_7745c5c3_Err, FileName: `features/snakegame/components/status.templ`, Line: 82, Col: 64}
}
_, templ_7745c5c3_Err = templ_7745c5c3_Buffer.WriteString(templ.EscapeString(templ_7745c5c3_Var9))
if templ_7745c5c3_Err != nil {
diff --git a/features/snakegame/handlers.go b/features/snakegame/handlers.go
index d97453c..03292a6 100644
--- a/features/snakegame/handlers.go
+++ b/features/snakegame/handlers.go
@@ -32,7 +32,7 @@ func getPlayerID(sessions *scs.SessionManager, r *http.Request) snake.PlayerID {
func HandleSnakePage(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Redirect(w, r, "/", http.StatusSeeOther)
@@ -81,7 +81,7 @@ func HandleSnakePage(snakeStore *snake.SnakeStore, sessions *scs.SessionManager)
func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Error(w, "game not found", http.StatusNotFound)
@@ -189,7 +189,7 @@ func HandleSnakeEvents(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *sc
func HandleSetDirection(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Error(w, "game not found", http.StatusNotFound)
@@ -221,7 +221,7 @@ type chatSignals struct {
func HandleSendChat(snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Error(w, "game not found", http.StatusNotFound)
@@ -268,7 +268,7 @@ type nicknameSignals struct {
func HandleSetNickname(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Error(w, "game not found", http.StatusNotFound)
@@ -308,7 +308,7 @@ func HandleSetNickname(snakeStore *snake.SnakeStore, sessions *scs.SessionManage
func HandleRematch(snakeStore *snake.SnakeStore, sessions *scs.SessionManager) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
- gameID := chi.URLParam(r, "game_id")
+ gameID := chi.URLParam(r, "id")
si, ok := snakeStore.Get(gameID)
if !ok {
http.Error(w, "game not found", http.StatusNotFound)
diff --git a/features/snakegame/pages/game_templ.go b/features/snakegame/pages/game_templ.go
index a7023cb..123aaab 100644
--- a/features/snakegame/pages/game_templ.go
+++ b/features/snakegame/pages/game_templ.go
@@ -27,10 +27,10 @@ func keydownScript(gameID string) string {
"else if(k==='s'||k==='ArrowDown'){evt.preventDefault();%s}"+
"else if(k==='a'||k==='ArrowLeft'){evt.preventDefault();%s}"+
"else if(k==='d'||k==='ArrowRight'){evt.preventDefault();%s}",
- datastar.PostSSE("/api/snake/%s/dir?d=0", gameID),
- datastar.PostSSE("/api/snake/%s/dir?d=1", gameID),
- datastar.PostSSE("/api/snake/%s/dir?d=2", gameID),
- datastar.PostSSE("/api/snake/%s/dir?d=3", gameID),
+ datastar.PostSSE("/snake/%s/dir?d=0", gameID),
+ datastar.PostSSE("/snake/%s/dir?d=1", gameID),
+ datastar.PostSSE("/snake/%s/dir?d=2", gameID),
+ datastar.PostSSE("/snake/%s/dir?d=3", gameID),
)
}
@@ -210,8 +210,8 @@ func JoinPage(gameID string) templ.Component {
}
ctx = templ.InitializeContext(ctx)
templ_7745c5c3_Err = components.GameJoinPrompt(
- fmt.Sprintf("/login?return=/snake/%s", gameID),
- fmt.Sprintf("/register?return=/snake/%s", gameID),
+ fmt.Sprintf("/login?return_url=/snake/%s", gameID),
+ fmt.Sprintf("/register?return_url=/snake/%s", gameID),
fmt.Sprintf("/snake/%s", gameID),
).Render(ctx, templ_7745c5c3_Buffer)
if templ_7745c5c3_Err != nil {
@@ -260,7 +260,7 @@ func NicknamePage(gameID string) templ.Component {
}()
}
ctx = templ.InitializeContext(ctx)
- templ_7745c5c3_Err = components.NicknamePrompt(fmt.Sprintf("/api/snake/%s/join", gameID)).Render(ctx, templ_7745c5c3_Buffer)
+ templ_7745c5c3_Err = components.NicknamePrompt(fmt.Sprintf("/snake/%s/join", gameID)).Render(ctx, templ_7745c5c3_Buffer)
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
diff --git a/features/snakegame/routes.go b/features/snakegame/routes.go
index d1ccc23..4cdceec 100644
--- a/features/snakegame/routes.go
+++ b/features/snakegame/routes.go
@@ -10,10 +10,9 @@ import (
)
func SetupRoutes(router chi.Router, snakeStore *snake.SnakeStore, nc *nats.Conn, sessions *scs.SessionManager) error {
- router.Get("/snake/{game_id}", HandleSnakePage(snakeStore, sessions))
- router.Get("/snake/{game_id}/events", HandleSnakeEvents(snakeStore, nc, sessions))
-
- router.Route("/api/snake/{game_id}", func(r chi.Router) {
+ router.Route("/snake/{id}", func(r chi.Router) {
+ r.Get("/", HandleSnakePage(snakeStore, sessions))
+ r.Get("/events", HandleSnakeEvents(snakeStore, nc, sessions))
r.Post("/dir", HandleSetDirection(snakeStore, sessions))
r.Post("/chat", HandleSendChat(snakeStore, nc, sessions))
r.Post("/join", HandleSetNickname(snakeStore, sessions))
From 587f392b8b0b8c8ed0895afa47e9912a29bbdeb1 Mon Sep 17 00:00:00 2001
From: Ryan Hamamura <58859899+ryanhamamura@users.noreply.github.com>
Date: Mon, 2 Mar 2026 14:34:39 -1000
Subject: [PATCH 8/9] fix: serve datastar locally and clean up session/route
config
Replace CDN-hosted datastar beta.11 with local v1.0.0-RC.7 to fix
client-side expression incompatibilities with the Go SDK. Also fix
quoted CSS class keys in data-class expressions, harden session cookie
settings (named cookie, Secure flag), simplify SetupRoutes to not
return an error, and regenerate templ output.
---
assets/js/datastar.js | 9 +++++++++
features/auth/pages/login_templ.go | 10 +++++-----
features/auth/pages/register_templ.go | 10 +++++-----
features/auth/routes.go | 4 +---
features/c4game/routes.go | 4 +---
features/common/layouts/base_templ.go | 2 +-
features/lobby/pages/lobby_templ.go | 8 ++++----
features/lobby/routes.go | 4 +---
features/snakegame/routes.go | 4 +---
main.go | 4 +---
router/router.go | 20 +++++---------------
sessions/sessions.go | 3 ++-
12 files changed, 36 insertions(+), 46 deletions(-)
create mode 100644 assets/js/datastar.js
diff --git a/assets/js/datastar.js b/assets/js/datastar.js
new file mode 100644
index 0000000..bfb85cc
--- /dev/null
+++ b/assets/js/datastar.js
@@ -0,0 +1,9 @@
+// Datastar v1.0.0-RC.7
+var at=/🖕JS_DS🚀/.source,je=at.slice(0,5),Ge=at.slice(4),q="datastar-fetch",Z="datastar-signal-patch";var C=Object.hasOwn??Object.prototype.hasOwnProperty.call;var U=e=>e!==null&&typeof e=="object"&&(Object.getPrototypeOf(e)===Object.prototype||Object.getPrototypeOf(e)===null),ct=e=>{for(let t in e)if(C(e,t))return!1;return!0},Y=(e,t)=>{for(let n in e){let r=e[n];U(r)||Array.isArray(r)?Y(r,t):e[n]=t(r)}},Me=e=>{let t={};for(let[n,r]of e){let s=n.split("."),o=s.pop(),i=s.reduce((a,c)=>a[c]??={},t);i[o]=r}return t};var xe=[],Be=[],Oe=0,Le=0,We=0,Ue,j,Ne=0,M=()=>{Oe++},x=()=>{--Oe||(ft(),J())},F=e=>{Ue=j,j=e},P=()=>{j=Ue,Ue=void 0},pe=e=>Ut.bind(0,{previousValue:e,t:e,e:1}),Je=Symbol("computed"),ke=e=>{let t=Jt.bind(0,{e:17,getter:e});return t[Je]=1,t},S=e=>{let t={d:e,e:2};j&&ze(t,j),F(t),M();try{t.d()}finally{x(),P()}return gt.bind(0,t)},ft=()=>{for(;Le"getter"in e?dt(e):mt(e,e.t),dt=e=>{F(e),ht(e);try{let t=e.t;return t!==(e.t=e.getter(t))}finally{P(),yt(e)}},mt=(e,t)=>(e.e=1,e.previousValue!==(e.previousValue=t)),Ke=e=>{let t=e.e;if(!(t&64)){e.e=t|64;let n=e.r;n?Ke(n.o):Be[We++]=e}},pt=(e,t)=>{if(t&16||t&32&&bt(e.s,e)){F(e),ht(e),M();try{e.d()}finally{x(),P(),yt(e)}return}t&32&&(e.e=t&-33);let n=e.s;for(;n;){let r=n.c,s=r.e;s&64&&pt(r,r.e=s&-65),n=n.i}},Ut=(e,...t)=>{if(t.length){if(e.t!==(e.t=t[0])){e.e=17;let r=e.r;return r&&(Kt(r),Oe||ft()),!0}return!1}let n=e.t;if(e.e&16&&mt(e,n)){let r=e.r;r&&Pe(r)}return j&&ze(e,j),n},Jt=e=>{let t=e.e;if(t&16||t&32&&bt(e.s,e)){if(dt(e)){let n=e.r;n&&Pe(n)}}else t&32&&(e.e=t&-33);return j&&ze(e,j),e.t},gt=e=>{let t=e.s;for(;t;)t=Fe(t,e);let n=e.r;n&&Fe(n),e.e=0},ze=(e,t)=>{let n=t.a;if(n&&n.c===e)return;let r=n?n.i:t.s;if(r&&r.c===e){r.m=Ne,t.a=r;return}let s=e.p;if(s&&s.m===Ne&&s.o===t)return;let o=t.a=e.p={m:Ne,c:e,o:t,l:n,i:r,u:s};r&&(r.l=o),n?n.i=o:t.s=o,s?s.n=o:e.r=o},Fe=(e,t=e.o)=>{let n=e.c,r=e.l,s=e.i,o=e.n,i=e.u;if(s?s.l=r:t.a=r,r?r.i=s:t.s=s,o?o.u=i:n.p=i,i)i.n=o;else if(!(n.r=o))if("getter"in n){let a=n.s;if(a){n.e=17;do a=Fe(a,n);while(a)}}else"previousValue"in n||gt(n);return s},Kt=e=>{let t=e.n,n;e:for(;;){let r=e.o,s=r.e;if(s&60?s&12?s&4?!(s&48)&&zt(e,r)?(r.e=s|40,s&=1):s=0:r.e=s&-9|32:s=0:r.e=s|32,s&2&&Ke(r),s&1){let o=r.r;if(o){let i=(e=o).n;i&&(n={t,f:n},t=i);continue}}if(e=t){t=e.n;continue}for(;n;)if(e=n.t,n=n.f,e){t=e.n;continue e}break}},ht=e=>{Ne++,e.a=void 0,e.e=e.e&-57|4},yt=e=>{let t=e.a,n=t?t.i:e.s;for(;n;)n=Fe(n,e);e.e&=-5},bt=(e,t)=>{let n,r=0,s=!1;e:for(;;){let o=e.c,i=o.e;if(t.e&16)s=!0;else if((i&17)===17){if(lt(o)){let a=o.r;a.n&&Pe(a),s=!0}}else if((i&33)===33){(e.n||e.u)&&(n={t:e,f:n}),e=o.s,t=o,++r;continue}if(!s){let a=e.i;if(a){e=a;continue}}for(;r--;){let a=t.r,c=a.n;if(c?(e=n.t,n=n.f):e=a,s){if(lt(t)){c&&Pe(a),t=e.o;continue}s=!1}else t.e&=-33;if(t=e.o,e.i){e=e.i;continue e}}return s}},Pe=e=>{do{let t=e.o,n=t.e;(n&48)===32&&(t.e=n|16,n&2&&Ke(t))}while(e=e.n)},zt=(e,t)=>{let n=t.a;for(;n;){if(n===e)return!0;n=n.l}return!1},oe=e=>{let t=X,n=e.split(".");for(let r of n){if(t==null||!C(t,r))return;t=t[r]}return t},Ce=(e,t="")=>{let n=Array.isArray(e);if(n||U(e)){let r=n?[]:{};for(let o in e)r[o]=pe(Ce(e[o],`${t+o}.`));let s=pe(0);return new Proxy(r,{get(o,i){if(!(i==="toJSON"&&!C(r,i)))return n&&i in Array.prototype?(s(),r[i]):typeof i=="symbol"?r[i]:((!C(r,i)||r[i]()==null)&&(r[i]=pe(""),J(t+i,""),s(s()+1)),r[i]())},set(o,i,a){let c=t+i;if(n&&i==="length"){let l=r[i]-a;if(r[i]=a,l>0){let u={};for(let d=a;d{if(e!==void 0&&t!==void 0&&xe.push([e,t]),!Oe&&xe.length){let n=Me(xe);xe.length=0,document.dispatchEvent(new CustomEvent(Z,{detail:n}))}},O=(e,{ifMissing:t}={})=>{M();for(let n in e)e[n]==null?t||delete X[n]:vt(e[n],n,X,"",t);x()},T=(e,t)=>O(Me(e),t),vt=(e,t,n,r,s)=>{if(U(e)){C(n,t)&&(U(n[t])||Array.isArray(n[t]))||(n[t]={});for(let o in e)e[o]==null?s||delete n[t][o]:vt(e[o],o,n[t],`${r+t}.`,s)}else s&&C(n,t)||(n[t]=e)},ut=e=>typeof e=="string"?RegExp(e.replace(/^\/|\/$/g,"")):e,_=({include:e=/.*/,exclude:t=/(?!)/}={},n=X)=>{let r=ut(e),s=ut(t),o=[],i=[[n,""]];for(;i.length;){let[a,c]=i.pop();for(let l in a){let u=c+l;U(a[l])?i.push([a[l],`${u}.`]):r.test(u)&&!s.test(u)&&o.push([u,oe(u)])}}return Me(o)},X=Ce({});var K=e=>e instanceof HTMLElement||e instanceof SVGElement||e instanceof MathMLElement;var ge=e=>e.replace(/([a-z0-9])([A-Z])/g,"$1-$2").replace(/([a-z])([0-9]+)/gi,"$1-$2").replace(/([0-9]+)([a-z])/gi,"$1-$2").replace(/[\s_]+/g,"-").toLowerCase();var Et=e=>ge(e).replace(/-/g,"_");var ae=e=>{try{return JSON.parse(e)}catch{return Function(`return (${e})`)()}},St={camel:e=>e.replace(/-[a-z]/g,t=>t[1].toUpperCase()),snake:e=>e.replace(/-/g,"_"),pascal:e=>e[0].toUpperCase()+St.camel(e.slice(1))},L=(e,t,n="camel")=>{for(let r of t.get("case")||[n])e=St[r]?.(e)||e;return e},G=e=>`data-${e}`;var Qt="https://data-star.dev/errors",he=(e,t,n={})=>{Object.assign(n,e);let r=new Error,s=Et(t),o=new URLSearchParams({metadata:JSON.stringify(n)}).toString(),i=JSON.stringify(n,null,2);return r.message=`${t}
+More info: ${Qt}/${s}?${o}
+Context: ${i}`,r},ye=new Map,Qe=new Map,At=new Map,Rt=new Proxy({},{get:(e,t)=>ye.get(t)?.apply,has:(e,t)=>ye.has(t),ownKeys:()=>Reflect.ownKeys(ye),set:()=>!1,deleteProperty:()=>!1}),be=new Map,He=[],Ze=new Set,Zt=new WeakSet,p=e=>{He.push(e),He.length===1&&setTimeout(()=>{for(let t of He)Ze.add(t.name),Qe.set(t.name,t);He.length=0,nn(),Ze.clear()})},k=e=>{ye.set(e.name,e)};document.addEventListener(q,e=>{let t=At.get(e.detail.type);t&&t.apply({error:he.bind(0,{plugin:{type:"watcher",name:t.name},element:{id:e.target.id,tag:e.target.tagName}})},e.detail.argsRaw)});var ve=e=>{At.set(e.name,e)},Tt=e=>{for(let t of e){let n=be.get(t);if(n&&be.delete(t))for(let r of n.values())for(let s of r.values())s()}},wt=G("ignore"),Yt=`[${wt}]`,Mt=e=>e.hasAttribute(`${wt}__self`)||!!e.closest(Yt),_e=(e,t)=>{for(let n of e)if(!Mt(n))for(let r in n.dataset)xt(n,r.replace(/[A-Z]/g,"-$&").toLowerCase(),n.dataset[r],t)},Xt=e=>{for(let{target:t,type:n,attributeName:r,addedNodes:s,removedNodes:o}of e)if(n==="childList"){for(let i of o)K(i)&&(Tt([i]),Tt(i.querySelectorAll("*")));for(let i of s)K(i)&&(_e([i]),_e(i.querySelectorAll("*")))}else if(n==="attributes"&&r.startsWith("data-")&&K(t)&&!Mt(t)){let i=r.slice(5),a=t.getAttribute(r);if(a===null){let c=be.get(t);if(c){let l=c.get(i);if(l){for(let u of l.values())u();c.delete(i)}}}else xt(t,i,a)}},en=new MutationObserver(Xt),tn=e=>{let[t,...n]=e.split("__"),[r,s]=t.split(/:(.+)/),o=new Map;for(let i of n){let[a,...c]=i.split(".");o.set(a,new Set(c))}return{pluginName:r,key:s,mods:o}};var nn=(e=document.documentElement,t=!0)=>{K(e)&&_e([e],!0),_e(e.querySelectorAll("*"),!0),t&&(en.observe(e,{subtree:!0,childList:!0,attributes:!0}),Zt.add(e))},xt=(e,t,n,r)=>{{let s=t,{pluginName:o,key:i,mods:a}=tn(s),c=Qe.get(o);if((!r||Ze.has(o))&&c){let l={el:e,rawKey:s,mods:a,error:he.bind(0,{plugin:{type:"attribute",name:c.name},element:{id:e.id,tag:e.tagName},expression:{rawKey:s,key:i,value:n}}),key:i,value:n,loadedPluginNames:{actions:new Set(ye.keys()),attributes:new Set(Qe.keys())},rx:void 0},u=c.requirement&&(typeof c.requirement=="string"?c.requirement:c.requirement.key)||"allowed",d=c.requirement&&(typeof c.requirement=="string"?c.requirement:c.requirement.value)||"allowed",h=i!=null&&i!=="",f=n!=null&&n!=="";if(h){if(u==="denied")throw l.error("KeyNotAllowed")}else if(u==="must")throw l.error("KeyRequired");if(f){if(d==="denied")throw l.error("ValueNotAllowed")}else if(d==="must")throw l.error("ValueRequired");if(u==="exclusive"||d==="exclusive"){if(h&&f)throw l.error("KeyAndValueProvided");if(!h&&!f)throw l.error("KeyOrValueRequired")}let m=new Map;if(f){let v;l.rx=(...A)=>(v||(v=rn(n,{returnsValue:c.returnsValue,argNames:c.argNames,cleanups:m})),v(e,...A))}let y=c.apply(l);y&&m.set("attribute",y);let b=be.get(e);if(b){let v=b.get(s);if(v)for(let A of v.values())A()}else b=new Map,be.set(e,b);b.set(s,m)}}},rn=(e,{returnsValue:t=!1,argNames:n=[],cleanups:r=new Map}={})=>{let s="";if(t){let c=/(\/(\\\/|[^/])*\/|"(\\"|[^"])*"|'(\\'|[^'])*'|`(\\`|[^`])*`|\(\s*((function)\s*\(\s*\)|(\(\s*\))\s*=>)\s*(?:\{[\s\S]*?\}|[^;){]*)\s*\)\s*\(\s*\)|[^;])+/gm,l=e.trim().match(c);if(l){let u=l.length-1,d=l[u].trim();d.startsWith("return")||(l[u]=`return (${d});`),s=l.join(`;
+`)}}else s=e.trim();let o=new Map,i=RegExp(`(?:${je})(.*?)(?:${Ge})`,"gm"),a=0;for(let c of s.matchAll(i)){let l=c[1],u=`__escaped${a++}`;o.set(u,l),s=s.replace(je+l+Ge,u)}s=s.replace(/\$\['([a-zA-Z_$\d][\w$]*)'\]/g,"$$$1").replace(/\$([a-zA-Z_\d]\w*(?:[.-]\w+)*)/g,(c,l)=>l.split(".").reduce((u,d)=>`${u}['${d}']`,"$")),s=s.replaceAll(/@([A-Za-z_$][\w$]*)\(/g,'__action("$1",evt,');for(let[c,l]of o)s=s.replace(c,l);try{let c=Function("el","$","__action","evt",...n,s);return(l,...u)=>{let d=(h,f,...m)=>{let y=he.bind(0,{plugin:{type:"action",name:h},element:{id:l.id,tag:l.tagName},expression:{fnContent:s,value:e}}),b=Rt[h];if(b)return b({el:l,evt:f,error:y,cleanups:r},...m);throw y("UndefinedAction")};try{return c(l,X,d,void 0,...u)}catch(h){throw console.error(h),he({element:{id:l.id,tag:l.tagName},expression:{fnContent:s,value:e},error:h.message},"ExecuteExpression")}}}catch(c){throw console.error(c),he({expression:{fnContent:s,value:e},error:c.message},"GenerateExpression")}};k({name:"peek",apply(e,t){F();try{return t()}finally{P()}}});k({name:"setAll",apply(e,t,n){F();let r=_(n);Y(r,()=>t),O(r),P()}});k({name:"toggleAll",apply(e,t){F();let n=_(t);Y(n,r=>!r),O(n),P()}});var Ee=(e,t,n=!0)=>k({name:e,apply:async({el:r,evt:s,error:o,cleanups:i},a,{selector:c,headers:l,contentType:u="json",filterSignals:{include:d=/.*/,exclude:h=/(^|\.)_/}={},openWhenHidden:f=n,payload:m,requestCancellation:y="auto",retry:b="auto",retryInterval:v=1e3,retryScaler:A=2,retryMaxWaitMs:I=3e4,retryMaxCount:ne=10}={})=>{let de=y instanceof AbortController?y:new AbortController;y==="auto"&&(i.get(`@${e}`)?.(),i.set(`@${e}`,async()=>{de.abort(),await Promise.resolve()}));let D=null;try{if(!a?.length)throw o("FetchNoUrlProvided",{action:k});let V={Accept:"text/event-stream, text/html, application/json","Datastar-Request":!0};u==="json"&&(V["Content-Type"]="application/json");let Ae=Object.assign({},V,l),re={method:t,headers:Ae,openWhenHidden:f,retry:b,retryInterval:v,retryScaler:A,retryMaxWaitMs:I,retryMaxCount:ne,signal:de.signal,onopen:async g=>{g.status>=400&&ee(sn,r,{status:g.status.toString()})},onmessage:g=>{if(!g.event.startsWith("datastar"))return;let B=g.event,E={};for(let R of g.data.split(`
+`)){let w=R.indexOf(" "),W=R.slice(0,w),De=R.slice(w+1);(E[W]||=[]).push(De)}let N=Object.fromEntries(Object.entries(E).map(([R,w])=>[R,w.join(`
+`)]));ee(B,r,N)},onerror:g=>{if(Lt(g))throw g("FetchExpectedTextEventStream",{url:a});g&&(console.error(g.message),ee(on,r,{message:g.message}))}},se=new URL(a,document.baseURI),ie=new URLSearchParams(se.search);if(u==="json"){F(),m=m!==void 0?m:_({include:d,exclude:h}),P();let g=JSON.stringify(m);t==="GET"?ie.set("datastar",g):re.body=g}else if(u==="form"){let g=c?document.querySelector(c):r.closest("form");if(!g)throw o("FetchFormNotFound",{action:k,selector:c});if(!g.noValidate&&!g.checkValidity()){g.reportValidity();return}let B=new FormData(g),E=r;if(r===g&&s instanceof SubmitEvent)E=s.submitter;else{let w=W=>W.preventDefault();g.addEventListener("submit",w),D=()=>{g.removeEventListener("submit",w)}}if(E instanceof HTMLButtonElement){let w=E.getAttribute("name");w&&B.append(w,E.value)}let N=g.getAttribute("enctype")==="multipart/form-data";N||(Ae["Content-Type"]="application/x-www-form-urlencoded");let R=new URLSearchParams(B);if(t==="GET")for(let[w,W]of R)ie.append(w,W);else N?re.body=B:re.body=R}else throw o("FetchInvalidContentType",{action:k,contentType:u});ee(Ye,r,{}),se.search=ie.toString();try{await dn(se.toString(),r,re)}catch(g){if(!Lt(g))throw o("FetchFailed",{method:t,url:a,error:g.message})}}finally{ee(Xe,r,{}),D?.(),i.delete(`@${e}`)}}});Ee("get","GET",!1);Ee("patch","PATCH");Ee("post","POST");Ee("put","PUT");Ee("delete","DELETE");var Ye="started",Xe="finished",sn="error",on="retrying",an="retries-failed",ee=(e,t,n)=>document.dispatchEvent(new CustomEvent(q,{detail:{type:e,el:t,argsRaw:n}})),Lt=e=>`${e}`.includes("text/event-stream"),cn=async(e,t)=>{let n=e.getReader(),r=await n.read();for(;!r.done;)t(r.value),r=await n.read()},ln=e=>{let t,n,r,s=!1;return o=>{t?t=fn(t,o):(t=o,n=0,r=-1);let i=t.length,a=0;for(;n{let r=Nt(),s=new TextDecoder;return(o,i)=>{if(!o.length)n?.(r),r=Nt();else if(i>0){let a=s.decode(o.subarray(0,i)),c=i+(o[i+1]===32?2:1),l=s.decode(o.subarray(c));switch(a){case"data":r.data=r.data?`${r.data}
+${l}`:l;break;case"event":r.event=l;break;case"id":e(r.id=l);break;case"retry":{let u=+l;Number.isNaN(u)||t(r.retry=u);break}}}}},fn=(e,t)=>{let n=new Uint8Array(e.length+t.length);return n.set(e),n.set(t,e.length),n},Nt=()=>({data:"",event:"",id:"",retry:void 0}),dn=(e,t,{signal:n,headers:r,onopen:s,onmessage:o,onclose:i,onerror:a,openWhenHidden:c,fetch:l,retry:u="auto",retryInterval:d=1e3,retryScaler:h=2,retryMaxWaitMs:f=3e4,retryMaxCount:m=10,responseOverrides:y,...b})=>new Promise((v,A)=>{let I={...r},ne,de=()=>{ne.abort(),document.hidden||g()};c||document.addEventListener("visibilitychange",de);let D,V=()=>{document.removeEventListener("visibilitychange",de),clearTimeout(D),ne.abort()};n?.addEventListener("abort",()=>{V(),v()});let Ae=l||window.fetch,re=s||(()=>{}),se=0,ie=d,g=async()=>{ne=new AbortController;let B=ne.signal;try{let E=await Ae(e,{...b,headers:I,signal:B});await re(E);let N=async($,me,$e,Re,...Wt)=>{let ot={[$e]:await me.text()};for(let Ie of Wt){let qe=me.headers.get(`datastar-${ge(Ie)}`);if(Re){let we=Re[Ie];we&&(qe=typeof we=="string"?we:JSON.stringify(we))}qe&&(ot[Ie]=qe)}ee($,t,ot),V(),v()},R=E.status,w=R===204,W=R>=300&&R<400,De=R>=400&&R<600;if(R!==200){if(i?.(),u!=="never"&&!w&&!W&&(u==="always"||u==="error"&&De)){clearTimeout(D),D=setTimeout(g,d);return}V(),v();return}se=0,d=ie;let Ve=E.headers.get("Content-Type");if(Ve?.includes("text/html"))return await N("datastar-patch-elements",E,"elements",y,"selector","mode","namespace","useViewTransition");if(Ve?.includes("application/json"))return await N("datastar-patch-signals",E,"signals",y,"onlyIfMissing");if(Ve?.includes("text/javascript")){let $=document.createElement("script"),me=E.headers.get("datastar-script-attributes");if(me)for(let[$e,Re]of Object.entries(JSON.parse(me)))$.setAttribute($e,Re);$.textContent=await E.text(),document.head.appendChild($),V();return}if(await cn(E.body,ln(un($=>{$?I["last-event-id"]=$:delete I["last-event-id"]},$=>{ie=d=$},o))),i?.(),u==="always"&&!W){clearTimeout(D),D=setTimeout(g,d);return}V(),v()}catch(E){if(!B.aborted)try{let N=a?.(E)||d;clearTimeout(D),D=setTimeout(g,N),d=Math.min(d*h,f),++se>=m?(ee(an,t,{}),V(),A("Max retries reached.")):console.error(`Datastar failed to reach ${e.toString()} retrying in ${N}ms.`)}catch(N){V(),A(N)}}};g()});p({name:"attr",requirement:{value:"must"},returnsValue:!0,apply({el:e,key:t,rx:n}){let r=(a,c)=>{c===""||c===!0?e.setAttribute(a,""):c===!1||c==null?e.removeAttribute(a):typeof c=="string"?e.setAttribute(a,c):e.setAttribute(a,JSON.stringify(c))},s=t?()=>{o.disconnect();let a=n();r(t,a),o.observe(e,{attributeFilter:[t]})}:()=>{o.disconnect();let a=n(),c=Object.keys(a);for(let l of c)r(l,a[l]);o.observe(e,{attributeFilter:c})},o=new MutationObserver(s),i=S(s);return()=>{o.disconnect(),i()}}});var mn=/^data:(?[^;]+);base64,(?.*)$/,Ct=Symbol("empty"),Ft=G("bind");p({name:"bind",requirement:"exclusive",apply({el:e,key:t,mods:n,value:r,error:s}){let o=t!=null?L(t,n):r,i=(f,m)=>m==="number"?+f.value:f.value,a=f=>{e.value=`${f}`};if(e instanceof HTMLInputElement)switch(e.type){case"range":case"number":i=(f,m)=>m==="string"?f.value:+f.value;break;case"checkbox":i=(f,m)=>f.value!=="on"?m==="boolean"?f.checked:f.checked?f.value:"":m==="string"?f.checked?f.value:"":f.checked,a=f=>{e.checked=typeof f=="string"?f===e.value:f};break;case"radio":e.getAttribute("name")?.length||e.setAttribute("name",o),i=(f,m)=>f.checked?m==="number"?+f.value:f.value:Ct,a=f=>{e.checked=f===(typeof f=="number"?+e.value:e.value)};break;case"file":{let f=()=>{let m=[...e.files||[]],y=[];Promise.all(m.map(b=>new Promise(v=>{let A=new FileReader;A.onload=()=>{if(typeof A.result!="string")throw s("InvalidFileResultType",{resultType:typeof A.result});let I=A.result.match(mn);if(!I?.groups)throw s("InvalidDataUri",{result:A.result});y.push({name:b.name,contents:I.groups.contents,mime:I.groups.mime})},A.onloadend=()=>v(),A.readAsDataURL(b)}))).then(()=>{T([[o,y]])})};return e.addEventListener("change",f),e.addEventListener("input",f),()=>{e.removeEventListener("change",f),e.removeEventListener("input",f)}}}else if(e instanceof HTMLSelectElement){if(e.multiple){let f=new Map;i=m=>[...m.selectedOptions].map(y=>{let b=f.get(y.value);return b==="string"||b==null?y.value:+y.value}),a=m=>{for(let y of e.options)m.includes(y.value)?(f.set(y.value,"string"),y.selected=!0):m.includes(+y.value)?(f.set(y.value,"number"),y.selected=!0):y.selected=!1}}}else e instanceof HTMLTextAreaElement||(i=f=>"value"in f?f.value:f.getAttribute("value"),a=f=>{"value"in e?e.value=f:e.setAttribute("value",f)});let c=oe(o),l=typeof c,u=o;if(Array.isArray(c)&&!(e instanceof HTMLSelectElement&&e.multiple)){let f=t||r,m=document.querySelectorAll(`[${Ft}\\:${CSS.escape(f)}],[${Ft}="${CSS.escape(f)}"]`),y=[],b=0;for(let v of m){if(y.push([`${u}.${b}`,i(v,"none")]),e===v)break;b++}T(y,{ifMissing:!0}),u=`${u}.${b}`}else T([[u,i(e,l)]],{ifMissing:!0});let d=()=>{let f=oe(u);if(f!=null){let m=i(e,typeof f);m!==Ct&&T([[u,m]])}};e.addEventListener("input",d),e.addEventListener("change",d);let h=S(()=>{a(oe(u))});return()=>{h(),e.removeEventListener("input",d),e.removeEventListener("change",d)}}});p({name:"class",requirement:{value:"must"},returnsValue:!0,apply({key:e,el:t,mods:n,rx:r}){e&&=L(e,n,"kebab");let s,o=()=>{i.disconnect(),s=e?{[e]:r()}:r();for(let c in s){let l=c.split(/\s+/).filter(u=>u.length>0);if(s[c])for(let u of l)t.classList.contains(u)||t.classList.add(u);else for(let u of l)t.classList.contains(u)&&t.classList.remove(u)}i.observe(t,{attributeFilter:["class"]})},i=new MutationObserver(o),a=S(o);return()=>{i.disconnect(),a();for(let c in s){let l=c.split(/\s+/).filter(u=>u.length>0);for(let u of l)t.classList.remove(u)}}}});p({name:"computed",requirement:{value:"must"},returnsValue:!0,apply({key:e,mods:t,rx:n,error:r}){if(e)T([[L(e,t),ke(n)]]);else{let s=Object.assign({},n());Y(s,o=>{if(typeof o=="function")return ke(o);throw r("ComputedExpectedFunction")}),O(s)}}});p({name:"effect",requirement:{key:"denied",value:"must"},apply:({rx:e})=>S(e)});p({name:"indicator",requirement:"exclusive",apply({el:e,key:t,mods:n,value:r}){let s=t!=null?L(t,n):r;T([[s,!1]]);let o=i=>{let{type:a,el:c}=i.detail;if(c===e)switch(a){case Ye:T([[s,!0]]);break;case Xe:T([[s,!1]]);break}};return document.addEventListener(q,o),()=>{T([[s,!1]]),document.removeEventListener(q,o)}}});var z=e=>{if(!e||e.size<=0)return 0;for(let t of e){if(t.endsWith("ms"))return+t.replace("ms","");if(t.endsWith("s"))return+t.replace("s","")*1e3;try{return Number.parseFloat(t)}catch{}}return 0},te=(e,t,n=!1)=>e?e.has(t.toLowerCase()):n;var et=(e,t)=>(...n)=>{setTimeout(()=>{e(...n)},t)},Pt=(e,t,n=!0,r=!1,s=!1)=>{let o=null,i=0;return(...a)=>{n&&!i?(e(...a),o=null):o=a,(!i||s)&&(i&&clearTimeout(i),i=setTimeout(()=>{r&&o!==null&&e(...o),o=null,i=0},t))}},ce=(e,t)=>{let n=t.get("delay");if(n){let o=z(n);e=et(e,o)}let r=t.get("debounce");if(r){let o=z(r),i=te(r,"leading",!1),a=!te(r,"notrailing",!1);e=Pt(e,o,i,a,!0)}let s=t.get("throttle");if(s){let o=z(s),i=!te(s,"noleading",!1),a=te(s,"trailing",!1);e=Pt(e,o,i,a)}return e};var tt=!!document.startViewTransition,Q=(e,t)=>{if(t.has("viewtransition")&&tt){let n=e;e=(...r)=>document.startViewTransition(()=>n(...r))}return e};p({name:"init",requirement:{key:"denied",value:"must"},apply({rx:e,mods:t}){let n=()=>{M(),e(),x()};n=Q(n,t);let r=0,s=t.get("delay");s&&(r=z(s),r>0&&(n=et(n,r))),n()}});p({name:"json-signals",requirement:{key:"denied"},apply({el:e,value:t,mods:n}){let r=n.has("terse")?0:2,s={};t&&(s=ae(t));let o=()=>{i.disconnect(),e.textContent=JSON.stringify(_(s),null,r),i.observe(e,{childList:!0,characterData:!0,subtree:!0})},i=new MutationObserver(o),a=S(o);return()=>{i.disconnect(),a()}}});p({name:"on",requirement:"must",argNames:["evt"],apply({el:e,key:t,mods:n,rx:r}){let s=e;n.has("window")&&(s=window);let o=c=>{c&&(n.has("prevent")&&c.preventDefault(),n.has("stop")&&c.stopPropagation()),M(),r(c),x()};o=Q(o,n),o=ce(o,n);let i={capture:n.has("capture"),passive:n.has("passive"),once:n.has("once")};if(n.has("outside")){s=document;let c=o;o=l=>{e.contains(l?.target)||c(l)}}let a=L(t,n,"kebab");if((a===q||a===Z)&&(s=document),e instanceof HTMLFormElement&&a==="submit"){let c=o;o=l=>{l?.preventDefault(),c(l)}}return s.addEventListener(a,o,i),()=>{s.removeEventListener(a,o)}}});var Ot=(e,t,n)=>Math.max(t,Math.min(n,e));var nt=new WeakSet;p({name:"on-intersect",requirement:{key:"denied",value:"must"},apply({el:e,mods:t,rx:n}){let r=()=>{M(),n(),x()};r=Q(r,t),r=ce(r,t);let s={threshold:0};t.has("full")?s.threshold=1:t.has("half")?s.threshold=.5:t.get("threshold")&&(s.threshold=Ot(Number(t.get("threshold")),0,100)/100);let o=t.has("exit"),i=new IntersectionObserver(a=>{for(let c of a)c.isIntersecting!==o&&(r(),i&&nt.has(e)&&i.disconnect())},s);return i.observe(e),t.has("once")&&nt.add(e),()=>{t.has("once")||nt.delete(e),i&&(i.disconnect(),i=null)}}});p({name:"on-interval",requirement:{key:"denied",value:"must"},apply({mods:e,rx:t}){let n=()=>{M(),t(),x()};n=Q(n,e);let r=1e3,s=e.get("duration");s&&(r=z(s),te(s,"leading",!1)&&n());let o=setInterval(n,r);return()=>{clearInterval(o)}}});p({name:"on-signal-patch",requirement:{value:"must"},argNames:["patch"],returnsValue:!0,apply({el:e,key:t,mods:n,rx:r,error:s}){if(t&&t!=="filter")throw s("KeyNotAllowed");let o=G(`${this.name}-filter`),i=e.getAttribute(o),a={};i&&(a=ae(i));let c=!1,l=ce(u=>{if(c)return;let d=_(a,u.detail);if(!ct(d)){c=!0,M();try{r(d)}finally{x(),c=!1}}},n);return document.addEventListener(Z,l),()=>{document.removeEventListener(Z,l)}}});p({name:"ref",requirement:"exclusive",apply({el:e,key:t,mods:n,value:r}){let s=t!=null?L(t,n):r;T([[s,e]])}});var kt="none",Ht="display";p({name:"show",requirement:{key:"denied",value:"must"},returnsValue:!0,apply({el:e,rx:t}){let n=()=>{r.disconnect(),t()?e.style.display===kt&&e.style.removeProperty(Ht):e.style.setProperty(Ht,kt),r.observe(e,{attributeFilter:["style"]})},r=new MutationObserver(n),s=S(n);return()=>{r.disconnect(),s()}}});p({name:"signals",returnsValue:!0,apply({key:e,mods:t,rx:n}){let r=t.has("ifmissing");if(e)e=L(e,t),T([[e,n?.()]],{ifMissing:r});else{let s=Object.assign({},n?.());O(s,{ifMissing:r})}}});p({name:"style",requirement:{value:"must"},returnsValue:!0,apply({key:e,el:t,rx:n}){let{style:r}=t,s=new Map,o=(l,u)=>{let d=s.get(l);!u&&u!==0?d!==void 0&&(d?r.setProperty(l,d):r.removeProperty(l)):(d===void 0&&s.set(l,r.getPropertyValue(l)),r.setProperty(l,String(u)))},i=()=>{if(a.disconnect(),e)o(e,n());else{let l=n();for(let[u,d]of s)u in l||(d?r.setProperty(u,d):r.removeProperty(u));for(let u in l)o(ge(u),l[u])}a.observe(t,{attributeFilter:["style"]})},a=new MutationObserver(i),c=S(i);return()=>{a.disconnect(),c();for(let[l,u]of s)u?r.setProperty(l,u):r.removeProperty(l)}}});p({name:"text",requirement:{key:"denied",value:"must"},returnsValue:!0,apply({el:e,rx:t}){let n=()=>{r.disconnect(),e.textContent=`${t()}`,r.observe(e,{childList:!0,characterData:!0,subtree:!0})},r=new MutationObserver(n),s=S(n);return()=>{r.disconnect(),s()}}});var _t=(e,t)=>e.includes(t),pn=["remove","outer","inner","replace","prepend","append","before","after"],gn=["html","svg","mathml"];ve({name:"datastar-patch-elements",apply(e,{selector:t="",mode:n="outer",namespace:r="html",useViewTransition:s="",elements:o=""}){if(!_t(pn,n))throw e.error("PatchElementsInvalidMode",{mode:n});if(!t&&n!=="outer"&&n!=="replace")throw e.error("PatchElementsExpectedSelector");if(!_t(gn,r))throw e.error("PatchElementsInvalidNamespace",{namespace:r});let i={selector:t,mode:n,namespace:r,useViewTransition:s.trim()==="true",elements:o};tt&&s?document.startViewTransition(()=>Dt(e,i)):Dt(e,i)}});var Dt=({error:e},{selector:t,mode:n,namespace:r,elements:s})=>{let o=s.replace(/]*>|>)([\s\S]*?)<\/svg>/gim,""),i=/<\/html>/.test(o),a=/<\/head>/.test(o),c=/<\/body>/.test(o),l=r==="svg"?"svg":r==="mathml"?"math":"",u=l?`<${l}>${s}${l}>`:s,d=new DOMParser().parseFromString(i||a||c?s:`${u} `,"text/html"),h=document.createDocumentFragment();if(i)h.appendChild(d.documentElement);else if(a&&c)h.appendChild(d.head),h.appendChild(d.body);else if(a)h.appendChild(d.head);else if(c)h.appendChild(d.body);else if(l){let f=d.querySelector("template").content.querySelector(l);for(let m of f.childNodes)h.appendChild(m)}else h=d.querySelector("template").content;if(!t&&(n==="outer"||n==="replace"))for(let f of h.children){let m;if(f instanceof HTMLHtmlElement)m=document.documentElement;else if(f instanceof HTMLBodyElement)m=document.body;else if(f instanceof HTMLHeadElement)m=document.head;else if(m=document.getElementById(f.id),!m){console.warn(e("PatchElementsNoTargetsFound"),{element:{id:f.id}});continue}$t(n,f,[m])}else{let f=document.querySelectorAll(t);if(!f.length){console.warn(e("PatchElementsNoTargetsFound"),{selector:t});return}$t(n,h,f)}},st=new WeakSet;for(let e of document.querySelectorAll("script"))st.add(e);var jt=e=>{let t=e instanceof HTMLScriptElement?[e]:e.querySelectorAll("script");for(let n of t)if(!st.has(n)){let r=document.createElement("script");for(let{name:s,value:o}of n.attributes)r.setAttribute(s,o);r.text=n.text,n.replaceWith(r),st.add(r)}},Vt=(e,t,n)=>{for(let r of e){let s=t.cloneNode(!0);jt(s),r[n](s)}},$t=(e,t,n)=>{switch(e){case"remove":for(let r of n)r.remove();break;case"outer":case"inner":for(let r of n)yn(r,t.cloneNode(!0),e),jt(r);break;case"replace":Vt(n,t,"replaceWith");break;case"prepend":case"append":case"before":case"after":Vt(n,t,e)}},H=new Map,ue=new Set,le=new Map,Se=new Set,fe=document.createElement("div");fe.hidden=!0;var Te=G("ignore-morph"),hn=`[${Te}]`,yn=(e,t,n="outer")=>{if(K(e)&&K(t)&&e.hasAttribute(Te)&&t.hasAttribute(Te)||e.parentElement?.closest(hn))return;let r=document.createElement("div");r.append(t),document.body.insertAdjacentElement("afterend",fe);let s=e.querySelectorAll("[id]");for(let{id:a,tagName:c}of s)le.has(a)?Se.add(a):le.set(a,c);e instanceof Element&&e.id&&(le.has(e.id)?Se.add(e.id):le.set(e.id,e.tagName)),ue.clear();let o=r.querySelectorAll("[id]");for(let{id:a,tagName:c}of o)ue.has(a)?Se.add(a):le.get(a)===c&&ue.add(a);for(let a of Se)ue.delete(a);le.clear(),Se.clear(),H.clear();let i=n==="outer"?e.parentElement:e;qt(i,s),qt(r,o),Gt(i,r,n==="outer"?e:null,e.nextSibling),fe.remove()},Gt=(e,t,n=null,r=null)=>{e instanceof HTMLTemplateElement&&t instanceof HTMLTemplateElement&&(e=e.content,t=t.content),n??=e.firstChild;for(let s of t.childNodes){if(n&&n!==r){let o=bn(s,n,r);if(o){if(o!==n){let i=n;for(;i&&i!==o;){let a=i;i=i.nextSibling,it(a)}}rt(o,s),n=o.nextSibling;continue}}if(s instanceof Element&&ue.has(s.id)){let o=document.getElementById(s.id),i=o;for(;i=i.parentNode;){let a=H.get(i);a&&(a.delete(s.id),a.size||H.delete(i))}Bt(e,o,n),rt(o,s),n=o.nextSibling;continue}if(H.has(s)){let o=s.namespaceURI,i=s.tagName,a=o&&o!=="http://www.w3.org/1999/xhtml"?document.createElementNS(o,i):document.createElement(i);e.insertBefore(a,n),rt(a,s),n=a.nextSibling}else{let o=document.importNode(s,!0);e.insertBefore(o,n),n=o.nextSibling}}for(;n&&n!==r;){let s=n;n=n.nextSibling,it(s)}},bn=(e,t,n)=>{let r=null,s=e.nextSibling,o=0,i=0,a=H.get(e)?.size||0,c=t;for(;c&&c!==n;){if(It(c,e)){let l=!1,u=H.get(c),d=H.get(e);if(d&&u){for(let h of u)if(d.has(h)){l=!0;break}}if(l)return c;if(!r&&!H.has(c)){if(!a)return c;r=c}}if(i+=H.get(c)?.size||0,i>a)break;r===null&&s&&It(c,s)&&(o++,s=s.nextSibling,o>=2&&(r=void 0)),c=c.nextSibling}return r||null},It=(e,t)=>e.nodeType===t.nodeType&&e.tagName===t.tagName&&(!e.id||e.id===t.id),it=e=>{H.has(e)?Bt(fe,e,null):e.parentNode?.removeChild(e)},Bt=it.call.bind(fe.moveBefore??fe.insertBefore),vn=G("preserve-attr"),rt=(e,t)=>{let n=t.nodeType;if(n===1){let r=e,s=t,o=r.hasAttribute("data-scope-children");if(r.hasAttribute(Te)&&s.hasAttribute(Te))return e;r instanceof HTMLInputElement&&s instanceof HTMLInputElement&&s.type!=="file"?s.getAttribute("value")!==r.getAttribute("value")&&(r.value=s.getAttribute("value")??""):r instanceof HTMLTextAreaElement&&s instanceof HTMLTextAreaElement&&(s.value!==r.value&&(r.value=s.value),r.firstChild&&r.firstChild.nodeValue!==s.value&&(r.firstChild.nodeValue=s.value));let i=(t.getAttribute(vn)??"").split(" ");for(let{name:a,value:c}of s.attributes)r.getAttribute(a)!==c&&!i.includes(a)&&r.setAttribute(a,c);for(let a=r.attributes.length-1;a>=0;a--){let{name:c}=r.attributes[a];!s.hasAttribute(c)&&!i.includes(c)&&r.removeAttribute(c)}o&&!r.hasAttribute("data-scope-children")&&r.setAttribute("data-scope-children",""),r.isEqualNode(s)||Gt(r,s),o&&r.dispatchEvent(new CustomEvent("datastar:scope-children",{bubbles:!1}))}return(n===8||n===3)&&e.nodeValue!==t.nodeValue&&(e.nodeValue=t.nodeValue),e},qt=(e,t)=>{for(let n of t)if(ue.has(n.id)){let r=n;for(;r&&r!==e;){let s=H.get(r);s||(s=new Set,H.set(r,s)),s.add(n.id),r=r.parentElement}}};ve({name:"datastar-patch-signals",apply({error:e},{signals:t,onlyIfMissing:n}){if(t){let r=n?.trim()==="true";O(ae(t),{ifMissing:r})}else throw e("PatchSignalsExpectedSignals")}});export{k as action,Rt as actions,p as attribute,M as beginBatch,ke as computed,S as effect,x as endBatch,_ as filtered,oe as getPath,O as mergePatch,T as mergePaths,X as root,pe as signal,F as startPeeking,P as stopPeeking,ve as watcher};
+//# sourceMappingURL=datastar.js.map
diff --git a/features/auth/pages/login_templ.go b/features/auth/pages/login_templ.go
index 3640508..5d0cec4 100644
--- a/features/auth/pages/login_templ.go
+++ b/features/auth/pages/login_templ.go
@@ -46,33 +46,33 @@ func LoginPage() templ.Component {
}()
}
ctx = templ.InitializeContext(ctx)
- templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "Login Sign in to your account
")
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
diff --git a/features/auth/pages/register_templ.go b/features/auth/pages/register_templ.go
index 61c8f21..7486adb 100644
--- a/features/auth/pages/register_templ.go
+++ b/features/auth/pages/register_templ.go
@@ -46,33 +46,33 @@ func RegisterPage() templ.Component {
}()
}
ctx = templ.InitializeContext(ctx)
- templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 1, "Register Create a new account
")
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
diff --git a/features/auth/routes.go b/features/auth/routes.go
index ac50938..16e5e85 100644
--- a/features/auth/routes.go
+++ b/features/auth/routes.go
@@ -8,11 +8,9 @@ import (
"github.com/ryanhamamura/c4/db/repository"
)
-func SetupRoutes(router chi.Router, queries *repository.Queries, sessions *scs.SessionManager) error {
+func SetupRoutes(router chi.Router, queries *repository.Queries, sessions *scs.SessionManager) {
router.Get("/login", HandleLoginPage())
router.Get("/register", HandleRegisterPage())
router.Post("/auth/login", HandleLogin(queries, sessions))
router.Post("/auth/register", HandleRegister(queries, sessions))
-
- return nil
}
diff --git a/features/c4game/routes.go b/features/c4game/routes.go
index 4cc4641..2ffd2dc 100644
--- a/features/c4game/routes.go
+++ b/features/c4game/routes.go
@@ -16,7 +16,7 @@ func SetupRoutes(
nc *nats.Conn,
sessions *scs.SessionManager,
queries *repository.Queries,
-) error {
+) {
router.Route("/games/{id}", func(r chi.Router) {
r.Get("/", HandleGamePage(store, sessions, queries))
r.Get("/events", HandleGameEvents(store, nc, sessions, queries))
@@ -25,6 +25,4 @@ func SetupRoutes(
r.Post("/join", HandleSetNickname(store, sessions))
r.Post("/rematch", HandleRematch(store, sessions))
})
-
- return nil
}
diff --git a/features/common/layouts/base_templ.go b/features/common/layouts/base_templ.go
index 2f5437f..083a510 100644
--- a/features/common/layouts/base_templ.go
+++ b/features/common/layouts/base_templ.go
@@ -44,7 +44,7 @@ func Base(title string) templ.Component {
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
- templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, " ")
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 2, " ")
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
diff --git a/features/lobby/pages/lobby_templ.go b/features/lobby/pages/lobby_templ.go
index ffae496..9e11bd7 100644
--- a/features/lobby/pages/lobby_templ.go
+++ b/features/lobby/pages/lobby_templ.go
@@ -100,7 +100,7 @@ func LobbyPage(data LobbyData) templ.Component {
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
- templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "")
+ templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 7, "")
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
@@ -108,7 +108,7 @@ func LobbyPage(data LobbyData) templ.Component {
if templ_7745c5c3_Err != nil {
return templ_7745c5c3_Err
}
- templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 8, " ~~~~