ChannelDirectTcpipTest_Dispose_SessionIsConnectedAndChannelIsOpen.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. using System;
  2. using System.Globalization;
  3. using System.Net;
  4. using System.Net.Sockets;
  5. using System.Threading;
  6. using Microsoft.VisualStudio.TestTools.UnitTesting;
  7. using Moq;
  8. using Renci.SshNet.Channels;
  9. using Renci.SshNet.Messages.Connection;
  10. using Renci.SshNet.Tests.Common;
  11. namespace Renci.SshNet.Tests.Classes.Channels
  12. {
  13. [TestClass]
  14. public class ChannelDirectTcpipTest_Dispose_SessionIsConnectedAndChannelIsOpen
  15. {
  16. private Mock<ISession> _sessionMock;
  17. private Mock<IConnectionInfo> _connectionInfoMock;
  18. private Mock<IForwardedPort> _forwardedPortMock;
  19. private ChannelDirectTcpip _channel;
  20. private uint _localChannelNumber;
  21. private uint _localWindowSize;
  22. private uint _localPacketSize;
  23. private uint _remoteWindowSize;
  24. private uint _remotePacketSize;
  25. private uint _remoteChannelNumber;
  26. private TimeSpan _channelCloseTimeout;
  27. private string _remoteHost;
  28. private uint _port;
  29. private AsyncSocketListener _listener;
  30. private EventWaitHandle _channelBindFinishedWaitHandle;
  31. private EventWaitHandle _clientReceivedFinishedWaitHandle;
  32. private Socket _client;
  33. private Exception _channelException;
  34. [TestInitialize]
  35. public void Initialize()
  36. {
  37. Arrange();
  38. Act();
  39. }
  40. [TestCleanup]
  41. public void CleanUp()
  42. {
  43. if (_client != null)
  44. {
  45. _client.Dispose();
  46. _client = null;
  47. }
  48. if (_listener != null)
  49. {
  50. _listener.Stop();
  51. _listener = null;
  52. }
  53. }
  54. private void Arrange()
  55. {
  56. var random = new Random();
  57. _localChannelNumber = (uint)random.Next(0, int.MaxValue);
  58. _localWindowSize = (uint)random.Next(2000, 3000);
  59. _localPacketSize = (uint)random.Next(1000, 2000);
  60. _channelCloseTimeout = TimeSpan.FromSeconds(random.Next(10, 20));
  61. _remoteHost = random.Next().ToString(CultureInfo.InvariantCulture);
  62. _port = (uint)random.Next(IPEndPoint.MinPort, IPEndPoint.MaxPort);
  63. _channelBindFinishedWaitHandle = new ManualResetEvent(false);
  64. _clientReceivedFinishedWaitHandle = new ManualResetEvent(false);
  65. _channelException = null;
  66. _remoteChannelNumber = (uint)random.Next(0, int.MaxValue);
  67. _remoteWindowSize = (uint)random.Next(0, int.MaxValue);
  68. _remotePacketSize = (uint)random.Next(100, 200);
  69. _sessionMock = new Mock<ISession>(MockBehavior.Strict);
  70. _connectionInfoMock = new Mock<IConnectionInfo>(MockBehavior.Strict);
  71. _forwardedPortMock = new Mock<IForwardedPort>(MockBehavior.Strict);
  72. var sequence = new MockSequence();
  73. _ = _sessionMock.InSequence(sequence).Setup(p => p.IsConnected).Returns(true);
  74. _ = _sessionMock.InSequence(sequence)
  75. .Setup(p => p.SendMessage(It.Is<ChannelOpenMessage>(m => AssertExpectedMessage(m))));
  76. _ = _sessionMock.InSequence(sequence)
  77. .Setup(p => p.WaitOnHandle(It.IsNotNull<WaitHandle>()))
  78. .Callback<WaitHandle>(
  79. w =>
  80. {
  81. _sessionMock.Raise(
  82. s => s.ChannelOpenConfirmationReceived += null,
  83. new MessageEventArgs<ChannelOpenConfirmationMessage>(
  84. new ChannelOpenConfirmationMessage(
  85. _localChannelNumber,
  86. _remoteWindowSize,
  87. _remotePacketSize,
  88. _remoteChannelNumber)));
  89. _ = w.WaitOne();
  90. });
  91. _ = _sessionMock.InSequence(sequence)
  92. .Setup(p => p.IsConnected)
  93. .Returns(true);
  94. _ = _sessionMock.InSequence(sequence)
  95. .Setup(p => p.TrySendMessage(It.Is<ChannelEofMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)))
  96. .Returns(true);
  97. _ = _sessionMock.InSequence(sequence)
  98. .Setup(p => p.IsConnected)
  99. .Returns(true);
  100. _ = _sessionMock.InSequence(sequence)
  101. .Setup(p => p.TrySendMessage(It.Is<ChannelCloseMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)))
  102. .Returns(true);
  103. _ = _sessionMock.InSequence(sequence)
  104. .Setup(p => p.ConnectionInfo)
  105. .Returns(_connectionInfoMock.Object);
  106. _ = _connectionInfoMock.InSequence(sequence)
  107. .Setup(p => p.ChannelCloseTimeout)
  108. .Returns(_channelCloseTimeout);
  109. _ = _sessionMock.InSequence(sequence)
  110. .Setup(p => p.TryWait(It.IsAny<EventWaitHandle>(), _channelCloseTimeout))
  111. .Callback<WaitHandle, TimeSpan>((waitHandle, channelCloseTimeout) =>
  112. {
  113. _sessionMock.Raise(
  114. s => s.ChannelCloseReceived += null,
  115. new MessageEventArgs<ChannelCloseMessage>(new ChannelCloseMessage(_localChannelNumber)));
  116. _ = waitHandle.WaitOne();
  117. })
  118. .Returns(WaitResult.Success);
  119. using var barrier = new Barrier(2);
  120. var localEndpoint = new IPEndPoint(IPAddress.Loopback, 8122);
  121. _listener = new AsyncSocketListener(localEndpoint);
  122. _listener.Connected += socket =>
  123. {
  124. try
  125. {
  126. // We need the Connect side and the Accept side to be
  127. // fully completed before continuing: we are implicitly
  128. // checking that RemoteEndPoint on the Accept socket
  129. // matches LocalEndPoint on the Connect socket when
  130. // checking the correctness of the ChannelOpenMessage
  131. // in the mock.
  132. _ = barrier.SignalAndWait(TimeSpan.FromSeconds(1));
  133. _channel = new ChannelDirectTcpip(_sessionMock.Object,
  134. _localChannelNumber,
  135. _localWindowSize,
  136. _localPacketSize);
  137. _channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);
  138. _channel.Bind();
  139. }
  140. catch (Exception ex)
  141. {
  142. _channelException = ex;
  143. }
  144. finally
  145. {
  146. _ = _channelBindFinishedWaitHandle.Set();
  147. }
  148. };
  149. _listener.Start();
  150. _client = new Socket(localEndpoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
  151. _client.Connect(localEndpoint);
  152. _ = barrier.SignalAndWait(TimeSpan.FromSeconds(1));
  153. var clientReceiveThread = new Thread(
  154. () =>
  155. {
  156. var buffer = new byte[16];
  157. var bytesReceived = _client.Receive(buffer, 0, buffer.Length, SocketFlags.None);
  158. if (bytesReceived == 0)
  159. {
  160. _client.Shutdown(SocketShutdown.Send);
  161. _ = _clientReceivedFinishedWaitHandle.Set();
  162. }
  163. }
  164. );
  165. clientReceiveThread.Start();
  166. // give channel time to bind to socket
  167. Thread.Sleep(200);
  168. }
  169. private void Act()
  170. {
  171. _channel?.Dispose();
  172. Thread.Sleep(200);
  173. }
  174. [TestMethod]
  175. public void BindShouldHaveFinishedWithoutException()
  176. {
  177. Assert.IsTrue(_channelBindFinishedWaitHandle.WaitOne(0));
  178. Assert.IsNull(_channelException, _channelException?.ToString());
  179. }
  180. [TestMethod]
  181. public void ClientShouldHaveFinished()
  182. {
  183. Assert.IsTrue(_clientReceivedFinishedWaitHandle.WaitOne(0));
  184. }
  185. [TestMethod]
  186. public void ChannelEofMessageShouldBeSentOnce()
  187. {
  188. _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelEofMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);
  189. }
  190. [TestMethod]
  191. public void ChannelCloseMessageShouldBeSentOnce()
  192. {
  193. _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelCloseMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);
  194. }
  195. [TestMethod]
  196. public void IsOpenShouldReturnFalse()
  197. {
  198. Assert.IsFalse(_channel.IsOpen);
  199. }
  200. private bool AssertExpectedMessage(ChannelOpenMessage channelOpenMessage)
  201. {
  202. if (channelOpenMessage == null)
  203. {
  204. return false;
  205. }
  206. if (channelOpenMessage.LocalChannelNumber != _localChannelNumber)
  207. {
  208. return false;
  209. }
  210. if (channelOpenMessage.InitialWindowSize != _localWindowSize)
  211. {
  212. return false;
  213. }
  214. if (channelOpenMessage.MaximumPacketSize != _localPacketSize)
  215. {
  216. return false;
  217. }
  218. if (channelOpenMessage.Info is not DirectTcpipChannelInfo directTcpipChannelInfo)
  219. {
  220. return false;
  221. }
  222. if (directTcpipChannelInfo.HostToConnect != _remoteHost)
  223. {
  224. return false;
  225. }
  226. if (directTcpipChannelInfo.PortToConnect != _port)
  227. {
  228. return false;
  229. }
  230. if (_client.LocalEndPoint is not IPEndPoint clientEndpoint)
  231. {
  232. return false;
  233. }
  234. if (directTcpipChannelInfo.OriginatorAddress != clientEndpoint.Address.ToString())
  235. {
  236. return false;
  237. }
  238. if (directTcpipChannelInfo.OriginatorPort != clientEndpoint.Port)
  239. {
  240. return false;
  241. }
  242. return true;
  243. }
  244. }
  245. }