mirror of
https://github.com/wabbajack-tools/wabbajack.git
synced 2024-08-30 18:42:17 +00:00
295 lines
11 KiB
C#
295 lines
11 KiB
C#
using System;
|
|
using System.ComponentModel.DataAnnotations;
|
|
using System.Linq;
|
|
using System.Reactive;
|
|
using System.Reactive.Linq;
|
|
using System.Threading.Tasks;
|
|
using F23.StringSimilarity;
|
|
using Newtonsoft.Json;
|
|
using ReactiveUI;
|
|
using Wabbajack.Common;
|
|
using Wabbajack.Common.Serialization.Json;
|
|
using Wabbajack.Common.StatusFeed.Errors;
|
|
using Wabbajack.Lib.NexusApi;
|
|
using Wabbajack.Lib.Validation;
|
|
using Game = Wabbajack.Common.Game;
|
|
|
|
namespace Wabbajack.Lib.Downloaders
|
|
{
|
|
public class NexusDownloader : IDownloader, INeedsLogin
|
|
{
|
|
private bool _prepared;
|
|
private AsyncLock _lock = new AsyncLock();
|
|
private UserStatus? _status;
|
|
public INexusApi? Client;
|
|
|
|
public IObservable<bool> IsLoggedIn => Utils.HaveEncryptedJsonObservable("nexusapikey");
|
|
|
|
public string SiteName => "Nexus Mods";
|
|
|
|
public IObservable<string> MetaInfo => Observable.Return("");
|
|
|
|
public Uri SiteURL => new Uri("https://www.nexusmods.com");
|
|
|
|
public Uri IconUri => new Uri("https://www.nexusmods.com/favicon.ico");
|
|
|
|
public ReactiveCommand<Unit, Unit> TriggerLogin { get; }
|
|
public ReactiveCommand<Unit, Unit> ClearLogin { get; }
|
|
|
|
public NexusDownloader()
|
|
{
|
|
TriggerLogin = ReactiveCommand.CreateFromTask(
|
|
execute: () => Utils.CatchAndLog(NexusApiClient.RequestAndCacheAPIKey),
|
|
canExecute: IsLoggedIn.Select(b => !b).ObserveOnGuiThread());
|
|
ClearLogin = ReactiveCommand.CreateFromTask(
|
|
execute: () => Utils.CatchAndLog(async () => await Utils.DeleteEncryptedJson("nexusapikey")),
|
|
canExecute: IsLoggedIn.ObserveOnGuiThread());
|
|
}
|
|
|
|
public async Task<AbstractDownloadState?> GetDownloaderState(dynamic archiveINI, bool quickMode)
|
|
{
|
|
var general = archiveINI.General;
|
|
|
|
if (general.modID != null && general.fileID != null && general.gameName != null)
|
|
{
|
|
var game = GameRegistry.GetByFuzzyName((string)general.gameName).Game;
|
|
|
|
if (quickMode)
|
|
{
|
|
return new State
|
|
{
|
|
Game = GameRegistry.GetByFuzzyName((string)general.gameName).Game,
|
|
ModID = long.Parse(general.modID),
|
|
FileID = long.Parse(general.fileID),
|
|
};
|
|
}
|
|
|
|
var client = await NexusApiClient.Get();
|
|
ModInfo info;
|
|
try
|
|
{
|
|
info = await client.GetModInfo(game, long.Parse((string)general.modID));
|
|
}
|
|
catch (Exception)
|
|
{
|
|
Utils.Error($"Error getting mod info for Nexus mod with {general.modID}");
|
|
throw;
|
|
}
|
|
|
|
try
|
|
{
|
|
return new State
|
|
{
|
|
Name = NexusApiUtils.FixupSummary(info.name),
|
|
Author = NexusApiUtils.FixupSummary(info.author),
|
|
Version = general.version ?? "0.0.0.0",
|
|
ImageURL = info.picture_url,
|
|
IsNSFW = info.contains_adult_content,
|
|
Description = NexusApiUtils.FixupSummary(info.summary),
|
|
Game = GameRegistry.GetByFuzzyName((string)general.gameName).Game,
|
|
ModID = long.Parse(general.modID),
|
|
FileID = long.Parse(general.fileID)
|
|
};
|
|
}
|
|
catch (FormatException)
|
|
{
|
|
Utils.Log(
|
|
$"Cannot parse ModID/FileID from {(string)general.gameName} {(string)general.modID} {(string)general.fileID}");
|
|
throw;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public async Task Prepare()
|
|
{
|
|
if (!_prepared)
|
|
{
|
|
using var _ = await _lock.WaitAsync();
|
|
// Could have become prepared while we waited for the lock
|
|
if (!_prepared)
|
|
{
|
|
if (CLIArguments.ApiKey != null)
|
|
{
|
|
await CLIArguments.ApiKey.ToEcryptedJson("nexusapikey");
|
|
}
|
|
|
|
Client = await NexusApiClient.Get();
|
|
_status = await Client.GetUserStatus();
|
|
if (!Client.IsAuthenticated)
|
|
{
|
|
Utils.ErrorThrow(new UnconvertedError(
|
|
$"Authenticating for the Nexus failed. A nexus account is required to automatically download mods."));
|
|
return;
|
|
}
|
|
|
|
|
|
/* Disabled for better User experience
|
|
if (!await _client.IsPremium())
|
|
{
|
|
var result = await Utils.Log(new YesNoIntervention(
|
|
"Wabbajack can operate without a premium account, but downloads will be slower and the install process will require more user interactions (you will have to start each download by hand). Are you sure you wish to continue?",
|
|
"Continue without Premium?")).Task;
|
|
if (result == ConfirmationIntervention.Choice.Abort)
|
|
{
|
|
Utils.ErrorThrow(new UnconvertedError($"Aborting at the request of the user"));
|
|
}
|
|
}
|
|
*/
|
|
_prepared = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
[JsonName("NexusDownloader")]
|
|
public class State : AbstractDownloadState, IMetaState, IUpgradingState
|
|
{
|
|
[JsonIgnore]
|
|
public Uri URL => new Uri($"http://nexusmods.com/{Game.MetaData().NexusName}/mods/{ModID}");
|
|
|
|
public string? Name { get; set; }
|
|
|
|
public string? Author { get; set; }
|
|
|
|
public string? Version { get; set; }
|
|
|
|
public Uri? ImageURL { get; set; }
|
|
|
|
public bool IsNSFW { get; set; }
|
|
|
|
public string? Description { get; set; }
|
|
|
|
[JsonProperty("GameName")]
|
|
[JsonConverter(typeof(Utils.GameConverter))]
|
|
public Game Game { get; set; }
|
|
|
|
public long ModID { get; set; }
|
|
public long FileID { get; set; }
|
|
|
|
public async Task<bool> LoadMetaData()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
[JsonIgnore]
|
|
public override object[] PrimaryKey { get => new object[]{Game, ModID, FileID};}
|
|
|
|
public override bool IsWhitelisted(ServerWhitelist whitelist)
|
|
{
|
|
// Nexus files are always whitelisted
|
|
return true;
|
|
}
|
|
|
|
public override async Task<bool> Download(Archive a, AbsolutePath destination)
|
|
{
|
|
string url;
|
|
try
|
|
{
|
|
var client = await NexusApiClient.Get();
|
|
url = await client.GetNexusDownloadLink(this);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Utils.Log($"{a.Name} - Error getting Nexus download URL - {ex.Message}");
|
|
return false;
|
|
}
|
|
|
|
Utils.Log($"Downloading Nexus Archive - {a.Name} - {Game} - {ModID} - {FileID}");
|
|
|
|
return await new HTTPDownloader.State(url).Download(a, destination);
|
|
}
|
|
|
|
public override async Task<bool> Verify(Archive a)
|
|
{
|
|
try
|
|
{
|
|
var client = await NexusApiClient.Get();
|
|
var modInfo = await client.GetModInfo(Game, ModID);
|
|
if (!modInfo.available) return false;
|
|
var modFiles = await client.GetModFiles(Game, ModID);
|
|
|
|
var found = modFiles.files
|
|
.FirstOrDefault(file => file.file_id == FileID && file.category_name != null);
|
|
|
|
return found != null;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
Utils.Log($"{Name} - {Game} - {ModID} - {FileID} - Error getting Nexus download URL - {ex}");
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
public override IDownloader GetDownloader()
|
|
{
|
|
return DownloadDispatcher.GetInstance<NexusDownloader>();
|
|
}
|
|
|
|
public override string GetManifestURL(Archive a)
|
|
{
|
|
return $"http://nexusmods.com/{Game.MetaData().NexusName}/mods/{ModID}";
|
|
}
|
|
|
|
public override string[] GetMetaIni()
|
|
{
|
|
return new[] {"[General]", $"gameName={Game.MetaData().MO2ArchiveName}", $"modID={ModID}", $"fileID={FileID}"};
|
|
}
|
|
|
|
public override async Task<(Archive? Archive, TempFile NewFile)> FindUpgrade(Archive a, Func<Archive, Task<AbsolutePath>> downloadResolver)
|
|
{
|
|
var client = await NexusApiClient.Get();
|
|
|
|
var mod = await client.GetModInfo(Game, ModID);
|
|
if (!mod.available)
|
|
return default;
|
|
|
|
var files = await client.GetModFiles(Game, ModID);
|
|
var oldFile = files.files.FirstOrDefault(f => f.file_id == FileID);
|
|
var nl = new Levenshtein();
|
|
var newFile = files.files.Where(f => f.category_name != null)
|
|
.OrderBy(f => nl.Distance(oldFile.name.ToLowerInvariant(), f.name.ToLowerInvariant())).FirstOrDefault();
|
|
|
|
if (!mod.available || oldFile == default || newFile == default)
|
|
{
|
|
return default;
|
|
}
|
|
// Size is in KB
|
|
if (oldFile.size > 4_500_000 || newFile.size > 4_500_000 || oldFile.file_id == newFile.file_id)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
var newArchive = new Archive(new State {Game = Game, ModID = ModID, FileID = newFile.file_id})
|
|
{
|
|
Name = newFile.file_name,
|
|
};
|
|
|
|
var fastPath = await downloadResolver(newArchive);
|
|
if (fastPath != default)
|
|
{
|
|
newArchive.Size = fastPath.Size;
|
|
newArchive.Hash = await fastPath.FileHashAsync();
|
|
return (newArchive, new TempFile());
|
|
}
|
|
|
|
var tempFile = new TempFile();
|
|
|
|
await newArchive.State.Download(newArchive, tempFile.Path);
|
|
|
|
newArchive.Size = tempFile.Path.Size;
|
|
newArchive.Hash = await tempFile.Path.FileHashAsync();
|
|
|
|
return (newArchive, tempFile);
|
|
}
|
|
|
|
public async Task<bool> ValidateUpgrade(Hash srcHash, AbstractDownloadState newArchiveState)
|
|
{
|
|
var state = (State)newArchiveState;
|
|
return Game == state.Game && ModID == state.ModID;
|
|
}
|
|
}
|
|
}
|
|
}
|