moth/cmd/mothd/server_test.go

213 lines
5.2 KiB
Go
Raw Normal View History

2020-08-19 18:01:21 -06:00
package main
import (
"io/ioutil"
2022-05-10 13:20:54 -06:00
"os"
2020-08-19 18:01:21 -06:00
"testing"
"time"
)
const TestMaintenanceInterval = time.Millisecond * 1
const TestTeamID = "teamID"
2022-05-10 13:20:54 -06:00
type TestMothServer struct {
*MothServer
stateDir string
}
func NewTestServer() (*TestMothServer, error) {
2020-08-19 18:01:21 -06:00
puzzles := NewTestMothballs()
go puzzles.Maintain(TestMaintenanceInterval)
2022-05-10 13:20:54 -06:00
stateDir, err := ioutil.TempDir("", "state")
if err != nil {
return nil, err
}
state := NewState(stateDir)
os.WriteFile(state.path("teamids.txt"), []byte("teamID\n"), 0644)
os.WriteFile(state.path("messages.html"), []byte("messages.html"), 0644)
2020-08-19 18:01:21 -06:00
go state.Maintain(TestMaintenanceInterval)
2022-05-10 13:20:54 -06:00
theme := NewTheme("testdata/theme")
2020-08-19 18:01:21 -06:00
go theme.Maintain(TestMaintenanceInterval)
2022-05-10 13:20:54 -06:00
return &TestMothServer{
MothServer: NewMothServer(Configuration{}, theme, state, puzzles),
stateDir: stateDir,
}, nil
}
func (m *TestMothServer) cleanup() {
if m.stateDir != "" {
os.RemoveAll(m.stateDir)
}
2020-08-19 18:01:21 -06:00
}
2020-10-16 14:18:44 -06:00
func TestDevelServer(t *testing.T) {
2022-05-10 13:20:54 -06:00
server, err := NewTestServer()
if err != nil {
t.Fatal(err)
}
defer server.cleanup()
2020-10-16 14:18:44 -06:00
server.Config.Devel = true
anonHandler := server.NewHandler("badParticipantId", "badTeamId")
{
es := anonHandler.ExportState()
if !es.Config.Devel {
t.Error("Not marked as development server")
}
if len(es.Puzzles) != 1 {
t.Error("Wrong puzzles for anonymous state on devel server:", es.Puzzles)
}
}
}
func TestProdServer(t *testing.T) {
2020-08-19 18:01:21 -06:00
teamName := "OurTeam"
participantID := "participantID"
teamID := TestTeamID
2022-05-10 13:20:54 -06:00
server, err := NewTestServer()
if err != nil {
t.Fatal(err)
}
defer server.cleanup()
2020-08-19 18:01:21 -06:00
handler := server.NewHandler(participantID, teamID)
anonHandler := server.NewHandler("badParticipantId", "badTeamId")
{
es := handler.ExportState()
if es.Config.Devel {
t.Error("Marked as development server", es.Config)
}
if len(es.Puzzles) != 0 {
t.Log("State", es)
t.Error("Unauthenticated state has non-empty puzzles list")
}
}
2020-08-19 18:01:21 -06:00
if err := handler.Register(teamName); err != nil {
t.Error(err)
}
if err := handler.Register(teamName); err == nil {
t.Error("Registering twice should have raised an error")
} else if err != ErrAlreadyRegistered {
t.Error("Wrong error for duplicate registration:", err)
}
2020-08-21 17:02:38 -06:00
if r, _, err := handler.ThemeOpen("/index.html"); err != nil {
2020-08-19 18:01:21 -06:00
t.Error(err)
} else if contents, err := ioutil.ReadAll(r); err != nil {
t.Error(err)
} else if string(contents) != "index.html" {
t.Error("index.html wrong contents", contents)
}
2021-10-26 12:48:23 -06:00
// Wait for refresh to pick everything up
time.Sleep(TestMaintenanceInterval)
{
es := handler.ExportState()
if es.Config.Devel {
t.Error("Marked as development server", es.Config)
}
if len(es.Puzzles) != 1 {
2021-10-26 12:48:23 -06:00
t.Error("Puzzle categories wrong length", len(es.Puzzles))
}
if es.Messages != "messages.html" {
t.Error("Messages has wrong contents")
}
if len(es.PointsLog) != 0 {
t.Error("Points log not empty")
}
if len(es.TeamNames) != 1 {
t.Error("Wrong number of team names")
}
if es.TeamNames["self"] != teamName {
t.Error("TeamNames['self'] wrong")
}
2020-08-19 18:01:21 -06:00
}
if r, _, err := handler.PuzzlesOpen("pategory", 1, "moo.txt"); err != nil {
t.Error(err)
} else if contents, err := ioutil.ReadAll(r); err != nil {
2020-08-21 17:02:38 -06:00
r.Close()
2020-08-19 18:01:21 -06:00
t.Error(err)
} else if string(contents) != "moo" {
2020-08-21 17:02:38 -06:00
r.Close()
2020-08-19 18:01:21 -06:00
t.Error("moo.txt has wrong contents", contents)
2020-08-21 17:02:38 -06:00
} else {
r.Close()
}
if r, _, err := handler.PuzzlesOpen("pategory", 2, "puzzle.json"); err == nil {
2020-08-21 17:02:38 -06:00
t.Error("Opening locked puzzle shouldn't work")
r.Close()
}
if r, _, err := handler.PuzzlesOpen("pategory", 20, "puzzle.json"); err == nil {
2020-08-21 17:02:38 -06:00
t.Error("Opening non-existent puzzle shouldn't work")
r.Close()
2020-08-19 18:01:21 -06:00
}
if err := anonHandler.CheckAnswer("pategory", 1, "answer123"); err == nil {
t.Error("Invalid team ID was able to get points with correct answer")
}
2020-08-19 18:01:21 -06:00
if err := handler.CheckAnswer("pategory", 1, "answer123"); err != nil {
t.Error("Right answer marked wrong", err)
}
time.Sleep(TestMaintenanceInterval)
{
es := handler.ExportState()
if len(es.PointsLog) != 1 {
t.Error("I didn't get my points!")
}
if len(es.Puzzles["pategory"]) != 2 {
t.Error("The next puzzle didn't unlock!")
} else if es.Puzzles["pategory"][1] != 2 {
t.Error("The 2-point puzzle should have unlocked!")
}
}
if r, _, err := handler.PuzzlesOpen("pategory", 2, "puzzle.json"); err != nil {
t.Error("Opening unlocked puzzle should work")
} else {
r.Close()
}
if r, _, err := anonHandler.PuzzlesOpen("pategory", 2, "puzzle.json"); err != nil {
t.Error("Opening unlocked puzzle anonymously should work")
} else {
r.Close()
}
2020-08-21 17:02:38 -06:00
2020-10-14 09:46:51 -06:00
if err := handler.CheckAnswer("pategory", 2, "wat"); err != nil {
t.Error("Right answer marked wrong:", err)
}
time.Sleep(TestMaintenanceInterval)
{
es := anonHandler.ExportState()
if len(es.TeamNames) != 1 {
t.Error("Anonymous TeamNames is wrong:", es.TeamNames)
}
if len(es.PointsLog) != 2 {
2021-10-18 18:30:11 -06:00
t.Errorf("Points log wrong length: got %d, wanted 2", len(es.PointsLog))
} else if es.PointsLog[1].TeamID != "0" {
t.Error("Second point log didn't anonymize team ID correctly:", es.PointsLog[1])
}
}
2020-10-14 09:46:51 -06:00
{
es := handler.ExportState()
if len(es.TeamNames) != 1 {
t.Error("TeamNames is wrong:", es.TeamNames)
}
2020-10-14 09:46:51 -06:00
}
2020-08-21 17:02:38 -06:00
// BUG(neale): We aren't currently testing the various ways to disable the server
2020-08-19 18:01:21 -06:00
}