using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Net.Http; using System.Net.Http.Headers; using System.Net.Http.Json; using System.Text.Json; using System.Threading; using System.Threading.Tasks; using Microsoft.Extensions.Logging; using Wabbajack.Common; using Wabbajack.DTOs; using Wabbajack.DTOs.Logins; using Wabbajack.DTOs.OAuth; using Wabbajack.Networking.Http; using Wabbajack.Networking.Http.Interfaces; using Wabbajack.Networking.NexusApi.DTOs; using Wabbajack.Paths.IO; using Wabbajack.RateLimiter; using Wabbajack.Server.DTOs; namespace Wabbajack.Networking.NexusApi; public class NexusApi { private readonly ApplicationInfo _appInfo; private readonly HttpClient _client; private readonly JsonSerializerOptions _jsonOptions; private readonly IResource _limiter; private readonly ILogger _logger; public readonly ITokenProvider AuthInfo; private DateTime _lastValidated; private (ValidateInfo info, ResponseMetadata header) _lastValidatedInfo; public NexusApi(ITokenProvider authInfo, ILogger logger, HttpClient client, IResource limiter, ApplicationInfo appInfo, JsonSerializerOptions jsonOptions) { AuthInfo = authInfo; _logger = logger; _client = client; _appInfo = appInfo; _jsonOptions = jsonOptions; _limiter = limiter; _lastValidated = DateTime.MinValue; _lastValidatedInfo = default; } public virtual async Task<(ValidateInfo info, ResponseMetadata header)> Validate( CancellationToken token = default) { var (isApi, code) = await GetAuthInfo(); if (isApi) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.Validate); _lastValidatedInfo = await Send(msg, token); } else { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.OAuthValidate); var (data, header) = await Send(msg, token); var validateInfo = new ValidateInfo { IsPremium = data.MembershipRoles.Contains("premium"), Name = data.Name, }; _lastValidatedInfo = (validateInfo, header); } _lastValidated = DateTime.Now; return _lastValidatedInfo; } public async Task<(ValidateInfo info, ResponseMetadata header)> ValidateCached( CancellationToken token = default) { if (DateTime.Now - _lastValidated < TimeSpan.FromMinutes(10)) { return _lastValidatedInfo; } await Validate(token); return _lastValidatedInfo; } public virtual async Task<(ModInfo info, ResponseMetadata header)> ModInfo(string nexusGameName, long modId, CancellationToken token = default) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.ModInfo, nexusGameName, modId); return await Send(msg, token); } public virtual async Task<(ModFiles info, ResponseMetadata header)> ModFiles(string nexusGameName, long modId, CancellationToken token = default) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.ModFiles, nexusGameName, modId); return await Send(msg, token); } public virtual async Task<(ModFile info, ResponseMetadata header)> FileInfo(string nexusGameName, long modId, long fileId, CancellationToken token = default) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.ModFile, nexusGameName, modId, fileId); return await Send(msg, token); } public virtual async Task<(DownloadLink[] info, ResponseMetadata header)> DownloadLink(string nexusGameName, long modId, long fileId, CancellationToken token = default) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.DownloadLink, nexusGameName, modId, fileId); return await Send(msg, token); } protected virtual async Task<(T data, ResponseMetadata header)> Send(HttpRequestMessage msg, CancellationToken token = default) { using var job = await _limiter.Begin($"API call to the Nexus {msg.RequestUri!.PathAndQuery}", 0, token); using var result = await _client.SendAsync(msg, token); if (!result.IsSuccessStatusCode) throw new HttpException(result); var headers = ParseHeaders(result); job.Size = result.Content.Headers.ContentLength ?? 0; await job.Report((int) (result.Content.Headers.ContentLength ?? 0), token); var body = await result.Content.ReadAsByteArrayAsync(token); return (JsonSerializer.Deserialize(body, _jsonOptions)!, headers); } protected virtual ResponseMetadata ParseHeaders(HttpResponseMessage result) { var metaData = new ResponseMetadata(); { if (result.Headers.TryGetValues("x-rl-daily-limit", out var limits)) if (int.TryParse(limits.First(), out var limit)) metaData.DailyLimit = limit; } { if (result.Headers.TryGetValues("x-rl-daily-remaining", out var limits)) if (int.TryParse(limits.First(), out var limit)) metaData.DailyRemaining = limit; } { if (result.Headers.TryGetValues("x-rl-daily-reset", out var resets)) if (DateTime.TryParse(resets.First(), out var reset)) metaData.DailyReset = reset; } { if (result.Headers.TryGetValues("x-rl-hourly-limit", out var limits)) if (int.TryParse(limits.First(), out var limit)) metaData.HourlyLimit = limit; } { if (result.Headers.TryGetValues("x-rl-hourly-remaining", out var limits)) if (int.TryParse(limits.First(), out var limit)) metaData.HourlyRemaining = limit; } { if (result.Headers.TryGetValues("x-rl-hourly-reset", out var resets)) if (DateTime.TryParse(resets.First(), out var reset)) metaData.HourlyReset = reset; } { if (result.Headers.TryGetValues("x-runtime", out var runtimes)) if (double.TryParse(runtimes.First(), out var reset)) metaData.Runtime = reset; } _logger.LogInformation("Nexus API call finished: {Runtime} - Remaining Limit: {RemainingLimit}", metaData.Runtime, Math.Max(metaData.DailyRemaining, metaData.HourlyRemaining)); return metaData; } protected virtual async ValueTask GenerateMessage(HttpMethod method, string uri, params object?[] parameters) { var msg = new HttpRequestMessage(); msg.Method = method; var userAgent = $"{_appInfo.ApplicationSlug}/{_appInfo.Version} ({_appInfo.OSVersion}; {_appInfo.Platform})"; if (!AuthInfo.HaveToken()) throw new Exception("Please log into the Nexus before attempting to use Wabbajack"); var token = (await AuthInfo.Get())!; if (uri.StartsWith("http")) { msg.RequestUri = new Uri($"{string.Format(uri, parameters)}"); } else { msg.RequestUri = new Uri($"https://api.nexusmods.com/{string.Format(uri, parameters)}"); } msg.Headers.Add("User-Agent", userAgent); msg.Headers.Add("Application-Name", _appInfo.ApplicationSlug); msg.Headers.Add("Application-Version", _appInfo.Version); await AddAuthHeaders(msg); msg.Headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); return msg; } private async ValueTask AddAuthHeaders(HttpRequestMessage msg) { var (isApi, code) = await GetAuthInfo(); if (string.IsNullOrWhiteSpace(code)) throw new Exception("No API Key or OAuth Token found for NexusMods"); if (isApi) msg.Headers.Add("apikey", code); else { msg.Headers.Authorization = new AuthenticationHeaderValue("Bearer", code); } } private async ValueTask<(bool IsApiKey, string code)> GetAuthInfo() { if (AuthInfo.HaveToken()) { var info = await AuthInfo.Get(); if (info!.OAuth != null) { if (info!.OAuth.IsExpired) info = await RefreshToken(info, CancellationToken.None); return (false, info.OAuth!.AccessToken!); } if (!string.IsNullOrWhiteSpace(info.ApiKey)) { return (true, info.ApiKey); } } else { if (Environment.GetEnvironmentVariable("NEXUS_API_KEY") is { } apiKey) { return (true, apiKey); } } return default; } private async Task RefreshToken(NexusOAuthState state, CancellationToken cancel) { _logger.LogInformation("Refreshing OAuth Token"); var request = new Dictionary { { "grant_type", "refresh_token" }, { "client_id", "wabbajack" }, { "refresh_token", state.OAuth!.RefreshToken }, }; var content = new FormUrlEncodedContent(request); var response = await _client.PostAsync($"https://users.nexusmods.com/oauth/token", content, cancel); var responseString = await response.Content.ReadAsStringAsync(cancel); var newJwt = JsonSerializer.Deserialize(responseString); state.OAuth = newJwt; await AuthInfo.SetToken(state); return state; } public async Task<(UpdateEntry[], ResponseMetadata headers)> GetUpdates(Game game, CancellationToken token) { var msg = await GenerateMessage(HttpMethod.Get, Endpoints.Updates, game.MetaData().NexusName, "1m"); return await Send(msg, token); } public async Task ChunkStatus(UploadDefinition definition, Chunk chunk) { var msg = new HttpRequestMessage(); msg.Method = HttpMethod.Get; var query = $"resumableChunkNumber={chunk.Index + 1}&resumableCurrentChunkSize={chunk.Size}&resumableTotalSize={definition.FileSize}" + $"&resumableType=&resumableIdentifier={definition.ResumableIdentifier}&resumableFilename={definition.ResumableRelativePath}" + $"&resumableRelativePath={definition.ResumableRelativePath}&resumableTotalChunks={definition.Chunks().Count()}"; msg.RequestUri = new Uri($"https://upload.nexusmods.com/uploads/chunk?{query}"); using var result = await _client.SendAsync(msg); if (!result.IsSuccessStatusCode) throw new HttpException(result); if (result.StatusCode == HttpStatusCode.NoContent) return DTOs.ChunkStatus.NoContent; var status = await result.Content.ReadFromJsonAsync(); return status?.Status ?? false ? DTOs.ChunkStatus.Done : DTOs.ChunkStatus.Waiting; } public async Task UploadChunk(UploadDefinition d, Chunk chunk) { var form = new MultipartFormDataContent(); form.Add(new StringContent((chunk.Index+1).ToString()), "resumableChunkNumber"); form.Add(new StringContent(UploadDefinition.ChunkSize.ToString()), "resumableChunkSize"); form.Add(new StringContent(chunk.Size.ToString()), "resumableCurrentChunkSize"); form.Add(new StringContent(d.FileSize.ToString()), "resumableTotalSize"); form.Add(new StringContent(""), "resumableType"); form.Add(new StringContent(d.ResumableIdentifier), "resumableIdentifier"); form.Add(new StringContent(d.ResumableRelativePath), "resumableFilename"); form.Add(new StringContent(d.ResumableRelativePath), "resumableRelativePath"); form.Add(new StringContent(d.Chunks().Count().ToString()), "resumableTotalChunks"); await using var ms = new MemoryStream(); await using var fs = d.Path.Open(FileMode.Open, FileAccess.Read, FileShare.Read); fs.Position = chunk.Offset; await fs.CopyToLimitAsync(ms, (int)chunk.Size, CancellationToken.None); ms.Position = 0; form.Add(new StreamContent(ms), "file", "blob"); var msg = new HttpRequestMessage(HttpMethod.Post, "https://upload.nexusmods.com/uploads/chunk"); msg.Content = form; var result = await _client.SendAsync(msg); if (result.StatusCode != HttpStatusCode.OK) throw new HttpException(result); var response = await result.Content.ReadFromJsonAsync(_jsonOptions); return response!; } public async Task UploadFile(UploadDefinition d) { _logger.LogInformation("Checking Access"); await CheckAccess(); _logger.LogInformation("Checking chunk status"); var numberOfChunks = d.Chunks().Count(); var chunkStatus = new ChunkStatusResult(); foreach (var chunk in d.Chunks()) { var status = await ChunkStatus(d, chunk); _logger.LogInformation("({Index}/{MaxChunks}) Chunk status: {Status}", chunk.Index, numberOfChunks, status); if (status == DTOs.ChunkStatus.NoContent) { _logger.LogInformation("({Index}/{MaxChunks}) Uploading", chunk.Index, numberOfChunks); chunkStatus = await UploadChunk(d, chunk); } } await WaitForFileStatus(chunkStatus); await AddFile(d, chunkStatus); } private async Task CheckAccess() { var msg = new HttpRequestMessage(HttpMethod.Get, "https://www.nexusmods.com/users/myaccount"); throw new NotSupportedException("Uploading to NexusMods is currently disabled"); using var response = await _client.SendAsync(msg); var body = await response.Content.ReadAsStringAsync(); if (body.Contains("You are not allowed to access this area!")) throw new HttpException(403, "Nexus Cookies are incorrect"); } private async Task AddFile(UploadDefinition d, ChunkStatusResult status) { _logger.LogInformation("Saving file update {Name} to {Game}:{ModId}", d.Path.FileName, d.Game, d.ModId); var msg = new HttpRequestMessage(HttpMethod.Post, "https://www.nexusmods.com/Core/Libs/Common/Managers/Mods?AddFile"); msg.Headers.Referrer = new Uri( $"https://www.nexusmods.com/{d.Game.MetaData().NexusName}/mods/edit/?id={d.ModId}&game_id={d.GameId}&step=files"); throw new NotSupportedException("Uploading to NexusMods is currently disabled"); var form = new MultipartFormDataContent(); form.Add(new StringContent(d.GameId.ToString()), "game_id"); form.Add(new StringContent(d.Name), "name"); form.Add(new StringContent(d.Version), "file-version"); form.Add(new StringContent((d.RemoveOldVersion ? 1 : 0).ToString()), "update-version"); form.Add(new StringContent(((int)Enum.Parse(d.Category, true)).ToString()), "category"); form.Add(new StringContent((d.NewExisting ? 1 : 0).ToString()), "new-existing"); form.Add(new StringContent(d.OldFileId.ToString()), "old_file_id"); form.Add(new StringContent((d.RemoveOldVersion ? 1 : 0).ToString()), "remove-old-version"); form.Add(new StringContent(d.BriefOverview), "brief-overview"); form.Add(new StringContent((d.SetAsMain ? 1 : 0).ToString()), "set_as_main_nmm"); form.Add(new StringContent(status.UUID), "file_uuid"); form.Add(new StringContent(d.FileSize.ToString()), "file_size"); form.Add(new StringContent(d.ModId.ToString()), "mod_id"); form.Add(new StringContent(d.ModId.ToString()), "id"); form.Add(new StringContent("save"), "action"); form.Add(new StringContent(status.Filename), "uploaded_file"); form.Add(new StringContent(d.Path.FileName.ToString()), "original_file"); msg.Content = form; using var result = await _client.SendAsync(msg); if (!result.IsSuccessStatusCode) throw new HttpException(result); } private async Task WaitForFileStatus(ChunkStatusResult chunkStatus) { while (true) { _logger.LogInformation("Checking file status of {Uuid}", chunkStatus.UUID); var data = await _client.GetFromJsonAsync( $"https://upload.nexusmods.com/uploads/check_status?id={chunkStatus.UUID}"); if (data!.FileChunksAssembled) return data; await Task.Delay(TimeSpan.FromSeconds(5)); } } public async Task IsPremium(CancellationToken token) { var validated = await ValidateCached(token); return validated.info.IsPremium; } }