aafdd46a4b
Server-rendered admin console in the backend at /_gm (internal/adminconsole), fronted on the gateway's public listener by Basic-Auth + a verbatim reverse proxy (mounted on the edge mux below the h2c wrap). A same-origin check guards its POSTs; no operator identity is tracked. This supersedes the Stage 6 gateway-fronts- /api/v1/admin model: GATEWAY_ADMIN_ADDR and the backend /api/v1/admin ping are dropped and gateway/internal/admin is repurposed to the verbatim proxy. - Complaints: migration 00008 (+ jetgen) adds disposition/resolution_note/ resolved_at/applied_in_version + the deferred status CHECK; resolution feeds a query-derived pending dictionary-change pipeline (marked applied after a reload). - Dictionary hot-reload: per-version subdir BACKEND_DICT_DIR/<version>/ via the new Registry.LoadAvailable; engine.OpenWithVersions restores resident versions on restart. Partially addresses TODO-2. - Broadcasts: a backend Telegram-connector client (internal/connector, BACKEND_CONNECTOR_ADDR) for SendToUser / SendToGameChannel (discharges the Stage 9 forward-note). - Admin reads: account.ListAccounts/CountAccounts/Identities and game.ListGames/CountGames/GameByID/ListComplaints/GetComplaint/CountComplaints/ ResolveComplaint/DictionaryChanges/MarkChangesApplied. - Tests: adminconsole render, engine reload, same-origin guard, gateway verbatim proxy + h2c console mount, inttest complaint pipeline + list/count + /_gm console. - Docs: PLAN (Stage 10 done + refinements + TODO-2), ARCHITECTURE §1/§5/§6/§12/§13, FUNCTIONAL (+_ru), TESTING, backend/gateway READMEs.
81 lines
2.2 KiB
Go
81 lines
2.2 KiB
Go
package admin_test
|
|
|
|
import (
|
|
"io"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
|
|
"scrabble/gateway/internal/admin"
|
|
)
|
|
|
|
// newAdmin fronts a fake backend with the admin proxy. The fake backend records the
|
|
// path it receives so a test can assert the proxy forwards /_gm verbatim.
|
|
func newAdmin(t *testing.T) (front *httptest.Server, gotPath *string, cleanup func()) {
|
|
t.Helper()
|
|
var path string
|
|
backend := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
path = r.URL.Path
|
|
_, _ = w.Write([]byte("console"))
|
|
}))
|
|
proxy, err := admin.NewProxy(backend.URL, "ops", "secret", nil)
|
|
if err != nil {
|
|
t.Fatalf("new proxy: %v", err)
|
|
}
|
|
front = httptest.NewServer(proxy)
|
|
return front, &path, func() { front.Close(); backend.Close() }
|
|
}
|
|
|
|
func TestAdminRejectsMissingCredentials(t *testing.T) {
|
|
front, _, cleanup := newAdmin(t)
|
|
defer cleanup()
|
|
|
|
resp, err := http.Get(front.URL + "/_gm/")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer func() { _ = resp.Body.Close() }()
|
|
if resp.StatusCode != http.StatusUnauthorized {
|
|
t.Fatalf("status = %d, want 401", resp.StatusCode)
|
|
}
|
|
if resp.Header.Get("WWW-Authenticate") == "" {
|
|
t.Error("missing WWW-Authenticate challenge")
|
|
}
|
|
}
|
|
|
|
func TestAdminProxiesVerbatimWithCredentials(t *testing.T) {
|
|
front, gotPath, cleanup := newAdmin(t)
|
|
defer cleanup()
|
|
|
|
req, _ := http.NewRequest(http.MethodGet, front.URL+"/_gm/complaints", nil)
|
|
req.SetBasicAuth("ops", "secret")
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer func() { _ = resp.Body.Close() }()
|
|
body, _ := io.ReadAll(resp.Body)
|
|
if resp.StatusCode != http.StatusOK || string(body) != "console" {
|
|
t.Fatalf("status = %d body = %q, want 200 console", resp.StatusCode, body)
|
|
}
|
|
if *gotPath != "/_gm/complaints" {
|
|
t.Errorf("backend path = %q, want /_gm/complaints (verbatim)", *gotPath)
|
|
}
|
|
}
|
|
|
|
func TestAdminRejectsWrongPassword(t *testing.T) {
|
|
front, _, cleanup := newAdmin(t)
|
|
defer cleanup()
|
|
|
|
req, _ := http.NewRequest(http.MethodGet, front.URL+"/_gm/", nil)
|
|
req.SetBasicAuth("ops", "wrong")
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer func() { _ = resp.Body.Close() }()
|
|
if resp.StatusCode != http.StatusUnauthorized {
|
|
t.Fatalf("status = %d, want 401", resp.StatusCode)
|
|
}
|
|
}
|