Revert "Serve game content and API"

This reverts commit 6bd7deb3de.
pull/4/head
maru 7 months ago
parent 6bd7deb3de
commit 0fec7cb4c3
No known key found for this signature in database
GPG Key ID: 37689350E9CD0F0D

1
.gitignore vendored

@ -3,4 +3,3 @@
pokerogue-server* pokerogue-server*
userdata/* userdata/*
secret.key secret.key
www/

@ -10,31 +10,9 @@ import (
"github.com/pagefaultgames/pokerogue-server/db" "github.com/pagefaultgames/pokerogue-server/db"
) )
func Init(mux *http.ServeMux) { func Init() {
scheduleStatRefresh() scheduleStatRefresh()
daily.Init() daily.Init()
// account
mux.HandleFunc("GET /api/account/info", handleAccountInfo)
mux.HandleFunc("POST /api/account/register", handleAccountRegister)
mux.HandleFunc("POST /api/account/login", handleAccountLogin)
mux.HandleFunc("GET /api/account/logout", handleAccountLogout)
// game
mux.HandleFunc("GET /api/game/playercount", handleGamePlayerCount)
mux.HandleFunc("GET /api/game/titlestats", handleGameTitleStats)
mux.HandleFunc("GET /api/game/classicsessioncount", handleGameClassicSessionCount)
// savedata
mux.HandleFunc("GET /api/savedata/get", handleSaveData)
mux.HandleFunc("POST /api/savedata/update", handleSaveData)
mux.HandleFunc("GET /api/savedata/delete", handleSaveData)
mux.HandleFunc("POST /api/savedata/clear", handleSaveData)
// daily
mux.HandleFunc("GET /api/daily/seed", handleDailySeed)
mux.HandleFunc("GET /api/daily/rankings", handleDailyRankings)
mux.HandleFunc("GET /api/daily/rankingpagecount", handleDailyRankingPageCount)
} }
func getUsernameFromRequest(r *http.Request) (string, error) { func getUsernameFromRequest(r *http.Request) (string, error) {

@ -7,6 +7,7 @@ import (
"log" "log"
"net/http" "net/http"
"strconv" "strconv"
"sync"
"github.com/pagefaultgames/pokerogue-server/api/account" "github.com/pagefaultgames/pokerogue-server/api/account"
"github.com/pagefaultgames/pokerogue-server/api/daily" "github.com/pagefaultgames/pokerogue-server/api/daily"
@ -14,252 +15,262 @@ import (
"github.com/pagefaultgames/pokerogue-server/defs" "github.com/pagefaultgames/pokerogue-server/defs"
) )
type Server struct {
Debug bool
Exit *sync.RWMutex
}
/* /*
The caller of endpoint handler functions are responsible for extracting the necessary data from the request. The caller of endpoint handler functions are responsible for extracting the necessary data from the request.
Handler functions are responsible for checking the validity of this data and returning a result or error. Handler functions are responsible for checking the validity of this data and returning a result or error.
Handlers should not return serialized JSON, instead return the struct itself. Handlers should not return serialized JSON, instead return the struct itself.
*/ */
func handleAccountInfo(w http.ResponseWriter, r *http.Request) { func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
username, err := getUsernameFromRequest(r) // kind of misusing the RWMutex but it doesn't matter
if err != nil { s.Exit.RLock()
httpError(w, r, err, http.StatusBadRequest) defer s.Exit.RUnlock()
return
}
uuid, err := getUUIDFromRequest(r) // lazy
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
response, err := account.Info(username, uuid) if s.Debug {
if err != nil { w.Header().Set("Access-Control-Allow-Headers", "*")
httpError(w, r, err, http.StatusInternalServerError) w.Header().Set("Access-Control-Allow-Methods", "*")
return w.Header().Set("Access-Control-Allow-Origin", "*")
}
err = json.NewEncoder(w).Encode(response) if r.Method == "OPTIONS" {
if err != nil { w.WriteHeader(http.StatusOK)
httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError) return
return }
}
}
func handleAccountRegister(w http.ResponseWriter, r *http.Request) {
err := r.ParseForm()
if err != nil {
httpError(w, r, fmt.Errorf("failed to parse request form: %s", err), http.StatusBadRequest)
return
}
err = account.Register(r.Form.Get("username"), r.Form.Get("password"))
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
} }
w.WriteHeader(http.StatusOK) switch r.URL.Path {
} // /account
case "/account/info":
func handleAccountLogin(w http.ResponseWriter, r *http.Request) { username, err := getUsernameFromRequest(r)
err := r.ParseForm() if err != nil {
if err != nil { httpError(w, r, err, http.StatusBadRequest)
httpError(w, r, fmt.Errorf("failed to parse request form: %s", err), http.StatusBadRequest) return
return }
}
response, err := account.Login(r.Form.Get("username"), r.Form.Get("password")) uuid, err := getUUIDFromRequest(r) // lazy
if err != nil { if err != nil {
httpError(w, r, err, http.StatusInternalServerError) httpError(w, r, err, http.StatusBadRequest)
return return
} }
err = json.NewEncoder(w).Encode(response) response, err := account.Info(username, uuid)
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError) httpError(w, r, err, http.StatusInternalServerError)
return return
} }
}
func handleAccountLogout(w http.ResponseWriter, r *http.Request) { err = json.NewEncoder(w).Encode(response)
token, err := base64.StdEncoding.DecodeString(r.Header.Get("Authorization")) if err != nil {
if err != nil { httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
httpError(w, r, fmt.Errorf("failed to decode token: %s", err), http.StatusBadRequest) return
return }
} case "/account/register":
err := r.ParseForm()
if err != nil {
httpError(w, r, fmt.Errorf("failed to parse request form: %s", err), http.StatusBadRequest)
return
}
err = account.Logout(token) err = account.Register(r.Form.Get("username"), r.Form.Get("password"))
if err != nil { if err != nil {
httpError(w, r, err, http.StatusInternalServerError) httpError(w, r, err, http.StatusInternalServerError)
return return
} }
w.WriteHeader(http.StatusOK) w.WriteHeader(http.StatusOK)
} case "/account/login":
err := r.ParseForm()
if err != nil {
httpError(w, r, fmt.Errorf("failed to parse request form: %s", err), http.StatusBadRequest)
return
}
func handleGamePlayerCount(w http.ResponseWriter, r *http.Request) { response, err := account.Login(r.Form.Get("username"), r.Form.Get("password"))
w.Write([]byte(strconv.Itoa(playerCount))) if err != nil {
} httpError(w, r, err, http.StatusInternalServerError)
return
}
func handleGameTitleStats(w http.ResponseWriter, r *http.Request) { err = json.NewEncoder(w).Encode(response)
err := json.NewEncoder(w).Encode(defs.TitleStats{ if err != nil {
PlayerCount: playerCount, httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
BattleCount: battleCount, return
}) }
if err != nil { case "/account/logout":
httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError) token, err := base64.StdEncoding.DecodeString(r.Header.Get("Authorization"))
return if err != nil {
} httpError(w, r, fmt.Errorf("failed to decode token: %s", err), http.StatusBadRequest)
} return
}
func handleGameClassicSessionCount(w http.ResponseWriter, r *http.Request) { err = account.Logout(token)
w.Write([]byte(strconv.Itoa(classicSessionCount))) if err != nil {
} httpError(w, r, err, http.StatusInternalServerError)
return
}
func handleSaveData(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK)
uuid, err := getUUIDFromRequest(r)
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
datatype := -1 // /game
if r.URL.Query().Has("datatype") { case "/game/playercount":
datatype, err = strconv.Atoi(r.URL.Query().Get("datatype")) w.Write([]byte(strconv.Itoa(playerCount)))
case "/game/titlestats":
err := json.NewEncoder(w).Encode(defs.TitleStats{
PlayerCount: playerCount,
BattleCount: battleCount,
})
if err != nil { if err != nil {
httpError(w, r, err, http.StatusBadRequest) httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
return return
} }
} case "/game/classicsessioncount":
w.Write([]byte(strconv.Itoa(classicSessionCount)))
var slot int // /savedata
if r.URL.Query().Has("slot") { case "/savedata/get", "/savedata/update", "/savedata/delete", "/savedata/clear":
slot, err = strconv.Atoi(r.URL.Query().Get("slot")) uuid, err := getUUIDFromRequest(r)
if err != nil { if err != nil {
httpError(w, r, err, http.StatusBadRequest) httpError(w, r, err, http.StatusBadRequest)
return return
} }
}
var save any datatype := -1
// /savedata/get and /savedata/delete specify datatype, but don't expect data in body if r.URL.Query().Has("datatype") {
if r.URL.Path != "/api/savedata/get" && r.URL.Path != "/api/savedata/delete" { datatype, err = strconv.Atoi(r.URL.Query().Get("datatype"))
if datatype == 0 {
var system defs.SystemSaveData
err = json.NewDecoder(r.Body).Decode(&system)
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest) httpError(w, r, err, http.StatusBadRequest)
return return
} }
}
save = system var slot int
// /savedata/clear doesn't specify datatype, it is assumed to be 1 (session) if r.URL.Query().Has("slot") {
} else if datatype == 1 || r.URL.Path == "/api/savedata/clear" { slot, err = strconv.Atoi(r.URL.Query().Get("slot"))
var session defs.SessionSaveData
err = json.NewDecoder(r.Body).Decode(&session)
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest) httpError(w, r, err, http.StatusBadRequest)
return return
} }
}
save = session var save any
// /savedata/get and /savedata/delete specify datatype, but don't expect data in body
if r.URL.Path != "/savedata/get" && r.URL.Path != "/savedata/delete" {
if datatype == 0 {
var system defs.SystemSaveData
err = json.NewDecoder(r.Body).Decode(&system)
if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest)
return
}
save = system
// /savedata/clear doesn't specify datatype, it is assumed to be 1 (session)
} else if datatype == 1 || r.URL.Path == "/savedata/clear" {
var session defs.SessionSaveData
err = json.NewDecoder(r.Body).Decode(&session)
if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest)
return
}
save = session
}
} }
}
switch r.URL.Path { switch r.URL.Path {
case "/api/savedata/get": case "/savedata/get":
save, err = savedata.Get(uuid, datatype, slot) save, err = savedata.Get(uuid, datatype, slot)
case "/api/savedata/update": case "/savedata/update":
err = savedata.Update(uuid, slot, save) err = savedata.Update(uuid, slot, save)
case "/api/savedata/delete": case "/savedata/delete":
err = savedata.Delete(uuid, datatype, slot) err = savedata.Delete(uuid, datatype, slot)
case "/api/savedata/clear": case "/savedata/clear":
s, ok := save.(defs.SessionSaveData) s, ok := save.(defs.SessionSaveData)
if !ok { if !ok {
httpError(w, r, fmt.Errorf("save data is not type SessionSaveData"), http.StatusBadRequest) httpError(w, r, fmt.Errorf("save data is not type SessionSaveData"), http.StatusBadRequest)
return
}
// doesn't return a save, but it works
save, err = savedata.Clear(uuid, slot, daily.Seed(), s)
}
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return return
} }
// doesn't return a save, but it works if save == nil || r.URL.Path == "/savedata/update" {
save, err = savedata.Clear(uuid, slot, daily.Seed(), s) w.WriteHeader(http.StatusOK)
} return
if err != nil { }
httpError(w, r, err, http.StatusInternalServerError)
return
}
if save == nil || r.URL.Path == "/api/savedata/update" { err = json.NewEncoder(w).Encode(save)
w.WriteHeader(http.StatusOK) if err != nil {
return httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
} return
}
err = json.NewEncoder(w).Encode(save) // /daily
if err != nil { case "/daily/seed":
httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError) w.Write([]byte(daily.Seed()))
return case "/daily/rankings":
} uuid, err := getUUIDFromRequest(r)
} if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
func handleDailySeed(w http.ResponseWriter, r *http.Request) { var category int
w.Write([]byte(daily.Seed())) if r.URL.Query().Has("category") {
} category, err = strconv.Atoi(r.URL.Query().Get("category"))
if err != nil {
httpError(w, r, fmt.Errorf("failed to convert category: %s", err), http.StatusBadRequest)
return
}
}
func handleDailyRankings(w http.ResponseWriter, r *http.Request) { page := 1
uuid, err := getUUIDFromRequest(r) if r.URL.Query().Has("page") {
if err != nil { page, err = strconv.Atoi(r.URL.Query().Get("page"))
httpError(w, r, err, http.StatusBadRequest) if err != nil {
return httpError(w, r, fmt.Errorf("failed to convert page: %s", err), http.StatusBadRequest)
} return
}
}
var category int rankings, err := daily.Rankings(uuid, category, page)
if r.URL.Query().Has("category") {
category, err = strconv.Atoi(r.URL.Query().Get("category"))
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to convert category: %s", err), http.StatusBadRequest) httpError(w, r, err, http.StatusInternalServerError)
return return
} }
}
page := 1 err = json.NewEncoder(w).Encode(rankings)
if r.URL.Query().Has("page") {
page, err = strconv.Atoi(r.URL.Query().Get("page"))
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to convert page: %s", err), http.StatusBadRequest) httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
return return
} }
} case "/daily/rankingpagecount":
var category int
rankings, err := daily.Rankings(uuid, category, page) if r.URL.Query().Has("category") {
if err != nil { var err error
httpError(w, r, err, http.StatusInternalServerError) category, err = strconv.Atoi(r.URL.Query().Get("category"))
return if err != nil {
} httpError(w, r, fmt.Errorf("failed to convert category: %s", err), http.StatusBadRequest)
return
err = json.NewEncoder(w).Encode(rankings) }
if err != nil { }
httpError(w, r, fmt.Errorf("failed to encode response json: %s", err), http.StatusInternalServerError)
return
}
}
func handleDailyRankingPageCount(w http.ResponseWriter, r *http.Request) { count, err := daily.RankingPageCount(category)
var category int
if r.URL.Query().Has("category") {
var err error
category, err = strconv.Atoi(r.URL.Query().Get("category"))
if err != nil { if err != nil {
httpError(w, r, fmt.Errorf("failed to convert category: %s", err), http.StatusBadRequest) httpError(w, r, err, http.StatusInternalServerError)
return
} }
}
count, err := daily.RankingPageCount(category) w.Write([]byte(strconv.Itoa(count)))
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
} }
w.Write([]byte(strconv.Itoa(count)))
} }
func httpError(w http.ResponseWriter, r *http.Request, err error, code int) { func httpError(w http.ResponseWriter, r *http.Request, err error, code int) {

@ -4,7 +4,12 @@ import (
"encoding/gob" "encoding/gob"
"flag" "flag"
"log" "log"
"net"
"net/http" "net/http"
"os"
"os/signal"
"sync"
"syscall"
"github.com/pagefaultgames/pokerogue-server/api" "github.com/pagefaultgames/pokerogue-server/api"
"github.com/pagefaultgames/pokerogue-server/db" "github.com/pagefaultgames/pokerogue-server/db"
@ -12,10 +17,10 @@ import (
func main() { func main() {
// flag stuff // flag stuff
addr := flag.String("addr", "0.0.0.0:80", "network address for api to listen on") debug := flag.Bool("debug", false, "debug mode")
wwwpath := flag.String("wwwpath", "www", "path to static content to serve")
tlscert := flag.String("tlscert", "", "path to tls certificate to use for https") proto := flag.String("proto", "tcp", "protocol for api to use (tcp, unix)")
tlskey := flag.String("tlskey", "", "path to tls private key to use for https") addr := flag.String("addr", "0.0.0.0", "network address for api to listen on")
dbuser := flag.String("dbuser", "pokerogue", "database username") dbuser := flag.String("dbuser", "pokerogue", "database username")
dbpass := flag.String("dbpass", "", "database password") dbpass := flag.String("dbpass", "", "database password")
@ -25,6 +30,7 @@ func main() {
flag.Parse() flag.Parse()
// register gob types // register gob types
gob.Register([]interface{}{}) gob.Register([]interface{}{})
gob.Register(map[string]interface{}{}) gob.Register(map[string]interface{}{})
@ -35,19 +41,55 @@ func main() {
log.Fatalf("failed to initialize database: %s", err) log.Fatalf("failed to initialize database: %s", err)
} }
// start web server // create listener
mux := http.NewServeMux() listener, err := createListener(*proto, *addr)
if err != nil {
log.Fatalf("failed to create net listener: %s", err)
}
api.Init(mux) // create exit handler
var exit sync.RWMutex
createExitHandler(&exit)
mux.Handle("/", http.FileServer(http.Dir(*wwwpath))) // init api
api.Init()
if *tlscert != "" && *tlskey != "" {
err = http.ListenAndServeTLS(*addr, *tlscert, *tlskey, mux) // start web server
} else { err = http.Serve(listener, &api.Server{Debug: *debug, Exit: &exit})
err = http.ListenAndServe(*addr, mux)
}
if err != nil { if err != nil {
log.Fatalf("failed to create http server or server errored: %s", err) log.Fatalf("failed to create http server or server errored: %s", err)
} }
} }
func createListener(proto, addr string) (net.Listener, error) {
if proto == "unix" {
os.Remove(addr)
}
listener, err := net.Listen(proto, addr)
if err != nil {
return nil, err
}
if proto == "unix" {
os.Chmod(addr, 0777)
}
return listener, nil
}
func createExitHandler(mtx *sync.RWMutex) {
s := make(chan os.Signal, 1)
signal.Notify(s, syscall.SIGINT, syscall.SIGTERM)
go func() {
// wait for exit signal of some kind
<-s
// block new requests and wait for existing ones to finish
mtx.Lock()
// bail
os.Exit(0)
}()
}

Loading…
Cancel
Save