wabbajack/Wabbajack.Lib/Downloaders/DownloadDispatcher.cs

161 lines
5.4 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Linq;
2019-12-07 03:50:50 +00:00
using System.Threading.Tasks;
using Alphaleonis.Win32.Filesystem;
2019-11-04 23:21:58 +00:00
using Wabbajack.Common;
using Wabbajack.Lib.Downloaders.UrlDownloaders;
namespace Wabbajack.Lib.Downloaders
{
public static class DownloadDispatcher
{
public static readonly List<IDownloader> Downloaders = new List<IDownloader>()
{
2019-12-14 17:30:52 +00:00
new GameFileSourceDownloader(),
new MegaDownloader(),
new DropboxDownloader(),
new GoogleDriveDownloader(),
2019-10-12 22:31:07 +00:00
new ModDBDownloader(),
new NexusDownloader(),
new MediaFireDownloader(),
2019-12-08 17:00:22 +00:00
new LoversLabDownloader(),
new VectorPlexusDownloader(),
new DeadlyStreamDownloader(),
new BethesdaNetDownloader(),
2020-01-22 09:41:28 +00:00
new AFKModsDownloader(),
2020-01-22 09:50:09 +00:00
new TESAllianceDownloader(),
new YouTubeDownloader(),
new HTTPDownloader(),
2019-10-12 22:31:07 +00:00
new ManualDownloader(),
};
public static readonly List<IUrlInferencer> Inferencers = new List<IUrlInferencer>()
{
new BethesdaNetInferencer(),
new YoutubeInferencer()
};
2019-10-12 22:31:07 +00:00
private static readonly Dictionary<Type, IDownloader> IndexedDownloaders;
static DownloadDispatcher()
{
2019-10-12 22:31:07 +00:00
IndexedDownloaders = Downloaders.ToDictionary(d => d.GetType());
}
public static async Task<AbstractDownloadState> Infer(Uri uri)
{
foreach (var inf in Inferencers)
{
var state = await inf.Infer(uri);
if (state != null)
return state;
}
return null;
}
2019-12-08 17:00:22 +00:00
public static T GetInstance<T>() where T : IDownloader
{
2019-12-08 17:00:22 +00:00
var inst = (T)IndexedDownloaders[typeof(T)];
inst.Prepare();
return inst;
}
2019-12-07 03:50:50 +00:00
public static async Task<AbstractDownloadState> ResolveArchive(dynamic ini)
{
2019-12-07 03:50:50 +00:00
var states = await Task.WhenAll(Downloaders.Select(d => (Task<AbstractDownloadState>)d.GetDownloaderState(ini)));
return states.FirstOrDefault(result => result != null);
}
/// <summary>
/// Reduced version of Resolve archive that requires less information, but only works
/// with a single URL string
/// </summary>
/// <param name="ini"></param>
/// <returns></returns>
public static AbstractDownloadState ResolveArchive(string url)
{
return Downloaders.OfType<IUrlDownloader>().Select(d => d.GetDownloaderState(url)).FirstOrDefault(result => result != null);
}
2019-11-04 23:21:58 +00:00
public static void PrepareAll(IEnumerable<AbstractDownloadState> states)
{
states.Select(s => s.GetDownloader().GetType())
.Distinct()
.Do(t => Downloaders.First(d => d.GetType() == t).Prepare());
}
2020-03-25 22:30:43 +00:00
public static async Task<bool> DownloadWithPossibleUpgrade(Archive archive, AbsolutePath destination)
{
var success = await Download(archive, destination);
if (success)
{
await destination.FileHashCachedAsync();
return true;
}
Utils.Log($"Download failed, looking for upgrade");
var upgrade = await ClientAPI.GetModUpgrade(archive.Hash);
if (upgrade == null)
{
Utils.Log($"No upgrade found for {archive.Hash}");
return false;
}
Utils.Log($"Upgrading {archive.Hash}");
2020-03-25 22:30:43 +00:00
var upgradePath = destination.Parent.Combine("_Upgrade_" + archive.Name);
var upgradeResult = await Download(upgrade, upgradePath);
if (!upgradeResult) return false;
2020-03-22 15:50:53 +00:00
var patchName = $"{archive.Hash.ToHex()}_{upgrade.Hash.ToHex()}";
2020-03-25 22:30:43 +00:00
var patchPath = destination.Parent.Combine("_Patch_" + patchName);
var patchState = new Archive
{
Name = patchName,
State = new HTTPDownloader.State
{
Url = $"https://wabbajackcdn.b-cdn.net/updates/{patchName}"
}
};
var patchResult = await Download(patchState, patchPath);
if (!patchResult) return false;
Utils.Status($"Applying Upgrade to {archive.Hash}");
2020-03-25 22:30:43 +00:00
await using (var patchStream = patchPath.OpenRead())
await using (var srcStream = upgradePath.OpenRead())
await using (var destStream = destination.Create())
{
OctoDiff.Apply(srcStream, patchStream, destStream);
}
await destination.FileHashCachedAsync();
return true;
}
2020-03-25 22:30:43 +00:00
private static async Task<bool> Download(Archive archive, AbsolutePath destination)
{
try
{
var result = await archive.State.Download(archive, destination);
if (!result) return false;
2020-03-27 03:10:23 +00:00
if (!archive.Hash.IsValid) return true;
var hash = await destination.FileHashCachedAsync();
if (hash == archive.Hash) return true;
Utils.Log($"Hashed download is incorrect");
return false;
}
catch (Exception ex)
{
return false;
}
}
}
}