-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathpokebot.py
158 lines (126 loc) · 5.56 KB
/
pokebot.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
149
150
151
152
153
154
155
156
157
158
"""Main program entrypoint."""
import argparse
import atexit
import os
import pathlib
import platform
from dataclasses import dataclass
from modules.runtime import is_bundled_app, get_base_path
from modules.version import pokebot_name, pokebot_version
OS_NAME = platform.system()
gui = None
# On Windows, the bot can be started by clicking this Python file. In that case, the terminal
# window is only open for as long as the bot runs, which would make it impossible to see error
# messages during a crash.
# For those cases, we register an `atexit` handler that will wait for user input before closing
# the terminal window.
def on_exit() -> None:
if OS_NAME == "Windows":
import psutil
import os
parent_process_name = psutil.Process(os.getppid()).name()
if parent_process_name == "py.exe" or is_bundled_app():
if gui is not None and gui.window is not None:
gui.window.withdraw()
input("\nPress Enter to close...")
atexit.register(on_exit)
@dataclass
class StartupSettings:
profile: "Profile | None"
debug: bool
bot_mode: str
headless: bool
no_video: bool
no_audio: bool
no_theme: bool
emulation_speed: int
always_on_top: bool
config_path: str
def directory_arg(value: str) -> pathlib.Path:
"""Determine if the value is a valid readable directory.
:param value: Directory to verify.
"""
path_obj = pathlib.Path(value)
if not path_obj.is_dir() or not path_obj.exists():
from modules import exceptions
raise exceptions.CriticalDirectoryMissing(value)
return path_obj
def parse_arguments(bot_mode_names: list[str]) -> StartupSettings:
"""Parses command-line arguments."""
parser = argparse.ArgumentParser(description=f"{pokebot_name} {pokebot_version}")
parser.add_argument(
"profile",
nargs="?",
help="Profile to initialize. Otherwise, the profile selection menu will appear.",
)
parser.add_argument("-m", "--bot-mode", choices=bot_mode_names, help="Initial bot mode (default: Manual).")
parser.add_argument(
"-s",
"--emulation-speed",
choices=["0", "1", "2", "3", "4", "8", "16", "32"],
help="Initial emulation speed (0 for unthrottled; default: 1)",
)
parser.add_argument("-hl", "--headless", action="store_true", help="Run without a GUI, only using the console.")
parser.add_argument("-nv", "--no-video", action="store_true", help="Turn off video output by default.")
parser.add_argument("-na", "--no-audio", action="store_true", help="Turn off audio output by default.")
parser.add_argument("-nt", "--no-theme", action="store_true", help="Turn off the fancy GUI theme.")
parser.add_argument(
"-t", "--always-on-top", action="store_true", help="Keep the bot window always on top of other windows."
)
parser.add_argument("-d", "--debug", action="store_true", help="Enable extra debug options and a debug menu.")
parser.add_argument("-c", "--config", type=directory_arg, dest="config_path", help=argparse.SUPPRESS)
args = parser.parse_args()
preselected_profile: Profile | None = None
if args.profile and profile_directory_exists(args.profile):
preselected_profile = load_profile_by_name(args.profile)
return StartupSettings(
profile=preselected_profile,
debug=bool(args.debug),
bot_mode=args.bot_mode or "Manual",
headless=bool(args.headless),
no_video=bool(args.no_video),
no_audio=bool(args.no_audio),
no_theme=bool(args.no_theme),
emulation_speed=int(args.emulation_speed or "1"),
always_on_top=bool(args.always_on_top),
config_path=args.config_path,
)
if __name__ == "__main__":
if not is_bundled_app():
from requirements import check_requirements
check_requirements()
from modules.context import context
from modules.console import console
from modules.exceptions_hook import register_exception_hook
from modules.main import main_loop
from modules.modes import get_bot_mode_names
from modules.plugins import load_plugins
from modules.profiles import Profile, profile_directory_exists, load_profile_by_name
from updater import run_updater
register_exception_hook()
load_plugins()
# This catches the signal Windows emits when the underlying console window is closed
# by the user. We still want to save the emulator state in that case, which would not
# happen by default!
if OS_NAME == "Windows":
import win32api
def win32_signal_handler(signal_type):
if signal_type == 2 and context.emulator is not None:
context.emulator.shutdown()
win32api.SetConsoleCtrlHandler(win32_signal_handler, True)
startup_settings = parse_arguments(get_bot_mode_names())
console.print(f"Starting [bold cyan]{pokebot_name} {pokebot_version}![/]")
if not is_bundled_app() and not (get_base_path() / ".git").is_dir():
run_updater()
if startup_settings.headless:
from modules.gui.headless import PokebotHeadless
gui = PokebotHeadless(main_loop, on_exit)
else:
from modules.gui import PokebotGui
# Previously, theming could _only_ be disabled through an environment flag. Now, it can
# be disabled using a command-line argument but for backward-compatibility reasons we
# accept either.
no_theme = os.getenv("POKEBOT_UNTHEMED") == "1" or startup_settings.no_theme
gui = PokebotGui(main_loop, on_exit, no_theme=no_theme)
context.gui = gui
gui.run(startup_settings)