blob: f6fb38ac6d5f59f99d3ccaa51f01e62de72c7fd2 [file] [log] [blame]
Philipp Schraderd9096a32022-02-24 17:53:09 -08001// This binary lets users interact with the scouting web server in order to
2// debug it. Run with `--help` to see all the options.
3
4package main
5
6import (
7 "flag"
8 "io/ioutil"
9 "log"
10 "os"
11 "os/exec"
12 "path/filepath"
13
Philipp Schraderd3fac192022-03-02 20:35:46 -080014 "github.com/davecgh/go-spew/spew"
Philipp Schraderd9096a32022-02-24 17:53:09 -080015 "github.com/frc971/971-Robot-Code/scouting/webserver/requests/debug"
16)
17
18// Returns the absolute path of the specified path. This is an unwrapped
19// version of `filepath.Abs`.
20func absPath(path string) string {
21 result, err := filepath.Abs(path)
22 if err != nil {
23 log.Fatal("Failed to determine absolute path for ", path, ": ", err)
24 }
25 return result
26}
27
28// Parses the specified JSON file into a binary version (i.e. serialized
29// flatbuffer). This uses the `flatc` binary and the JSON's corresponding
30// `.fbs` file.
31func parseJson(fbsPath string, jsonPath string) []byte {
32 // Work inside a temporary directory since `flatc` doesn't allow us to
33 // customize the name of the output file.
34 dir, err := ioutil.TempDir("", "webserver_debug_cli")
35 if err != nil {
36 log.Fatal("Failed to create temporary directory: ", err)
37 }
38 defer os.RemoveAll(dir)
39
40 // Turn these paths absolute so that it everything still works from
41 // inside the temporary directory.
42 absFlatcPath := absPath("external/com_github_google_flatbuffers/flatc")
43 absFbsPath := absPath(fbsPath)
44
45 // Create a symlink to the .fbs file so that the output filename that
46 // `flatc` generates is predictable. I.e. `fb.json` gets serialized
47 // into `fb.bin`.
48 jsonSymlink := filepath.Join(dir, "fb.json")
49 os.Symlink(jsonPath, jsonSymlink)
50
51 // Execute the `flatc` command.
52 flatcCommand := exec.Command(absFlatcPath, "--binary", absFbsPath, jsonSymlink)
53 flatcCommand.Dir = dir
Philipp Schradercbf5c6a2022-02-27 23:25:19 -080054 output, err := flatcCommand.CombinedOutput()
Philipp Schraderd9096a32022-02-24 17:53:09 -080055 if err != nil {
Philipp Schradercbf5c6a2022-02-27 23:25:19 -080056 log.Fatal("Failed to execute flatc: ", err, ": ", string(output))
Philipp Schraderd9096a32022-02-24 17:53:09 -080057 }
58
59 // Read the serialized flatbuffer and return it.
60 binaryPath := filepath.Join(dir, "fb.bin")
61 binaryFb, err := os.ReadFile(binaryPath)
62 if err != nil {
63 log.Fatal("Failed to read flatc output ", binaryPath, ": ", err)
64 }
65 return binaryFb
66}
67
Philipp Schrader02ec8e42022-03-26 16:37:18 -070068func maybePerformRequest[T interface{}](fbName, fbsPath, requestJsonPath, address string, requester func(string, []byte) (*T, error)) {
69 if requestJsonPath != "" {
70 log.Printf("Sending %s to %s", fbName, address)
71 binaryRequest := parseJson(fbsPath, requestJsonPath)
72 response, err := requester(address, binaryRequest)
73 if err != nil {
74 log.Fatalf("Failed %s: %v", fbName, err)
75 }
76 spew.Dump(*response)
77 }
78}
79
Philipp Schraderd9096a32022-02-24 17:53:09 -080080func main() {
81 // Parse command line arguments.
Philipp Schrader30005e42022-03-06 13:53:58 -080082 indentPtr := flag.String("indent", " ",
83 "The indentation to use for the result dumping. Default is a space.")
Philipp Schraderd9096a32022-02-24 17:53:09 -080084 addressPtr := flag.String("address", "http://localhost:8080",
85 "The end point where the server is listening.")
86 submitDataScoutingPtr := flag.String("submitDataScouting", "",
87 "If specified, parse the file as a SubmitDataScouting JSON request.")
Philipp Schradercbf5c6a2022-02-27 23:25:19 -080088 requestAllMatchesPtr := flag.String("requestAllMatches", "",
89 "If specified, parse the file as a RequestAllMatches JSON request.")
Philipp Schraderd1c4bef2022-02-28 22:51:30 -080090 requestMatchesForTeamPtr := flag.String("requestMatchesForTeam", "",
91 "If specified, parse the file as a RequestMatchesForTeam JSON request.")
Philipp Schraderacf96232022-03-01 22:03:30 -080092 requestDataScoutingPtr := flag.String("requestDataScouting", "",
93 "If specified, parse the file as a RequestDataScouting JSON request.")
Philipp Schraderd3fac192022-03-02 20:35:46 -080094 refreshMatchListPtr := flag.String("refreshMatchList", "",
95 "If specified, parse the file as a RefreshMatchList JSON request.")
Philipp Schraderd9096a32022-02-24 17:53:09 -080096 flag.Parse()
97
Philipp Schrader30005e42022-03-06 13:53:58 -080098 spew.Config.Indent = *indentPtr
99
Philipp Schraderd9096a32022-02-24 17:53:09 -0800100 // Handle the actual arguments.
Philipp Schrader02ec8e42022-03-26 16:37:18 -0700101 maybePerformRequest(
102 "SubmitDataScouting",
103 "scouting/webserver/requests/messages/submit_data_scouting.fbs",
104 *submitDataScoutingPtr,
105 *addressPtr,
106 debug.SubmitDataScouting)
107
108 maybePerformRequest(
109 "RequestAllMatches",
110 "scouting/webserver/requests/messages/request_all_matches.fbs",
111 *requestAllMatchesPtr,
112 *addressPtr,
113 debug.RequestAllMatches)
114
115 maybePerformRequest(
116 "RequestMatchesForTeam",
117 "scouting/webserver/requests/messages/request_matches_for_team.fbs",
118 *requestMatchesForTeamPtr,
119 *addressPtr,
120 debug.RequestMatchesForTeam)
121
122 maybePerformRequest(
123 "RequestDataScouting",
124 "scouting/webserver/requests/messages/request_data_scouting.fbs",
125 *requestDataScoutingPtr,
126 *addressPtr,
127 debug.RequestDataScouting)
128
129 maybePerformRequest(
130 "RefreshMatchList",
131 "scouting/webserver/requests/messages/refresh_match_list.fbs",
132 *refreshMatchListPtr,
133 *addressPtr,
134 debug.RefreshMatchList)
Philipp Schraderd9096a32022-02-24 17:53:09 -0800135}