| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- 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);
- using var barrier = new Barrier(2);
- var localEndpoint = new IPEndPoint(IPAddress.Loopback, 8122);
- _listener = new AsyncSocketListener(localEndpoint);
- _listener.Connected += socket =>
- {
- try
- {
- // We need the Connect side and the Accept side to be
- // fully completed before continuing: we are implicitly
- // checking that RemoteEndPoint on the Accept socket
- // matches LocalEndPoint on the Connect socket when
- // checking the correctness of the ChannelOpenMessage
- // in the mock.
- _ = barrier.SignalAndWait(TimeSpan.FromSeconds(1));
- _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);
- _ = barrier.SignalAndWait(TimeSpan.FromSeconds(1));
- 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();
- Thread.Sleep(200);
- }
- [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;
- }
- }
- }
|