Fixes for when authors create lists with broken download links

This commit is contained in:
Timothy Baldridge 2020-05-13 06:09:20 -06:00
parent 4357564347
commit 8d8af953ac
5 changed files with 61 additions and 19 deletions

View File

@ -38,12 +38,16 @@ namespace Wabbajack.BuildServer.Test
Consts.ModlistMetadataURL = modlist.ToString(); Consts.ModlistMetadataURL = modlist.ToString();
var sql = Fixture.GetService<SqlService>(); var sql = Fixture.GetService<SqlService>();
var downloader = Fixture.GetService<ModListDownloader>(); var downloader = Fixture.GetService<ModListDownloader>();
await downloader.CheckForNewLists(); Assert.Equal(2, await downloader.CheckForNewLists());
foreach (var list in ModListMetaData) foreach (var list in ModListMetaData)
{ {
Assert.True(await sql.HaveIndexedModlist(list.Links.MachineURL, list.DownloadMetadata.Hash)); Assert.True(await sql.HaveIndexedModlist(list.Links.MachineURL, list.DownloadMetadata.Hash));
} }
// Nothing has changed so we shouldn't be downloading anything this time
Assert.Equal(0, await downloader.CheckForNewLists());
} }
private async Task<Uri> MakeModList() private async Task<Uri> MakeModList()
@ -90,6 +94,22 @@ namespace Wabbajack.BuildServer.Test
MachineURL = "test_list", MachineURL = "test_list",
Download = MakeURL("test_modlist.wabbajack") Download = MakeURL("test_modlist.wabbajack")
} }
},
new ModlistMetadata
{
Official = true,
Author = "Test Suite",
Description = "A list with a broken hash",
DownloadMetadata = new DownloadMetadata()
{
Hash = Hash.FromLong(42),
Size = 42
},
Links = new ModlistMetadata.LinksObject
{
MachineURL = "broken_list",
Download = MakeURL("test_modlist.wabbajack")
}
} }
}; };

View File

@ -323,6 +323,7 @@ CREATE TABLE [dbo].[ModLists](
[Hash] [bigint] NOT NULL, [Hash] [bigint] NOT NULL,
[Metadata] [nvarchar](max) NOT NULL, [Metadata] [nvarchar](max) NOT NULL,
[Modlist] [nvarchar](max) NOT NULL, [Modlist] [nvarchar](max) NOT NULL,
[BrokenDownload] [tinyint] NOT NULL,
CONSTRAINT [PK_ModLists] PRIMARY KEY CLUSTERED CONSTRAINT [PK_ModLists] PRIMARY KEY CLUSTERED
( (
[MachineURL] ASC [MachineURL] ASC

View File

@ -9,7 +9,7 @@ namespace Wabbajack.Server.DataLayer
{ {
public partial class SqlService public partial class SqlService
{ {
public async Task IngestModList(Hash hash, ModlistMetadata metadata, ModList modlist) public async Task IngestModList(Hash hash, ModlistMetadata metadata, ModList modlist, bool brokenDownload)
{ {
await using var conn = await Open(); await using var conn = await Open();
await using var tran = await conn.BeginTransactionAsync(); await using var tran = await conn.BeginTransactionAsync();
@ -18,13 +18,14 @@ namespace Wabbajack.Server.DataLayer
new {MachineUrl = metadata.Links.MachineURL}, tran); new {MachineUrl = metadata.Links.MachineURL}, tran);
await conn.ExecuteAsync( await conn.ExecuteAsync(
@"INSERT INTO dbo.ModLists (MachineUrl, Hash, Metadata, ModList) VALUES (@MachineUrl, @Hash, @Metadata, @ModList)", @"INSERT INTO dbo.ModLists (MachineUrl, Hash, Metadata, ModList, BrokenDownload) VALUES (@MachineUrl, @Hash, @Metadata, @ModList, @BrokenDownload)",
new new
{ {
MachineUrl = metadata.Links.MachineURL, MachineUrl = metadata.Links.MachineURL,
Hash = hash, Hash = hash,
MetaData = metadata.ToJson(), MetaData = metadata.ToJson(),
ModList = modlist.ToJson() ModList = modlist.ToJson(),
BrokenDownload = brokenDownload
}, tran); }, tran);
var entries = modlist.Archives.Select(a => var entries = modlist.Archives.Select(a =>

View File

@ -12,7 +12,7 @@ using File = System.IO.File;
namespace Wabbajack.Server.Services namespace Wabbajack.Server.Services
{ {
/// <summary> /// <summary>
/// Maintains a concurrent cache of all the files we've downloaded, indexed by Hash /// Maintains a concurrent cache of all the files we've downloaded, indexed by Hash.
/// </summary> /// </summary>
public class ArchiveMaintainer public class ArchiveMaintainer
{ {

View File

@ -1,5 +1,6 @@
using System; using System;
using System.IO.Compression; using System.IO.Compression;
using System.Runtime.CompilerServices;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Builder;
using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging;
@ -38,6 +39,7 @@ namespace Wabbajack.Server.Services
{ {
try try
{ {
_logger.Log(LogLevel.Information, "Checking for updated mod lists");
await CheckForNewLists(); await CheckForNewLists();
} }
catch (Exception ex) catch (Exception ex)
@ -52,31 +54,47 @@ namespace Wabbajack.Server.Services
} }
} }
public async Task CheckForNewLists() public async Task<int> CheckForNewLists()
{ {
int downloaded = 0;
var lists = await ModlistMetadata.LoadFromGithub(); var lists = await ModlistMetadata.LoadFromGithub();
foreach (var list in lists) foreach (var list in lists)
{ {
try try
{ {
if (_maintainer.HaveArchive(list.DownloadMetadata!.Hash)) if (await _sql.HaveIndexedModlist(list.Links.MachineURL, list.DownloadMetadata.Hash))
continue; continue;
_logger.Log(LogLevel.Information, $"Downloading {list.Links.MachineURL}");
var tf = new TempFile(); if (!_maintainer.HaveArchive(list.DownloadMetadata!.Hash))
var state = DownloadDispatcher.ResolveArchive(list.Links.Download);
if (state == null)
{ {
_logger.Log(LogLevel.Error, _logger.Log(LogLevel.Information, $"Downloading {list.Links.MachineURL}");
$"Now downloader found for list {list.Links.MachineURL} : {list.Links.Download}"); var tf = new TempFile();
continue; var state = DownloadDispatcher.ResolveArchive(list.Links.Download);
if (state == null)
{
_logger.Log(LogLevel.Error,
$"Now downloader found for list {list.Links.MachineURL} : {list.Links.Download}");
continue;
}
downloaded += 1;
await state.Download(new Archive(state) {Name = $"{list.Links.MachineURL}.wabbajack"}, tf.Path);
var hash = await tf.Path.FileHashAsync();
if (hash != list.DownloadMetadata.Hash)
{
_logger.Log(LogLevel.Error,
$"Downloaded modlist {list.Links.MachineURL} {list.DownloadMetadata.Hash} didn't match metadata hash of {hash}");
await _sql.IngestModList(list.DownloadMetadata.Hash, list, new ModList(), true);
continue;
}
await _maintainer.Ingest(tf.Path);
} }
await state.Download(new Archive(state) {Name = $"{list.Links.MachineURL}.wabbajack"}, tf.Path); _maintainer.TryGetPath(list.DownloadMetadata.Hash, out var modlistPath);
var modistPath = await _maintainer.Ingest(tf.Path);
ModList modlist; ModList modlist;
await using (var fs = modistPath.OpenRead()) await using (var fs = modlistPath.OpenRead())
using (var zip = new ZipArchive(fs, ZipArchiveMode.Read)) using (var zip = new ZipArchive(fs, ZipArchiveMode.Read))
await using (var entry = zip.GetEntry("modlist")?.Open()) await using (var entry = zip.GetEntry("modlist")?.Open())
{ {
@ -97,13 +115,15 @@ namespace Wabbajack.Server.Services
} }
} }
await _sql.IngestModList(list.DownloadMetadata!.Hash, list, modlist); await _sql.IngestModList(list.DownloadMetadata!.Hash, list, modlist, false);
} }
catch (Exception ex) catch (Exception ex)
{ {
_logger.LogError(ex, $"Error downloading modlist {list.Links.MachineURL}"); _logger.LogError(ex, $"Error downloading modlist {list.Links.MachineURL}");
} }
} }
_logger.Log(LogLevel.Information, $"Done checking modlists. Downloaded {downloaded} new lists");
return downloaded;
} }
} }