-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy paththor.py
114 lines (96 loc) · 4.03 KB
/
thor.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
109
110
111
112
113
114
"""
Thor CLI.
CLI that bootstraps a SSPBFT application according to configuration found
in conf/custom|default.ini and command line arguments.
"""
# standard
import argparse
import signal
import threading
import logging
import subprocess
import sys
# local
from local.bootstrap import bootstrap as local_bootstrap
from planetlab.bootstrap import bootstrap as planetlab_bootstrap
from helpers.enums import Mode
from conf import config
import helpers.ps as ps
logger = logging.getLogger(__name__)
def check_mode(mode):
"""Validates that the supplied mode is a valid mode."""
if mode not in Mode.__members__:
raise argparse.ArgumentTypeError("Only modes [local] and " +
"[planetlab] are allowed.")
return Mode[mode]
def setup_argparse():
"""Configures CLI and updates conf if options/args are supplied."""
parser = argparse.ArgumentParser()
parser.add_argument("mode", help="either [local] or [planetlab]",
type=check_mode)
parser.add_argument("-d", "--debug", help="all output to current shell",
action="store_true")
parser.add_argument("-n", "--nodes", help="total number of nodes",
type=int)
parser.add_argument("-f", "--faulty", help="number of byzantine nodes",
type=int)
parser.add_argument("-p", "--path", help="absolute path to app to run")
parser.add_argument("-e", "--entrypoint", help="start script for app")
parser.add_argument("-m", "--metrics", help="start heimdall metrics",
action="store_true")
parser.add_argument("-i", "--id", help="id of the current node", type=int)
parser.add_argument("-lp", "--logpath", help="where to write app log")
parser.add_argument("-rs", "--runsleep", help="s to sleep in module.run")
parser.add_argument("-nss", "--non-selfstab",
help="run without self-stabilization",
action="store_true")
parser.add_argument("-ss", "--start-state", help="tell app that it " +
"should load start_state.json in conf/",
action="store_true")
parser.add_argument("-s", "--scale", help="scale instances on this node",
type=int, default=1)
parser.add_argument("-c", "--clients", help="number of clients",
type=int, default=6)
parser.add_argument("-pf", "--profiling",
help="run app with profiling",
action="store_true")
args = parser.parse_args()
if args.nodes is not None:
config.set_node_count(args.nodes)
if args.faulty is not None:
config.set_byzantine_count(args.faulty)
if args.path is not None:
config.set_app_path(args.path)
if args.entrypoint is not None:
config.set_entrypoint(args.entrypoint)
return args
def on_sig_term(signal, frame):
"""Kills subprocess and terminates Thor on CTRL + C."""
ps.kill_all_subprocesses()
# run docker-compose down in heimdall directory to kill heimdall as well
path = config.get_heimdall_root()
try:
p = subprocess.Popen("docker-compose down", shell=True, cwd=path)
p.wait()
except FileNotFoundError:
logger.info(f"Invalid Heimdall path {path} configured")
sys.exit(0)
def setup_logging():
"""Configures the logging for Thor."""
FORMAT = "\033[1mThor.%(name)s ==> [%(levelname)s] : " + \
"%(message)s\033[0m"
logging.basicConfig(format=FORMAT, level=logging.INFO)
if __name__ == "__main__":
args = setup_argparse()
setup_logging()
if args.mode == Mode.local:
local_bootstrap(args)
logger.info("All processes launched in background... (CTRL+C to kill)")
elif args.mode == Mode.planetlab:
planetlab_bootstrap(args)
logger.info("Running process in background... (CTRL+C to kill)")
else:
raise NotImplementedError
signal.signal(signal.SIGINT, on_sig_term)
forever = threading.Event()
forever.wait()