rogueserver/api/generic.go

286 lines
7.2 KiB
Go
Raw Normal View History

2023-12-05 10:28:08 -08:00
package api
import (
2024-04-08 17:44:36 -07:00
"encoding/base64"
2024-04-06 14:43:11 -07:00
"encoding/gob"
2024-04-08 17:44:36 -07:00
"encoding/json"
"fmt"
"net/http"
2024-04-08 17:44:36 -07:00
"strconv"
"github.com/Flashfyre/pokerogue-server/defs"
)
2023-12-29 11:30:47 -08:00
type Server struct {
Debug bool
}
2024-04-08 17:44:36 -07:00
/*
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.
Handlers should not return serialized JSON, instead return the struct itself.
*/
2023-12-29 11:30:47 -08:00
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
2024-04-06 14:43:11 -07:00
gob.Register([]interface{}{})
gob.Register(map[string]interface{}{})
2023-12-29 11:30:47 -08:00
if s.Debug {
2023-12-29 11:34:58 -08:00
w.Header().Set("Access-Control-Allow-Headers", "*")
w.Header().Set("Access-Control-Allow-Methods", "*")
2023-12-29 11:33:11 -08:00
w.Header().Set("Access-Control-Allow-Origin", "*")
2023-12-29 11:30:47 -08:00
if r.Method == "OPTIONS" {
w.WriteHeader(http.StatusOK)
return
}
}
switch r.URL.Path {
2024-04-08 17:44:36 -07:00
// /account
2023-12-29 11:30:47 -08:00
case "/account/info":
2024-04-08 17:44:36 -07:00
username, err := getUsernameFromRequest(r)
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
uuid, err := getUUIDFromRequest(r) // lazy
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
info, err := handleAccountInfo(username, uuid)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
response, err := json.Marshal(info)
if err != nil {
httpError(w, r, fmt.Errorf("failed to marshal response json: %s", err), http.StatusInternalServerError)
return
}
w.Write(response)
2023-12-29 11:30:47 -08:00
case "/account/register":
2024-04-08 17:44:36 -07:00
var request AccountRegisterRequest
err := json.NewDecoder(r.Body).Decode(&request)
if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest)
return
}
err = handleAccountRegister(request.Username, request.Password)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
w.WriteHeader(http.StatusOK)
2023-12-29 11:30:47 -08:00
case "/account/login":
2024-04-08 17:44:36 -07:00
var request AccountLoginRequest
err := json.NewDecoder(r.Body).Decode(&request)
if err != nil {
httpError(w, r, fmt.Errorf("failed to decode request body: %s", err), http.StatusBadRequest)
return
}
token, err := handleAccountLogin(request.Username, request.Password)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
response, err := json.Marshal(token)
if err != nil {
httpError(w, r, fmt.Errorf("failed to marshal response json: %s", err), http.StatusInternalServerError)
return
}
w.Write(response)
2023-12-29 11:30:47 -08:00
case "/account/logout":
2024-04-08 17:44:36 -07:00
token, err := base64.StdEncoding.DecodeString(r.Header.Get("Authorization"))
if err != nil {
httpError(w, r, fmt.Errorf("failed to decode token: %s", err), http.StatusBadRequest)
return
}
err = handleAccountLogout(token)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
2023-12-29 11:30:47 -08:00
2024-04-08 17:44:36 -07:00
w.WriteHeader(http.StatusOK)
// /game
2024-03-23 18:34:18 -07:00
case "/game/playercount":
2024-04-08 17:44:36 -07:00
w.Write([]byte(strconv.Itoa(playerCount)))
2024-04-06 15:15:47 -07:00
case "/game/titlestats":
2024-04-08 17:44:36 -07:00
response, err := json.Marshal(&defs.TitleStats{
PlayerCount: playerCount,
BattleCount: battleCount,
})
if err != nil {
httpError(w, r, fmt.Errorf("failed to marshal response json: %s", err), http.StatusInternalServerError)
return
}
w.Write(response)
2024-04-06 15:15:47 -07:00
case "/game/classicsessioncount":
2024-04-08 17:44:36 -07:00
w.Write([]byte(strconv.Itoa(classicSessionCount)))
2024-03-23 18:34:18 -07:00
2024-04-08 17:44:36 -07:00
// /savedata
case "/savedata/get", "/savedata/update", "/savedata/delete", "/savedata/clear":
uuid, err := getUUIDFromRequest(r)
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
datatype := -1
if r.URL.Query().Has("datatype") {
datatype, err = strconv.Atoi(r.URL.Query().Get("datatype"))
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
}
var slot int
if r.URL.Query().Has("slot") {
slot, err = strconv.Atoi(r.URL.Query().Get("slot"))
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
}
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" {
2024-04-08 17:44:36 -07:00
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 {
case "/savedata/get":
save, err = handleSavedataGet(uuid, datatype, slot)
case "/savedata/update":
err = handleSavedataUpdate(uuid, slot, save)
case "/savedata/delete":
err = handleSavedataDelete(uuid, datatype, slot)
case "/savedata/clear":
// doesn't return a save, but it works
save, err = handleSavedataClear(uuid, slot, save.(defs.SessionSaveData))
}
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
2024-04-08 18:34:42 -07:00
if save == nil || r.URL.Path == "/savedata/update" {
2024-04-08 17:44:36 -07:00
w.WriteHeader(http.StatusOK)
2024-04-08 17:47:23 -07:00
return
2024-04-08 17:44:36 -07:00
}
response, err := json.Marshal(save)
if err != nil {
httpError(w, r, fmt.Errorf("failed to marshal response json: %s", err), http.StatusInternalServerError)
return
}
w.Write(response)
2024-03-16 18:51:13 -07:00
2024-04-08 17:44:36 -07:00
// /daily
2024-03-16 18:51:13 -07:00
case "/daily/seed":
2024-04-08 17:44:36 -07:00
w.Write([]byte(dailyRunSeed))
2024-03-17 10:18:51 -07:00
case "/daily/rankings":
2024-04-08 17:44:36 -07:00
uuid, err := getUUIDFromRequest(r)
if err != nil {
httpError(w, r, err, http.StatusBadRequest)
return
}
var category int
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
}
}
page := 1
if r.URL.Query().Has("page") {
page, err = strconv.Atoi(r.URL.Query().Get("page"))
if err != nil {
httpError(w, r, fmt.Errorf("failed to convert page: %s", err), http.StatusBadRequest)
return
}
}
rankings, err := handleRankings(uuid, category, page)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
return
}
response, err := json.Marshal(rankings)
if err != nil {
httpError(w, r, fmt.Errorf("failed to marshal response json: %s", err), http.StatusInternalServerError)
return
}
w.Write(response)
case "/daily/rankingpagecount":
2024-04-08 17:44:36 -07:00
var category int
if r.URL.Query().Has("category") {
var err error
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
}
}
count, err := handleRankingPageCount(category)
if err != nil {
httpError(w, r, err, http.StatusInternalServerError)
}
w.Write([]byte(strconv.Itoa(count)))
2023-12-29 11:30:47 -08:00
}
}
2023-12-05 10:28:08 -08:00
// auth
type GenericAuthRequest struct {
Username string `json:"username"`
Password string `json:"password"`
}
type GenericAuthResponse struct {
Token string `json:"token"`
}