using ReactiveUI; using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Linq; using System.Net; using System.Net.Http; using System.Net.Http.Headers; using System.Reflection; using System.Security.Authentication; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; using System.Windows.Documents; using Syroot.Windows.IO; using Wabbajack.Common; using Wabbajack.Lib.Downloaders; using Wabbajack.Lib.LibCefHelpers; using WebSocketSharp; using Xilium.CefGlue; using Xilium.CefGlue.Common; using Xilium.CefGlue.Common.Handlers; using Xilium.CefGlue.WPF; using static Wabbajack.Lib.NexusApi.NexusApiUtils; using System.Threading; namespace Wabbajack.Lib.NexusApi { public class NexusApiClient : ViewModel { private static readonly string API_KEY_CACHE_FILE = "nexus.key_cache"; private static string _additionalEntropy = "vtP2HF6ezg"; public HttpClient HttpClient { get; } = new HttpClient(); #region Authentication public string ApiKey { get; } public bool IsAuthenticated => ApiKey != null; private Task _userStatus; public Task UserStatus { get { if (_userStatus == null) _userStatus = GetUserStatus(); return _userStatus; } } public async Task IsPremium() { return IsAuthenticated && (await UserStatus).is_premium; } public async Task Username() => (await UserStatus).name; private static SemaphoreSlim _getAPIKeyLock = new SemaphoreSlim(1, 1); private static async Task GetApiKey() { await _getAPIKeyLock.WaitAsync(); try { // Clean up old location if (File.Exists(API_KEY_CACHE_FILE)) { File.Delete(API_KEY_CACHE_FILE); } try { return Utils.FromEncryptedJson("nexusapikey"); } catch (Exception) { } var env_key = Environment.GetEnvironmentVariable("NEXUSAPIKEY"); if (env_key != null) { return env_key; } var result = await Utils.Log(new RequestNexusAuthorization()).Task; result.ToEcryptedJson("nexusapikey"); return result; } finally { _getAPIKeyLock.Release(); } } class RefererHandler : RequestHandler { private string _referer; public RefererHandler(string referer) { _referer = referer; } protected override bool OnBeforeBrowse(CefBrowser browser, CefFrame frame, CefRequest request, bool userGesture, bool isRedirect) { base.OnBeforeBrowse(browser, frame, request, userGesture, isRedirect); if (request.ReferrerURL == null) request.SetReferrer(_referer, CefReferrerPolicy.Default); return false; } } public static async Task SetupNexusLogin(BaseCefBrowser browser, Action updateStatus, CancellationToken cancel) { updateStatus("Please Log Into the Nexus"); browser.Address = "https://users.nexusmods.com/auth/continue?client_id=nexus&redirect_uri=https://www.nexusmods.com/oauth/callback&response_type=code&referrer=//www.nexusmods.com"; while (true) { var cookies = await Helpers.GetCookies("nexusmods.com"); if (cookies.Any(c => c.Name == "member_id")) break; cancel.ThrowIfCancellationRequested(); await Task.Delay(500, cancel); } // open a web socket to receive the api key var guid = Guid.NewGuid(); using (var websocket = new WebSocket("wss://sso.nexusmods.com") { SslConfiguration = { EnabledSslProtocols = SslProtocols.Tls12 } }) { updateStatus("Please Authorize Wabbajack to Download Mods"); var api_key = new TaskCompletionSource(); websocket.OnMessage += (sender, msg) => { api_key.SetResult(msg.Data); }; websocket.Connect(); websocket.Send("{\"id\": \"" + guid + "\", \"appid\": \"" + Consts.AppName + "\"}"); await Task.Delay(1000, cancel); // open a web browser to get user permission browser.Address = $"https://www.nexusmods.com/sso?id={guid}&application={Consts.AppName}"; using (cancel.Register(() => { api_key.SetCanceled(); })) { return await api_key.Task; } } } public async Task GetUserStatus() { var url = "https://api.nexusmods.com/v1/users/validate.json"; return await Get(url); } #endregion #region Rate Tracking private readonly object RemainingLock = new object(); private int _dailyRemaining; public int DailyRemaining { get { lock (RemainingLock) { return _dailyRemaining; } } } private int _hourlyRemaining; public int HourlyRemaining { get { lock (RemainingLock) { return _hourlyRemaining; } } } private void UpdateRemaining(HttpResponseMessage response) { try { var dailyRemaining = int.Parse(response.Headers.GetValues("x-rl-daily-remaining").First()); var hourlyRemaining = int.Parse(response.Headers.GetValues("x-rl-hourly-remaining").First()); lock (RemainingLock) { _dailyRemaining = Math.Min(dailyRemaining, hourlyRemaining); _hourlyRemaining = Math.Min(dailyRemaining, hourlyRemaining); } this.RaisePropertyChanged(nameof(DailyRemaining)); this.RaisePropertyChanged(nameof(HourlyRemaining)); } catch (Exception) { } } #endregion private NexusApiClient(string apiKey = null) { ApiKey = apiKey; // set default headers for all requests to the Nexus API var headers = HttpClient.DefaultRequestHeaders; headers.Add("User-Agent", Consts.UserAgent); headers.Add("apikey", ApiKey); headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); headers.Add("Application-Name", Consts.AppName); headers.Add("Application-Version", $"{Assembly.GetEntryAssembly()?.GetName()?.Version ?? new Version(0, 1)}"); if (!Directory.Exists(Consts.NexusCacheDirectory)) Directory.CreateDirectory(Consts.NexusCacheDirectory); } public static async Task Get(string apiKey = null) { apiKey = apiKey ?? await GetApiKey(); return new NexusApiClient(apiKey); } private async Task Get(string url) { var response = await HttpClient.GetAsync(url, HttpCompletionOption.ResponseHeadersRead); UpdateRemaining(response); using (var stream = await response.Content.ReadAsStreamAsync()) { return stream.FromJSON(); } } private async Task GetCached(string url) { var code = Encoding.UTF8.GetBytes(url).ToHex() + ".json"; if (UseLocalCache) { if (!Directory.Exists(LocalCacheDir)) Directory.CreateDirectory(LocalCacheDir); var cache_file = Path.Combine(LocalCacheDir, code); if (File.Exists(cache_file)) { return cache_file.FromJSON(); } var result = await Get(url); if (result != null) result.ToJSON(cache_file); return result; } try { return await Get(Consts.WabbajackCacheLocation + code); } catch (Exception) { return await Get(url); } } public async Task GetNexusDownloadLink(NexusDownloader.State archive, bool cache = false) { if (cache) { var result = await TryGetCachedLink(archive); if (result.Succeeded) { return result.Value; } } ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls12; var url = $"https://api.nexusmods.com/v1/games/{ConvertGameName(archive.GameName)}/mods/{archive.ModID}/files/{archive.FileID}/download_link.json"; return (await Get>(url)).First().URI; } private async Task> TryGetCachedLink(NexusDownloader.State archive) { if (!Directory.Exists(Consts.NexusCacheDirectory)) Directory.CreateDirectory(Consts.NexusCacheDirectory); var path = Path.Combine(Consts.NexusCacheDirectory, $"link-{archive.GameName}-{archive.ModID}-{archive.FileID}.txt"); if (!File.Exists(path) || (DateTime.Now - new FileInfo(path).LastWriteTime).TotalHours > 24) { File.Delete(path); var result = await GetNexusDownloadLink(archive); File.WriteAllText(path, result); return GetResponse.Succeed(result); } return GetResponse.Succeed(File.ReadAllText(path)); } public async Task GetFileInfo(NexusDownloader.State mod) { var url = $"https://api.nexusmods.com/v1/games/{ConvertGameName(mod.GameName)}/mods/{mod.ModID}/files/{mod.FileID}.json"; return await GetCached(url); } public class GetModFilesResponse { public List files; } public async Task GetModFiles(Game game, int modid) { var url = $"https://api.nexusmods.com/v1/games/{game.MetaData().NexusName}/mods/{modid}/files.json"; var result = await GetCached(url); if (result.files == null) throw new InvalidOperationException("Got Null data from the Nexus while finding mod files"); return result; } public async Task> GetModInfoFromMD5(Game game, string md5Hash) { var url = $"https://api.nexusmods.com/v1/games/{game.MetaData().NexusName}/mods/md5_search/{md5Hash}.json"; return await Get>(url); } public async Task GetModInfo(Game game, string modId) { var url = $"https://api.nexusmods.com/v1/games/{game.MetaData().NexusName}/mods/{modId}.json"; return await GetCached(url); } public async Task EndorseMod(NexusDownloader.State mod) { Utils.Status($"Endorsing ${mod.GameName} - ${mod.ModID}"); var url = $"https://api.nexusmods.com/v1/games/{ConvertGameName(mod.GameName)}/mods/{mod.ModID}/endorse.json"; var content = new FormUrlEncodedContent(new Dictionary { { "version", mod.Version } }); using (var stream = await HttpClient.PostStream(url, content)) { return stream.FromJSON(); } } private class DownloadLink { public string URI { get; set; } } private class UpdatedMod { public long mod_id; public long latest_file_update; public long latest_mod_activity; } private static bool? _useLocalCache; public static bool UseLocalCache { get { if (_useLocalCache == null) return LocalCacheDir != null; return _useLocalCache ?? false; } set => _useLocalCache = value; } private static string _localCacheDir; public static string LocalCacheDir { get { if (_localCacheDir == null) _localCacheDir = Environment.GetEnvironmentVariable("NEXUSCACHEDIR"); return _localCacheDir; } set => _localCacheDir = value; } public async Task ClearUpdatedModsInCache() { if (!UseLocalCache) return; var gameTasks = GameRegistry.Games.Values .Where(game => game.NexusName != null) .Select(async game => { return (game, mods: await Get>( $"https://api.nexusmods.com/v1/games/{game.NexusName}/mods/updated.json?period=1m")); }) .Select(async rTask => { var (game, mods) = await rTask; return mods.Select(mod => new { game = game, mod = mod }); }); var purge = (await Task.WhenAll(gameTasks)) .SelectMany(i => i) .ToList(); Utils.Log($"Found {purge.Count} updated mods in the last month"); using (var queue = new WorkQueue()) { var to_purge = (await Directory.EnumerateFiles(LocalCacheDir, "*.json") .PMap(queue, f => { Utils.Status("Cleaning Nexus cache for"); var uri = new Uri(Encoding.UTF8.GetString(Path.GetFileNameWithoutExtension(f).FromHex())); var parts = uri.PathAndQuery.Split('/', '.').ToHashSet(); var found = purge.FirstOrDefault(p => parts.Contains(p.game.NexusName) && parts.Contains(p.mod.mod_id.ToString())); if (found != null) { var should_remove = File.GetLastWriteTimeUtc(f) <= found.mod.latest_file_update.AsUnixTime(); return (should_remove, f); } // ToDo // Can improve to not read the entire file to see if it starts with null if (File.ReadAllText(f).StartsWith("null")) return (true, f); return (false, f); })) .Where(p => p.Item1) .ToList(); Utils.Log($"Purging {to_purge.Count} cache entries"); await to_purge.PMap(queue, f => { var uri = new Uri(Encoding.UTF8.GetString(Path.GetFileNameWithoutExtension(f.f).FromHex())); Utils.Log($"Purging {uri}"); File.Delete(f.f); }); } } } }