forked from 07th-mod/python-patcher
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
302 lines (248 loc) · 11.6 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
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
#!/usr/bin/python
from __future__ import print_function, unicode_literals, with_statement
import argparse
import locale
import os
import sys
import platform
# Embedded python doesn't have current directory as path
import tempfile
import time
if os.getcwd() not in sys.path:
print("Startup: Adding {} to path".format(os.getcwd()))
sys.path.append(os.getcwd())
import datetime
import pprint
import socket
import threading
import traceback
import common
import httpGUI
import installConfiguration
import logger
import fileVersionManagement
try: input = raw_input
except NameError: pass
pp = pprint.PrettyPrinter(indent=4)
try:
from urllib.request import urlopen, Request
from urllib.error import HTTPError
except ImportError:
from urllib2 import urlopen, Request, HTTPError
def check07thModServerConnection():
"""
Makes sure that we can connect to the 07th-mod server, where the mod downloads are stored.
This check is only required because Japan is blocked from downloading the 07th-mod website - the installer
already downloads a file from Github during startup, which suffices to check the internet connection.
This function/check can be removed if the block is removed.
"""
try:
_ = common.downloadFile("https://07th-mod.com/", is_text=True)
except Exception as error:
traceback.print_exc()
raise Exception("""------------------------------------------------------------------------
Error: Couldn't reach 07th Mod Server! (https://07th-mod.com/)
If you have a working internet connection, most likely you are in Japan, which is blocked from the 07th-mod server.
Please visit https://www.07th-mod.com - if you get a "406 Not Acceptable" error, you are definitely blocked.
As a workaround, VPNs like ProtonVPN could be used...
Otherwise, please check the following:
- You have a working internet connection
- Check if our website is down (https://07th-mod.com/)
- Check our Wiki for more solutions: https://07th-mod.com/wiki/Installer/faq/#connection-troubleshooting
------------------------------------------------------------------------
Dev Error Message: {}
""".format(error))
def getModList(is_developer=True):
if is_developer and os.path.exists('installData.json'):
return common.getModList("installData.json", isURL=False)
else:
return common.getModList(common.Globals.GITHUB_MASTER_BASE_URL + "installData.json", isURL=True)
def getSubModConfigList(modList):
subModconfigList = []
for mod in modList:
for submod in mod['submods']:
conf = installConfiguration.SubModConfig(mod, submod)
subModconfigList.append(conf)
return subModconfigList
def installerCommonStartupTasks():
"""
Peforms tasks common to both the normal GUI installer and the CLI Installer
- Change current directory to path of the launched python file
- Setup logging
- Read the launcher path (works only on Windows)
- Enable developer mode if installData.json found on disk
- Log information about the environment (current dir, python version etc.)
- On Windows, check for hostname problems
"""
errors = []
print("Installer script name (argv[0]): [{}]".format(sys.argv[0]))
# If you double-click on the file in Finder on macOS, it will not open with a path that is near the .py file
# Since we want to properly find things like `./aria2c`, we should move to that path first.
dirname = os.path.dirname(sys.argv[0])
if dirname.strip():
os.chdir(dirname)
# redirect stdout to both a file and console
# TODO: on MAC using a .app file, not sure if this logfile will be writeable
# could do a try-catch, and then only begin logging once the game path has been set?
sys.stdout = logger.Logger(common.Globals.LOG_FILE_PATH)
logger.setGlobalLogger(sys.stdout)
sys.stderr = logger.StdErrRedirector(sys.stdout)
parser = argparse.ArgumentParser()
parser.add_argument(
'--launcher-path',
dest="launcher_path",
default=None,
help=('Optionally specify the path to the Windows install launcher. '
'This Python script may call the install launcher with special arguments (eg. to open the file picker)')
)
parser.add_argument(
"-ao",
"--asset-os",
action="store",
dest="force_asset_os_string",
metavar="ASSET_OS",
default=None,
choices=['windows', 'linux', 'mac'],
help=(
'Force the installer to install assets from another operating system'
'Mainly used on Linux to install Windows assets for use under Wine'
),
)
parser.add_argument(
'--no-launch-browser',
action='store_true',
help=(
'Launch the browser automatically after web server started'
)
)
args = parser.parse_args()
if args.no_launch_browser:
common.Globals.LAUNCH_BROWSER = False
# Optional first argument tells the script the path of the launcher (currently only used with Windows launcher)
if args.launcher_path is not None:
common.Globals.NATIVE_LAUNCHER_PATH = args.launcher_path
print("Launcher is located at [{}]".format(common.Globals.NATIVE_LAUNCHER_PATH))
else:
if common.Globals.IS_WINDOWS:
print("WARNING: Launcher path not given to Python script. Will try to use PowerShell file chooser instead of native one.")
common.Globals.FORCE_ASSET_OS_STRING = args.force_asset_os_string
if common.Globals.FORCE_ASSET_OS_STRING is not None:
print("Warning: Force asset argument passed - will install {} assets despite os being {}".format(common.Globals.FORCE_ASSET_OS_STRING, common.Globals.OS_STRING))
# Enable developer mode if we detect the program is run from the git repository
# Comment out this line to simulate a 'normal' installation - files will be fetched from the web.
if os.path.exists("installData.json"):
common.Globals.DEVELOPER_MODE = True
print("""------ NOTE: Developer mode is enabled (will use installData.json from disk) ----""")
print("> Install Started On {}".format(datetime.datetime.now()))
common.Globals.getBuildInfo()
print("Python {}".format(sys.version))
print("Operating System: {} {} {} ({})".format(platform.system(), platform.release(), platform.machine(), platform.version()))
print("Locale - Default: {} | Preferred: {} | Filesystem: {}".format(sys.getdefaultencoding(), locale.getpreferredencoding(), sys.getfilesystemencoding()))
print("Installer Build Information: {}".format(common.Globals.BUILD_INFO))
print("Installer is being run from: [{}]".format(os.getcwd()))
# Windows only checks
if common.Globals.IS_WINDOWS:
# Check for non-ascii characters in hostname, which prevent the server starting up
if not all(ord(c) < 128 for c in socket.gethostname()):
errors.append(
"ERROR: It looks like your hostname [{}] contains non-ASCII characters. This may prevent the installer from starting up.\n"
"Please change your hostname to only contain ASCII characters, then restart the installer.".format(socket.gethostname())
)
# Check if installer is being run from system root (C:\Windows for example)
system_root = os.environ.get('SYSTEMROOT')
if system_root:
if os.path.realpath(os.getcwd()).startswith(os.path.realpath(system_root)):
errors.append("ERROR: You are trying to run the installer from the system folder [{}]. Please do not use the start menu to launch the installer. Please run the installer from a user writeable folder instead".format(dirname))
# Check if the current folder is writeable
try:
# Write some dummy data to a temp file
test_data = "test"
temp_file_handle, temp_file_path = tempfile.mkstemp(dir='.')
with os.fdopen(temp_file_handle, 'w') as temp_file:
temp_file.write(test_data)
# Wait for the file to appear on the filesystem (in most cases this happens immediately)
for i in range(3):
if os.path.exists(temp_file_path):
break
time.sleep(.5)
# Read back the file and check its contents is the same that was written earlier
with open(temp_file_path) as temp_file:
if temp_file.read() != test_data:
errors.append(
"ERROR: File written to installer folder was not readable [{}]. Please run the installer from a user writeable folder instead".format(temp_file_path))
# Remove the temp file
os.remove(temp_file_path)
except Exception as e:
traceback.print_exc()
errors.append("ERROR: Installer folder is not writeable [{}]. Please run the installer from a user writeable folder instead. Full error:\n{}".format(os.getcwd(), e))
if errors:
print('\n--------------------------------------------------------------')
print('The following problems were found during startup:')
print('- ', end='')
print('\n- '.join(errors))
print('--------------------------------------------------------------')
print('Please try to fix these errors before continuing, then restart the installer.')
input('If you think the error was a false positive, press ENTER to continue anyway')
input("If you're absolutely sure you want to continue, press ENTER again")
if __name__ == "__main__":
installerCommonStartupTasks()
print("\n\n----------------------------------------- PLEASE READ -----------------------------------------\n")
print(" - Do not close this window until you are finished with the installer! Closing this window will\n"
" stop the installer!")
print(" - A web page should have opened - do not close it! The web page is the installer's user interface.")
print(" - On the web page, click the game you want to mod to start the installation.")
print("\n----------------------------------------- PLEASE READ -----------------------------------------\n")
installerGUI = httpGUI.InstallerGUI()
def thread_getSubModConfigList():
modList = getModList(common.Globals.DEVELOPER_MODE)
common.Globals.loadCachedDownloadSizes(modList)
return getSubModConfigList(modList)
def thread_unimportantTasks():
t_loadDonations = common.makeThread(installerGUI.loadDonationStatus)
t_loadLatestInstallerStatus = common.makeThread(common.Globals.loadInstallerLatestStatus)
t_preloadModUpdatesHTML = common.makeThread(installerGUI.preloadModUpdatesHTML)
t_loadDonations.start()
t_loadLatestInstallerStatus.start()
t_preloadModUpdatesHTML.start()
try:
t_loadDonations.join(timeout=6)
except Exception as e:
print(e)
try:
t_loadLatestInstallerStatus.join(timeout=6)
except Exception as e:
print(e)
try:
t_preloadModUpdatesHTML.join(timeout=6)
except Exception as e:
print(e)
def doInstallerInit():
try:
if common.Globals.IS_MAC:
common.Globals.macUnQuarantineExecutable("./.aria2c")
common.Globals.macUnQuarantineExecutable("./.7za")
# Executable scanning must happen first, as other init operations might require Aria or CURL to download
common.Globals.scanForExecutables()
common.Globals.scanCertLocation()
common.Globals.chooseCurlCertificate()
# Run remaining init tasks concurrently
t_getSubModConfig = common.makeThread(thread_getSubModConfigList)
t_unimportantTasks = common.makeThread(thread_unimportantTasks)
t_check07thModServer = common.makeThread(check07thModServerConnection)
common.startAndJoinThreads([t_getSubModConfig, t_unimportantTasks, t_check07thModServer])
if common.Globals.DEVELOPER_MODE:
fileVersionManagement.Developer_ValidateVersionDataJSON(t_getSubModConfig.result)
# Indicate init is complete. This causes the browser to advance from loading_screen.html to index.html
installerGUI.setSubModconfigs(t_getSubModConfig.result)
except Exception as e:
print(traceback.format_exc())
# Indicate init failed. This causes the browser to show an error message.
installerGUI.setInitError(e, traceback.format_exc())
# The installer initialization (scan for executables, check network, retrieve mod list) is launched
# concurrently with the Web GUI. The Web GUI shows a loading screen until init is complete.
threading.Thread(target=doInstallerInit).start()
try:
installerGUI.server_test()
except KeyboardInterrupt:
installerGUI.shutdown()
logger.Logger.globalLogger.close_all_logs()