Files
cheesy-arena-lite/web/api.go

314 lines
8.6 KiB
Go
Raw Normal View History

// Copyright 2014 Team 254. All Rights Reserved.
// Author: pat@patfairbank.com (Patrick Fairbank)
//
// Web API for providing JSON-formatted event data.
package web
import (
"encoding/json"
"fmt"
2021-05-16 11:00:29 -07:00
"github.com/Team254/cheesy-arena-lite/game"
"github.com/Team254/cheesy-arena-lite/model"
"github.com/Team254/cheesy-arena-lite/partner"
"github.com/Team254/cheesy-arena-lite/websocket"
2014-07-29 23:31:28 -07:00
"github.com/gorilla/mux"
2022-08-20 21:23:00 -07:00
"io"
"net/http"
"os"
"strconv"
)
2017-07-16 14:06:42 -07:00
type MatchResultWithSummary struct {
model.MatchResult
RedSummary *game.ScoreSummary
BlueSummary *game.ScoreSummary
2017-07-16 14:06:42 -07:00
}
2014-07-29 23:31:28 -07:00
type MatchWithResult struct {
model.Match
2017-07-16 14:06:42 -07:00
Result *MatchResultWithSummary
2014-07-29 23:31:28 -07:00
}
2014-07-13 01:07:21 -07:00
type RankingWithNickname struct {
game.Ranking
2014-07-13 01:07:21 -07:00
Nickname string
}
type allianceMatchup struct {
Round int
Group int
DisplayName string
RedAllianceSource string
BlueAllianceSource string
RedAlliance *model.Alliance
BlueAlliance *model.Alliance
IsActive bool
SeriesLeader string
SeriesStatus string
IsComplete bool
}
2014-07-29 23:31:28 -07:00
// Generates a JSON dump of the matches and results.
func (web *Web) matchesApiHandler(w http.ResponseWriter, r *http.Request) {
2014-07-29 23:31:28 -07:00
vars := mux.Vars(r)
matches, err := web.arena.Database.GetMatchesByType(vars["type"])
2014-07-29 23:31:28 -07:00
if err != nil {
handleWebErr(w, err)
return
}
matchesWithResults := make([]MatchWithResult, len(matches))
for i, match := range matches {
matchesWithResults[i].Match = match
matchResult, err := web.arena.Database.GetMatchResultForMatch(match.Id)
2014-07-29 23:31:28 -07:00
if err != nil {
handleWebErr(w, err)
return
}
2017-07-16 14:06:42 -07:00
var matchResultWithSummary *MatchResultWithSummary
if matchResult != nil {
matchResultWithSummary = &MatchResultWithSummary{MatchResult: *matchResult}
2020-04-14 19:38:14 -05:00
matchResultWithSummary.RedSummary = matchResult.RedScoreSummary()
matchResultWithSummary.BlueSummary = matchResult.BlueScoreSummary()
2017-07-16 14:06:42 -07:00
}
matchesWithResults[i].Result = matchResultWithSummary
2014-07-29 23:31:28 -07:00
}
jsonData, err := json.MarshalIndent(matchesWithResults, "", " ")
if err != nil {
handleWebErr(w, err)
return
}
w.Header().Set("Content-Type", "application/json")
_, err = w.Write(jsonData)
if err != nil {
handleWebErr(w, err)
return
}
}
// Generates a JSON dump of the sponsor slides for use by the audience display.
func (web *Web) sponsorSlidesApiHandler(w http.ResponseWriter, r *http.Request) {
sponsors, err := web.arena.Database.GetAllSponsorSlides()
if err != nil {
handleWebErr(w, err)
return
}
if sponsors == nil {
// Go marshals an empty slice to null, so explicitly create it so that it appears as an empty JSON array.
sponsors = make([]model.SponsorSlide, 0)
}
jsonData, err := json.MarshalIndent(sponsors, "", " ")
if err != nil {
handleWebErr(w, err)
return
}
w.Header().Set("Content-Type", "application/json")
_, err = w.Write(jsonData)
if err != nil {
handleWebErr(w, err)
return
}
}
2022-08-20 20:03:55 -07:00
// Generates a JSON dump of the qualification rankings, primarily for use by the rankings display.
func (web *Web) rankingsApiHandler(w http.ResponseWriter, r *http.Request) {
rankings, err := web.arena.Database.GetAllRankings()
if err != nil {
handleWebErr(w, err)
return
}
2014-07-13 01:07:21 -07:00
var rankingsWithNicknames []RankingWithNickname
if rankings == nil {
// Go marshals an empty slice to null, so explicitly create it so that it appears as an empty JSON array.
2014-07-13 01:07:21 -07:00
rankingsWithNicknames = make([]RankingWithNickname, 0)
} else {
rankingsWithNicknames = make([]RankingWithNickname, len(rankings))
}
// Get team info so that nicknames can be displayed.
teams, err := web.arena.Database.GetAllTeams()
2014-07-13 01:07:21 -07:00
if err != nil {
handleWebErr(w, err)
return
}
teamNicknames := make(map[int]string)
for _, team := range teams {
teamNicknames[team.Id] = team.Nickname
}
for i, ranking := range rankings {
rankingsWithNicknames[i] = RankingWithNickname{ranking, teamNicknames[ranking.TeamId]}
}
2014-07-12 15:51:24 -07:00
// Get the last match scored so we can report that on the display.
matches, err := web.arena.Database.GetMatchesByType("qualification")
2014-07-12 15:51:24 -07:00
if err != nil {
handleWebErr(w, err)
return
}
highestPlayedMatch := ""
for _, match := range matches {
if match.IsComplete() {
2014-07-12 15:51:24 -07:00
highestPlayedMatch = match.DisplayName
}
}
data := struct {
2014-07-13 01:07:21 -07:00
Rankings []RankingWithNickname
2014-07-12 15:51:24 -07:00
HighestPlayedMatch string
2014-07-13 01:07:21 -07:00
}{rankingsWithNicknames, highestPlayedMatch}
2014-07-12 15:51:24 -07:00
jsonData, err := json.MarshalIndent(data, "", " ")
if err != nil {
handleWebErr(w, err)
return
}
w.Header().Set("Content-Type", "application/json")
2014-07-12 15:51:24 -07:00
_, err = w.Write(jsonData)
if err != nil {
handleWebErr(w, err)
return
}
}
2017-09-23 22:25:36 -07:00
// Generates a JSON dump of the alliances.
func (web *Web) alliancesApiHandler(w http.ResponseWriter, r *http.Request) {
alliances, err := web.arena.Database.GetAllAlliances()
if err != nil {
handleWebErr(w, err)
return
}
jsonData, err := json.MarshalIndent(alliances, "", " ")
if err != nil {
handleWebErr(w, err)
return
}
w.Header().Set("Content-Type", "application/json")
_, err = w.Write(jsonData)
if err != nil {
handleWebErr(w, err)
return
}
}
// Websocket API for receiving arena status updates.
func (web *Web) arenaWebsocketApiHandler(w http.ResponseWriter, r *http.Request) {
ws, err := websocket.NewWebsocket(w, r)
if err != nil {
handleWebErr(w, err)
return
}
defer ws.Close()
// Subscribe the websocket to the notifiers whose messages will be passed on to the client.
ws.HandleNotifiers(web.arena.MatchTimingNotifier, web.arena.MatchLoadNotifier, web.arena.MatchTimeNotifier)
}
// Serves the avatar for a given team, or a default if none exists.
func (web *Web) teamAvatarsApiHandler(w http.ResponseWriter, r *http.Request) {
vars := mux.Vars(r)
teamId, err := strconv.Atoi(vars["teamId"])
if err != nil {
handleWebErr(w, err)
return
}
avatarPath := fmt.Sprintf("%s/%d.png", partner.AvatarsDir, teamId)
if _, err := os.Stat(avatarPath); os.IsNotExist(err) {
avatarPath = fmt.Sprintf("%s/0.png", partner.AvatarsDir)
}
http.ServeFile(w, r, avatarPath)
}
func (web *Web) bracketSvgApiHandler(w http.ResponseWriter, r *http.Request) {
var activeMatch *model.Match
showTemporaryConnectors := false
if activeMatchValue, ok := r.URL.Query()["activeMatch"]; ok {
if activeMatchValue[0] == "current" {
activeMatch = web.arena.CurrentMatch
} else if activeMatchValue[0] == "saved" {
activeMatch = web.arena.SavedMatch
showTemporaryConnectors = true
}
}
2022-08-20 21:23:00 -07:00
w.Header().Set("Content-Type", "image/svg+xml")
if err := web.generateBracketSvg(w, activeMatch, showTemporaryConnectors); err != nil {
handleWebErr(w, err)
return
}
2022-08-20 21:23:00 -07:00
}
func (web *Web) generateBracketSvg(w io.Writer, activeMatch *model.Match, showTemporaryConnectors bool) error {
2022-08-20 21:23:00 -07:00
alliances, err := web.arena.Database.GetAllAlliances()
if err != nil {
return err
}
matchups := make(map[string]*allianceMatchup)
if web.arena.PlayoffBracket != nil {
for _, matchup := range web.arena.PlayoffBracket.GetAllMatchups() {
allianceMatchup := allianceMatchup{
Round: matchup.Round,
Group: matchup.Group,
DisplayName: matchup.LongDisplayName(),
RedAllianceSource: matchup.RedAllianceSourceDisplayName(),
BlueAllianceSource: matchup.BlueAllianceSourceDisplayName(),
IsComplete: matchup.IsComplete(),
}
if matchup.RedAllianceId > 0 {
if len(alliances) > 0 {
allianceMatchup.RedAlliance = &alliances[matchup.RedAllianceId-1]
} else {
allianceMatchup.RedAlliance = &model.Alliance{Id: matchup.RedAllianceId}
}
}
if matchup.BlueAllianceId > 0 {
if len(alliances) > 0 {
allianceMatchup.BlueAlliance = &alliances[matchup.BlueAllianceId-1]
} else {
allianceMatchup.BlueAlliance = &model.Alliance{Id: matchup.BlueAllianceId}
}
}
if activeMatch != nil {
allianceMatchup.IsActive = activeMatch.ElimRound == matchup.Round &&
activeMatch.ElimGroup == matchup.Group
}
allianceMatchup.SeriesLeader, allianceMatchup.SeriesStatus = matchup.StatusText()
matchups[fmt.Sprintf("%d_%d", matchup.Round, matchup.Group)] = &allianceMatchup
}
}
bracketType := "double"
numAlliances := web.arena.EventSettings.NumElimAlliances
if web.arena.EventSettings.ElimType == "single" {
if numAlliances > 8 {
bracketType = "16"
} else if numAlliances > 4 {
bracketType = "8"
} else if numAlliances > 2 {
bracketType = "4"
} else {
bracketType = "2"
}
}
template, err := web.parseFiles("templates/bracket.svg")
if err != nil {
2022-08-20 21:23:00 -07:00
return err
}
data := struct {
BracketType string
Matchups map[string]*allianceMatchup
ShowTemporaryConnectors bool
}{bracketType, matchups, showTemporaryConnectors}
2022-08-20 21:23:00 -07:00
return template.ExecuteTemplate(w, "bracket", data)
}