Files
server/MareSynchronosServer/MareSynchronosStaticFilesServer/Utils/BlockFileDataSubstream.cs
2024-11-04 11:51:13 +01:00

92 lines
3.3 KiB
C#

using System.Globalization;
using System.Text;
namespace MareSynchronosStaticFilesServer.Utils;
public sealed class BlockFileDataSubstream : IDisposable
{
private readonly MemoryStream _headerStream;
private bool _disposed = false;
private readonly Lazy<FileStream> _dataStreamLazy;
private FileStream DataStream => _dataStreamLazy.Value;
public BlockFileDataSubstream(FileInfo file)
{
_dataStreamLazy = new(() => File.Open(file.FullName, FileMode.Open, FileAccess.Read, FileShare.Read | FileShare.Inheritable));
_headerStream = new MemoryStream(Encoding.ASCII.GetBytes("#" + file.Name + ":" + file.Length.ToString(CultureInfo.InvariantCulture) + "#"));
}
public int Read(byte[] inputBuffer, int offset, int count)
{
int bytesRead = 0;
// Read from header stream if it has remaining data
if (_headerStream.Position < _headerStream.Length)
{
int headerBytesToRead = (int)Math.Min(count, _headerStream.Length - _headerStream.Position);
bytesRead += _headerStream.Read(inputBuffer, offset, headerBytesToRead);
offset += bytesRead;
count -= bytesRead;
}
// Read from data stream if there is still space in buffer
if (count > 0 && DataStream.Position < DataStream.Length)
{
bytesRead += DataStream.Read(inputBuffer, offset, count);
}
return bytesRead;
}
public async Task<int> ReadAsync(byte[] inputBuffer, int offset, int count, CancellationToken cancellationToken = default)
{
int bytesRead = 0;
// Async read from header stream
if (_headerStream.Position < _headerStream.Length)
{
int headerBytesToRead = (int)Math.Min(count, _headerStream.Length - _headerStream.Position);
bytesRead += await _headerStream.ReadAsync(inputBuffer.AsMemory(offset, headerBytesToRead), cancellationToken).ConfigureAwait(false);
offset += bytesRead;
count -= bytesRead;
}
// Async read from data stream
if (count > 0 && DataStream.Position < DataStream.Length)
{
bytesRead += await DataStream.ReadAsync(inputBuffer.AsMemory(offset, count), cancellationToken).ConfigureAwait(false);
}
return bytesRead;
}
public async ValueTask<int> ReadAsync(Memory<byte> buffer, CancellationToken cancellationToken = default)
{
int bytesRead = 0;
// Async read from header stream
if (_headerStream.Position < _headerStream.Length)
{
int headerBytesToRead = (int)Math.Min(buffer.Length, _headerStream.Length - _headerStream.Position);
bytesRead += await _headerStream.ReadAsync(buffer.Slice(0, headerBytesToRead), cancellationToken).ConfigureAwait(false);
buffer = buffer.Slice(headerBytesToRead);
}
// Async read from data stream
if (buffer.Length > 0 && DataStream.Position < DataStream.Length)
{
bytesRead += await DataStream.ReadAsync(buffer, cancellationToken).ConfigureAwait(false);
}
return bytesRead;
}
public void Dispose()
{
if (_disposed) return;
_headerStream.Dispose();
if (_dataStreamLazy.IsValueCreated)
DataStream.Dispose();
_disposed = true;
}
}