forked from 07th-mod/python-patcher
-
Notifications
You must be signed in to change notification settings - Fork 0
/
gameScanner.py
425 lines (349 loc) · 16.7 KB
/
gameScanner.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
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
from __future__ import unicode_literals
import glob
import json
import re
import common
import os
import subprocess
import traceback
import installConfiguration
import logger
try:
from typing import List, Optional
except ImportError:
pass # Just needed for pycharm comments
def findAdditionalSteamLibraries(mainSteamPath):
# type: (str) -> List[str]
r"""
Try to locate additional steam libraries using the steamapps/libraryfolders.vdf file
This function should never throw an exception, and instead return an empty list
NOTE: we had some old code for scanning vdf files, but looks like the format and
vdf file path has changed since then.
"""
try:
vdf = os.path.join(mainSteamPath, "steamapps", "libraryfolders.vdf")
if not os.path.exists(vdf):
return []
import io
baseInstallFolderRegex = re.compile(r'^\s*"path"\s+"([^"]+)"', re.MULTILINE)
with io.open(vdf, 'r', encoding='UTF-8') as vdfFile:
return baseInstallFolderRegex.findall(vdfFile.read())
except:
traceback.print_exc()
return []
def getSecondarySteamPaths(baseSteamPaths):
# type: (List[str]) -> List[str]
r"""
Given a list of possible steam paths (like 'c:\\program files (x86)\\steam)) returns a list of secondary steam paths
"""
libraries = []
for basePath in baseSteamPaths:
libraries += findAdditionalSteamLibraries(basePath)
return libraries
def deDuplicatePaths(originalPaths):
# type: (List[str]) -> List[str]
r"""
Returns the input list with any duplicate paths removed
Paths are considered duplicates if they compare equal after os.path.abspath and os.path.normcase are called on them
"""
try:
# Normalize the paths
paths = [os.path.normcase(os.path.abspath(p)) for p in originalPaths]
# Eliminate duplicate paths
return list(dict.fromkeys(paths))
except:
return originalPaths
def findPossibleGamePathsWindows():
r"""
Blindly retrieve all game folders in the `Steam\steamappps\common` folder (no filtering is performed)
TODO: scan other locations than just the steamapps folder
:return: a list of absolute paths, which are the folders in the `Steam\steamappps\common` folder
:rtype: list[str]
"""
try:
import winreg
except ImportError:
import _winreg as winreg
allSteamPaths = []
try:
try:
registryKey = winreg.OpenKey(winreg.HKEY_CURRENT_USER, r'Software\Valve\Steam')
except WindowsError:
# I installed Steam on a Win 10 64-bit machine and it used this alternate registry key location. Not sure why.
registryKey = winreg.OpenKey(winreg.HKEY_LOCAL_MACHINE, r'Software\Wow6432Node\Valve\Steam')
defaultSteamPath, _regType = winreg.QueryValueEx(registryKey, 'SteamPath')
allSteamPaths.append(defaultSteamPath)
winreg.CloseKey(registryKey)
except WindowsError:
print("findPossibleGamePathsWindows: Couldn't read Steam registry key - Steam not installed?")
return []
# NOTE: this code only works with the old steam config.vdf file. I've added some more code
# to handle the new format, but left this code in incase it ends up finding more paths.
#
# now that we know the steam path, search the "Steam\config\config.vdf" file for extra install paths
# this is a purely optional step, so it's OK if it fails
try:
import io
baseInstallFolderRegex = re.compile(r'^\s*"BaseInstallFolder_\d+"\s*"([^"]+)"', re.MULTILINE)
steamConfigVDFLocation = os.path.join(defaultSteamPath, r'config\config.vdf')
with io.open(steamConfigVDFLocation, 'r', encoding='UTF-8') as configVDFFile:
allSteamPaths += baseInstallFolderRegex.findall(configVDFFile.read())
except Exception as e:
traceback.print_exc()
# Find additional steam libraries
allSteamPaths += getSecondarySteamPaths(allSteamPaths)
allSteamPaths = deDuplicatePaths(allSteamPaths)
print("Scanning Steam install locations: {}".format(allSteamPaths))
# normpath added so returned paths have consistent slash directions (registry key has forward slashes on Win...)
allPossibleGamePaths = []
for steamCommonPath in (os.path.join(steamPath, r'steamapps\common') for steamPath in allSteamPaths):
try:
for gameFolderName in os.listdir(steamCommonPath):
gameFolderPath = os.path.join(steamCommonPath, gameFolderName)
if os.path.isdir(gameFolderPath):
allPossibleGamePaths.append(
os.path.normpath(
gameFolderPath
)
)
except:
traceback.print_exc()
print("findPossibleGamePathsWindows: Failed to scan steam folder {}".format(steamCommonPath))
return allPossibleGamePaths
# Get paths which COULD be game paths.
def getMaybeGamePaths():
"""
If supported, searches the computer for things that might be Higurashi games
Currently only does things on Mac OS and Windows
TODO: Find ways to search for games on Linux
:param str gameName: The name of the game to search for (should be either "Higurashi" or "Umineko"), used to reduce the time spent searching on Mac OS
:return: A list of game paths that might be Higurashi games
:rtype: list[str]
"""
allPossibleGamePaths = []
if common.Globals.IS_WINDOWS:
allPossibleGamePaths.extend(findPossibleGamePathsWindows())
if common.Globals.IS_MAC:
# mdfind is kind of slow, don't run it more than we have to
allPossibleGamePaths.extend(
x for x in subprocess.check_output(["mdfind", "kMDItemContentType == com.apple.application-bundle && ** == '*Higurashi*'"])
.decode("utf-8")
.split("\n") if x
)
for gamePath in subprocess.check_output(["mdfind", "kMDItemContentType == com.apple.application-bundle && ** == '*Umineko*'"]).decode("utf-8").split("\n"):
allPossibleGamePaths.append(gamePath)
# GOG installer makes a `.app` that contains the actual game at `/Contents/Resources/game`
gogPath = os.path.join(gamePath, "Contents/Resources/game")
if os.path.exists(gogPath):
allPossibleGamePaths.append(gogPath)
# Scan hardcoded paths for game subfolders
hardCodedGameContainingPaths = []
if common.Globals.IS_MAC:
hardCodedGameContainingPaths.append("~/Library/Application Support/Steam/steamapps/common")
hardCodedGameContainingPaths.append("~/GOG Games") # Not sure if this is correct for MacOS
if common.Globals.IS_WINDOWS:
hardCodedGameContainingPaths.append("c:/games/Mangagamer")
hardCodedGameContainingPaths.append("c:/GOG Games")
if common.Globals.IS_LINUX:
hardCodedGameContainingPaths.append("~/.steam/steam/steamapps/common")
hardCodedGameContainingPaths.append("~/.steam/steambeta/steamapps/common")
hardCodedGameContainingPaths.append("~/.var/app/com.valvesoftware.Steam/data/Steam/steamapps/common") # Steam Flatpak
hardCodedGameContainingPaths.append("~/GOG Games") # GOG's website states this, but is unconfirmed
# Expand '~' before continuing to the next steps so os.path.* functions work correctly
hardCodedGameContainingPaths = [os.path.realpath(os.path.expanduser(p)) for p in hardCodedGameContainingPaths]
# Try to find secondary steam folders. Need to remove the 'steamapps/common' part of path to get base steam path
try:
baseHardCodedSteamPaths = [os.path.split(os.path.split(p)[0])[0] for p in hardCodedGameContainingPaths if 'steam' in p.lower()]
hardCodedGameContainingPaths += [os.path.realpath(os.path.join(p, "steamapps", "common")) for p in getSecondarySteamPaths(baseHardCodedSteamPaths)]
except:
traceback.print_exc()
# Remove any duplicated paths
hardCodedGameContainingPaths = deDuplicatePaths(hardCodedGameContainingPaths)
print("Scanning Secondary install locations:")
for hardCodedPath in hardCodedGameContainingPaths:
try:
for gameFolderName in os.listdir(hardCodedPath):
gameFolderPath = os.path.normpath(os.path.join(hardCodedPath, gameFolderName))
if os.path.isdir(gameFolderPath):
allPossibleGamePaths.append(gameFolderPath)
print(" - [{}] - OK".format(hardCodedPath))
except Exception as e:
print(" - [{}] - {}".format(hardCodedPath, e))
# Remove any duplicate game paths
allPossibleGamePaths = deDuplicatePaths(allPossibleGamePaths)
# if all methods fail, return empty list
return sorted(allPossibleGamePaths)
def getPossibleIdentifiersFromFolder(folderPath):
# type: (str) -> List[str]
# Given a folder, retrieves the possible identifiers for that folder
if not os.path.exists(folderPath):
print("WARNING: getPossibleIdentifiersFromPath() on path [{}] but path didn't exist".format(folderPath))
return []
if not os.path.isdir(folderPath):
print("WARNING: getPossibleIdentifiersFromPath() on path [{}] but path is not a folder".format(folderPath))
return []
infoPlist = os.path.join(folderPath, "Contents/Info.plist")
if common.Globals.IS_MAC and os.path.exists(infoPlist):
try:
info = subprocess.check_output(
["plutil", "-convert", "json", "-o", "-", infoPlist]
)
parsed = json.loads(common.ensureUnicodeOrStr(info))
name = parsed["CFBundleExecutable"] + "_Data" # type: str
# GoG Umineko installs will be formatted like this but we *don't* want to use it
if name.startswith("Higurashi"):
return [name]
else:
return []
except (subprocess.CalledProcessError, KeyError):
pass
return os.listdir(folderPath)
def gamePathIsPartiallyUninstalled(gamePath):
# type: (str) -> bool
try:
filesInGamePath = os.listdir(gamePath)
def regexExistsInGamePath(regexString):
return any(re.match(regexString, name) for name in filesInGamePath)
higuFolderExists = regexExistsInGamePath(r"^HigurashiEp\d\d_Data$")
higuExeExists = regexExistsInGamePath(r"^HigurashiEp\d\d((.exe)|(.app)|(.x86)|(.x86_64))?$")
if higuFolderExists and not higuExeExists:
print("Warning: Detected partial Higurashi install at {}. You may want to delete this manually.".format(gamePath))
return True
umiScriptExists = '0.u' in filesInGamePath
umiExeExists = regexExistsInGamePath(r"^Umineko\dto\d((.exe)|(.app))?$")
if umiScriptExists and not umiExeExists:
print("Warning: Detected partial Umineko install at {}. You may want to delete this manually".format(gamePath))
return True
except Exception as e:
traceback.print_exc()
print("Failed to scan for partially uninstalled paths")
return False
def scanForFullInstallConfigs(subModConfigList, possiblePaths=None, scanExtraPaths=True):
# type: (List[installConfiguration.SubModConfig], [str], bool) -> [installConfiguration.FullInstallConfiguration, List[str]]
"""
This function has two purposes:
- When given a specific game path ('possiblePaths' argument), it checks if any of the given SubModConfig
can be installed into that path. Each SubModConfig which can be installed into that path will be returned
as a FullInstallConfiguration object.
- When not given a specific game path, it searches the computer for valid installations where the given
SubModConfig could be installed to. Each valid (installation + SubModConfig) combination will be returned
as a FullInstallConfiguration object.
:param subModConfigList: A **list** of SubModConfig which are to be searched for on disk
:param possiblePaths: (Optional) Specify folders to check if the given SubModConfig can be installed into that path.
:return: 1. A list of FullInstallConfig, each representing a valid install path that the
given SubModConfig(s) couldbe installed into.
2. A list of games which were "partially uninstalled" by Steam - steam deletes game files, but not the mod
files. The user should be notified to delete these files manually.
"""
returnedFullConfigs = []
returnedPartiallyUninstalledPaths = []
pathsToBeScanned = possiblePaths
if not pathsToBeScanned:
pathsToBeScanned = getMaybeGamePaths()
# Build a mapping from subModIdentifier -> List[subMod]
# This tells us, for each identifier, which subMods are compatible with that identifier (can be installed)
# In all our games, the identifiers are the same for each subMod (but different for each Mod),
# but it is easier to work with in the installer if we work with subMods
from collections import defaultdict
subModConfigDictionary = defaultdict(list) #type: defaultdict[List[installConfiguration.SubModConfig]]
for subMod in subModConfigList:
# If autodetection is disabled, and autodetection requested, do not scan for this submod
if not subMod.autodetect and possiblePaths is None:
continue
for identifier in subMod.identifiers:
subModConfigDictionary[identifier].append(subMod)
# If there are no identifiers to be matched, give up immediately as we'll never find a match
if not subModConfigDictionary:
return [], []
if scanExtraPaths:
extraPaths = []
for gamePath in pathsToBeScanned:
# MacOS: Any subpath with '.app' is also checked in case the containing path was manually entered
extraPaths.extend(glob.glob(os.path.join(gamePath, "*.app")))
# GOG Linux: Higurashi might be inside a 'game' subfolder
extraPaths.extend(glob.glob(os.path.join(gamePath, "game")))
pathsToBeScanned += extraPaths
if common.Globals.DEBUG_GAME_SCAN_VERBOSE:
print("Scanning:\n\t- " + "\n\t- ".join(pathsToBeScanned))
# Remove any duplicate game paths
pathsToBeScanned = deDuplicatePaths(pathsToBeScanned)
print("\nThe following games were found:")
for gamePath in pathsToBeScanned:
possibleIdentifiers = getPossibleIdentifiersFromFolder(gamePath)
subModConfigsInThisGamePath = set()
possibleSteamPaths = [
os.path.join(gamePath, "steam_api.dll"),
os.path.join(gamePath, "Contents/Plugins/CSteamworks.bundle"),
os.path.join(gamePath, "libsteam_api.so")
]
isSteam = False
for possibleSteamPath in possibleSteamPaths:
if os.path.exists(possibleSteamPath):
isSteam = True
# Skip any game paths which look like they're corrupted / partially uninstalled
if gamePathIsPartiallyUninstalled(gamePath):
returnedPartiallyUninstalledPaths.append(gamePath)
continue
for possibleIdentifier in possibleIdentifiers:
try:
# Add each submod which is compatible with the found identifier, unless it has already been detected at this path.
for subModConfig in subModConfigDictionary[possibleIdentifier]:
if subModConfig not in subModConfigsInThisGamePath:
subModConfigsInThisGamePath.add(subModConfig)
returnedFullConfigs.append(installConfiguration.FullInstallConfiguration(subModConfig, gamePath, isSteam))
print(" - {} ({}) at [{}]".format(subModConfig.modName, possibleIdentifier, gamePath))
except KeyError:
pass
return returnedFullConfigs, returnedPartiallyUninstalledPaths
def scanUserSelectedPath(subModConfigList, gameExecutablePath):
# type: (List[SubModConfig], [str]) -> ([FullInstallConfiguration], str)
"""
Scans a user-selected path for configs. Unlike the normal "scanForFullInstallConfigs()" function,
this will attempt to search all parent directories, incase a user has selected a subdirectory of the game directory
by accident.
:param subModConfigList:
:param gameExecutablePath: A path to the game executable, or to a folder containing the game.
:return: A tuple - The first is an array of valid FullInstallConfigurations.
- The second is an error message (on success a 'success' message is generated)
"""
if gameExecutablePath:
if os.path.isfile(gameExecutablePath):
gameExecutablePath = os.path.dirname(gameExecutablePath)
# Search upwards for the game path, in case user has selected a deep subfolder of the game path
alreadyScanned = set()
for scanAttempt in range(10):
fullInstallConfigs, _ = scanForFullInstallConfigs(subModConfigList=subModConfigList,
possiblePaths=[gameExecutablePath],
scanExtraPaths= scanAttempt==0)
if fullInstallConfigs:
return fullInstallConfigs, "scanUserSelectedPath(): Path [{}] Ok".format(gameExecutablePath)
alreadyScanned.add(gameExecutablePath)
gameExecutablePath = os.path.dirname(gameExecutablePath)
if gameExecutablePath in alreadyScanned:
break
# Failed to find path. Notify user which paths tried to be searched to find the file.
errorStrings = ["scanUserSelectedPath(): Can't install the mod. Searched:"] + sorted(list(alreadyScanned))
errorMessage = '\n - '.join(errorStrings)
return None, errorMessage
return None, "scanUserSelectedPath(): game executable path is falsey: [{}]".format(gameExecutablePath)
def gameIsUnsupported(subMod, installPath):
#type: (installConfiguration.SubModConfig, str) -> (bool, str)
"""
Check for if user selected path is not supported by our mod
system.arc = Old MG version of Higurashi
nscript.dat = Old Nscripter version of games
:returns if unsupported, returns (True, identifier), giving the identifier that caused the unsupported detection
if supported, returns (False, None)
"""
identifiers = []
if subMod.family == 'higurashi':
identifiers = ['system.arc', 'nscript.dat']
elif subMod.family == 'umineko':
identifiers = ['nscript.dat']
# If user selected a file, need to check parent directory instead. Just check both.
installPathsToCheck = [installPath, os.path.dirname(installPath)]
for identifier in identifiers:
for path in installPathsToCheck:
if os.path.exists(os.path.join(path, identifier)):
return True, identifier
return False, None