Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 1 | """This library is here to run the various servers involved in the scouting app. |
| 2 | |
| 3 | The servers are: |
| 4 | - The fake TBA server |
| 5 | - The actual web server |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 6 | - The postgres database |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 7 | """ |
| 8 | |
| 9 | import argparse |
| 10 | import json |
| 11 | import os |
| 12 | from pathlib import Path |
| 13 | import shutil |
| 14 | import signal |
| 15 | import socket |
| 16 | import subprocess |
| 17 | import sys |
| 18 | import time |
| 19 | from typing import List |
| 20 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 21 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 22 | def wait_for_server(port: int): |
| 23 | """Waits for the server at the specified port to respond to TCP connections.""" |
| 24 | while True: |
| 25 | try: |
| 26 | connection = socket.socket(socket.AF_INET, socket.SOCK_STREAM) |
| 27 | connection.connect(("localhost", port)) |
| 28 | connection.close() |
| 29 | break |
| 30 | except ConnectionRefusedError: |
| 31 | connection.close() |
| 32 | time.sleep(0.01) |
| 33 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 34 | |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 35 | def create_db_config(tmpdir: Path) -> Path: |
| 36 | config = tmpdir / "db_config.json" |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 37 | config.write_text( |
| 38 | json.dumps({ |
| 39 | "username": "test", |
| 40 | "password": "password", |
| 41 | "port": 5432, |
| 42 | })) |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 43 | return config |
| 44 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 45 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 46 | def create_tba_config(tmpdir: Path) -> Path: |
| 47 | # Configure the scouting webserver to scrape data from our fake TBA |
| 48 | # server. |
| 49 | config = tmpdir / "scouting_config.json" |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 50 | config.write_text( |
| 51 | json.dumps({ |
| 52 | "api_key": "dummy_key_that_is_not_actually_used_in_this_test", |
| 53 | "base_url": "http://localhost:7000", |
| 54 | })) |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 55 | return config |
| 56 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 57 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 58 | def set_up_tba_api_dir(tmpdir: Path, year: int, event_code: str): |
| 59 | tba_api_dir = tmpdir / "api" / "v3" / "event" / f"{year}{event_code}" |
| 60 | tba_api_dir.mkdir(parents=True, exist_ok=True) |
| 61 | (tba_api_dir / "matches").write_text( |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 62 | Path(f"scouting/scraping/test_data/{year}_{event_code}.json"). |
| 63 | read_text()) |
| 64 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 65 | |
| 66 | class Runner: |
| 67 | """Helps manage the services we need for testing the scouting app.""" |
| 68 | |
| 69 | def start(self, port: int): |
| 70 | """Starts the services needed for testing the scouting app.""" |
| 71 | self.tmpdir = Path(os.environ["TEST_TMPDIR"]) / "servers" |
| 72 | self.tmpdir.mkdir(exist_ok=True) |
| 73 | |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 74 | db_config = create_db_config(self.tmpdir) |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 75 | tba_config = create_tba_config(self.tmpdir) |
| 76 | |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 77 | # The database needs to be running and addressable before the scouting |
| 78 | # webserver can start. |
| 79 | self.testdb_server = subprocess.Popen( |
| 80 | ["scouting/db/testdb_server/testdb_server_/testdb_server"]) |
| 81 | wait_for_server(5432) |
| 82 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 83 | self.webserver = subprocess.Popen([ |
| 84 | "scouting/scouting", |
| 85 | f"--port={port}", |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 86 | f"--db_config={db_config}", |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 87 | f"--tba_config={tba_config}", |
| 88 | ]) |
| 89 | |
| 90 | # Create a fake TBA server to serve the static match list. |
| 91 | set_up_tba_api_dir(self.tmpdir, year=2016, event_code="nytr") |
| 92 | set_up_tba_api_dir(self.tmpdir, year=2020, event_code="fake") |
| 93 | self.fake_tba_api = subprocess.Popen( |
| 94 | ["python3", "-m", "http.server", "7000"], |
| 95 | cwd=self.tmpdir, |
| 96 | ) |
| 97 | |
| 98 | # Wait for the TBA server and the scouting webserver to start up. |
| 99 | wait_for_server(7000) |
| 100 | wait_for_server(port) |
| 101 | |
| 102 | def stop(self): |
| 103 | """Stops the services needed for testing the scouting app.""" |
Philipp Schrader | 7365d32 | 2022-03-06 16:40:08 -0800 | [diff] [blame] | 104 | servers = (self.webserver, self.testdb_server, self.fake_tba_api) |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 105 | for server in servers: |
| 106 | server.terminate() |
| 107 | for server in servers: |
| 108 | server.wait() |
| 109 | |
| 110 | try: |
| 111 | shutil.rmtree(self.tmpdir) |
| 112 | except FileNotFoundError: |
| 113 | pass |
| 114 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 115 | |
Philipp Schrader | aa76a69 | 2022-05-29 23:55:16 -0700 | [diff] [blame] | 116 | def discard_signal(signum, frame): |
| 117 | """A NOP handler to ignore certain signals. |
| 118 | |
| 119 | We use signal.pause() to wait for a signal. That means we can't use the default handler. The |
| 120 | default handler would tear the application down without stopping child processes. |
| 121 | """ |
| 122 | pass |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 123 | |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 124 | |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 125 | def main(argv: List[str]): |
| 126 | parser = argparse.ArgumentParser() |
Ravago Jones | 5127ccc | 2022-07-31 16:32:45 -0700 | [diff] [blame] | 127 | parser.add_argument("--port", |
| 128 | type=int, |
| 129 | help="The port for the actual web server.") |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 130 | args = parser.parse_args(argv[1:]) |
| 131 | |
| 132 | runner = Runner() |
| 133 | runner.start(args.port) |
| 134 | |
| 135 | # Wait until we're asked to shut down via CTRL-C or SIGTERM. |
Philipp Schrader | aa76a69 | 2022-05-29 23:55:16 -0700 | [diff] [blame] | 136 | signal.signal(signal.SIGINT, discard_signal) |
| 137 | signal.signal(signal.SIGTERM, discard_signal) |
Philipp Schrader | 9430572 | 2022-03-13 12:59:21 -0700 | [diff] [blame] | 138 | signal.pause() |
| 139 | |
| 140 | runner.stop() |
| 141 | |
| 142 | |
| 143 | if __name__ == "__main__": |
| 144 | sys.exit(main(sys.argv)) |