forked from The-Alpha-Project/alpha-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
148 lines (122 loc) · 4.65 KB
/
main.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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
import multiprocessing
import os
import argparse
from sys import platform
from time import sleep
from game.realm.RealmManager import RealmManager
from game.world import WorldManager
from game.world.managers.maps.MapManager import MapManager
from game.world.managers.maps.MapTile import MapTile
from tools.map_extractor.MapExtractor import MapExtractor
from utils.ConfigManager import config, ConfigManager
from utils.Logger import Logger
from utils.ChatLogManager import ChatLogManager
from utils.PathManager import PathManager
from utils.constants import EnvVars
# Initialize argument parser.
parser = argparse.ArgumentParser()
parser.add_argument(
'-l', '--launch',
help='-l realm to launch realm or -l to launch world, if nothing is specified both are launched',
dest='launch',
action='store',
default=None
)
parser.add_argument(
'-e', '--extract',
help='-e in order to extract .map files',
dest='extract',
action='store_true',
default=False
)
args = parser.parse_args()
def release_process(process):
while process.is_alive():
try:
process.join(timeout=2) # Seconds.
if process.is_alive():
process.terminate()
except (ValueError, KeyboardInterrupt):
sleep(0.1)
if __name__ == '__main__':
# Initialize path.
PathManager.set_root_path(os.path.dirname(os.path.realpath(__file__)))
# Validate configuration file version.
# (Not using Logger since it can fail due to missing config options too).
try:
if config.Version.current != ConfigManager.EXPECTED_VERSION:
print(f'Invalid config.yml version. Expected {ConfigManager.EXPECTED_VERSION} '
f'found {config.Version.current}.')
exit()
except AttributeError:
print(f'Invalid config.yml version. Expected {ConfigManager.EXPECTED_VERSION}, none found.')
exit()
if args.extract:
MapExtractor.run()
exit()
# Validate if maps available and if version match.
if not MapManager.validate_maps():
Logger.error(f'Invalid maps version or maps missing, expected version {MapTile.EXPECTED_VERSION}')
exit()
# if platform != 'win32':
# from signal import signal, SIGPIPE, SIG_DFL
# # https://stackoverflow.com/a/30091579
# signal(SIGPIPE, SIG_DFL)
# Semaphore objects are leaked on shutdown in macOS if using spawn for some reason.
if platform == 'darwin':
context = multiprocessing.get_context('fork')
else:
context = multiprocessing.get_context('spawn')
# Print active env vars.
for env_var_name in EnvVars.EnvironmentalVariables.ACTIVE_ENV_VARS:
env_var = os.getenv(env_var_name, '')
if env_var:
Logger.info(f'Environment variable {env_var_name}: {env_var}')
# Process launching starts here.
launch_realm = not args.launch or args.launch == 'realm'
launch_world = not args.launch or args.launch == 'world'
login_process = None
proxy_process = None
world_process = None
if launch_realm:
login_process = context.Process(target=RealmManager.start_realm)
login_process.start()
proxy_process = context.Process(target=RealmManager.start_proxy)
proxy_process.start()
if not launch_world:
try:
login_process.join()
except:
Logger.info('Terminating login processes...')
if launch_world:
world_process = context.Process(target=WorldManager.WorldServerSessionHandler.start)
world_process.start()
# noinspection PyBroadException
try:
if os.getenv(EnvVars.EnvironmentalVariables.CONSOLE_MODE,
config.Server.Settings.console_mode) in [True, 'True', 'true']:
while input() != 'exit':
Logger.error('Invalid command.')
else:
world_process.join()
except:
Logger.info('Shutting down the core...')
ChatLogManager.exit()
# Send SIGTERM to processes.
# Add checks to send SIGTERM to only running process
if launch_world:
world_process.terminate()
Logger.info('World process terminated.')
if launch_realm:
login_process.terminate()
Logger.info('Login process terminated.')
proxy_process.terminate()
Logger.info('Proxy process terminated.')
# Release process resources.
Logger.info('Waiting to release resources...')
if launch_world:
release_process(world_process)
if launch_realm:
release_process(proxy_process)
release_process(login_process)
Logger.success('Core gracefully shut down.')