2020-03-04 02:19:11 +01:00
|
|
|
|
using System;
|
|
|
|
|
using LibHac.Common;
|
|
|
|
|
using LibHac.Fs;
|
2020-06-10 07:55:59 +02:00
|
|
|
|
using LibHac.Fs.Fsa;
|
2020-03-04 02:19:11 +01:00
|
|
|
|
using LibHac.FsSystem;
|
2020-10-11 05:17:07 +02:00
|
|
|
|
using LibHac.Util;
|
2020-03-04 02:19:11 +01:00
|
|
|
|
using Xunit;
|
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
namespace LibHac.Tests.Fs;
|
|
|
|
|
|
|
|
|
|
public class LayeredFileSystemTests
|
2020-03-04 02:19:11 +01:00
|
|
|
|
{
|
2021-11-14 20:08:57 +01:00
|
|
|
|
private IFileSystem CreateFileSystem()
|
2020-03-04 02:19:11 +01:00
|
|
|
|
{
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var lowerLayerFs = new InMemoryFileSystem();
|
|
|
|
|
var upperLayerFs = new InMemoryFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var layeredFs = new LayeredFileSystem(lowerLayerFs, upperLayerFs);
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateDirectory("/dir").ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateDirectory("/dir").ThrowIfFailure();
|
|
|
|
|
lowerLayerFs.CreateDirectory("/dir2").ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateDirectory("/dir2").ThrowIfFailure();
|
|
|
|
|
lowerLayerFs.CreateDirectory("/dir3").ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateDirectory("/dir3").ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateDirectory("/lowerDir").ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateDirectory("/upperDir").ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateFile("/dir/replacedFile", 1, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateFile("/dir/replacedFile", 2, CreateFileOptions.None).ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateFile("/dir2/lowerFile", 0, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateFile("/dir2/upperFile", 0, CreateFileOptions.None).ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateFile("/dir3/lowerFile", 0, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateFile("/dir3/upperFile", 2, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
lowerLayerFs.CreateFile("/dir3/replacedFile", 1, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateFile("/dir3/replacedFile", 2, CreateFileOptions.None).ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
lowerLayerFs.CreateFile("/replacedWithDir", 0, CreateFileOptions.None).ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateDirectory("/replacedWithDir").ThrowIfFailure();
|
|
|
|
|
upperLayerFs.CreateFile("/replacedWithDir/subFile", 0, CreateFileOptions.None).ThrowIfFailure();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
return layeredFs;
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
private IFileSystem CreateEmptyFileSystem()
|
|
|
|
|
{
|
|
|
|
|
var baseLayerFs = new InMemoryFileSystem();
|
|
|
|
|
var topLayerFs = new InMemoryFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
return new LayeredFileSystem(baseLayerFs, topLayerFs);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenFile_FileDoesNotExist_ReturnsPathNotFound()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var file = new UniqueRef<IFile>();
|
|
|
|
|
Assert.Result(ResultFs.PathNotFound, fs.OpenFile(ref file.Ref(), "/fakefile", OpenMode.All));
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenFile_FileIsInBothSources_ReturnsFileFromTopSource()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var file = new UniqueRef<IFile>();
|
|
|
|
|
Assert.Success(fs.OpenFile(ref file.Ref(), "/dir/replacedFile", OpenMode.All));
|
|
|
|
|
Assert.Success(file.Get.GetSize(out long fileSize));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Equal(2, fileSize);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenFile_InsideMergedDirectory_CanOpenFilesFromBothSources()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var file = new UniqueRef<IFile>();
|
|
|
|
|
Assert.Success(fs.OpenFile(ref file.Ref(), "/dir2/lowerFile", OpenMode.All));
|
|
|
|
|
Assert.Success(fs.OpenFile(ref file.Ref(), "/dir2/upperFile", OpenMode.All));
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenDirectory_DirDoesNotExist_ReturnsPathNotFound()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Result(ResultFs.PathNotFound, fs.OpenDirectory(ref directory.Ref(), "/fakedir", OpenDirectoryMode.All));
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenDirectory_ExistsInSingleLayer_ReturnsNonMergedDirectory()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/lowerDir", OpenDirectoryMode.All));
|
|
|
|
|
Assert.Equal(typeof(InMemoryFileSystem), directory.Get.GetType().DeclaringType);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void OpenDirectory_ExistsInMultipleLayers_ReturnsMergedDirectory()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/dir", OpenDirectoryMode.All));
|
|
|
|
|
Assert.Equal(typeof(LayeredFileSystem), directory.Get.GetType().DeclaringType);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void GetEntryType_InsideMergedDirectory_CanGetEntryTypesFromBothSources()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(fs.GetEntryType(out _, "/dir2/lowerFile"));
|
|
|
|
|
Assert.Success(fs.GetEntryType(out _, "/dir2/upperFile"));
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryRead_DuplicatedEntriesAreReturnedOnlyOnce()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
|
|
|
|
Span<DirectoryEntry> entries = stackalloc DirectoryEntry[4];
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/dir3", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.Read(out long entriesRead, entries));
|
|
|
|
|
Assert.Equal(3, entriesRead);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryRead_DuplicatedEntryReturnsFromTopLayer()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
|
|
|
|
var entry = new DirectoryEntry();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/dir", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.Read(out _, SpanHelpers.AsSpan(ref entry)));
|
|
|
|
|
Assert.Equal("replacedFile", StringUtils.Utf8ZToString(entry.Name));
|
|
|
|
|
Assert.Equal(2, entry.Size);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryRead_EmptyFs_NoEntriesAreRead()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateEmptyFileSystem();
|
|
|
|
|
var entry = new DirectoryEntry();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.Read(out long entriesRead, SpanHelpers.AsSpan(ref entry)));
|
|
|
|
|
Assert.Equal(0, entriesRead);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryGetEntryCount_DuplicatedEntriesAreCountedOnlyOnce()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/dir3", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.GetEntryCount(out long entryCount));
|
|
|
|
|
Assert.Equal(3, entryCount);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryGetEntryCount_MergedDirectoryAfterRead_AllEntriesAreCounted()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateFileSystem();
|
|
|
|
|
var entry = new DirectoryEntry();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/dir3", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
// Read all entries
|
|
|
|
|
long entriesRead;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
Assert.Success(directory.Get.Read(out entriesRead, SpanHelpers.AsSpan(ref entry)));
|
|
|
|
|
} while (entriesRead != 0);
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.GetEntryCount(out long entryCount));
|
|
|
|
|
Assert.Equal(3, entryCount);
|
|
|
|
|
}
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void IDirectoryGetEntryCount_EmptyFs_EntryCountIsZero()
|
|
|
|
|
{
|
|
|
|
|
IFileSystem fs = CreateEmptyFileSystem();
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
using var directory = new UniqueRef<IDirectory>();
|
|
|
|
|
Assert.Success(fs.OpenDirectory(ref directory.Ref(), "/", OpenDirectoryMode.All));
|
2020-03-04 02:19:11 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Success(directory.Get.GetEntryCount(out long entryCount));
|
|
|
|
|
Assert.Equal(0, entryCount);
|
2020-03-04 02:19:11 +01:00
|
|
|
|
}
|
|
|
|
|
}
|