forked from devsoc-unsw/circles
-
Notifications
You must be signed in to change notification settings - Fork 0
/
run_app.py
108 lines (95 loc) · 3.14 KB
/
run_app.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
# pylint: disable=cyclic-import
""" run all of circles in one terminal, assuming a unix environment """
import logging
import os
import sys
import threading
from subprocess import Popen, check_call
from typing import TextIO
from dotenv import load_dotenv
class LogPipe(threading.Thread, TextIO):
""" boilerplate abstraction for redirecting the logs of a process """
def __init__(self, level):
"""Setup the object with a logger and a loglevel
and start the thread
"""
threading.Thread.__init__(self)
self.daemon = False
self.level = level
self.fdRead, self.fdWrite = os.pipe()
self.pipeReader = os.fdopen(self.fdRead)
self.start()
def fileno(self):
"""Return the write file descriptor of the pipe"""
return self.fdWrite
def run(self):
"""Run the thread, logging everything."""
for line in iter(self.pipeReader.readline, ''):
logging.log(self.level, line.strip('\n'))
self.pipeReader.close()
def close(self):
"""Close the write end of the pipe."""
os.close(self.fdWrite)
def write(self, message):
""" do write """
logging.log(self.level, message)
def get_backend_env():
"""
reads backend.env for mongodb username and password and python
version.
"""
load_dotenv("./env/backend.env")
username = os.getenv("MONGODB_USERNAME")
password = os.getenv("MONGODB_PASSWORD")
python = os.getenv("PYTHON_VERSION") or "python"
return (username, password, python)
def get_frontend_env():
"""
reads frontend.env for mongodb username and password and python
version.
"""
load_dotenv("./env/frontend.env")
baseurl = os.getenv("VITE_BACKEND_API_BASE_URL")
return baseurl
def main():
if os.system("docker ps") != 0:
print("please run docker first!")
sys.exit(1)
logging.basicConfig(level=logging.INFO,format='%(asctime)s %(message)s',
handlers=[
logging.FileHandler("debug.log", mode='w'),
logging.StreamHandler()
]
)
# this is actually kooky if you think about it
sys.stdout = LogPipe(logging.INFO)
sys.stderr = LogPipe(logging.ERROR)
username, password, python_ver = get_backend_env()
base_url = get_frontend_env()
os.system('docker compose run --rm init-mongo')
try:
Popen(
f'MONGODB_SERVICE_HOSTNAME=localhost MONGODB_PASSWORD={password} MONGODB_USERNAME={username} nodemon --exec {python_ver} runserver.py',
stdout=sys.stdout,
stderr=sys.stderr,
shell=True,
cwd='backend/'
)
check_call(
f'VITE_BACKEND_API_BASE_URL={base_url} npm start',
shell=True,
stdout=sys.stdout,
stderr=sys.stderr,
cwd='frontend/'
)
# pylint: disable=broad-except
except Exception as e:
sys.stdout.write(f'exception - {e}')
finally:
sys.stdout.close()
sys.stderr.close()
sys.stdout = sys.__stdout__
sys.stderr = sys.__stderr__
logging.shutdown()
if __name__ == '__main__':
main()