reimplment ReadAsync for BlockFileDataStream/SubStream
This commit is contained in:
@@ -36,9 +36,9 @@ public class CacheController : ControllerBase
|
|||||||
long requestSize = 0;
|
long requestSize = 0;
|
||||||
List<BlockFileDataSubstream> substreams = new();
|
List<BlockFileDataSubstream> substreams = new();
|
||||||
|
|
||||||
foreach (var file in request.FileIds)
|
foreach (var fileHash in request.FileIds)
|
||||||
{
|
{
|
||||||
var fs = await _cachedFileProvider.GetAndDownloadFileStream(file);
|
var fs = await _cachedFileProvider.DownloadAndGetLocalFileInfo(fileHash).ConfigureAwait(false);
|
||||||
if (fs == null) continue;
|
if (fs == null) continue;
|
||||||
|
|
||||||
substreams.Add(new(fs));
|
substreams.Add(new(fs));
|
||||||
|
|||||||
@@ -17,13 +17,13 @@ public class DistributionController : ControllerBase
|
|||||||
|
|
||||||
[HttpGet(MareFiles.Distribution_Get)]
|
[HttpGet(MareFiles.Distribution_Get)]
|
||||||
[Authorize(Policy = "Internal")]
|
[Authorize(Policy = "Internal")]
|
||||||
public async Task<IActionResult> GetFile(string file)
|
public async Task<IActionResult> GetFile(string fileHash)
|
||||||
{
|
{
|
||||||
_logger.LogInformation($"GetFile:{MareUser}:{file}");
|
_logger.LogInformation($"GetFile:{MareUser}:{fileHash}");
|
||||||
|
|
||||||
var fs = await _cachedFileProvider.GetAndDownloadFileStream(file);
|
var fs = await _cachedFileProvider.DownloadAndGetLocalFileInfo(fileHash);
|
||||||
if (fs == null) return NotFound();
|
if (fs == null) return NotFound();
|
||||||
|
|
||||||
return File(fs, "application/octet-stream");
|
return PhysicalFile(fs.FullName, "application/octet-stream");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -167,7 +167,7 @@ public sealed class CachedFileProvider : IDisposable
|
|||||||
_downloadSemaphore.Release();
|
_downloadSemaphore.Release();
|
||||||
}
|
}
|
||||||
|
|
||||||
public FileStream? GetLocalFileStream(string hash)
|
public FileInfo? GetLocalFilePath(string hash)
|
||||||
{
|
{
|
||||||
var fi = FilePathUtil.GetFileInfoForHash(_hotStoragePath, hash);
|
var fi = FilePathUtil.GetFileInfoForHash(_hotStoragePath, hash);
|
||||||
if (fi == null) return null;
|
if (fi == null) return null;
|
||||||
@@ -176,10 +176,10 @@ public sealed class CachedFileProvider : IDisposable
|
|||||||
|
|
||||||
_fileStatisticsService.LogFile(hash, fi.Length);
|
_fileStatisticsService.LogFile(hash, fi.Length);
|
||||||
|
|
||||||
return new FileStream(fi.FullName, FileMode.Open, FileAccess.Read, FileShare.Inheritable | FileShare.Read);
|
return new FileInfo(fi.FullName);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<FileStream?> GetAndDownloadFileStream(string hash)
|
public async Task<FileInfo?> DownloadAndGetLocalFileInfo(string hash)
|
||||||
{
|
{
|
||||||
await DownloadFileWhenRequired(hash).ConfigureAwait(false);
|
await DownloadFileWhenRequired(hash).ConfigureAwait(false);
|
||||||
|
|
||||||
@@ -203,7 +203,7 @@ public sealed class CachedFileProvider : IDisposable
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return GetLocalFileStream(hash);
|
return GetLocalFilePath(hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool AnyFilesDownloading(List<string> hashes)
|
public bool AnyFilesDownloading(List<string> hashes)
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
namespace MareSynchronosStaticFilesServer.Utils;
|
|
||||||
|
namespace MareSynchronosStaticFilesServer.Utils;
|
||||||
|
|
||||||
public sealed class BlockFileDataStream : Stream
|
public sealed class BlockFileDataStream : Stream
|
||||||
{
|
{
|
||||||
private readonly List<BlockFileDataSubstream> _substreams;
|
private readonly IEnumerable<BlockFileDataSubstream> _substreams;
|
||||||
private int _currentStreamIndex = 0;
|
private int _currentStreamIndex = 0;
|
||||||
|
|
||||||
public BlockFileDataStream(List<BlockFileDataSubstream> substreams)
|
public BlockFileDataStream(IEnumerable<BlockFileDataSubstream> substreams)
|
||||||
{
|
{
|
||||||
_substreams = substreams;
|
_substreams = substreams;
|
||||||
}
|
}
|
||||||
@@ -16,17 +17,17 @@ public sealed class BlockFileDataStream : Stream
|
|||||||
public override long Length => throw new NotSupportedException();
|
public override long Length => throw new NotSupportedException();
|
||||||
public override long Position { get => throw new NotSupportedException(); set => throw new NotSupportedException(); }
|
public override long Position { get => throw new NotSupportedException(); set => throw new NotSupportedException(); }
|
||||||
|
|
||||||
public override int Read(byte[] buffer, int offset, int count)
|
public override async Task<int> ReadAsync(byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
int totalRead = 0;
|
int totalRead = 0;
|
||||||
int currentOffset = 0;
|
int currentOffset = 0;
|
||||||
int remainingCount = count;
|
int remainingCount = count;
|
||||||
while (totalRead < count && _currentStreamIndex < _substreams.Count)
|
while (totalRead < count && _currentStreamIndex < _substreams.Count())
|
||||||
{
|
{
|
||||||
var lastReadBytes = _substreams[_currentStreamIndex].Read(buffer, currentOffset, remainingCount);
|
var lastReadBytes = await _substreams.ElementAt(_currentStreamIndex).ReadAsync(buffer, currentOffset, remainingCount, cancellationToken).ConfigureAwait(false);
|
||||||
if (lastReadBytes < remainingCount)
|
if (lastReadBytes < remainingCount)
|
||||||
{
|
{
|
||||||
_substreams[_currentStreamIndex].Dispose();
|
_substreams.ElementAt(_currentStreamIndex).Dispose();
|
||||||
_currentStreamIndex++;
|
_currentStreamIndex++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -38,6 +39,51 @@ public sealed class BlockFileDataStream : Stream
|
|||||||
return totalRead;
|
return totalRead;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public override int Read(byte[] buffer, int offset, int count)
|
||||||
|
{
|
||||||
|
int totalRead = 0;
|
||||||
|
int currentOffset = 0;
|
||||||
|
int remainingCount = count;
|
||||||
|
while (totalRead < count && _currentStreamIndex < _substreams.Count())
|
||||||
|
{
|
||||||
|
var lastReadBytes = _substreams.ElementAt(_currentStreamIndex).Read(buffer, currentOffset, remainingCount);
|
||||||
|
if (lastReadBytes < remainingCount)
|
||||||
|
{
|
||||||
|
_substreams.ElementAt(_currentStreamIndex).Dispose();
|
||||||
|
_currentStreamIndex++;
|
||||||
|
}
|
||||||
|
|
||||||
|
totalRead += lastReadBytes;
|
||||||
|
currentOffset += lastReadBytes;
|
||||||
|
remainingCount -= lastReadBytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
return totalRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override async ValueTask<int> ReadAsync(Memory<byte> buffer, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
int totalRead = 0;
|
||||||
|
|
||||||
|
while (totalRead < buffer.Length && _currentStreamIndex < _substreams.Count())
|
||||||
|
{
|
||||||
|
var substream = _substreams.ElementAt(_currentStreamIndex);
|
||||||
|
int bytesRead = await substream.ReadAsync(buffer.Slice(totalRead), cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (bytesRead == 0)
|
||||||
|
{
|
||||||
|
substream.Dispose();
|
||||||
|
_currentStreamIndex++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
totalRead += bytesRead;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return totalRead;
|
||||||
|
}
|
||||||
|
|
||||||
protected override void Dispose(bool disposing)
|
protected override void Dispose(bool disposing)
|
||||||
{
|
{
|
||||||
if (disposing)
|
if (disposing)
|
||||||
@@ -52,23 +98,11 @@ public sealed class BlockFileDataStream : Stream
|
|||||||
base.Dispose(disposing);
|
base.Dispose(disposing);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Flush()
|
public override void Flush() => throw new NotSupportedException();
|
||||||
{
|
|
||||||
throw new NotSupportedException();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override long Seek(long offset, SeekOrigin origin)
|
public override long Seek(long offset, SeekOrigin origin) => throw new NotSupportedException();
|
||||||
{
|
|
||||||
throw new NotSupportedException();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void SetLength(long value)
|
public override void SetLength(long value) => throw new NotSupportedException();
|
||||||
{
|
|
||||||
throw new NotSupportedException();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void Write(byte[] buffer, int offset, int count)
|
public override void Write(byte[] buffer, int offset, int count) => throw new NotSupportedException();
|
||||||
{
|
|
||||||
throw new NotSupportedException();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,70 +6,87 @@ namespace MareSynchronosStaticFilesServer.Utils;
|
|||||||
public sealed class BlockFileDataSubstream : IDisposable
|
public sealed class BlockFileDataSubstream : IDisposable
|
||||||
{
|
{
|
||||||
private readonly MemoryStream _headerStream;
|
private readonly MemoryStream _headerStream;
|
||||||
private readonly BinaryWriter _headerStreamWriter;
|
|
||||||
private readonly FileStream _dataStream;
|
|
||||||
private int _headerPosition = 0;
|
|
||||||
private long _dataPosition = 0;
|
|
||||||
private bool _disposed = false;
|
private bool _disposed = false;
|
||||||
|
private readonly Lazy<FileStream> _dataStreamLazy;
|
||||||
|
private FileStream DataStream => _dataStreamLazy.Value;
|
||||||
|
|
||||||
private long RemainingHeaderLength => _headerStream.Length - _headerPosition;
|
public BlockFileDataSubstream(FileInfo file)
|
||||||
private long RemainingDataLength => _dataStream.Length - _dataPosition;
|
|
||||||
|
|
||||||
public BlockFileDataSubstream(FileStream dataStream)
|
|
||||||
{
|
{
|
||||||
_headerStream = new MemoryStream();
|
_dataStreamLazy = new(() => File.Open(file.FullName, FileMode.Open, FileAccess.Read, FileShare.Read | FileShare.Inheritable));
|
||||||
_headerStreamWriter = new BinaryWriter(_headerStream);
|
_headerStream = new MemoryStream(Encoding.ASCII.GetBytes("#" + file.Name + ":" + file.Length.ToString(CultureInfo.InvariantCulture) + "#"));
|
||||||
_headerStreamWriter.Write(Encoding.ASCII.GetBytes("#" + new FileInfo(dataStream.Name).Name + ":" + dataStream.Length.ToString(CultureInfo.InvariantCulture) + "#"));
|
|
||||||
_headerStreamWriter.Flush();
|
|
||||||
_headerStream.Position = 0;
|
|
||||||
_dataStream = dataStream;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Read(byte[] inputBuffer, int offset, int count)
|
public int Read(byte[] inputBuffer, int offset, int count)
|
||||||
{
|
{
|
||||||
int currentOffset = offset;
|
int bytesRead = 0;
|
||||||
int currentCount = count;
|
|
||||||
int readHeaderBytes = 0;
|
|
||||||
|
|
||||||
if (RemainingHeaderLength > 0)
|
// Read from header stream if it has remaining data
|
||||||
|
if (_headerStream.Position < _headerStream.Length)
|
||||||
{
|
{
|
||||||
bool readOnlyHeader = currentCount <= RemainingHeaderLength;
|
int headerBytesToRead = (int)Math.Min(count, _headerStream.Length - _headerStream.Position);
|
||||||
byte[] readHeaderBuffer = new byte[Math.Min(currentCount, RemainingHeaderLength)];
|
bytesRead += _headerStream.Read(inputBuffer, offset, headerBytesToRead);
|
||||||
|
offset += bytesRead;
|
||||||
readHeaderBytes = _headerStream.Read(readHeaderBuffer, 0, readHeaderBuffer.Length);
|
count -= bytesRead;
|
||||||
_headerPosition += readHeaderBytes;
|
|
||||||
|
|
||||||
Buffer.BlockCopy(readHeaderBuffer, 0, inputBuffer, currentOffset, readHeaderBytes);
|
|
||||||
|
|
||||||
if (readOnlyHeader)
|
|
||||||
{
|
|
||||||
return readHeaderBytes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
currentOffset += readHeaderBytes;
|
// Read from data stream if there is still space in buffer
|
||||||
currentCount -= readHeaderBytes;
|
if (count > 0 && DataStream.Position < DataStream.Length)
|
||||||
}
|
|
||||||
|
|
||||||
if (RemainingDataLength > 0)
|
|
||||||
{
|
{
|
||||||
byte[] readDataBuffer = new byte[currentCount];
|
bytesRead += DataStream.Read(inputBuffer, offset, count);
|
||||||
var readDataBytes = _dataStream.Read(readDataBuffer, 0, readDataBuffer.Length);
|
|
||||||
_dataPosition += readDataBytes;
|
|
||||||
|
|
||||||
Buffer.BlockCopy(readDataBuffer, 0, inputBuffer, currentOffset, readDataBytes);
|
|
||||||
|
|
||||||
return readDataBytes + readHeaderBytes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
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()
|
public void Dispose()
|
||||||
{
|
{
|
||||||
if (_disposed) return;
|
if (_disposed) return;
|
||||||
_headerStream.Dispose();
|
_headerStream.Dispose();
|
||||||
_headerStreamWriter.Dispose();
|
if (_dataStreamLazy.IsValueCreated)
|
||||||
_dataStream.Dispose();
|
DataStream.Dispose();
|
||||||
_disposed = true;
|
_disposed = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user