2
0
mirror of synced 2025-01-12 17:44:08 +00:00

158 lines
4.6 KiB
C#
Raw Normal View History

2023-04-12 16:06:04 +02:00
using Logging;
using Utils;
2023-04-12 16:06:04 +02:00
namespace FileUtils
2023-04-12 16:06:04 +02:00
{
public interface IFileManager
{
2023-09-13 10:03:11 +02:00
TrackedFile CreateEmptyFile(string label = "");
TrackedFile GenerateFile(ByteSize size, string label = "");
void DeleteAllFiles();
void ScopedFiles(Action action);
T ScopedFiles<T>(Func<T> action);
2023-04-12 16:06:04 +02:00
}
public class FileManager : IFileManager
{
2023-06-07 09:59:00 +02:00
public const int ChunkSize = 1024 * 1024 * 100;
private static NumberSource folderNumberSource = new NumberSource(0);
2023-04-12 16:06:04 +02:00
private readonly Random random = new Random();
2023-09-12 10:31:55 +02:00
private readonly ILog log;
2023-09-21 10:33:09 +02:00
private readonly string rootFolder;
2023-04-12 16:06:04 +02:00
private readonly string folder;
2023-09-12 13:32:06 +02:00
private readonly List<List<TrackedFile>> fileSetStack = new List<List<TrackedFile>>();
2023-04-12 16:06:04 +02:00
2023-09-12 10:31:55 +02:00
public FileManager(ILog log, string rootFolder)
2023-04-12 16:06:04 +02:00
{
folder = Path.Combine(rootFolder, folderNumberSource.GetNextNumber().ToString("D5"));
2023-04-12 16:06:04 +02:00
this.log = log;
2023-09-21 10:33:09 +02:00
this.rootFolder = rootFolder;
2023-04-12 16:06:04 +02:00
}
2023-09-13 10:03:11 +02:00
public TrackedFile CreateEmptyFile(string label = "")
2023-04-12 16:06:04 +02:00
{
2023-09-21 10:33:09 +02:00
var path = Path.Combine(folder, Guid.NewGuid().ToString() + ".bin");
EnsureDirectory();
2023-09-12 13:32:06 +02:00
var result = new TrackedFile(log, path, label);
2023-04-12 16:06:04 +02:00
File.Create(result.Filename).Close();
if (fileSetStack.Any()) fileSetStack.Last().Add(result);
2023-04-12 16:06:04 +02:00
return result;
}
2023-09-13 10:03:11 +02:00
public TrackedFile GenerateFile(ByteSize size, string label)
2023-04-12 16:06:04 +02:00
{
2023-06-07 09:59:00 +02:00
var sw = Stopwatch.Begin(log);
2023-09-13 10:03:11 +02:00
var result = GenerateRandomFile(size, label);
2023-06-07 09:59:00 +02:00
sw.End($"Generated file '{result.Describe()}'.");
2023-04-12 16:06:04 +02:00
return result;
}
2023-09-13 10:03:11 +02:00
public void DeleteAllFiles()
2023-04-12 16:06:04 +02:00
{
DeleteDirectory();
2023-04-12 16:06:04 +02:00
}
public void ScopedFiles(Action action)
{
PushFileSet();
action();
PopFileSet();
}
public T ScopedFiles<T>(Func<T> action)
{
PushFileSet();
var result = action();
PopFileSet();
return result;
}
private void PushFileSet()
{
2023-09-12 13:32:06 +02:00
fileSetStack.Add(new List<TrackedFile>());
}
private void PopFileSet()
{
if (!fileSetStack.Any()) return;
var pop = fileSetStack.Last();
fileSetStack.Remove(pop);
foreach (var file in pop)
{
2023-09-21 10:33:09 +02:00
File.Delete(file.Filename);
}
2023-09-21 10:33:09 +02:00
// If the folder is now empty, delete it too.
if (!Directory.GetFiles(folder).Any()) DeleteDirectory();
}
2023-09-13 10:03:11 +02:00
private TrackedFile GenerateRandomFile(ByteSize size, string label)
2023-06-07 09:59:00 +02:00
{
2023-09-13 10:03:11 +02:00
var result = CreateEmptyFile(label);
2023-06-07 09:59:00 +02:00
CheckSpaceAvailable(result, size);
GenerateFileBytes(result, size);
return result;
}
2023-09-12 13:32:06 +02:00
private void CheckSpaceAvailable(TrackedFile testFile, ByteSize size)
2023-06-07 09:59:00 +02:00
{
var file = new FileInfo(testFile.Filename);
var drive = new DriveInfo(file.Directory!.Root.FullName);
var spaceAvailable = drive.TotalFreeSpace;
if (spaceAvailable < size.SizeInBytes)
{
2023-09-20 10:51:47 +02:00
var msg = $"Not enough disk space. " +
2023-06-07 09:59:00 +02:00
$"{Formatter.FormatByteSize(size.SizeInBytes)} required. " +
$"{Formatter.FormatByteSize(spaceAvailable)} available.";
log.Log(msg);
2023-09-20 10:51:47 +02:00
throw new Exception(msg);
2023-06-07 09:59:00 +02:00
}
}
2023-09-12 13:32:06 +02:00
private void GenerateFileBytes(TrackedFile result, ByteSize size)
2023-04-12 16:06:04 +02:00
{
long bytesLeft = size.SizeInBytes;
2023-06-07 09:59:00 +02:00
int chunkSize = ChunkSize;
2023-04-12 16:06:04 +02:00
while (bytesLeft > 0)
{
2023-06-07 09:59:00 +02:00
try
{
var length = Math.Min(bytesLeft, chunkSize);
AppendRandomBytesToFile(result, length);
bytesLeft -= length;
}
catch
{
chunkSize = chunkSize / 2;
if (chunkSize < 1024) throw;
}
2023-04-12 16:06:04 +02:00
}
}
2023-09-12 13:32:06 +02:00
private void AppendRandomBytesToFile(TrackedFile result, long length)
2023-04-12 16:06:04 +02:00
{
var bytes = new byte[length];
random.NextBytes(bytes);
using var stream = new FileStream(result.Filename, FileMode.Append);
stream.Write(bytes, 0, bytes.Length);
}
private void EnsureDirectory()
{
2023-09-21 10:33:09 +02:00
Directory.CreateDirectory(folder);
}
private void DeleteDirectory()
{
2023-09-21 10:33:09 +02:00
if (Directory.Exists(folder)) Directory.Delete(folder, true);
}
2023-04-12 16:06:04 +02:00
}
}