blob: b3df5183c230218dbf5237a1730da1afa122aee1 [file] [log] [blame]
Philipp Schraderd9096a32022-02-24 17:53:09 -08001package debug
2
3import (
4 "bytes"
Philipp Schraderfae8a7e2022-03-13 22:51:54 -07005 "encoding/base64"
Philipp Schraderd9096a32022-02-24 17:53:09 -08006 "errors"
7 "fmt"
8 "io"
9 "log"
10 "net/http"
11
12 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/error_response"
Philipp Schraderd3fac192022-03-02 20:35:46 -080013 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/refresh_match_list_response"
Philipp Schradercbf5c6a2022-02-27 23:25:19 -080014 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/request_all_matches_response"
Philipp Schraderacf96232022-03-01 22:03:30 -080015 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/request_data_scouting_response"
Philipp Schraderd1c4bef2022-02-28 22:51:30 -080016 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/request_matches_for_team_response"
Alex Perry81f96ba2022-03-13 18:26:19 -070017 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/request_notes_for_team_response"
Milo Lin1d59f0c2022-06-22 20:30:58 -070018 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/request_shift_schedule_response"
Philipp Schraderd9096a32022-02-24 17:53:09 -080019 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/submit_data_scouting_response"
Alex Perry81f96ba2022-03-13 18:26:19 -070020 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/submit_notes_response"
Milo Lin1d59f0c2022-06-22 20:30:58 -070021 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/messages/submit_shift_schedule_response"
Philipp Schrader86bf2b92022-03-30 20:57:30 -070022 flatbuffers "github.com/google/flatbuffers/go"
Philipp Schraderd9096a32022-02-24 17:53:09 -080023)
24
Philipp Schraderfae8a7e2022-03-13 22:51:54 -070025// The username to submit the various requests as.
26const DefaultUsername = "debug_cli"
27
Philipp Schraderd9096a32022-02-24 17:53:09 -080028// A struct that can be used as an `error`. It contains information about the
29// why the server was unhappy and what the corresponding request was.
30type ResponseError struct {
31 Url string
32 StatusCode int
33 ErrorResponse *error_response.ErrorResponse
34}
35
36// Required to implement the `error` interface.
37func (err *ResponseError) Error() string {
38 return fmt.Sprintf(
39 "%s returned %d %s: %s", err.Url, err.StatusCode,
40 http.StatusText(err.StatusCode), err.ErrorResponse.ErrorMessage())
41}
42
43// Parse an `ErrorResponse` message that the server sent back. This happens
44// whenever the status code is something other than 200. If the message is
45// successfully parsed, it's turned into a `ResponseError` which implements the
46// `error` interface.
47func parseErrorResponse(url string, statusCode int, responseBytes []byte) error {
48 getRootErrMessage := ""
49 defer func() {
50 if r := recover(); r != nil {
51 getRootErrMessage = fmt.Sprintf("%v", r)
52 }
53 }()
54 errorMessage := error_response.GetRootAsErrorResponse(responseBytes, 0)
55 if getRootErrMessage != "" {
56 return errors.New(fmt.Sprintf(
57 "Failed to parse response from %s with status %d %s (bytes %v) as ErrorResponse: %s",
58 url, statusCode, http.StatusText(statusCode), responseBytes, getRootErrMessage))
59 }
60
61 return &ResponseError{
62 Url: url,
63 StatusCode: statusCode,
64 ErrorResponse: errorMessage,
65 }
66}
67
68// Performs a POST request with the specified payload. The bytes that the
69// server responds with are returned.
70func performPost(url string, requestBytes []byte) ([]byte, error) {
Philipp Schraderfae8a7e2022-03-13 22:51:54 -070071 req, err := http.NewRequest("POST", url, bytes.NewReader(requestBytes))
72 if err != nil {
73 log.Printf("Failed to create a new POST request to %s: %v", url, err)
74 return nil, err
75 }
76 req.Header.Add("Authorization", "Basic "+
77 base64.StdEncoding.EncodeToString([]byte(DefaultUsername+":")))
78
79 client := &http.Client{}
80 resp, err := client.Do(req)
Philipp Schraderd9096a32022-02-24 17:53:09 -080081 if err != nil {
82 log.Printf("Failed to send POST request to %s: %v", url, err)
83 return nil, err
84 }
85 responseBytes, err := io.ReadAll(resp.Body)
86 if err != nil {
87 log.Printf("Failed to parse response bytes from POST to %s: %v", url, err)
88 return nil, err
89 }
90 if resp.StatusCode != http.StatusOK {
91 return nil, parseErrorResponse(url, resp.StatusCode, responseBytes)
92 }
93 return responseBytes, nil
94}
95
Philipp Schrader86bf2b92022-03-30 20:57:30 -070096// Sends a message to the server and returns the deserialized response.
97// The first generic argument must be specified.
98func sendMessage[FbT interface{}, Fb interface{ UnPack() *FbT }](url string, requestBytes []byte, parser func([]byte, flatbuffers.UOffsetT) Fb) (*FbT, error) {
99 responseBytes, err := performPost(url, requestBytes)
100 if err != nil {
101 return nil, err
102 }
103 response := parser(responseBytes, 0)
104 return response.UnPack(), nil
105}
106
Philipp Schrader20440f82022-03-16 20:07:09 -0700107func SubmitDataScouting(server string, requestBytes []byte) (*submit_data_scouting_response.SubmitDataScoutingResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700108 return sendMessage[submit_data_scouting_response.SubmitDataScoutingResponseT](
109 server+"/requests/submit/data_scouting", requestBytes,
110 submit_data_scouting_response.GetRootAsSubmitDataScoutingResponse)
Philipp Schraderd9096a32022-02-24 17:53:09 -0800111}
Philipp Schradercbf5c6a2022-02-27 23:25:19 -0800112
Philipp Schrader20440f82022-03-16 20:07:09 -0700113func RequestAllMatches(server string, requestBytes []byte) (*request_all_matches_response.RequestAllMatchesResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700114 return sendMessage[request_all_matches_response.RequestAllMatchesResponseT](
115 server+"/requests/request/all_matches", requestBytes,
116 request_all_matches_response.GetRootAsRequestAllMatchesResponse)
Philipp Schradercbf5c6a2022-02-27 23:25:19 -0800117}
Philipp Schraderd1c4bef2022-02-28 22:51:30 -0800118
Philipp Schrader20440f82022-03-16 20:07:09 -0700119func RequestMatchesForTeam(server string, requestBytes []byte) (*request_matches_for_team_response.RequestMatchesForTeamResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700120 return sendMessage[request_matches_for_team_response.RequestMatchesForTeamResponseT](
121 server+"/requests/request/matches_for_team", requestBytes,
122 request_matches_for_team_response.GetRootAsRequestMatchesForTeamResponse)
Philipp Schraderd1c4bef2022-02-28 22:51:30 -0800123}
Philipp Schraderacf96232022-03-01 22:03:30 -0800124
Philipp Schrader20440f82022-03-16 20:07:09 -0700125func RequestDataScouting(server string, requestBytes []byte) (*request_data_scouting_response.RequestDataScoutingResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700126 return sendMessage[request_data_scouting_response.RequestDataScoutingResponseT](
127 server+"/requests/request/data_scouting", requestBytes,
128 request_data_scouting_response.GetRootAsRequestDataScoutingResponse)
Philipp Schraderacf96232022-03-01 22:03:30 -0800129}
Philipp Schraderd3fac192022-03-02 20:35:46 -0800130
Philipp Schrader20440f82022-03-16 20:07:09 -0700131func RefreshMatchList(server string, requestBytes []byte) (*refresh_match_list_response.RefreshMatchListResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700132 return sendMessage[refresh_match_list_response.RefreshMatchListResponseT](
133 server+"/requests/refresh_match_list", requestBytes,
134 refresh_match_list_response.GetRootAsRefreshMatchListResponse)
Philipp Schraderd3fac192022-03-02 20:35:46 -0800135}
Alex Perry81f96ba2022-03-13 18:26:19 -0700136
137func SubmitNotes(server string, requestBytes []byte) (*submit_notes_response.SubmitNotesResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700138 return sendMessage[submit_notes_response.SubmitNotesResponseT](
139 server+"/requests/submit/submit_notes", requestBytes,
140 submit_notes_response.GetRootAsSubmitNotesResponse)
Alex Perry81f96ba2022-03-13 18:26:19 -0700141}
142
143func RequestNotes(server string, requestBytes []byte) (*request_notes_for_team_response.RequestNotesForTeamResponseT, error) {
Philipp Schrader86bf2b92022-03-30 20:57:30 -0700144 return sendMessage[request_notes_for_team_response.RequestNotesForTeamResponseT](
145 server+"/requests/request/notes_for_team", requestBytes,
146 request_notes_for_team_response.GetRootAsRequestNotesForTeamResponse)
Alex Perry81f96ba2022-03-13 18:26:19 -0700147}
Milo Lin1d59f0c2022-06-22 20:30:58 -0700148
149func RequestShiftSchedule(server string, requestBytes []byte) (*request_shift_schedule_response.RequestShiftScheduleResponseT, error) {
150 return sendMessage[request_shift_schedule_response.RequestShiftScheduleResponseT](
151 server+"/requests/request/shift_schedule", requestBytes,
152 request_shift_schedule_response.GetRootAsRequestShiftScheduleResponse)
153}
154
155func SubmitShiftSchedule(server string, requestBytes []byte) (*submit_shift_schedule_response.SubmitShiftScheduleResponseT, error) {
156 return sendMessage[submit_shift_schedule_response.SubmitShiftScheduleResponseT](
157 server+"/requests/submit/shift_schedule", requestBytes,
158 submit_shift_schedule_response.GetRootAsSubmitShiftScheduleResponse)
159}