using System; using System.Collections.Generic; using System.Diagnostics; using System.Net; using System.Net.Sockets; using System.Threading; using Microsoft.VisualStudio.TestTools.UnitTesting; using Moq; using Renci.SshNet.Channels; using Renci.SshNet.Common; namespace Renci.SshNet.Tests.Classes { [TestClass] public class ForwardedPortLocalTest_Dispose_PortStarted_ChannelBound { private Mock _sessionMock; private Mock _connectionInfoMock; private Mock _channelMock; private ForwardedPortLocal _forwardedPort; private IList _closingRegister; private IList _exceptionRegister; private IPEndPoint _localEndpoint; private IPEndPoint _remoteEndpoint; private Socket _client; private TimeSpan _expectedElapsedTime; private TimeSpan _elapsedTimeOfStop; private Stopwatch _stopwatch; [TestInitialize] public void Setup() { Arrange(); Act(); } [TestCleanup] public void Cleanup() { if (_client != null) { _client.Dispose(); _client = null; } if (_forwardedPort != null) { _forwardedPort.Dispose(); _forwardedPort = null; } } protected void Arrange() { var random = new Random(); _closingRegister = new List(); _exceptionRegister = new List(); _localEndpoint = new IPEndPoint(IPAddress.Loopback, 8122); _remoteEndpoint = new IPEndPoint(IPAddress.Parse("193.168.1.5"), random.Next(IPEndPoint.MinPort, IPEndPoint.MaxPort)); _expectedElapsedTime = TimeSpan.FromMilliseconds(random.Next(100, 500)); _forwardedPort = new ForwardedPortLocal(_localEndpoint.Address.ToString(), (uint)_localEndpoint.Port, _remoteEndpoint.Address.ToString(), (uint)_remoteEndpoint.Port); _connectionInfoMock = new Mock(MockBehavior.Strict); _sessionMock = new Mock(MockBehavior.Strict); _channelMock = new Mock(MockBehavior.Strict); Socket handlerSocket = null; _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.CreateChannelDirectTcpip()).Returns(_channelMock.Object); _channelMock.Setup(p => p.Open(_forwardedPort.Host, _forwardedPort.Port, _forwardedPort, It.IsAny())).Callback((address, port, forwardedPort, socket) => handlerSocket = socket); _channelMock.Setup(p => p.Bind()).Callback(() => { Thread.Sleep(_expectedElapsedTime); if (handlerSocket != null && handlerSocket.Connected) handlerSocket.Shutdown(SocketShutdown.Both); }); _channelMock.Setup(p => p.Close()); _channelMock.Setup(p => p.Dispose()); _forwardedPort.Closing += (sender, args) => _closingRegister.Add(args); _forwardedPort.Exception += (sender, args) => _exceptionRegister.Add(args); _forwardedPort.Session = _sessionMock.Object; _forwardedPort.Start(); _client = new Socket(_localEndpoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp) { ReceiveTimeout = 500, SendTimeout = 500, SendBufferSize = 0 }; _stopwatch = new Stopwatch(); _stopwatch.Start(); _client.Connect(_localEndpoint); // give client socket time to establish connection Thread.Sleep(50); } protected void Act() { _forwardedPort.Dispose(); _stopwatch.Stop(); _elapsedTimeOfStop = _stopwatch.Elapsed; } [TestMethod] public void StopShouldBlockUntilBoundChannelHasClosed() { Assert.IsTrue(_elapsedTimeOfStop >= _expectedElapsedTime, string.Format("Expected {0} or greater but was {1}.", _expectedElapsedTime.TotalMilliseconds, _elapsedTimeOfStop.TotalMilliseconds)); Assert.IsTrue(_elapsedTimeOfStop < _expectedElapsedTime.Add(TimeSpan.FromMilliseconds(200))); } [TestMethod] public void IsStartedShouldReturnFalse() { Assert.IsFalse(_forwardedPort.IsStarted); } [TestMethod] public void ForwardedPortShouldRefuseNewConnections() { using (var client = new Socket(_localEndpoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp)) { try { client.Connect(_localEndpoint); Assert.Fail(); } catch (SocketException ex) { Assert.AreEqual(SocketError.ConnectionRefused, ex.SocketErrorCode); } } } [TestMethod] public void ExistingConnectionShouldBeClosed() { try { _client.Send(new byte[] { 0x0a }, 0, 1, SocketFlags.None); Assert.Fail(); } catch (SocketException ex) { Assert.AreEqual(SocketError.ConnectionReset, ex.SocketErrorCode); } } [TestMethod] public void ClosingShouldHaveFiredOnce() { Assert.AreEqual(1, _closingRegister.Count); } [TestMethod] public void ExceptionShouldNotHaveFired() { Assert.AreEqual(0, _exceptionRegister.Count); } [TestMethod] public void OpenOnChannelShouldBeInvokedOnce() { _channelMock.Verify(p => p.Open(_forwardedPort.Host, _forwardedPort.Port, _forwardedPort, It.IsAny()), Times.Once); } [TestMethod] public void BindOnChannelShouldBeInvokedOnce() { _channelMock.Verify(p => p.Bind(), Times.Once); } [TestMethod] public void CloseOnChannelShouldBeInvokedOnce() { _channelMock.Verify(p => p.Close(), Times.Once); } [TestMethod] public void DisposeOnChannelShouldBeInvokedOnce() { _channelMock.Verify(p => p.Dispose(), Times.Once); } } }