2026-02-13 12:44:12 +00:00
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using SharpCompress.IO;
|
|
|
|
|
using SharpCompress.Test.Mocks;
|
|
|
|
|
using Xunit;
|
|
|
|
|
|
|
|
|
|
namespace SharpCompress.Test.Streams;
|
|
|
|
|
|
|
|
|
|
public class SharpCompressStreamSeekTest
|
|
|
|
|
{
|
|
|
|
|
private class NonSeekableStreamWrapper : Stream
|
|
|
|
|
{
|
|
|
|
|
private readonly Stream _baseStream;
|
|
|
|
|
|
|
|
|
|
public NonSeekableStreamWrapper(Stream baseStream)
|
|
|
|
|
{
|
|
|
|
|
_baseStream = baseStream;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override bool CanRead => _baseStream.CanRead;
|
|
|
|
|
public override bool CanSeek => false;
|
|
|
|
|
public override bool CanWrite => _baseStream.CanWrite;
|
|
|
|
|
public override long Length => _baseStream.Length;
|
|
|
|
|
public override long Position
|
|
|
|
|
{
|
|
|
|
|
get => throw new NotSupportedException();
|
|
|
|
|
set => throw new NotSupportedException();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void Flush() => _baseStream.Flush();
|
|
|
|
|
|
|
|
|
|
public override int Read(byte[] buffer, int offset, int count) =>
|
|
|
|
|
_baseStream.Read(buffer, offset, count);
|
|
|
|
|
|
|
|
|
|
public override long Seek(long offset, SeekOrigin origin) =>
|
|
|
|
|
throw new NotSupportedException();
|
|
|
|
|
|
|
|
|
|
public override void SetLength(long value) => _baseStream.SetLength(value);
|
|
|
|
|
|
|
|
|
|
public override void Write(byte[] buffer, int offset, int count) =>
|
|
|
|
|
_baseStream.Write(buffer, offset, count);
|
|
|
|
|
|
|
|
|
|
protected override void Dispose(bool disposing)
|
|
|
|
|
{
|
|
|
|
|
if (disposing)
|
2026-02-13 14:26:21 +00:00
|
|
|
{
|
2026-02-13 12:44:12 +00:00
|
|
|
_baseStream.Dispose();
|
2026-02-13 14:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
2026-02-13 12:44:12 +00:00
|
|
|
base.Dispose(disposing);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Fact]
|
|
|
|
|
public void Seek_CurrentOrigin_MovesRelativeToCurrent()
|
|
|
|
|
{
|
|
|
|
|
var ms = new MemoryStream(new byte[] { 1, 2, 3, 4, 5, 6, 7, 8 });
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
|
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs, 128);
|
|
|
|
|
stream.StartRecording();
|
|
|
|
|
var buffer = new byte[4];
|
|
|
|
|
stream.Read(buffer, 0, 4);
|
|
|
|
|
Assert.Equal(4, stream.Position);
|
|
|
|
|
|
|
|
|
|
stream.Seek(-2, SeekOrigin.Current);
|
|
|
|
|
Assert.Equal(2, stream.Position);
|
|
|
|
|
|
|
|
|
|
stream.Read(buffer, 0, 2);
|
|
|
|
|
Assert.Equal(3, buffer[0]);
|
|
|
|
|
Assert.Equal(4, buffer[1]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Fact]
|
|
|
|
|
public void Seek_BeginOrigin_MovesToAbsolutePosition()
|
|
|
|
|
{
|
|
|
|
|
var ms = new MemoryStream(new byte[] { 1, 2, 3, 4, 5, 6, 7, 8 });
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
|
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs, 128);
|
|
|
|
|
stream.StartRecording();
|
|
|
|
|
var buffer = new byte[8];
|
|
|
|
|
stream.Read(buffer, 0, 8);
|
|
|
|
|
|
|
|
|
|
stream.Seek(2, SeekOrigin.Begin);
|
|
|
|
|
Assert.Equal(2, stream.Position);
|
|
|
|
|
|
|
|
|
|
var readBuffer = new byte[2];
|
|
|
|
|
stream.Read(readBuffer, 0, 2);
|
|
|
|
|
Assert.Equal(3, readBuffer[0]);
|
|
|
|
|
Assert.Equal(4, readBuffer[1]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Fact]
|
|
|
|
|
public void Seek_ToExactBufferBoundary_Succeeds()
|
|
|
|
|
{
|
|
|
|
|
var ms = new MemoryStream(new byte[] { 1, 2, 3, 4, 5, 6, 7, 8 });
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
|
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs, 128);
|
|
|
|
|
stream.StartRecording();
|
|
|
|
|
var buffer = new byte[4];
|
|
|
|
|
stream.Read(buffer, 0, 4);
|
|
|
|
|
|
|
|
|
|
stream.Seek(4, SeekOrigin.Begin);
|
|
|
|
|
Assert.Equal(4, stream.Position);
|
|
|
|
|
|
|
|
|
|
stream.Read(buffer, 0, 4);
|
|
|
|
|
Assert.Equal(5, buffer[0]);
|
|
|
|
|
Assert.Equal(6, buffer[1]);
|
|
|
|
|
Assert.Equal(7, buffer[2]);
|
|
|
|
|
Assert.Equal(8, buffer[3]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Fact]
|
|
|
|
|
public void Position_SetWithinRecordedRange_Succeeds()
|
|
|
|
|
{
|
|
|
|
|
var ms = new MemoryStream(new byte[] { 1, 2, 3, 4, 5, 6, 7, 8 });
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
|
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs, 128);
|
|
|
|
|
stream.StartRecording();
|
|
|
|
|
var buffer = new byte[8];
|
|
|
|
|
stream.Read(buffer, 0, 8);
|
|
|
|
|
|
|
|
|
|
stream.Position = 2;
|
|
|
|
|
Assert.Equal(2, stream.Position);
|
|
|
|
|
|
|
|
|
|
var readBuffer = new byte[2];
|
|
|
|
|
stream.Read(readBuffer, 0, 2);
|
|
|
|
|
Assert.Equal(3, readBuffer[0]);
|
|
|
|
|
Assert.Equal(4, readBuffer[1]);
|
|
|
|
|
}
|
2026-03-31 09:01:21 +00:00
|
|
|
|
|
|
|
|
[Fact]
|
2026-04-01 11:19:17 +00:00
|
|
|
public void StartRecording_WithLargerMinBufferSize_AllowsLargeRewind()
|
2026-03-31 09:01:21 +00:00
|
|
|
{
|
2026-04-01 11:19:17 +00:00
|
|
|
// Simulates the BZip2 scenario: the ring buffer must be large enough
|
|
|
|
|
// from the moment StartRecording is called so that a large probe read
|
|
|
|
|
// (up to 900 KB for BZip2) can be rewound without buffer overflow.
|
|
|
|
|
const int largeSize = 100;
|
2026-03-31 09:01:21 +00:00
|
|
|
const int largeReadSize = 80;
|
|
|
|
|
|
|
|
|
|
var data = new byte[100];
|
|
|
|
|
for (var i = 0; i < data.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
data[i] = (byte)(i + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var ms = new MemoryStream(data);
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
2026-04-01 11:19:17 +00:00
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs, largeSize);
|
2026-03-31 09:01:21 +00:00
|
|
|
|
2026-04-01 11:19:17 +00:00
|
|
|
// Pass the required size upfront — no expansion needed later
|
|
|
|
|
stream.StartRecording(largeSize);
|
2026-03-31 09:01:21 +00:00
|
|
|
|
2026-04-01 11:19:17 +00:00
|
|
|
// Read a large amount (simulating BZip2 block decompression during IsTarFile probe)
|
2026-03-31 09:01:21 +00:00
|
|
|
var largeBuffer = new byte[largeReadSize];
|
|
|
|
|
stream.Read(largeBuffer, 0, largeReadSize);
|
|
|
|
|
|
2026-04-01 11:19:17 +00:00
|
|
|
// Rewind must succeed because the buffer was large enough from the start
|
2026-03-31 09:01:21 +00:00
|
|
|
stream.Rewind();
|
|
|
|
|
|
|
|
|
|
var verifyBuffer = new byte[largeReadSize];
|
|
|
|
|
stream.Read(verifyBuffer, 0, largeReadSize);
|
|
|
|
|
Assert.Equal(data[0], verifyBuffer[0]);
|
|
|
|
|
Assert.Equal(data[largeReadSize - 1], verifyBuffer[largeReadSize - 1]);
|
|
|
|
|
}
|
2026-04-01 11:19:17 +00:00
|
|
|
|
|
|
|
|
[Fact]
|
|
|
|
|
public void StartRecording_DefaultSize_UsesConstantsRewindableBufferSize()
|
|
|
|
|
{
|
|
|
|
|
// When no minimum is specified StartRecording uses the global default.
|
|
|
|
|
var ms = new MemoryStream(new byte[] { 1, 2, 3, 4, 5 });
|
|
|
|
|
var nonSeekableMs = new NonSeekableStreamWrapper(ms);
|
|
|
|
|
var stream = SharpCompressStream.Create(nonSeekableMs);
|
|
|
|
|
stream.StartRecording();
|
|
|
|
|
|
|
|
|
|
var buffer = new byte[5];
|
|
|
|
|
stream.Read(buffer, 0, 5);
|
|
|
|
|
stream.Rewind();
|
|
|
|
|
|
|
|
|
|
var readBuffer = new byte[5];
|
|
|
|
|
stream.Read(readBuffer, 0, 5);
|
|
|
|
|
Assert.Equal(1, readBuffer[0]);
|
|
|
|
|
Assert.Equal(5, readBuffer[4]);
|
|
|
|
|
}
|
2026-02-13 12:44:12 +00:00
|
|
|
}
|