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] 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, "
") + 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, "") + 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, "
") + 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.

Login Continue as 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, "
") + if templ_7745c5c3_Err != nil { + return templ_7745c5c3_Err + } + } else { + templ_7745c5c3_Err = templruntime.WriteString(templ_7745c5c3_Buffer, 5, "
Playing as guest. Login or Register to save your games.
") + 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, "

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, "
") + 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 + } + } + 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 + } + } + 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 + } + } + 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, "
") + 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, "") + 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, "
") + 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+"')"), - ), - ) -}