| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 | using System;using System.Globalization;using System.IO;using System.Threading;using System.Threading.Tasks;using Microsoft.VisualStudio.TestTools.UnitTesting;using Moq;using Renci.SshNet.Common;using Renci.SshNet.Sftp;namespace Renci.SshNet.Tests.Classes.Sftp{    [TestClass]    public class SftpFileStreamTest_WriteAsync_SessionOpen_CountGreatherThanTwoTimesTheWriteBufferSize : SftpFileStreamAsyncTestBase    {        private SftpFileStream _target;        private string _path;        private byte[] _handle;        private uint _bufferSize;        private uint _readBufferSize;        private uint _writeBufferSize;        private byte[] _data;        private int _count;        private int _offset;        private Random _random;        private uint _expectedWrittenByteCount;        private int _expectedBufferedByteCount;        private byte[] _expectedBufferedBytes;        private CancellationToken _cancellationToken;        protected override void SetupData()        {            base.SetupData();            _random = new Random();            _path = _random.Next().ToString(CultureInfo.InvariantCulture);            _handle = GenerateRandom(5, _random);            _bufferSize = (uint)_random.Next(1, 1000);            _readBufferSize = (uint)_random.Next(0, 1000);            _writeBufferSize = (uint)_random.Next(500, 1000);            _data = new byte[(_writeBufferSize * 2) + 15];            _random.NextBytes(_data);            _offset = _random.Next(1, 5);            // to get multiple SSH_FXP_WRITE messages (and verify the offset is updated correctly), we make sure            // the number of bytes to write is at least two times the write buffer size; we write a few extra bytes to            // ensure the buffer is not empty after the writes so we can verify whether Length, Dispose and Flush            // flush the buffer            _count = ((int)_writeBufferSize * 2) + _random.Next(1, 5);            _expectedWrittenByteCount = (2 * _writeBufferSize);            _expectedBufferedByteCount = (int)(_count - _expectedWrittenByteCount);            _expectedBufferedBytes = _data.Take(_offset + (int)_expectedWrittenByteCount, _expectedBufferedByteCount);            _cancellationToken = new CancellationToken();        }        protected override void SetupMocks()        {            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.RequestOpenAsync(_path, Flags.Write | Flags.CreateNewOrOpen | Flags.Truncate, _cancellationToken))                           .ReturnsAsync(_handle);            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.CalculateOptimalReadLength(_bufferSize))                           .Returns(_readBufferSize);            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.CalculateOptimalWriteLength(_bufferSize, _handle))                           .Returns(_writeBufferSize);            SftpSessionMock.InSequence(MockSequence).Setup(p => p.IsOpen).Returns(true);            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.RequestWriteAsync(_handle, 0, _data, _offset, (int)_writeBufferSize, _cancellationToken))                           .Returns(Task.CompletedTask);            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.RequestWriteAsync(_handle, _writeBufferSize, _data, _offset + (int)_writeBufferSize, (int)_writeBufferSize, _cancellationToken))                           .Returns(Task.CompletedTask);        }        [TestCleanup]        public void TearDown()        {            if (SftpSessionMock != null)            {                // allow Dispose to complete successfully                SftpSessionMock.InSequence(MockSequence)                               .Setup(p => p.IsOpen)                               .Returns(true);                SftpSessionMock.InSequence(MockSequence)                               .Setup(p => p.RequestWriteAsync(_handle, _expectedWrittenByteCount, It.IsAny<byte[]>(), 0, _expectedBufferedByteCount, _cancellationToken))                               .Returns(Task.CompletedTask);                SftpSessionMock.InSequence(MockSequence)                               .Setup(p => p.RequestClose(_handle));            }        }        protected override async Task ArrangeAsync()        {            await base.ArrangeAsync();            _target = await SftpFileStream.OpenAsync(SftpSessionMock.Object, _path, FileMode.Create, FileAccess.Write, (int)_bufferSize, _cancellationToken);        }        protected override Task ActAsync()        {            return _target.WriteAsync(_data, _offset, _count);        }        [TestMethod]        public void RequestWriteOnSftpSessionShouldBeInvokedTwice()        {            SftpSessionMock.Verify(p => p.RequestWriteAsync(_handle, 0, _data, _offset, (int)_writeBufferSize, _cancellationToken), Times.Once);            SftpSessionMock.Verify(p => p.RequestWriteAsync(_handle, _writeBufferSize, _data, _offset + (int)_writeBufferSize, (int)_writeBufferSize, _cancellationToken), Times.Once);        }        [TestMethod]        public void PositionShouldBeNumberOfBytesWrittenToFileAndNUmberOfBytesInBuffer()        {            SftpSessionMock.InSequence(MockSequence).Setup(p => p.IsOpen).Returns(true);            Assert.AreEqual(_count, _target.Position);        }        [TestMethod]        public async Task FlushShouldFlushBuffer()        {            byte[] actualFlushedData = null;            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.IsOpen)                           .Returns(true);            SftpSessionMock.InSequence(MockSequence)                           .Setup(p => p.RequestWriteAsync(_handle, _expectedWrittenByteCount, It.IsAny<byte[]>(), 0, _expectedBufferedByteCount, _cancellationToken))                           .Callback<byte[], ulong, byte[], int, int, CancellationToken>((handle, serverFileOffset, data, offset, length, ct) => actualFlushedData = data.Take(offset, length))                           .Returns(Task.CompletedTask);            await _target.FlushAsync();            Assert.IsTrue(actualFlushedData.IsEqualTo(_expectedBufferedBytes));            SftpSessionMock.Verify(p => p.RequestWriteAsync(_handle, _expectedWrittenByteCount, It.IsAny<byte[]>(), 0, _expectedBufferedByteCount, _cancellationToken), Times.Once);        }    }}
 |