using System.Runtime.InteropServices; using GameFinder.RegistryUtils; using GameFinder.StoreHandlers.EGS; using GameFinder.StoreHandlers.GOG; using GameFinder.StoreHandlers.Origin; using GameFinder.StoreHandlers.Steam; using Microsoft.Extensions.Logging; using Wabbajack.DTOs; using Wabbajack.Paths; using Wabbajack.Paths.IO; namespace Wabbajack.Downloaders.GameFile; public class GameLocator : IGameLocator { private readonly SteamHandler _steam; private readonly GOGHandler? _gog; private readonly EGSHandler? _egs; private readonly OriginHandler? _origin; private readonly Dictionary _steamGames = new(); private readonly Dictionary _gogGames = new(); private readonly Dictionary _egsGames = new(); private readonly Dictionary _originGames = new(); private readonly Dictionary _locationCache; private readonly ILogger _logger; public GameLocator(ILogger logger) { _logger = logger; if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { var windowsRegistry = new WindowsRegistry(); _steam = new SteamHandler(windowsRegistry); _gog = new GOGHandler(windowsRegistry); _egs = new EGSHandler(windowsRegistry); _origin = new OriginHandler(); } else { _steam = new SteamHandler(null); } _locationCache = new Dictionary(); FindAllGames(); } private void FindAllGames() { FindStoreGames(_steam.FindAllGames(), _steamGames, steamGame => steamGame.Path, steamGame => steamGame.AppId); if (_gog is not null) { FindStoreGames(_gog.FindAllGames(), _gogGames, gogGame => gogGame.Path, gogGame => gogGame.Id); } if (_egs is not null) { FindStoreGames(_egs.FindAllGames(), _egsGames, egsGame => egsGame.InstallLocation, egsGame => egsGame.CatalogItemId); } if (_origin is not null) { FindStoreGames(_origin.FindAllGames(), _originGames, originGame => originGame.InstallPath, originGame => originGame.Id); } } private void FindStoreGames( IEnumerable<(TGame? game, string? error)> games, IDictionary paths, Func getPath, Func getId) where TGame : class { foreach (var (game, error) in games) { try { if (game is not null) { var path = getPath(game).ToAbsolutePath(); if (path.DirectoryExists()) { paths[getId(game)] = path; _logger.LogDebug("Found Game {} at {}", game, path); } else { _logger.LogError("Game {} does not exist at {}", game, path); } } else { _logger.LogError("{}", error); } } catch (Exception ex) { _logger.LogError(ex, "While locating game {Game}", game); } } } public AbsolutePath GameLocation(Game game) { if (TryFindLocation(game, out var path)) return path; throw new Exception($"Can't find game {game}"); } public bool IsInstalled(Game game) { return TryFindLocation(game, out _); } public bool TryFindLocation(Game game, out AbsolutePath path) { lock (_locationCache) { if (_locationCache.TryGetValue(game, out path)) return true; if (TryFindLocationInner(game, out path)) { _locationCache.Add(game, path); return true; } } return false; } private bool TryFindLocationInner(Game game, out AbsolutePath path) { var metaData = game.MetaData(); try { foreach (var id in metaData.SteamIDs) { if (!_steamGames.TryGetValue(id, out var found)) continue; path = found; return true; } } catch (Exception ex) { _logger.LogInformation(ex, "During Steam detection"); } try { foreach (var id in metaData.GOGIDs) { if (!_gogGames.TryGetValue(id, out var found)) continue; path = found; return true; } } catch (Exception ex) { _logger.LogInformation(ex, "During GOG detection"); } try { foreach (var id in metaData.EpicGameStoreIDs) { if (!_egsGames.TryGetValue(id, out var found)) continue; path = found; return true; } } catch (Exception ex) { _logger.LogInformation(ex, "During Epic detection"); } try { foreach (var id in metaData.OriginIDs) { if (!_originGames.TryGetValue(id, out var found)) continue; path = found; return true; } } catch (Exception ex) { _logger.LogInformation(ex, "During Origin Store detection"); } path = default; return false; } }