wabbajack/Wabbajack.VirtualFileSystem.Test/VirtualFileSystemTests.cs

220 lines
7.1 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.IO.Compression;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Wabbajack.Common;
using Wabbajack.Lib;
using Wabbajack.Lib.Downloaders;
2020-03-24 12:21:19 +00:00
using Xunit;
using Xunit.Abstractions;
namespace Wabbajack.VirtualFileSystem.Test
{
public class VFSTests : IAsyncLifetime
{
2020-03-24 02:46:30 +00:00
private static readonly AbsolutePath VFS_TEST_DIR = "vfs_test_dir".ToPath().RelativeToEntryPoint();
private static readonly AbsolutePath TEST_ZIP = "test.zip".RelativeTo(VFS_TEST_DIR);
private static readonly AbsolutePath TEST_TXT = "test.txt".RelativeTo(VFS_TEST_DIR);
2020-03-24 21:42:28 +00:00
private static readonly AbsolutePath ARCHIVE_TEST_TXT = "archive/test.txt".RelativeTo(VFS_TEST_DIR);
private Context context;
2020-03-24 12:21:19 +00:00
private readonly ITestOutputHelper _helper;
2020-09-09 11:28:34 +00:00
private IDisposable _unsub;
private WorkQueue Queue { get; } = new WorkQueue();
2020-04-11 04:22:10 +00:00
public VFSTests(ITestOutputHelper helper)
{
2020-03-24 12:21:19 +00:00
_helper = helper;
2020-09-09 11:28:34 +00:00
_unsub = Utils.LogMessages.Subscribe(f =>
{
try
{
_helper.WriteLine(f.ShortDescription);
}
2020-10-01 03:50:09 +00:00
catch (Exception)
2020-09-09 11:28:34 +00:00
{
// ignored
}
});
2019-11-17 04:16:42 +00:00
context = new Context(Queue);
}
public async Task InitializeAsync()
{
await VFS_TEST_DIR.DeleteDirectory();
VFS_TEST_DIR.CreateDirectory();
}
public async Task DisposeAsync()
{
2020-09-09 11:28:34 +00:00
_unsub.Dispose();
await VFS_TEST_DIR.DeleteDirectory();
}
2020-03-24 12:21:19 +00:00
[Fact]
public async Task FilesAreIndexed()
{
2020-03-24 02:46:30 +00:00
await AddFile(TEST_TXT, "This is a test");
await AddTestRoot();
var file = context.Index.ByRootPath["test.txt".ToPath().RelativeTo(VFS_TEST_DIR)];
2020-03-24 12:21:19 +00:00
Assert.NotNull(file);
2020-03-24 12:21:19 +00:00
Assert.Equal(14, file.Size);
Assert.Equal(file.Hash, Hash.FromBase64("qX0GZvIaTKM="));
}
2020-03-24 12:21:19 +00:00
private async Task AddTestRoot()
{
await context.AddRoot(VFS_TEST_DIR);
await context.WriteToFile("vfs_cache.bin".RelativeTo(VFS_TEST_DIR));
await context.IntegrateFromFile( "vfs_cache.bin".RelativeTo(VFS_TEST_DIR));
}
2020-03-24 21:42:28 +00:00
[Fact]
public async Task ArchiveContentsAreIndexed()
{
2020-03-24 02:46:30 +00:00
await AddFile(ARCHIVE_TEST_TXT, "This is a test");
await ZipUpFolder(ARCHIVE_TEST_TXT.Parent, TEST_ZIP);
await AddTestRoot();
2020-03-24 02:46:30 +00:00
var absPath = "test.zip".RelativeTo(VFS_TEST_DIR);
var file = context.Index.ByRootPath[absPath];
2020-03-24 21:42:28 +00:00
Assert.NotNull(file);
2020-03-24 21:42:28 +00:00
Assert.Equal(128, file.Size);
Assert.Equal(await absPath.FileHashAsync(), file.Hash);
2020-03-24 21:42:28 +00:00
Assert.True(file.IsArchive);
var innerFile = file.Children.First();
2020-03-24 21:42:28 +00:00
Assert.Equal(14, innerFile.Size);
Assert.Equal(Hash.FromBase64("qX0GZvIaTKM="), innerFile.Hash);
Assert.Same(file, file.Children.First().Parent);
}
2020-03-24 21:42:28 +00:00
2020-03-24 21:42:28 +00:00
[Fact]
public async Task DuplicateFileHashes()
{
2020-03-24 02:46:30 +00:00
await AddFile(ARCHIVE_TEST_TXT, "This is a test");
await ZipUpFolder(ARCHIVE_TEST_TXT.Parent, TEST_ZIP);
2020-03-24 02:46:30 +00:00
await AddFile(TEST_TXT, "This is a test");
await AddTestRoot();
2020-03-22 15:50:53 +00:00
var files = context.Index.ByHash[Hash.FromBase64("qX0GZvIaTKM=")];
2020-03-24 21:42:28 +00:00
Assert.Equal(2, files.Count());
}
2020-03-24 21:42:28 +00:00
[Fact]
public async Task DeletedFilesAreRemoved()
{
2020-03-24 02:46:30 +00:00
await AddFile(TEST_TXT, "This is a test");
await AddTestRoot();
2020-03-24 02:46:30 +00:00
var file = context.Index.ByRootPath[TEST_TXT];
2020-03-24 21:42:28 +00:00
Assert.NotNull(file);
2020-03-24 21:42:28 +00:00
Assert.Equal(14, file.Size);
Assert.Equal(Hash.FromBase64("qX0GZvIaTKM="), file.Hash);
2020-05-26 11:31:11 +00:00
await TEST_TXT.DeleteAsync();
await AddTestRoot();
2020-03-24 21:42:28 +00:00
Assert.DoesNotContain(TEST_TXT, context.Index.AllFiles.Select(f => f.AbsoluteName));
}
2020-03-24 21:42:28 +00:00
[Fact]
public async Task UnmodifiedFilesAreNotReIndexed()
{
2020-03-24 02:46:30 +00:00
await AddFile(TEST_TXT, "This is a test");
await AddTestRoot();
2020-03-24 02:46:30 +00:00
var old_file = context.Index.ByRootPath[TEST_TXT];
var old_time = old_file.LastAnalyzed;
await AddTestRoot();
2020-03-24 02:46:30 +00:00
var new_file = context.Index.ByRootPath[TEST_TXT];
2020-03-24 21:42:28 +00:00
Assert.Equal(old_time, new_file.LastAnalyzed);
}
2020-03-24 21:42:28 +00:00
[Fact]
public async Task CanStageSimpleArchives()
{
2020-03-24 02:46:30 +00:00
await AddFile(ARCHIVE_TEST_TXT, "This is a test");
await ZipUpFolder(ARCHIVE_TEST_TXT.Parent, TEST_ZIP);
await AddTestRoot();
2020-03-24 21:42:28 +00:00
var res = new FullPath(TEST_ZIP, new[] {(RelativePath)"test.txt"});
2020-09-04 21:00:29 +00:00
var files = new [] {context.Index.ByFullPath[res]};
2020-09-04 21:00:29 +00:00
var queue = new WorkQueue();
await context.Extract(queue, files.ToHashSet(), async (file, factory) =>
{
await using var s = await factory.GetStream();
Assert.Equal("This is a test", await s.ReadAllTextAsync());
});
}
2020-03-24 21:42:28 +00:00
[Fact]
public async Task CanStageNestedArchives()
{
2020-03-24 02:46:30 +00:00
await AddFile(ARCHIVE_TEST_TXT, "This is a test");
await ZipUpFolder(ARCHIVE_TEST_TXT.Parent, TEST_ZIP);
2020-03-24 02:46:30 +00:00
var inner_dir = @"archive\other\dir".RelativeTo(VFS_TEST_DIR);
inner_dir.CreateDirectory();
await TEST_ZIP.MoveToAsync( @"archive\other\dir\nested.zip".RelativeTo(VFS_TEST_DIR));
await ZipUpFolder(ARCHIVE_TEST_TXT.Parent, TEST_ZIP);
await AddTestRoot();
2020-03-22 15:50:53 +00:00
var files = context.Index.ByHash[Hash.FromBase64("qX0GZvIaTKM=")];
2020-09-04 21:00:29 +00:00
var queue = new WorkQueue();
await context.Extract(queue, files.ToHashSet(), async (file, factory) =>
{
2020-09-04 21:00:29 +00:00
await using var s = await factory.GetStream();
Assert.Equal("This is a test", await s.ReadAllTextAsync());
});
}
[Theory]
[InlineData(Game.SkyrimSpecialEdition, 20035, 130759)] // Lucian
public async Task CanAnalyzeMods(Game game, int modid, int fileId)
{
await using var tmpFolder = await TempFolder.Create();
var path = await FileExtractorTests.DownloadMod(game, modid, fileId);
await path.CopyToAsync(path.FileName.RelativeTo(tmpFolder.Dir));
var context = new Context(Queue);
await context.AddRoot(tmpFolder.Dir);
Assert.True(context.Index.ByFullPath.Count >= 3);
}
2020-03-24 02:46:30 +00:00
private static async Task AddFile(AbsolutePath filename, string text)
{
2020-03-24 02:46:30 +00:00
filename.Parent.CreateDirectory();
await filename.WriteAllTextAsync(text);
}
private static async Task ZipUpFolder(AbsolutePath folder, AbsolutePath output)
{
2020-03-24 02:46:30 +00:00
ZipFile.CreateFromDirectory((string)folder, (string)output);
await folder.DeleteDirectory();
}
}
}