2020-06-01 22:27:57 +00:00
|
|
|
|
using System;
|
2020-04-30 22:35:16 +00:00
|
|
|
|
using System.Collections.Generic;
|
2020-07-16 21:17:37 +00:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.IO.Compression;
|
|
|
|
|
using System.Linq;
|
2020-07-14 12:15:01 +00:00
|
|
|
|
using System.Net;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
using System.Net.Http;
|
|
|
|
|
using System.Text;
|
2020-04-30 22:35:16 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2020-07-16 21:17:37 +00:00
|
|
|
|
using K4os.Compression.LZ4.Internal;
|
2021-04-28 11:57:49 +00:00
|
|
|
|
using Newtonsoft.Json;
|
2020-07-15 04:20:56 +00:00
|
|
|
|
using Org.BouncyCastle.Crypto.Agreement.Srp;
|
|
|
|
|
using Wabbajack.Common;
|
2020-05-21 22:12:51 +00:00
|
|
|
|
using Wabbajack.Common.Exceptions;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
using Wabbajack.Common.Serialization.Json;
|
|
|
|
|
using Wabbajack.Lib.Downloaders;
|
2020-09-09 12:51:38 +00:00
|
|
|
|
using Wabbajack.Lib.LibCefHelpers;
|
2021-04-28 11:57:49 +00:00
|
|
|
|
using Wabbajack.Lib.ModListRegistry;
|
2020-09-09 12:51:38 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.Lib
|
2020-02-27 13:46:34 +00:00
|
|
|
|
{
|
2020-07-14 12:15:01 +00:00
|
|
|
|
public static class BuildServerStatus
|
|
|
|
|
{
|
|
|
|
|
private static bool _didCheck;
|
|
|
|
|
private static bool _isBuildServerDown;
|
|
|
|
|
|
|
|
|
|
private static bool CheckBuildServer()
|
|
|
|
|
{
|
2020-07-14 12:44:07 +00:00
|
|
|
|
var client = new Http.Client();
|
2020-07-14 12:15:01 +00:00
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var result = client.GetAsync($"{Consts.WabbajackBuildServerUri}heartbeat").Result;
|
2020-07-14 12:44:07 +00:00
|
|
|
|
_isBuildServerDown = result.StatusCode != HttpStatusCode.OK && result.StatusCode != HttpStatusCode.InternalServerError;
|
2020-07-14 12:15:01 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception)
|
|
|
|
|
{
|
|
|
|
|
_isBuildServerDown = true;
|
|
|
|
|
}
|
|
|
|
|
finally
|
|
|
|
|
{
|
|
|
|
|
_didCheck = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Utils.Log($"Build server is {(_isBuildServerDown ? "down" : "alive")}");
|
|
|
|
|
return _isBuildServerDown;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static bool IsBuildServerDown
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return _didCheck ? _isBuildServerDown : CheckBuildServer();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-20 03:25:41 +00:00
|
|
|
|
[JsonName("ModUpgradeRequest")]
|
|
|
|
|
public class ModUpgradeRequest
|
|
|
|
|
{
|
|
|
|
|
public Archive OldArchive { get; set; }
|
|
|
|
|
public Archive NewArchive { get; set; }
|
|
|
|
|
|
|
|
|
|
public ModUpgradeRequest(Archive oldArchive, Archive newArchive)
|
|
|
|
|
{
|
|
|
|
|
OldArchive = oldArchive;
|
|
|
|
|
NewArchive = newArchive;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
public async Task<bool> IsValid()
|
2020-05-20 03:25:41 +00:00
|
|
|
|
{
|
2020-07-13 22:10:05 +00:00
|
|
|
|
if (OldArchive.Size > 2_500_000_000 || NewArchive.Size > 2_500_000_000) return false;
|
|
|
|
|
if (OldArchive.Hash == NewArchive.Hash && OldArchive.State.PrimaryKeyString == NewArchive.State.PrimaryKeyString) return false;
|
|
|
|
|
if (OldArchive.State.GetType() != NewArchive.State.GetType())
|
2020-05-20 03:25:41 +00:00
|
|
|
|
return false;
|
2020-07-13 22:10:05 +00:00
|
|
|
|
if (OldArchive.State is IUpgradingState u)
|
|
|
|
|
{
|
|
|
|
|
return await u.ValidateUpgrade(OldArchive.Hash, NewArchive.State);
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
2020-07-13 22:10:05 +00:00
|
|
|
|
|
|
|
|
|
return false;
|
2020-05-20 03:25:41 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-04-28 11:57:49 +00:00
|
|
|
|
|
2020-02-27 13:46:34 +00:00
|
|
|
|
public class ClientAPI
|
|
|
|
|
{
|
2020-06-26 17:08:30 +00:00
|
|
|
|
public static async Task<Wabbajack.Lib.Http.Client> GetClient()
|
2020-02-27 13:46:34 +00:00
|
|
|
|
{
|
2020-06-26 17:08:30 +00:00
|
|
|
|
var client = new Wabbajack.Lib.Http.Client();
|
2020-06-01 22:27:57 +00:00
|
|
|
|
client.Headers.Add((Consts.MetricsKeyHeader, await Metrics.GetMetricsKey()));
|
2020-02-27 13:46:34 +00:00
|
|
|
|
return client;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
public static async Task<Uri> GetModUpgrade(Archive oldArchive, Archive newArchive, TimeSpan? maxWait = null, TimeSpan? waitBetweenTries = null, bool useAuthor = false)
|
2020-02-27 13:46:34 +00:00
|
|
|
|
{
|
2020-05-20 03:25:41 +00:00
|
|
|
|
maxWait ??= TimeSpan.FromMinutes(10);
|
|
|
|
|
waitBetweenTries ??= TimeSpan.FromSeconds(15);
|
|
|
|
|
|
|
|
|
|
var request = new ModUpgradeRequest( oldArchive, newArchive);
|
|
|
|
|
var start = DateTime.UtcNow;
|
|
|
|
|
|
|
|
|
|
RETRY:
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
var response = await (useAuthor ? await AuthorApi.Client.GetAuthorizedClient() : await GetClient())
|
2020-05-20 03:25:41 +00:00
|
|
|
|
.PostAsync($"{Consts.WabbajackBuildServerUri}mod_upgrade", new StringContent(request.ToJson(), Encoding.UTF8, "application/json"));
|
|
|
|
|
|
2020-04-10 03:54:02 +00:00
|
|
|
|
if (response.IsSuccessStatusCode)
|
|
|
|
|
{
|
2020-05-20 03:25:41 +00:00
|
|
|
|
switch (response.StatusCode)
|
|
|
|
|
{
|
|
|
|
|
case HttpStatusCode.OK:
|
|
|
|
|
return new Uri(await response.Content.ReadAsStringAsync());
|
|
|
|
|
case HttpStatusCode.Accepted:
|
|
|
|
|
Utils.Log($"Waiting for patch processing on the server for {oldArchive.Name}, sleeping for another 15 seconds");
|
|
|
|
|
await Task.Delay(TimeSpan.FromSeconds(15));
|
|
|
|
|
response.Dispose();
|
|
|
|
|
if (DateTime.UtcNow - start > maxWait)
|
|
|
|
|
throw new HttpException(response);
|
|
|
|
|
goto RETRY;
|
|
|
|
|
}
|
2020-04-10 03:54:02 +00:00
|
|
|
|
}
|
2020-05-20 03:25:41 +00:00
|
|
|
|
var ex = new HttpException(response);
|
|
|
|
|
response.Dispose();
|
|
|
|
|
throw ex;
|
2020-02-27 13:46:34 +00:00
|
|
|
|
}
|
2020-03-31 22:05:36 +00:00
|
|
|
|
|
2020-04-02 21:16:46 +00:00
|
|
|
|
public class NexusCacheStats
|
|
|
|
|
{
|
|
|
|
|
public long CachedCount { get; set; }
|
|
|
|
|
public long ForwardCount { get; set; }
|
|
|
|
|
public double CacheRatio { get; set; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static async Task<NexusCacheStats> GetNexusCacheStats()
|
|
|
|
|
{
|
2020-05-25 14:31:56 +00:00
|
|
|
|
return await (await GetClient())
|
2020-04-02 21:16:46 +00:00
|
|
|
|
.GetJsonAsync<NexusCacheStats>($"{Consts.WabbajackBuildServerUri}nexus_cache/stats");
|
|
|
|
|
}
|
2020-04-30 22:35:16 +00:00
|
|
|
|
|
2020-06-20 22:51:47 +00:00
|
|
|
|
public static async Task SendModListDefinition(ModList modList)
|
2020-04-30 22:35:16 +00:00
|
|
|
|
{
|
2020-06-20 22:51:47 +00:00
|
|
|
|
var client = await GetClient();
|
2020-07-14 12:15:01 +00:00
|
|
|
|
if (BuildServerStatus.IsBuildServerDown)
|
|
|
|
|
return;
|
2020-07-16 21:17:37 +00:00
|
|
|
|
var data = Encoding.UTF8.GetBytes(modList.ToJson());
|
|
|
|
|
|
|
|
|
|
await using var fs = new MemoryStream();
|
|
|
|
|
await using var gzip = new GZipStream(fs, CompressionLevel.Optimal, true);
|
|
|
|
|
await gzip.WriteAsync(data);
|
|
|
|
|
await gzip.DisposeAsync();
|
|
|
|
|
|
|
|
|
|
client.Headers.Add((Consts.CompressedBodyHeader, "gzip"));
|
|
|
|
|
await client.PostAsync($"{Consts.WabbajackBuildServerUri}list_definitions/ingest", new ByteArrayContent(fs.ToArray()));
|
2020-04-30 22:35:16 +00:00
|
|
|
|
}
|
2020-06-14 13:13:29 +00:00
|
|
|
|
|
2020-06-20 22:51:47 +00:00
|
|
|
|
public static async Task<Archive[]> GetExistingGameFiles(WorkQueue queue, Game game)
|
2020-06-14 13:13:29 +00:00
|
|
|
|
{
|
2020-06-20 22:51:47 +00:00
|
|
|
|
var metaData = game.MetaData();
|
2020-11-03 01:55:54 +00:00
|
|
|
|
var results = await GetGameFilesFromGithub(game, metaData.InstalledVersion);
|
2020-06-20 22:51:47 +00:00
|
|
|
|
|
|
|
|
|
return (await results.PMap(queue, async file => (await file.State.Verify(file), file))).Where(f => f.Item1)
|
|
|
|
|
.Select(f =>
|
|
|
|
|
{
|
|
|
|
|
f.file.Name = ((GameFileSourceDownloader.State)f.file.State).GameFile.Munge().ToString();
|
|
|
|
|
return f.file;
|
|
|
|
|
})
|
|
|
|
|
.ToArray();
|
2020-06-14 13:13:29 +00:00
|
|
|
|
}
|
2020-11-03 01:55:54 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<Archive[]> GetGameFilesFromGithub(Game game, string version)
|
|
|
|
|
{
|
|
|
|
|
var url =
|
|
|
|
|
$"https://raw.githubusercontent.com/wabbajack-tools/indexed-game-files/master/{game}/{version}.json";
|
|
|
|
|
Utils.Log($"Loading game file definition from {url}");
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
return await client.GetJsonAsync<Archive[]>(url);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static async Task<Archive[]> GetGameFilesFromServer(Game game, string version)
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
return await client.GetJsonAsync<Archive[]>(
|
|
|
|
|
$"{Consts.WabbajackBuildServerUri}game_files/{game}/{version}");
|
|
|
|
|
}
|
2020-06-21 22:03:54 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<AbstractDownloadState?> InferDownloadState(Hash hash)
|
|
|
|
|
{
|
2020-07-14 12:15:01 +00:00
|
|
|
|
if (BuildServerStatus.IsBuildServerDown)
|
|
|
|
|
return null;
|
|
|
|
|
|
2020-06-21 22:03:54 +00:00
|
|
|
|
var client = await GetClient();
|
2020-07-14 12:15:01 +00:00
|
|
|
|
|
|
|
|
|
var results = await client.GetJsonAsync<Archive[]>(
|
|
|
|
|
$"{Consts.WabbajackBuildServerUri}mod_files/by_hash/{hash.ToHex()}");
|
2020-06-21 22:03:54 +00:00
|
|
|
|
|
2020-07-12 02:18:58 +00:00
|
|
|
|
await DownloadDispatcher.PrepareAll(results.Select(r => r.State));
|
2020-06-21 22:03:54 +00:00
|
|
|
|
foreach (var result in results)
|
|
|
|
|
{
|
2020-07-12 02:18:58 +00:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
if (await result.State.Verify(result)) return result.State;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
Utils.Log($"Verification error for failed for inferenced archive {result.State.PrimaryKeyString}");
|
|
|
|
|
Utils.Log(ex.ToString());
|
|
|
|
|
}
|
2020-06-21 22:03:54 +00:00
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-07-13 22:10:05 +00:00
|
|
|
|
|
2020-12-31 06:44:42 +00:00
|
|
|
|
public static async Task<Archive[]> InferAllDownloadStates(Hash hash)
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
|
|
|
|
|
var results = await client.GetJsonAsync<Archive[]>(
|
|
|
|
|
$"{Consts.WabbajackBuildServerUri}mod_files/by_hash/{hash.ToHex()}");
|
|
|
|
|
return results;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-13 22:10:05 +00:00
|
|
|
|
public static async Task<Archive[]> GetModUpgrades(Hash src)
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
Utils.Log($"Looking for generic upgrade for {src} ({(long)src})");
|
|
|
|
|
var results = await client.GetJsonAsync<Archive[]>($"{Consts.WabbajackBuildServerUri}mod_upgrade/find/{src.ToHex()}");
|
|
|
|
|
return results;
|
|
|
|
|
}
|
2020-07-15 04:20:56 +00:00
|
|
|
|
|
2020-07-20 03:45:55 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<string[]> GetCDNMirrorList()
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
Utils.Log($"Looking for CDN mirrors");
|
|
|
|
|
var results = await client.GetJsonAsync<string[]>($"{Consts.WabbajackBuildServerUri}authored_files/mirrors");
|
|
|
|
|
return results;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-15 04:20:56 +00:00
|
|
|
|
public static async Task<VirusScanner.Result> GetVirusScanResult(AbsolutePath path)
|
|
|
|
|
{
|
2021-08-12 05:12:12 +00:00
|
|
|
|
return VirusScanner.Result.NotMalware;
|
2020-07-15 04:20:56 +00:00
|
|
|
|
var client = await GetClient();
|
|
|
|
|
Utils.Log($"Checking virus result for {path}");
|
|
|
|
|
|
|
|
|
|
var hash = await path.FileHashAsync();
|
2021-01-09 19:04:11 +00:00
|
|
|
|
if (hash == null)
|
|
|
|
|
{
|
|
|
|
|
throw new Exception("Hash is null!");
|
|
|
|
|
}
|
2020-07-15 04:20:56 +00:00
|
|
|
|
|
2021-01-09 19:04:11 +00:00
|
|
|
|
using var result = await client.GetAsync($"{Consts.WabbajackBuildServerUri}virus_scan/{hash.Value.ToHex()}", errorsAsExceptions: false);
|
2020-07-15 04:20:56 +00:00
|
|
|
|
if (result.StatusCode == HttpStatusCode.OK)
|
|
|
|
|
{
|
|
|
|
|
var data = await result.Content.ReadAsStringAsync();
|
|
|
|
|
return Enum.Parse<VirusScanner.Result>(data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result.StatusCode == HttpStatusCode.NotFound)
|
|
|
|
|
{
|
|
|
|
|
await using var input = await path.OpenRead();
|
|
|
|
|
using var postResult = await client.PostAsync($"{Consts.WabbajackBuildServerUri}virus_scan", errorsAsExceptions: false, content: new StreamContent(input));
|
|
|
|
|
if (postResult.StatusCode == HttpStatusCode.OK)
|
|
|
|
|
{
|
|
|
|
|
var data = await postResult.Content.ReadAsStringAsync();
|
|
|
|
|
return Enum.Parse<VirusScanner.Result>(data);
|
|
|
|
|
}
|
|
|
|
|
throw new HttpException(result);
|
|
|
|
|
}
|
|
|
|
|
throw new HttpException(result);
|
|
|
|
|
}
|
2020-08-08 03:40:03 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<Uri?> GetMirrorUrl(Hash archiveHash)
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var result =
|
2020-08-26 04:03:43 +00:00
|
|
|
|
await client.GetStringAsync($"{Consts.WabbajackBuildServerUri}mirror/{archiveHash.ToHex()}");
|
2020-08-08 03:40:03 +00:00
|
|
|
|
return new Uri(result);
|
|
|
|
|
}
|
2020-11-14 14:26:04 +00:00
|
|
|
|
catch (HttpException)
|
2020-08-08 03:40:03 +00:00
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-09 12:51:38 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<Helpers.Cookie[]> GetAuthInfo<T>(string key)
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
return await client.GetJsonAsync<Helpers.Cookie[]>(
|
|
|
|
|
$"{Consts.WabbajackBuildServerUri}site-integration/auth-info/{key}");
|
|
|
|
|
}
|
2020-11-03 01:55:54 +00:00
|
|
|
|
|
|
|
|
|
public static async Task<IEnumerable<(Game, string)>> GetServerGamesAndVersions()
|
|
|
|
|
{
|
|
|
|
|
var client = await GetClient();
|
|
|
|
|
var results =
|
|
|
|
|
await client.GetJsonAsync<(Game, string)[]>(
|
|
|
|
|
$"{Consts.WabbajackBuildServerUri}game_files");
|
|
|
|
|
return results;
|
|
|
|
|
}
|
2020-02-27 13:46:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|