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