Files
Aaru/Aaru.Tests/Images/BlockMediaImageTest.cs

295 lines
12 KiB
C#
Raw Normal View History

2022-03-07 07:36:44 +00:00
namespace Aaru.Tests.Images;
using System;
using System.Collections.Generic;
using System.IO;
using Aaru.Checksums;
using Aaru.CommonTypes;
2021-09-16 04:42:14 +01:00
using Aaru.CommonTypes.Enums;
using Aaru.CommonTypes.Interfaces;
using Aaru.Core;
using Aaru.Tests.Filesystems;
using FluentAssertions.Execution;
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;
using NUnit.Framework;
2022-03-06 13:29:38 +00:00
public abstract class BlockMediaImageTest : BaseMediaImageTest
{
2022-03-06 13:29:38 +00:00
// How many sectors to read at once
const uint SECTORS_TO_READ = 256;
public abstract BlockImageTestExpected[] Tests { get; }
[Test]
public void Info()
{
2022-03-06 13:29:38 +00:00
Environment.CurrentDirectory = DataFolder;
2022-03-06 13:29:38 +00:00
Assert.Multiple(() =>
{
2022-03-06 13:29:38 +00:00
foreach(BlockImageTestExpected test in Tests)
{
2022-03-06 13:29:38 +00:00
string testFile = test.TestFile;
2022-03-06 13:29:38 +00:00
bool exists = File.Exists(testFile);
Assert.True(exists, $"{testFile} not found");
2022-03-06 13:29:38 +00:00
// ReSharper disable once ConditionIsAlwaysTrueOrFalse
// It arrives here...
if(!exists)
continue;
2022-03-06 13:29:38 +00:00
var filtersList = new FiltersList();
IFilter filter = filtersList.GetFilter(testFile);
filter.Open(testFile);
2022-03-06 13:29:38 +00:00
var image = Activator.CreateInstance(_plugin.GetType()) as IMediaImage;
Assert.NotNull(image, $"Could not instantiate filesystem for {testFile}");
2022-03-06 13:29:38 +00:00
ErrorNumber opened = image.Open(filter);
Assert.AreEqual(ErrorNumber.NoError, opened, $"Open: {testFile}");
2022-03-06 13:29:38 +00:00
if(opened != ErrorNumber.NoError)
continue;
2022-03-06 13:29:38 +00:00
using(new AssertionScope())
Assert.Multiple(() =>
{
2022-03-06 13:29:38 +00:00
Assert.AreEqual(test.Sectors, image.Info.Sectors, $"Sectors: {testFile}");
Assert.AreEqual(test.SectorSize, image.Info.SectorSize, $"Sector size: {testFile}");
Assert.AreEqual(test.MediaType, image.Info.MediaType, $"Media type: {testFile}");
});
}
});
}
2022-03-06 13:29:38 +00:00
[Test]
public void Hashes()
{
Environment.CurrentDirectory = DataFolder;
ErrorNumber errno;
2022-03-06 13:29:38 +00:00
Assert.Multiple(() =>
{
foreach(BlockImageTestExpected test in Tests)
{
2022-03-06 13:29:38 +00:00
string testFile = test.TestFile;
2022-03-06 13:29:38 +00:00
bool exists = File.Exists(testFile);
Assert.True(exists, $"{testFile} not found");
2022-03-06 13:29:38 +00:00
// ReSharper disable once ConditionIsAlwaysTrueOrFalse
// It arrives here...
if(!exists)
continue;
2022-03-06 13:29:38 +00:00
var filtersList = new FiltersList();
IFilter filter = filtersList.GetFilter(testFile);
filter.Open(testFile);
2022-03-06 13:29:38 +00:00
var image = Activator.CreateInstance(_plugin.GetType()) as IMediaImage;
Assert.NotNull(image, $"Could not instantiate filesystem for {testFile}");
2022-03-06 13:29:38 +00:00
ErrorNumber opened = image.Open(filter);
Assert.AreEqual(ErrorNumber.NoError, opened, $"Open: {testFile}");
2022-03-06 13:29:38 +00:00
if(opened != ErrorNumber.NoError)
continue;
2022-03-06 13:29:38 +00:00
ulong doneSectors = 0;
var ctx = new Md5Context();
2022-03-06 13:29:38 +00:00
while(doneSectors < image.Info.Sectors)
{
byte[] sector;
2022-03-06 13:29:38 +00:00
if(image.Info.Sectors - doneSectors >= SECTORS_TO_READ)
{
errno = image.ReadSectors(doneSectors, SECTORS_TO_READ, out sector);
doneSectors += SECTORS_TO_READ;
}
else
{
2022-03-07 07:36:44 +00:00
errno = image.ReadSectors(doneSectors, (uint)(image.Info.Sectors - doneSectors), out sector);
2022-03-06 13:29:38 +00:00
doneSectors += image.Info.Sectors - doneSectors;
}
2022-03-06 13:29:38 +00:00
Assert.AreEqual(ErrorNumber.NoError, errno);
ctx.Update(sector);
}
2022-03-06 13:29:38 +00:00
Assert.AreEqual(test.MD5, ctx.End(), $"Hash: {testFile}");
}
});
}
2022-03-06 13:29:38 +00:00
[Test]
public void Contents()
{
Environment.CurrentDirectory = DataFolder;
PluginBase plugins = GetPluginBase.Instance;
Assert.Multiple(() =>
{
foreach(BlockImageTestExpected test in Tests)
{
2022-03-06 13:29:38 +00:00
if(test.Partitions is null)
continue;
2022-03-06 13:29:38 +00:00
string testFile = test.TestFile;
2022-03-06 13:29:38 +00:00
bool exists = File.Exists(testFile);
Assert.True(exists, $"{testFile} not found");
2022-03-06 13:29:38 +00:00
// ReSharper disable once ConditionIsAlwaysTrueOrFalse
// It arrives here...
if(!exists)
continue;
2022-03-06 13:29:38 +00:00
var filtersList = new FiltersList();
IFilter filter = filtersList.GetFilter(testFile);
filter.Open(testFile);
2022-03-06 13:29:38 +00:00
var image = Activator.CreateInstance(_plugin.GetType()) as IMediaImage;
Assert.NotNull(image, $"Could not instantiate filesystem for {testFile}");
2022-03-06 13:29:38 +00:00
ErrorNumber opened = image.Open(filter);
Assert.AreEqual(ErrorNumber.NoError, opened, $"Open: {testFile}");
2022-03-06 13:29:38 +00:00
if(opened != ErrorNumber.NoError)
continue;
2022-03-07 07:36:44 +00:00
List<Partition> partitions = Partitions.GetAll(image);
2022-03-06 13:29:38 +00:00
if(partitions.Count == 0)
partitions.Add(new Partition
{
2022-03-06 13:29:38 +00:00
Description = "Whole device",
Length = image.Info.Sectors,
Offset = 0,
Size = image.Info.SectorSize * image.Info.Sectors,
Sequence = 1,
Start = 0
});
2022-03-06 13:29:38 +00:00
Assert.AreEqual(test.Partitions.Length, partitions.Count,
$"Expected {test.Partitions.Length} partitions in {testFile} but found {partitions.Count}");
2022-03-06 13:29:38 +00:00
using(new AssertionScope())
Assert.Multiple(() =>
{
2022-03-07 07:36:44 +00:00
for(var i = 0; i < test.Partitions.Length; i++)
{
2022-03-06 13:29:38 +00:00
BlockPartitionVolumes expectedPartition = test.Partitions[i];
Partition foundPartition = partitions[i];
2022-03-06 13:29:38 +00:00
Assert.AreEqual(expectedPartition.Start, foundPartition.Start,
$"Expected partition {i} to start at sector {expectedPartition.Start} but found it starts at {foundPartition.Start} in {testFile}");
2022-03-06 13:29:38 +00:00
Assert.AreEqual(expectedPartition.Length, foundPartition.Length,
$"Expected partition {i} to have {expectedPartition.Length} sectors but found it has {foundPartition.Length} sectors in {testFile}");
2022-03-06 13:29:38 +00:00
string expectedDataFilename = $"{testFile}.contents.partition{i}.json";
2022-03-06 13:29:38 +00:00
if(!File.Exists(expectedDataFilename))
continue;
2022-03-06 13:29:38 +00:00
var serializer = new JsonSerializer
{
Formatting = Formatting.Indented,
MaxDepth = 16384,
NullValueHandling = NullValueHandling.Ignore
};
2022-03-06 13:29:38 +00:00
serializer.Converters.Add(new StringEnumConverter());
2022-03-06 13:29:38 +00:00
var sr = new StreamReader(expectedDataFilename);
2022-03-07 07:36:44 +00:00
VolumeData[] expectedData = serializer.Deserialize<VolumeData[]>(new JsonTextReader(sr));
2022-03-06 13:29:38 +00:00
Assert.NotNull(expectedData);
2022-03-07 07:36:44 +00:00
Filesystems.Identify(image, out List<string> idPlugins, partitions[i]);
2022-03-06 13:29:38 +00:00
if(expectedData.Length != idPlugins.Count)
continue;
2022-03-07 07:36:44 +00:00
// Uncomment to generate JSON file
/*
2022-03-06 13:29:38 +00:00
expectedData = new VolumeData[idPlugins.Count];
2022-03-06 13:29:38 +00:00
for(int j = 0; j < idPlugins.Count; j++)
{
string pluginName = idPlugins[j];
2022-03-06 13:29:38 +00:00
if(!plugins.ReadOnlyFilesystems.TryGetValue(pluginName,
out IReadOnlyFilesystem plugin))
continue;
2022-03-06 13:29:38 +00:00
Assert.IsNotNull(plugin, "Could not instantiate filesystem plugin");
2022-03-06 13:29:38 +00:00
var fs = (IReadOnlyFilesystem)plugin.GetType().GetConstructor(Type.EmptyTypes)?.
Invoke(new object[]
{});
2022-03-06 13:29:38 +00:00
Assert.IsNotNull(fs, $"Could not instantiate filesystem {pluginName}");
2022-03-06 13:29:38 +00:00
Errno error = fs.Mount(image, partitions[i], null, null, null);
2022-03-06 13:29:38 +00:00
Assert.AreEqual(ErrorNumber.NoError, error,
$"Could not mount {pluginName} in partition {i}.");
2022-03-06 13:29:38 +00:00
if(error != ErrorNumber.NoError)
continue;
2022-03-06 13:29:38 +00:00
expectedData[j] = new VolumeData
{
Files = ReadOnlyFilesystemTest.BuildDirectory(fs, "/")
};
}
2022-03-06 13:29:38 +00:00
var sw = new StreamWriter(expectedDataFilename);
serializer.Serialize(sw, expectedData);
sw.Close();
*/
2022-03-06 13:29:38 +00:00
if(idPlugins.Count == 0)
continue;
2022-03-06 13:29:38 +00:00
Assert.AreEqual(expectedData.Length, idPlugins.Count,
$"Expected {expectedData.Length} filesystems identified in partition {i} but found {idPlugins.Count} in {testFile}");
2022-03-07 07:36:44 +00:00
for(var j = 0; j < idPlugins.Count; j++)
2022-03-06 13:29:38 +00:00
{
string pluginName = idPlugins[j];
2022-03-07 07:36:44 +00:00
if(!plugins.ReadOnlyFilesystems.TryGetValue(pluginName, out IReadOnlyFilesystem plugin))
2022-03-06 13:29:38 +00:00
continue;
2022-03-06 13:29:38 +00:00
Assert.IsNotNull(plugin, "Could not instantiate filesystem plugin");
2022-03-06 13:29:38 +00:00
var fs = (IReadOnlyFilesystem)plugin.GetType().GetConstructor(Type.EmptyTypes)?.
Invoke(new object[]
{});
2022-03-07 07:36:44 +00:00
Assert.IsNotNull(fs, $"Could not instantiate filesystem {pluginName} in {testFile}");
2022-03-06 13:29:38 +00:00
ErrorNumber error = fs.Mount(image, partitions[i], null, null, null);
2022-03-06 13:29:38 +00:00
Assert.AreEqual(ErrorNumber.NoError, error,
$"Could not mount {pluginName} in partition {i} in {testFile}.");
2022-03-06 13:29:38 +00:00
if(error != ErrorNumber.NoError)
continue;
2022-03-06 13:29:38 +00:00
VolumeData volumeData = expectedData[j];
ReadOnlyFilesystemTest.TestDirectory(fs, "/", volumeData.Files, testFile, false);
}
2022-03-06 13:29:38 +00:00
}
});
}
});
}
}