2020-05-09 22:16:16 +00:00
|
|
|
|
using System;
|
2020-08-31 23:29:48 +00:00
|
|
|
|
using System.Collections.Generic;
|
2020-05-09 22:16:16 +00:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.IO.Compression;
|
2020-05-13 03:53:20 +00:00
|
|
|
|
using System.IO.MemoryMappedFiles;
|
2020-07-20 03:45:55 +00:00
|
|
|
|
using System.Net.Http;
|
|
|
|
|
using System.Threading;
|
2020-05-09 22:16:16 +00:00
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Wabbajack.Common;
|
2020-05-21 22:12:51 +00:00
|
|
|
|
using Wabbajack.Common.Exceptions;
|
2020-05-09 22:16:16 +00:00
|
|
|
|
using Wabbajack.Common.Serialization.Json;
|
|
|
|
|
using Wabbajack.Lib.AuthorApi;
|
|
|
|
|
using Wabbajack.Lib.Validation;
|
2021-01-11 08:42:03 +00:00
|
|
|
|
using System.Linq;
|
2020-05-09 22:16:16 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.Lib.Downloaders
|
|
|
|
|
{
|
2020-08-31 23:29:48 +00:00
|
|
|
|
|
2020-05-10 21:35:15 +00:00
|
|
|
|
public class WabbajackCDNDownloader : IDownloader, IUrlDownloader
|
2020-05-09 22:16:16 +00:00
|
|
|
|
{
|
2020-08-31 23:29:48 +00:00
|
|
|
|
public static Dictionary<string, string> DomainRemaps = new Dictionary<string, string>
|
|
|
|
|
{
|
|
|
|
|
{"wabbajack.b-cdn.net", "authored-files.wabbajack.org"},
|
|
|
|
|
{"wabbajack-mirror.b-cdn.net", "mirror.wabbajack.org"},
|
2020-11-29 07:42:50 +00:00
|
|
|
|
{"wabbajack-patches.b-cdn.net", "patches.wabbajack.org"},
|
2020-08-31 23:29:48 +00:00
|
|
|
|
{"wabbajacktest.b-cdn.net", "test-files.wabbajack.org"}
|
|
|
|
|
};
|
|
|
|
|
|
2021-03-11 02:28:28 +00:00
|
|
|
|
|
2020-07-20 03:45:55 +00:00
|
|
|
|
public string[]? Mirrors;
|
|
|
|
|
public long TotalRetries;
|
|
|
|
|
|
2020-05-09 22:16:16 +00:00
|
|
|
|
public async Task<AbstractDownloadState?> GetDownloaderState(dynamic archiveINI, bool quickMode = false)
|
|
|
|
|
{
|
|
|
|
|
var url = (Uri)DownloaderUtils.GetDirectURL(archiveINI);
|
|
|
|
|
return url == null ? null : StateFromUrl(url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task Prepare()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-10 21:35:15 +00:00
|
|
|
|
public AbstractDownloadState? GetDownloaderState(string url)
|
|
|
|
|
{
|
|
|
|
|
return StateFromUrl(new Uri(url));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-05-09 22:16:16 +00:00
|
|
|
|
public static AbstractDownloadState? StateFromUrl(Uri url)
|
|
|
|
|
{
|
2020-11-01 17:27:29 +00:00
|
|
|
|
if (DomainRemaps.ContainsKey(url.Host) || DomainRemaps.ContainsValue(url.Host))
|
2020-05-09 22:16:16 +00:00
|
|
|
|
{
|
|
|
|
|
return new State(url);
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[JsonName("WabbajackCDNDownloader+State")]
|
|
|
|
|
public class State : AbstractDownloadState
|
|
|
|
|
{
|
|
|
|
|
public Uri Url { get; set; }
|
|
|
|
|
public State(Uri url)
|
|
|
|
|
{
|
|
|
|
|
Url = url;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override object[] PrimaryKey => new object[] {Url};
|
|
|
|
|
public override bool IsWhitelisted(ServerWhitelist whitelist)
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-11 02:28:28 +00:00
|
|
|
|
public override async Task<(Archive? Archive, TempFile NewFile)> FindUpgrade(Archive a, Func<Archive, Task<AbsolutePath>> downloadResolver)
|
|
|
|
|
{
|
|
|
|
|
return default;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 22:16:16 +00:00
|
|
|
|
public override async Task<bool> Download(Archive a, AbsolutePath destination)
|
|
|
|
|
{
|
2020-05-10 22:45:48 +00:00
|
|
|
|
destination.Parent.CreateDirectory();
|
2020-05-09 22:16:16 +00:00
|
|
|
|
var definition = await GetDefinition();
|
2020-05-25 17:34:25 +00:00
|
|
|
|
await using var fs = await destination.Create();
|
2020-05-13 03:53:20 +00:00
|
|
|
|
using var mmfile = MemoryMappedFile.CreateFromFile(fs, null, definition.Size, MemoryMappedFileAccess.ReadWrite, HandleInheritability.None, false);
|
2020-06-26 17:08:30 +00:00
|
|
|
|
var client = new Wabbajack.Lib.Http.Client();
|
2021-03-19 03:55:43 +00:00
|
|
|
|
|
2021-05-17 23:20:03 +00:00
|
|
|
|
if (!DomainRemaps.ContainsKey(Url.Host))
|
2020-08-31 23:29:48 +00:00
|
|
|
|
client.Headers.Add(("Host", Url.Host));
|
|
|
|
|
|
2020-05-13 03:53:20 +00:00
|
|
|
|
using var queue = new WorkQueue();
|
|
|
|
|
await definition.Parts.PMap(queue, async part =>
|
2020-05-09 22:16:16 +00:00
|
|
|
|
{
|
2020-05-13 03:53:20 +00:00
|
|
|
|
Utils.Status($"Downloading {a.Name}", Percent.FactoryPutInRange(definition.Parts.Length - part.Index, definition.Parts.Length));
|
|
|
|
|
await using var ostream = mmfile.CreateViewStream(part.Offset, part.Size);
|
2020-08-31 23:29:48 +00:00
|
|
|
|
|
|
|
|
|
if (DomainRemaps.TryGetValue(Url.Host, out var remap))
|
|
|
|
|
{
|
|
|
|
|
var builder = new UriBuilder(Url) {Host = remap};
|
2021-01-11 08:42:03 +00:00
|
|
|
|
using var response = await GetWithCDNRetry(client, $"{builder}/parts/{part.Index}");
|
2020-08-31 23:29:48 +00:00
|
|
|
|
if (!response.IsSuccessStatusCode)
|
2021-01-01 00:06:56 +00:00
|
|
|
|
throw new HttpException((int)response.StatusCode, response.ReasonPhrase ?? "Unknown");
|
2020-08-31 23:29:48 +00:00
|
|
|
|
await response.Content.CopyToAsync(ostream);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-03-16 12:00:48 +00:00
|
|
|
|
using var response = await GetWithRetry(client, $"{Url}/parts/{part.Index}");
|
2020-08-31 23:29:48 +00:00
|
|
|
|
if (!response.IsSuccessStatusCode)
|
2021-01-01 00:06:56 +00:00
|
|
|
|
throw new HttpException((int)response.StatusCode, response.ReasonPhrase ?? "Unknown");
|
2020-08-31 23:29:48 +00:00
|
|
|
|
await response.Content.CopyToAsync(ostream);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 22:16:16 +00:00
|
|
|
|
});
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
public override async Task<bool> Verify(Archive archive, CancellationToken? token)
|
2020-05-09 22:16:16 +00:00
|
|
|
|
{
|
2020-12-31 06:44:42 +00:00
|
|
|
|
var definition = await GetDefinition(token);
|
2020-05-09 22:16:16 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-11 08:42:03 +00:00
|
|
|
|
private async Task<HttpResponseMessage> GetWithCDNRetry(Http.Client client, string url, CancellationToken? token = null)
|
|
|
|
|
{
|
|
|
|
|
int retries = 0;
|
|
|
|
|
|
|
|
|
|
TOP:
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return await client.GetAsync(url, retry: false, token: token);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
if (retries > 2)
|
|
|
|
|
{
|
|
|
|
|
var remap = url.Replace(new Uri(url).Host, DomainRemaps.FirstOrDefault(x => x.Value == new Uri(url).Host).Key);
|
|
|
|
|
return await client.GetAsync(remap, retry: false, token: token);
|
|
|
|
|
}
|
|
|
|
|
retries += 1;
|
|
|
|
|
Utils.Log($"Error reading {url} retrying [{retries}]");
|
|
|
|
|
Utils.Log(ex.ToString());
|
|
|
|
|
goto TOP;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-16 12:00:48 +00:00
|
|
|
|
private async Task<HttpResponseMessage> GetWithRetry(Http.Client client, string url)
|
2020-07-20 03:45:55 +00:00
|
|
|
|
{
|
2021-03-16 12:00:48 +00:00
|
|
|
|
return await client.GetAsync(url, retry: false);
|
2020-07-20 03:45:55 +00:00
|
|
|
|
}
|
2020-08-31 23:29:48 +00:00
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
private async Task<CDNFileDefinition> GetDefinition(CancellationToken? token = null)
|
2020-05-09 22:16:16 +00:00
|
|
|
|
{
|
2020-06-26 17:08:30 +00:00
|
|
|
|
var client = new Wabbajack.Lib.Http.Client();
|
2020-08-31 23:29:48 +00:00
|
|
|
|
if (DomainRemaps.TryGetValue(Url.Host, out var remap))
|
|
|
|
|
{
|
|
|
|
|
var builder = new UriBuilder(Url) {Host = remap};
|
2021-01-11 08:42:03 +00:00
|
|
|
|
using var data = await GetWithCDNRetry(client, builder + "/definition.json.gz", token: token);
|
2020-08-31 23:29:48 +00:00
|
|
|
|
await using var gz = new GZipStream(await data.Content.ReadAsStreamAsync(),
|
|
|
|
|
CompressionMode.Decompress);
|
|
|
|
|
return gz.FromJson<CDNFileDefinition>();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
client.Headers.Add(("Host", Url.Host));
|
2021-03-16 12:00:48 +00:00
|
|
|
|
using var data = await GetWithRetry(client, Url + "/definition.json.gz");
|
2020-08-31 23:29:48 +00:00
|
|
|
|
await using var gz = new GZipStream(await data.Content.ReadAsStreamAsync(),
|
|
|
|
|
CompressionMode.Decompress);
|
|
|
|
|
return gz.FromJson<CDNFileDefinition>();
|
|
|
|
|
}
|
2020-05-09 22:16:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override IDownloader GetDownloader()
|
|
|
|
|
{
|
|
|
|
|
return DownloadDispatcher.GetInstance<WabbajackCDNDownloader>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string? GetManifestURL(Archive a)
|
|
|
|
|
{
|
|
|
|
|
return Url.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string[] GetMetaIni()
|
|
|
|
|
{
|
|
|
|
|
return new[] {"[General]", $"directURL={Url}"};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|