2023-04-12 16:06:04 +02:00
|
|
|
|
using Logging;
|
|
|
|
|
using NUnit.Framework;
|
2023-05-03 14:55:26 +02:00
|
|
|
|
using Utils;
|
2023-04-12 16:06:04 +02:00
|
|
|
|
|
|
|
|
|
namespace DistTestCore
|
|
|
|
|
{
|
|
|
|
|
public interface IFileManager
|
|
|
|
|
{
|
|
|
|
|
TestFile CreateEmptyTestFile();
|
|
|
|
|
TestFile GenerateTestFile(ByteSize size);
|
|
|
|
|
void DeleteAllTestFiles();
|
2023-05-29 09:13:38 +02:00
|
|
|
|
void PushFileSet();
|
|
|
|
|
void PopFileSet();
|
2023-04-12 16:06:04 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class FileManager : IFileManager
|
|
|
|
|
{
|
|
|
|
|
public const int ChunkSize = 1024 * 1024;
|
2023-05-03 14:55:26 +02:00
|
|
|
|
private static NumberSource folderNumberSource = new NumberSource(0);
|
2023-04-12 16:06:04 +02:00
|
|
|
|
private readonly Random random = new Random();
|
|
|
|
|
private readonly TestLog log;
|
|
|
|
|
private readonly string folder;
|
2023-05-29 09:13:38 +02:00
|
|
|
|
private readonly List<List<TestFile>> fileSetStack = new List<List<TestFile>>();
|
2023-04-12 16:06:04 +02:00
|
|
|
|
|
|
|
|
|
public FileManager(TestLog log, Configuration configuration)
|
|
|
|
|
{
|
2023-05-03 14:55:26 +02:00
|
|
|
|
folder = Path.Combine(configuration.GetFileManagerFolder(), folderNumberSource.GetNextNumber().ToString("D5"));
|
2023-04-12 16:06:04 +02:00
|
|
|
|
|
2023-04-17 09:10:39 +02:00
|
|
|
|
EnsureDirectory();
|
2023-04-12 16:06:04 +02:00
|
|
|
|
this.log = log;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public TestFile CreateEmptyTestFile()
|
|
|
|
|
{
|
|
|
|
|
var result = new TestFile(Path.Combine(folder, Guid.NewGuid().ToString() + "_test.bin"));
|
|
|
|
|
File.Create(result.Filename).Close();
|
2023-05-29 09:13:38 +02:00
|
|
|
|
if (fileSetStack.Any()) fileSetStack.Last().Add(result);
|
2023-04-12 16:06:04 +02:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public TestFile GenerateTestFile(ByteSize size)
|
|
|
|
|
{
|
|
|
|
|
var result = CreateEmptyTestFile();
|
|
|
|
|
GenerateFileBytes(result, size);
|
|
|
|
|
log.Log($"Generated {size.SizeInBytes} bytes of content for file '{result.Filename}'.");
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void DeleteAllTestFiles()
|
|
|
|
|
{
|
2023-04-17 09:10:39 +02:00
|
|
|
|
DeleteDirectory();
|
2023-04-12 16:06:04 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-29 09:13:38 +02:00
|
|
|
|
public void PushFileSet()
|
|
|
|
|
{
|
|
|
|
|
fileSetStack.Add(new List<TestFile>());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void PopFileSet()
|
|
|
|
|
{
|
|
|
|
|
if (!fileSetStack.Any()) return;
|
|
|
|
|
var pop = fileSetStack.Last();
|
|
|
|
|
fileSetStack.Remove(pop);
|
|
|
|
|
|
|
|
|
|
foreach (var file in pop)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
File.Delete(file.Filename);
|
|
|
|
|
}
|
|
|
|
|
catch { }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 16:06:04 +02:00
|
|
|
|
private void GenerateFileBytes(TestFile result, ByteSize size)
|
|
|
|
|
{
|
|
|
|
|
long bytesLeft = size.SizeInBytes;
|
|
|
|
|
while (bytesLeft > 0)
|
|
|
|
|
{
|
|
|
|
|
var length = Math.Min(bytesLeft, ChunkSize);
|
|
|
|
|
AppendRandomBytesToFile(result, length);
|
|
|
|
|
bytesLeft -= length;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AppendRandomBytesToFile(TestFile result, long length)
|
|
|
|
|
{
|
|
|
|
|
var bytes = new byte[length];
|
|
|
|
|
random.NextBytes(bytes);
|
|
|
|
|
using var stream = new FileStream(result.Filename, FileMode.Append);
|
|
|
|
|
stream.Write(bytes, 0, bytes.Length);
|
|
|
|
|
}
|
2023-04-17 09:10:39 +02:00
|
|
|
|
|
|
|
|
|
private void EnsureDirectory()
|
|
|
|
|
{
|
|
|
|
|
if (!Directory.Exists(folder)) Directory.CreateDirectory(folder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void DeleteDirectory()
|
|
|
|
|
{
|
|
|
|
|
Directory.Delete(folder, true);
|
|
|
|
|
}
|
2023-04-12 16:06:04 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class TestFile
|
|
|
|
|
{
|
|
|
|
|
public TestFile(string filename)
|
|
|
|
|
{
|
|
|
|
|
Filename = filename;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string Filename { get; }
|
|
|
|
|
|
|
|
|
|
public long GetFileSize()
|
|
|
|
|
{
|
|
|
|
|
var info = new FileInfo(Filename);
|
|
|
|
|
return info.Length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void AssertIsEqual(TestFile? actual)
|
|
|
|
|
{
|
|
|
|
|
if (actual == null) Assert.Fail("TestFile is null.");
|
|
|
|
|
if (actual == this || actual!.Filename == Filename) Assert.Fail("TestFile is compared to itself.");
|
|
|
|
|
|
|
|
|
|
Assert.That(actual.GetFileSize(), Is.EqualTo(GetFileSize()), "Files are not of equal length.");
|
|
|
|
|
|
|
|
|
|
using var streamExpected = new FileStream(Filename, FileMode.Open, FileAccess.Read);
|
|
|
|
|
using var streamActual = new FileStream(actual.Filename, FileMode.Open, FileAccess.Read);
|
|
|
|
|
|
|
|
|
|
var bytesExpected = new byte[FileManager.ChunkSize];
|
|
|
|
|
var bytesActual = new byte[FileManager.ChunkSize];
|
|
|
|
|
|
|
|
|
|
var readExpected = 0;
|
|
|
|
|
var readActual = 0;
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
|
|
|
|
readExpected = streamExpected.Read(bytesExpected, 0, FileManager.ChunkSize);
|
|
|
|
|
readActual = streamActual.Read(bytesActual, 0, FileManager.ChunkSize);
|
|
|
|
|
|
|
|
|
|
if (readExpected == 0 && readActual == 0) return;
|
|
|
|
|
Assert.That(readActual, Is.EqualTo(readExpected), "Unable to read buffers of equal length.");
|
|
|
|
|
CollectionAssert.AreEqual(bytesExpected, bytesActual, "Files are not binary-equal.");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|