| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 | using System;using System.Globalization;using System.Net;using System.Net.Sockets;using System.Threading;using Microsoft.VisualStudio.TestTools.UnitTesting;using Moq;using Renci.SshNet.Channels;using Renci.SshNet.Messages.Connection;using Renci.SshNet.Tests.Common;namespace Renci.SshNet.Tests.Classes.Channels{    [TestClass]    public class ChannelDirectTcpipTest_Dispose_SessionIsConnectedAndChannelIsOpen    {        private Mock<ISession> _sessionMock;        private Mock<IConnectionInfo> _connectionInfoMock;        private Mock<IForwardedPort> _forwardedPortMock;        private ChannelDirectTcpip _channel;        private uint _localChannelNumber;        private uint _localWindowSize;        private uint _localPacketSize;        private uint _remoteWindowSize;        private uint _remotePacketSize;        private uint _remoteChannelNumber;        private TimeSpan _channelCloseTimeout;        private string _remoteHost;        private uint _port;        private AsyncSocketListener _listener;        private EventWaitHandle _channelBindFinishedWaitHandle;        private EventWaitHandle _clientReceivedFinishedWaitHandle;        private Socket _client;        private Exception _channelException;        [TestInitialize]        public void Initialize()        {            Arrange();            Act();        }        [TestCleanup]        public void CleanUp()        {            if (_client != null)            {                _client.Dispose();                _client = null;            }            if (_listener != null)            {                _listener.Stop();                _listener = null;            }        }        private void Arrange()        {            var random = new Random();            _localChannelNumber = (uint) random.Next(0, int.MaxValue);            _localWindowSize = (uint) random.Next(2000, 3000);            _localPacketSize = (uint) random.Next(1000, 2000);            _channelCloseTimeout = TimeSpan.FromSeconds(random.Next(10, 20));            _remoteHost = random.Next().ToString(CultureInfo.InvariantCulture);            _port = (uint) random.Next(IPEndPoint.MinPort, IPEndPoint.MaxPort);            _channelBindFinishedWaitHandle = new ManualResetEvent(false);            _clientReceivedFinishedWaitHandle = new ManualResetEvent(false);            _channelException = null;            _remoteChannelNumber = (uint)random.Next(0, int.MaxValue);            _remoteWindowSize = (uint)random.Next(0, int.MaxValue);            _remotePacketSize = (uint)random.Next(100, 200);            _sessionMock = new Mock<ISession>(MockBehavior.Strict);            _connectionInfoMock = new Mock<IConnectionInfo>(MockBehavior.Strict);            _forwardedPortMock = new Mock<IForwardedPort>(MockBehavior.Strict);            var sequence = new MockSequence();            _ = _sessionMock.InSequence(sequence).Setup(p => p.IsConnected).Returns(true);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.SendMessage(It.Is<ChannelOpenMessage>(m => AssertExpectedMessage(m))));            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.WaitOnHandle(It.IsNotNull<WaitHandle>()))                            .Callback<WaitHandle>(                                w =>                                {                                    _sessionMock.Raise(                                        s => s.ChannelOpenConfirmationReceived += null,                                        new MessageEventArgs<ChannelOpenConfirmationMessage>(                                            new ChannelOpenConfirmationMessage(                                                _localChannelNumber,                                                _remoteWindowSize,                                                _remotePacketSize,                                                _remoteChannelNumber)));                                    _ = w.WaitOne();                                });            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.IsConnected)                            .Returns(true);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.TrySendMessage(It.Is<ChannelEofMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)))                            .Returns(true);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.IsConnected)                            .Returns(true);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.TrySendMessage(It.Is<ChannelCloseMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)))                            .Returns(true);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.ConnectionInfo)                            .Returns(_connectionInfoMock.Object);            _ = _connectionInfoMock.InSequence(sequence)                                   .Setup(p => p.ChannelCloseTimeout)                                   .Returns(_channelCloseTimeout);            _ = _sessionMock.InSequence(sequence)                            .Setup(p => p.TryWait(It.IsAny<EventWaitHandle>(), _channelCloseTimeout))                            .Callback<WaitHandle, TimeSpan>((waitHandle, channelCloseTimeout) =>                                {                                    _sessionMock.Raise(                                        s => s.ChannelCloseReceived += null,                                        new MessageEventArgs<ChannelCloseMessage>(new ChannelCloseMessage(_localChannelNumber)));                                    _ = waitHandle.WaitOne();                                })                            .Returns(WaitResult.Success);            var localEndpoint = new IPEndPoint(IPAddress.Loopback, 8122);            _listener = new AsyncSocketListener(localEndpoint);            _listener.Connected += socket =>                {                    try                    {                        _channel = new ChannelDirectTcpip(_sessionMock.Object,                                                          _localChannelNumber,                                                          _localWindowSize,                                                          _localPacketSize);                        _channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);                        _channel.Bind();                    }                    catch (Exception ex)                    {                        _channelException = ex;                    }                    finally                    {                        _ = _channelBindFinishedWaitHandle.Set();                    }                };            _listener.Start();            _client = new Socket(localEndpoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);            _client.Connect(localEndpoint);            var clientReceiveThread = new Thread(                () =>                    {                        var buffer = new byte[16];                        var bytesReceived = _client.Receive(buffer, 0, buffer.Length, SocketFlags.None);                        if (bytesReceived == 0)                        {                            _client.Shutdown(SocketShutdown.Send);                            _ = _clientReceivedFinishedWaitHandle.Set();                        }                    }                );            clientReceiveThread.Start();            // give channel time to bind to socket            Thread.Sleep(200);        }        private void Act()        {            _channel?.Dispose();        }        [TestMethod]        public void BindShouldHaveFinishedWithoutException()        {            Assert.IsTrue(_channelBindFinishedWaitHandle.WaitOne(0));            Assert.IsNull(_channelException, _channelException?.ToString());        }        [TestMethod]        public void ClientShouldHaveFinished()        {            Assert.IsTrue(_clientReceivedFinishedWaitHandle.WaitOne(0));        }        [TestMethod]        public void ChannelEofMessageShouldBeSentOnce()        {            _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelEofMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);        }        [TestMethod]        public void ChannelCloseMessageShouldBeSentOnce()        {            _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelCloseMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);        }        [TestMethod]        public void IsOpenShouldReturnFalse()        {            Assert.IsFalse(_channel.IsOpen);        }        private bool AssertExpectedMessage(ChannelOpenMessage channelOpenMessage)        {            if (channelOpenMessage == null)            {                return false;            }            if (channelOpenMessage.LocalChannelNumber != _localChannelNumber)            {                return false;            }            if (channelOpenMessage.InitialWindowSize != _localWindowSize)            {                return false;            }            if (channelOpenMessage.MaximumPacketSize != _localPacketSize)            {                return false;            }            if (channelOpenMessage.Info is not DirectTcpipChannelInfo directTcpipChannelInfo)            {                return false;            }            if (directTcpipChannelInfo.HostToConnect != _remoteHost)            {                return false;            }            if (directTcpipChannelInfo.PortToConnect != _port)            {                return false;            }            if (_client.LocalEndPoint is not IPEndPoint clientEndpoint)            {                return false;            }            if (directTcpipChannelInfo.OriginatorAddress != clientEndpoint.Address.ToString())            {                return false;            }            if (directTcpipChannelInfo.OriginatorPort != clientEndpoint.Port)            {                return false;            }            return true;        }    }}
 |