ChannelDirectTcpipTest_Dispose_SessionIsConnectedAndChannelIsOpen.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  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. var localEndpoint = new IPEndPoint(IPAddress.Loopback, 8122);
  120. _listener = new AsyncSocketListener(localEndpoint);
  121. _listener.Connected += socket =>
  122. {
  123. try
  124. {
  125. _channel = new ChannelDirectTcpip(_sessionMock.Object,
  126. _localChannelNumber,
  127. _localWindowSize,
  128. _localPacketSize);
  129. _channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);
  130. _channel.Bind();
  131. }
  132. catch (Exception ex)
  133. {
  134. _channelException = ex;
  135. }
  136. finally
  137. {
  138. _ = _channelBindFinishedWaitHandle.Set();
  139. }
  140. };
  141. _listener.Start();
  142. _client = new Socket(localEndpoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
  143. _client.Connect(localEndpoint);
  144. var clientReceiveThread = new Thread(
  145. () =>
  146. {
  147. var buffer = new byte[16];
  148. var bytesReceived = _client.Receive(buffer, 0, buffer.Length, SocketFlags.None);
  149. if (bytesReceived == 0)
  150. {
  151. _client.Shutdown(SocketShutdown.Send);
  152. _ = _clientReceivedFinishedWaitHandle.Set();
  153. }
  154. }
  155. );
  156. clientReceiveThread.Start();
  157. // give channel time to bind to socket
  158. Thread.Sleep(200);
  159. }
  160. private void Act()
  161. {
  162. _channel?.Dispose();
  163. Thread.Sleep(200);
  164. }
  165. [TestMethod]
  166. public void BindShouldHaveFinishedWithoutException()
  167. {
  168. Assert.IsTrue(_channelBindFinishedWaitHandle.WaitOne(0));
  169. Assert.IsNull(_channelException, _channelException?.ToString());
  170. }
  171. [TestMethod]
  172. public void ClientShouldHaveFinished()
  173. {
  174. Assert.IsTrue(_clientReceivedFinishedWaitHandle.WaitOne(0));
  175. }
  176. [TestMethod]
  177. public void ChannelEofMessageShouldBeSentOnce()
  178. {
  179. _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelEofMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);
  180. }
  181. [TestMethod]
  182. public void ChannelCloseMessageShouldBeSentOnce()
  183. {
  184. _sessionMock.Verify(p => p.TrySendMessage(It.Is<ChannelCloseMessage>(m => m.LocalChannelNumber == _remoteChannelNumber)), Times.Once);
  185. }
  186. [TestMethod]
  187. public void IsOpenShouldReturnFalse()
  188. {
  189. Assert.IsFalse(_channel.IsOpen);
  190. }
  191. private bool AssertExpectedMessage(ChannelOpenMessage channelOpenMessage)
  192. {
  193. if (channelOpenMessage == null)
  194. {
  195. return false;
  196. }
  197. if (channelOpenMessage.LocalChannelNumber != _localChannelNumber)
  198. {
  199. return false;
  200. }
  201. if (channelOpenMessage.InitialWindowSize != _localWindowSize)
  202. {
  203. return false;
  204. }
  205. if (channelOpenMessage.MaximumPacketSize != _localPacketSize)
  206. {
  207. return false;
  208. }
  209. if (channelOpenMessage.Info is not DirectTcpipChannelInfo directTcpipChannelInfo)
  210. {
  211. return false;
  212. }
  213. if (directTcpipChannelInfo.HostToConnect != _remoteHost)
  214. {
  215. return false;
  216. }
  217. if (directTcpipChannelInfo.PortToConnect != _port)
  218. {
  219. return false;
  220. }
  221. if (_client.LocalEndPoint is not IPEndPoint clientEndpoint)
  222. {
  223. return false;
  224. }
  225. if (directTcpipChannelInfo.OriginatorAddress != clientEndpoint.Address.ToString())
  226. {
  227. return false;
  228. }
  229. if (directTcpipChannelInfo.OriginatorPort != clientEndpoint.Port)
  230. {
  231. return false;
  232. }
  233. return true;
  234. }
  235. }
  236. }