using System; using System.Collections.Generic; using System.Globalization; using System.Net; using System.Threading; using Microsoft.VisualStudio.TestTools.UnitTesting; using Moq; using Renci.SshNet.Channels; using Renci.SshNet.Common; using Renci.SshNet.Messages.Connection; namespace Renci.SshNet.Tests.Classes { [TestClass] public class ForwardedPortRemoteTest_Started { private Mock _sessionMock; private Mock _connectionInfoMock; private ForwardedPortRemote _forwardedPort; private IList _closingRegister; private IList _exceptionRegister; private IPEndPoint _bindEndpoint; private IPEndPoint _remoteEndpoint; [TestInitialize] public void Setup() { Arrange(); Act(); } [TestCleanup] public void Cleanup() { if (_forwardedPort != null) { _sessionMock.Setup( p => p.SendMessage( It.Is( g => g.AddressToBind == _forwardedPort.BoundHost && g.PortToBind == _forwardedPort.BoundPort))); _sessionMock.Setup(p => p.MessageListenerCompleted).Returns(new ManualResetEvent(true)); _forwardedPort.Dispose(); _forwardedPort = null; } } protected void Arrange() { var random = new Random(); _closingRegister = new List(); _exceptionRegister = new List(); _bindEndpoint = new IPEndPoint(IPAddress.Any, random.Next(IPEndPoint.MinPort, 1000)); _remoteEndpoint = new IPEndPoint(IPAddress.Parse("193.168.1.5"), random.Next(IPEndPoint.MinPort, IPEndPoint.MaxPort)); _forwardedPort = new ForwardedPortRemote(_bindEndpoint.Address, (uint)_bindEndpoint.Port, _remoteEndpoint.Address, (uint)_remoteEndpoint.Port); _connectionInfoMock = new Mock(MockBehavior.Strict); _sessionMock = new Mock(MockBehavior.Strict); _connectionInfoMock.Setup(p => p.Timeout).Returns(TimeSpan.FromSeconds(15)); _sessionMock.Setup(p => p.IsConnected).Returns(true); _sessionMock.Setup(p => p.ConnectionInfo).Returns(_connectionInfoMock.Object); _sessionMock.Setup(p => p.RegisterMessage("SSH_MSG_REQUEST_FAILURE")); _sessionMock.Setup(p => p.RegisterMessage("SSH_MSG_REQUEST_SUCCESS")); _sessionMock.Setup(p => p.RegisterMessage("SSH_MSG_CHANNEL_OPEN")); _sessionMock.Setup( p => p.SendMessage( It.Is( g => g.AddressToBind == _forwardedPort.BoundHost && g.PortToBind == _forwardedPort.BoundPort))) .Callback( () => _sessionMock.Raise(s => s.RequestSuccessReceived += null, new MessageEventArgs(new RequestSuccessMessage()))); _sessionMock.Setup(p => p.WaitOnHandle(It.IsAny())) .Callback(handle => handle.WaitOne()); _forwardedPort.Closing += (sender, args) => _closingRegister.Add(args); _forwardedPort.Exception += (sender, args) => _exceptionRegister.Add(args); _forwardedPort.Session = _sessionMock.Object; _forwardedPort.Start(); } protected void Act() { } [TestMethod] public void ForwardedPortShouldAcceptChannelOpenMessageForBoundAddressAndBoundPort() { var channelNumber = (uint) new Random().Next(1001, int.MaxValue); var initialWindowSize = (uint) new Random().Next(0, int.MaxValue); var maximumPacketSize = (uint) new Random().Next(0, int.MaxValue); var originatorAddress = new Random().Next().ToString(CultureInfo.InvariantCulture); var originatorPort = (uint) new Random().Next(0, int.MaxValue); var channelMock = new Mock(MockBehavior.Strict); var channelDisposed = new ManualResetEvent(false); _sessionMock.Setup( p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize)).Returns(channelMock.Object); channelMock.Setup( p => p.Bind( It.Is( ep => ep.Address.Equals(_remoteEndpoint.Address) && ep.Port == _remoteEndpoint.Port), _forwardedPort)); channelMock.Setup(p => p.Dispose()).Callback(() => channelDisposed.Set()); _sessionMock.Raise(p => p.ChannelOpenReceived += null, new MessageEventArgs(new ChannelOpenMessage(channelNumber, initialWindowSize, maximumPacketSize, new ForwardedTcpipChannelInfo(_forwardedPort.BoundHost, _forwardedPort.BoundPort, originatorAddress, originatorPort)))); // wait for channel to be disposed channelDisposed.WaitOne(TimeSpan.FromMilliseconds(200)); _sessionMock.Verify(p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize), Times.Once); channelMock.Verify(p => p.Bind(It.Is(ep => ep.Address.Equals(_remoteEndpoint.Address) && ep.Port == _remoteEndpoint.Port), _forwardedPort), Times.Once); channelMock.Verify(p => p.Dispose(), Times.Once); Assert.AreEqual(0, _closingRegister.Count); Assert.AreEqual(0, _exceptionRegister.Count); } [TestMethod] public void ForwardedPortShouldIgnoreChannelOpenMessageForBoundHostAndOtherPort() { var channelNumber = (uint)new Random().Next(1001, int.MaxValue); var initialWindowSize = (uint)new Random().Next(0, int.MaxValue); var maximumPacketSize = (uint)new Random().Next(0, int.MaxValue); var originatorAddress = new Random().Next().ToString(CultureInfo.InvariantCulture); var originatorPort = (uint)new Random().Next(0, int.MaxValue); var channelMock = new Mock(MockBehavior.Strict); _sessionMock.Setup( p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize)).Returns(channelMock.Object); _sessionMock.Raise(p => p.ChannelOpenReceived += null, new MessageEventArgs(new ChannelOpenMessage(channelNumber, initialWindowSize, maximumPacketSize, new ForwardedTcpipChannelInfo(_forwardedPort.BoundHost, _forwardedPort.BoundPort + 1, originatorAddress, originatorPort)))); _sessionMock.Verify(p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize), Times.Never); Assert.AreEqual(0, _closingRegister.Count); Assert.AreEqual(0, _exceptionRegister.Count); } [TestMethod] public void ForwardedPortShouldIgnoreChannelOpenMessageForOtherHostAndBoundPort() { var channelNumber = (uint)new Random().Next(1001, int.MaxValue); var initialWindowSize = (uint)new Random().Next(0, int.MaxValue); var maximumPacketSize = (uint)new Random().Next(0, int.MaxValue); var originatorAddress = new Random().Next().ToString(CultureInfo.InvariantCulture); var originatorPort = (uint)new Random().Next(0, int.MaxValue); var channelMock = new Mock(MockBehavior.Strict); _sessionMock.Setup( p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize)).Returns(channelMock.Object); _sessionMock.Raise(p => p.ChannelOpenReceived += null, new MessageEventArgs(new ChannelOpenMessage(channelNumber, initialWindowSize, maximumPacketSize, new ForwardedTcpipChannelInfo("111.111.111.111", _forwardedPort.BoundPort, originatorAddress, originatorPort)))); _sessionMock.Verify(p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize), Times.Never); Assert.AreEqual(0, _closingRegister.Count); Assert.AreEqual(0, _exceptionRegister.Count); } [TestMethod] public void ForwardedPortShouldIgnoreChannelOpenMessageWhenChannelOpenInfoIsNotForwardedTcpipChannelInfo() { var channelNumber = (uint)new Random().Next(1001, int.MaxValue); var initialWindowSize = (uint)new Random().Next(0, int.MaxValue); var maximumPacketSize = (uint)new Random().Next(0, int.MaxValue); var channelMock = new Mock(MockBehavior.Strict); _sessionMock.Setup( p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize)).Returns(channelMock.Object); _sessionMock.Raise(p => p.ChannelOpenReceived += null, new MessageEventArgs(new ChannelOpenMessage(channelNumber, initialWindowSize, maximumPacketSize, new DirectTcpipChannelInfo("HOST", 5, "ORIGIN", 4)))); _sessionMock.Verify(p => p.CreateChannelForwardedTcpip(channelNumber, initialWindowSize, maximumPacketSize), Times.Never); Assert.AreEqual(0, _closingRegister.Count); Assert.AreEqual(0, _exceptionRegister.Count); } } }