Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 4 | "encoding/json" |
Philipp Schrader | d3fac19 | 2022-03-02 20:35:46 -0800 | [diff] [blame] | 5 | "errors" |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 6 | "flag" |
| 7 | "fmt" |
Philipp Schrader | 4953cc3 | 2022-02-25 18:09:02 -0800 | [diff] [blame] | 8 | "io/ioutil" |
Philipp Schrader | 8747f1b | 2022-02-23 23:56:22 -0800 | [diff] [blame] | 9 | "log" |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 10 | "os" |
| 11 | "os/signal" |
Philipp Schrader | 4af5af4 | 2023-03-11 15:59:51 -0800 | [diff] [blame] | 12 | "path" |
Philipp Schrader | 4e661d6 | 2022-03-13 22:15:56 -0700 | [diff] [blame] | 13 | "strconv" |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 14 | "syscall" |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 15 | "time" |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 16 | |
Philipp Schrader | a9a7939 | 2023-03-25 13:28:31 -0700 | [diff] [blame] | 17 | "github.com/frc971/971-Robot-Code/scouting/background_task" |
Philipp Schrader | 8747f1b | 2022-02-23 23:56:22 -0800 | [diff] [blame] | 18 | "github.com/frc971/971-Robot-Code/scouting/db" |
Philipp Schrader | 0247300 | 2023-03-07 19:10:17 -0800 | [diff] [blame] | 19 | "github.com/frc971/971-Robot-Code/scouting/webserver/driver_ranking" |
Philipp Schrader | 43c730b | 2023-02-26 20:27:44 -0800 | [diff] [blame] | 20 | "github.com/frc971/971-Robot-Code/scouting/webserver/match_list" |
Yash Chainani | 37261d4 | 2022-04-19 16:21:27 -0700 | [diff] [blame] | 21 | "github.com/frc971/971-Robot-Code/scouting/webserver/rankings" |
Philipp Schrader | cdb5cfc | 2022-02-20 14:57:07 -0800 | [diff] [blame] | 22 | "github.com/frc971/971-Robot-Code/scouting/webserver/requests" |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 23 | "github.com/frc971/971-Robot-Code/scouting/webserver/server" |
| 24 | "github.com/frc971/971-Robot-Code/scouting/webserver/static" |
| 25 | ) |
| 26 | |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 27 | type DatabaseConfig struct { |
| 28 | Username string `json:"username"` |
| 29 | Password string `json:"password"` |
| 30 | Port int `json:"port"` |
| 31 | } |
| 32 | |
| 33 | func readDatabaseConfig(configPath string) (*DatabaseConfig, error) { |
| 34 | content, err := ioutil.ReadFile(configPath) |
| 35 | if err != nil { |
| 36 | return nil, errors.New(fmt.Sprint("Failed to open database config at ", configPath, ": ", err)) |
Philipp Schrader | 4953cc3 | 2022-02-25 18:09:02 -0800 | [diff] [blame] | 37 | } |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 38 | |
| 39 | var config DatabaseConfig |
| 40 | if err := json.Unmarshal([]byte(content), &config); err != nil { |
| 41 | return nil, errors.New(fmt.Sprint("Failed to parse database config file as JSON: ", err)) |
Philipp Schrader | 4953cc3 | 2022-02-25 18:09:02 -0800 | [diff] [blame] | 42 | } |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 43 | |
| 44 | return &config, nil |
Philipp Schrader | 4953cc3 | 2022-02-25 18:09:02 -0800 | [diff] [blame] | 45 | } |
| 46 | |
Philipp Schrader | 4e661d6 | 2022-03-13 22:15:56 -0700 | [diff] [blame] | 47 | // Gets the default port to use for the webserver. If wrapped by |
| 48 | // apache_wrapper(), we use the port dictated by the wrapper. |
| 49 | func getDefaultPort() int { |
| 50 | port_str := os.Getenv("APACHE_WRAPPED_PORT") |
| 51 | if port_str != "" { |
| 52 | port, err := strconv.Atoi(port_str) |
| 53 | if err != nil { |
| 54 | log.Fatalf("Failed to parse \"%s\" as integer: %v", port_str, err) |
| 55 | } |
| 56 | return port |
| 57 | } |
| 58 | |
| 59 | return 8080 |
| 60 | } |
| 61 | |
Philipp Schrader | 4af5af4 | 2023-03-11 15:59:51 -0800 | [diff] [blame] | 62 | func getDefaultBlueAllianceConfig() string { |
| 63 | workspaceDirectory := os.Getenv("BUILD_WORKSPACE_DIRECTORY") |
| 64 | if workspaceDirectory != "" { |
| 65 | return path.Join(workspaceDirectory, "scouting_config.json") |
| 66 | } |
| 67 | return "scouting_config.json" |
| 68 | } |
| 69 | |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 70 | func main() { |
Philipp Schrader | 4e661d6 | 2022-03-13 22:15:56 -0700 | [diff] [blame] | 71 | portPtr := flag.Int("port", getDefaultPort(), "The port number to bind to.") |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 72 | dirPtr := flag.String("directory", ".", "The directory to serve at /.") |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 73 | dbConfigPtr := flag.String("db_config", "", |
| 74 | "The postgres database JSON config. It needs the following keys: "+ |
| 75 | "\"username\", \"password\", and \"port\". "+ |
| 76 | "This option cannot be used in conjunction with -testdb_port.") |
| 77 | testDbPortPtr := flag.Int("testdb_port", 0, |
| 78 | "If set, connects to an instance of //scouting/db/testdb_server at the "+ |
| 79 | "specified port. This option cannot be used in conjunction with "+ |
| 80 | "-db_config.") |
| 81 | dbConnectRetries := flag.Int("db_retries", 5, |
| 82 | "The number of seconds to retry connecting to the database on startup.") |
Philipp Schrader | 4af5af4 | 2023-03-11 15:59:51 -0800 | [diff] [blame] | 83 | blueAllianceConfigPtr := flag.String("tba_config", getDefaultBlueAllianceConfig(), |
Philipp Schrader | d3fac19 | 2022-03-02 20:35:46 -0800 | [diff] [blame] | 84 | "The path to your The Blue Alliance JSON config. "+ |
| 85 | "It needs an \"api_key\" field with your TBA API key. "+ |
Philipp Schrader | 4af5af4 | 2023-03-11 15:59:51 -0800 | [diff] [blame] | 86 | "It needs a \"year\" field with the event year. "+ |
| 87 | "It needs an \"event_code\" field with the event code. "+ |
Philipp Schrader | 72beced | 2022-03-07 05:29:52 -0800 | [diff] [blame] | 88 | "Optionally, it can have a \"base_url\" field with the TBA API base URL.") |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 89 | flag.Parse() |
| 90 | |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 91 | // Set up the database config. It's either specified via a JSON file on |
| 92 | // disk (-db_config) or we can generate an in-memory config when the |
| 93 | // user wants to connect to a `testdb_server` instance (-testdb_port). |
| 94 | var dbConfig *DatabaseConfig |
| 95 | var err error |
| 96 | if *dbConfigPtr != "" { |
| 97 | if *testDbPortPtr != 0 { |
| 98 | log.Fatal("Cannot specify both -db_config and -testdb_port. Choose one.") |
| 99 | } |
| 100 | dbConfig, err = readDatabaseConfig(*dbConfigPtr) |
| 101 | if err != nil { |
| 102 | log.Fatal("Failed to read ", *dbConfigPtr, ": ", err) |
| 103 | } |
| 104 | } else if *testDbPortPtr != 0 { |
| 105 | dbConfig = &DatabaseConfig{ |
| 106 | Username: "test", |
| 107 | Password: "password", |
| 108 | Port: *testDbPortPtr, |
| 109 | } |
| 110 | } else { |
| 111 | log.Fatal("Must specify one of -db_config and -testdb_port. See " + |
| 112 | "https://github.com/frc971/971-Robot-Code/blob/master/scouting/README.md " + |
| 113 | "for more information.") |
| 114 | } |
| 115 | |
| 116 | // Connect to the database. It may still be starting up so we do a |
| 117 | // bunch of retries here. The number of retries can be set on the |
| 118 | // command line. |
| 119 | var database *db.Database |
| 120 | for i := 0; i < *dbConnectRetries*10; i++ { |
| 121 | database, err = db.NewDatabase(dbConfig.Username, dbConfig.Password, dbConfig.Port) |
| 122 | if err == nil { |
| 123 | break |
| 124 | } |
| 125 | if i%10 == 0 { |
| 126 | log.Println("Failed to connect to the database. Retrying in a bit.") |
| 127 | } |
| 128 | time.Sleep(1 * time.Millisecond) |
| 129 | } |
Philipp Schrader | 8747f1b | 2022-02-23 23:56:22 -0800 | [diff] [blame] | 130 | if err != nil { |
| 131 | log.Fatal("Failed to connect to database: ", err) |
| 132 | } |
Philipp Schrader | eecb896 | 2022-06-01 21:02:42 -0700 | [diff] [blame] | 133 | defer database.Delete() |
Philipp Schrader | 8747f1b | 2022-02-23 23:56:22 -0800 | [diff] [blame] | 134 | |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 135 | scoutingServer := server.NewScoutingServer() |
Emily Markova | 521725a | 2024-03-21 18:46:04 -0700 | [diff] [blame^] | 136 | var realClock requests.RealClock |
Emily Markova | faecfe1 | 2023-07-01 12:40:03 -0700 | [diff] [blame] | 137 | static.ServePages(scoutingServer, *dirPtr, database) |
Emily Markova | 521725a | 2024-03-21 18:46:04 -0700 | [diff] [blame^] | 138 | requests.HandleRequests(database, scoutingServer, realClock) |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 139 | scoutingServer.Start(*portPtr) |
| 140 | fmt.Println("Serving", *dirPtr, "on port", *portPtr) |
| 141 | |
Philipp Schrader | 43c730b | 2023-02-26 20:27:44 -0800 | [diff] [blame] | 142 | // Since Go doesn't support default arguments, we use 0 and "" to |
| 143 | // indicate that we want to source the values from the config. |
| 144 | |
Philipp Schrader | 3f0e36f | 2023-03-25 13:49:41 -0700 | [diff] [blame] | 145 | matchListScraper := background_task.New(10 * time.Minute) |
Philipp Schrader | 43c730b | 2023-02-26 20:27:44 -0800 | [diff] [blame] | 146 | matchListScraper.Start(func() { |
| 147 | match_list.GetMatchList(database, 0, "", *blueAllianceConfigPtr) |
| 148 | }) |
| 149 | |
Philipp Schrader | 3f0e36f | 2023-03-25 13:49:41 -0700 | [diff] [blame] | 150 | rankingsScraper := background_task.New(10 * time.Minute) |
Philipp Schrader | c49eaf7 | 2023-02-26 16:56:52 -0800 | [diff] [blame] | 151 | rankingsScraper.Start(func() { |
| 152 | rankings.GetRankings(database, 0, "", *blueAllianceConfigPtr) |
| 153 | }) |
Yash Chainani | 37261d4 | 2022-04-19 16:21:27 -0700 | [diff] [blame] | 154 | |
Philipp Schrader | 3f0e36f | 2023-03-25 13:49:41 -0700 | [diff] [blame] | 155 | driverRankingParser := background_task.New(10 * time.Minute) |
Philipp Schrader | 0247300 | 2023-03-07 19:10:17 -0800 | [diff] [blame] | 156 | driverRankingParser.Start(func() { |
| 157 | // Specify "" as the script path here so that the default is |
| 158 | // used. |
| 159 | driver_ranking.GenerateFullDriverRanking(database, "") |
| 160 | }) |
| 161 | |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 162 | // Block until the user hits Ctrl-C. |
| 163 | sigint := make(chan os.Signal, 1) |
| 164 | signal.Notify(sigint, syscall.SIGINT) |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 165 | signal.Notify(sigint, syscall.SIGTERM) |
| 166 | fmt.Println("Waiting for CTRL-C or SIGTERM.") |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 167 | <-sigint |
| 168 | |
| 169 | fmt.Println("Shutting down.") |
| 170 | scoutingServer.Stop() |
Philipp Schrader | c49eaf7 | 2023-02-26 16:56:52 -0800 | [diff] [blame] | 171 | rankingsScraper.Stop() |
Philipp Schrader | 0247300 | 2023-03-07 19:10:17 -0800 | [diff] [blame] | 172 | driverRankingParser.Stop() |
Philipp Schrader | 43c730b | 2023-02-26 20:27:44 -0800 | [diff] [blame] | 173 | matchListScraper.Stop() |
Philipp Schrader | 5562df7 | 2022-02-16 20:56:51 -0800 | [diff] [blame] | 174 | fmt.Println("Successfully shut down.") |
| 175 | } |