ChannelDirectTcpipTest.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  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.Common;
  10. using Renci.SshNet.Messages;
  11. using Renci.SshNet.Messages.Connection;
  12. using Renci.SshNet.Tests.Common;
  13. namespace Renci.SshNet.Tests.Classes.Channels
  14. {
  15. [TestClass]
  16. public class ChannelDirectTcpipTestTest : TestBase
  17. {
  18. private Mock<ISession> _sessionMock;
  19. private Mock<IForwardedPort> _forwardedPortMock;
  20. private Mock<IConnectionInfo> _connectionInfoMock;
  21. private uint _localChannelNumber;
  22. private uint _localWindowSize;
  23. private uint _localPacketSize;
  24. private string _remoteHost;
  25. private uint _port;
  26. private uint _remoteWindowSize;
  27. private uint _remotePacketSize;
  28. private uint _remoteChannelNumber;
  29. private TimeSpan _channelCloseTimeout;
  30. protected override void OnInit()
  31. {
  32. base.OnInit();
  33. var random = new Random();
  34. _localWindowSize = (uint)random.Next(2000, 3000);
  35. _localPacketSize = (uint)random.Next(1000, 2000);
  36. _remoteHost = random.Next().ToString(CultureInfo.InvariantCulture);
  37. _port = (uint)random.Next(IPEndPoint.MinPort, IPEndPoint.MaxPort);
  38. _localChannelNumber = (uint)random.Next(0, int.MaxValue);
  39. _remoteWindowSize = (uint)random.Next(0, int.MaxValue);
  40. _remotePacketSize = (uint)random.Next(100, 200);
  41. _remoteChannelNumber = (uint)random.Next(0, int.MaxValue);
  42. _channelCloseTimeout = TimeSpan.FromSeconds(random.Next(10, 20));
  43. _sessionMock = new Mock<ISession>(MockBehavior.Strict);
  44. _forwardedPortMock = new Mock<IForwardedPort>(MockBehavior.Strict);
  45. _connectionInfoMock = new Mock<IConnectionInfo>(MockBehavior.Strict);
  46. }
  47. [TestMethod]
  48. public void SocketShouldBeClosedAndBindShouldEndWhenForwardedPortSignalsClosingEvent()
  49. {
  50. _ = _sessionMock.Setup(p => p.IsConnected)
  51. .Returns(true);
  52. _ = _sessionMock.Setup(p => p.SendMessage(It.IsAny<ChannelOpenMessage>()))
  53. .Callback<Message>(m => _sessionMock.Raise(p => p.ChannelOpenConfirmationReceived += null,
  54. new MessageEventArgs<ChannelOpenConfirmationMessage>(
  55. new ChannelOpenConfirmationMessage(((ChannelOpenMessage)m).LocalChannelNumber,
  56. _remoteWindowSize,
  57. _remotePacketSize,
  58. _remoteChannelNumber))));
  59. _ = _sessionMock.Setup(p => p.WaitOnHandle(It.IsAny<EventWaitHandle>()))
  60. .Callback<WaitHandle>(p => p.WaitOne());
  61. var localPortEndPoint = new IPEndPoint(IPAddress.Loopback, 8122);
  62. using (var localPortListener = new AsyncSocketListener(localPortEndPoint))
  63. {
  64. localPortListener.Start();
  65. localPortListener.Connected += socket =>
  66. {
  67. var channel = new ChannelDirectTcpip(_sessionMock.Object,
  68. _localChannelNumber,
  69. _localWindowSize,
  70. _localPacketSize);
  71. channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);
  72. var closeForwardedPortThread =
  73. new Thread(() =>
  74. {
  75. // sleep for a short period to allow channel to actually start receiving from socket
  76. Thread.Sleep(100);
  77. // raise Closing event on forwarded port
  78. _forwardedPortMock.Raise(p => p.Closing += null, EventArgs.Empty);
  79. });
  80. closeForwardedPortThread.Start();
  81. channel.Bind();
  82. closeForwardedPortThread.Join();
  83. };
  84. var client = new Socket(localPortEndPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
  85. client.Connect(localPortEndPoint);
  86. // attempt to receive from socket to verify it was shut down by channel
  87. var buffer = new byte[16];
  88. var bytesReceived = client.Receive(buffer, 0, buffer.Length, SocketFlags.None);
  89. Assert.AreEqual(0, bytesReceived);
  90. Assert.IsTrue(client.Connected);
  91. // signal to server that we also shut down the socket at our end
  92. client.Shutdown(SocketShutdown.Send);
  93. }
  94. }
  95. [TestMethod]
  96. public void SocketShouldBeClosedAndBindShouldEndWhenOnErrorOccurredIsInvoked()
  97. {
  98. _ = _sessionMock.Setup(p => p.IsConnected)
  99. .Returns(true);
  100. _ = _sessionMock.Setup(p => p.SendMessage(It.IsAny<ChannelOpenMessage>()))
  101. .Callback<Message>(m => _sessionMock.Raise(p => p.ChannelOpenConfirmationReceived += null,
  102. new MessageEventArgs<ChannelOpenConfirmationMessage>(
  103. new ChannelOpenConfirmationMessage(((ChannelOpenMessage)m).LocalChannelNumber,
  104. _remoteWindowSize,
  105. _remotePacketSize,
  106. _remoteChannelNumber))));
  107. _ = _sessionMock.Setup(p => p.WaitOnHandle(It.IsAny<EventWaitHandle>()))
  108. .Callback<WaitHandle>(p => p.WaitOne());
  109. var localPortEndPoint = new IPEndPoint(IPAddress.Loopback, 8122);
  110. using (var localPortListener = new AsyncSocketListener(localPortEndPoint))
  111. {
  112. localPortListener.Start();
  113. localPortListener.Connected += socket =>
  114. {
  115. var channel = new ChannelDirectTcpip(_sessionMock.Object,
  116. _localChannelNumber,
  117. _localWindowSize,
  118. _localPacketSize);
  119. channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);
  120. var signalSessionErrorOccurredThread =
  121. new Thread(() =>
  122. {
  123. // sleep for a short period to allow channel to actually start receiving from socket
  124. Thread.Sleep(100);
  125. // raise ErrorOccured event on session
  126. _sessionMock.Raise(s => s.ErrorOccured += null,
  127. new ExceptionEventArgs(new SystemException()));
  128. });
  129. signalSessionErrorOccurredThread.Start();
  130. channel.Bind();
  131. signalSessionErrorOccurredThread.Join();
  132. };
  133. var client = new Socket(localPortEndPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
  134. client.Connect(localPortEndPoint);
  135. // attempt to receive from socket to verify it was shut down by channel
  136. var buffer = new byte[16];
  137. var bytesReceived = client.Receive(buffer, 0, buffer.Length, SocketFlags.None);
  138. Assert.AreEqual(0, bytesReceived);
  139. Assert.IsTrue(client.Connected);
  140. // signal to server that we also shut down the socket at our end
  141. client.Shutdown(SocketShutdown.Send);
  142. }
  143. }
  144. [TestMethod]
  145. public void SocketShouldBeClosedAndEofShouldBeSentToServerWhenClientShutsDownSocket()
  146. {
  147. var sequence = new MockSequence();
  148. _ = _sessionMock.InSequence(sequence).Setup(p => p.IsConnected)
  149. .Returns(true);
  150. _ = _sessionMock.InSequence(sequence)
  151. .Setup(p => p.SendMessage(It.IsAny<ChannelOpenMessage>()))
  152. .Callback<Message>(m => _sessionMock.Raise(p => p.ChannelOpenConfirmationReceived += null,
  153. new MessageEventArgs<ChannelOpenConfirmationMessage>(
  154. new ChannelOpenConfirmationMessage(((ChannelOpenMessage)m).LocalChannelNumber,
  155. _remoteWindowSize,
  156. _remotePacketSize,
  157. _remoteChannelNumber))));
  158. _ = _sessionMock.InSequence(sequence)
  159. .Setup(p => p.WaitOnHandle(It.IsAny<EventWaitHandle>()))
  160. .Callback<WaitHandle>(p => p.WaitOne());
  161. _ = _sessionMock.InSequence(sequence)
  162. .Setup(p => p.IsConnected)
  163. .Returns(true);
  164. _ = _sessionMock.InSequence(sequence)
  165. .Setup(p => p.TrySendMessage(It.IsAny<ChannelEofMessage>()))
  166. .Returns(true)
  167. .Callback<Message>(m => new Thread(() =>
  168. {
  169. Thread.Sleep(50);
  170. _sessionMock.Raise(s => s.ChannelEofReceived += null,
  171. new MessageEventArgs<ChannelEofMessage>(new ChannelEofMessage(_localChannelNumber)));
  172. }).Start());
  173. _ = _sessionMock.InSequence(sequence)
  174. .Setup(p => p.IsConnected)
  175. .Returns(true);
  176. _ = _sessionMock.InSequence(sequence)
  177. .Setup(p => p.TrySendMessage(It.IsAny<ChannelCloseMessage>()))
  178. .Returns(true)
  179. .Callback<Message>(m => new Thread(() =>
  180. {
  181. Thread.Sleep(50);
  182. _sessionMock.Raise(s => s.ChannelCloseReceived += null,
  183. new MessageEventArgs<ChannelCloseMessage>(new ChannelCloseMessage(_localChannelNumber)));
  184. }).Start());
  185. _ = _sessionMock.InSequence(sequence)
  186. .Setup(p => p.ConnectionInfo)
  187. .Returns(_connectionInfoMock.Object);
  188. _ = _connectionInfoMock.InSequence(sequence)
  189. .Setup(p => p.ChannelCloseTimeout)
  190. .Returns(_channelCloseTimeout);
  191. _ = _sessionMock.InSequence(sequence)
  192. .Setup(p => p.TryWait(It.IsAny<EventWaitHandle>(), _channelCloseTimeout))
  193. .Callback<WaitHandle, TimeSpan>((waitHandle, channelCloseTimeout) => waitHandle.WaitOne())
  194. .Returns(WaitResult.Success);
  195. var channelBindFinishedWaitHandle = new ManualResetEvent(false);
  196. Socket handler = null;
  197. ChannelDirectTcpip channel = null;
  198. var localPortEndPoint = new IPEndPoint(IPAddress.Loopback, 8122);
  199. using (var localPortListener = new AsyncSocketListener(localPortEndPoint))
  200. {
  201. localPortListener.Start();
  202. localPortListener.Connected += socket =>
  203. {
  204. channel = new ChannelDirectTcpip(_sessionMock.Object,
  205. _localChannelNumber,
  206. _localWindowSize,
  207. _localPacketSize);
  208. channel.Open(_remoteHost, _port, _forwardedPortMock.Object, socket);
  209. channel.Bind();
  210. channel.Dispose();
  211. handler = socket;
  212. _ = channelBindFinishedWaitHandle.Set();
  213. };
  214. var client = new Socket(localPortEndPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
  215. client.Connect(localPortEndPoint);
  216. client.Shutdown(SocketShutdown.Send);
  217. Assert.IsFalse(client.Connected);
  218. _ = channelBindFinishedWaitHandle.WaitOne();
  219. Assert.IsNotNull(handler);
  220. Assert.IsFalse(handler.Connected);
  221. _sessionMock.Verify(p => p.TrySendMessage(It.IsAny<ChannelEofMessage>()), Times.Once);
  222. _sessionMock.Verify(p => p.TrySendMessage(It.IsAny<ChannelCloseMessage>()), Times.Once);
  223. channel.Dispose();
  224. _sessionMock.Verify(p => p.TrySendMessage(It.IsAny<ChannelEofMessage>()), Times.Once);
  225. _sessionMock.Verify(p => p.TrySendMessage(It.IsAny<ChannelCloseMessage>()), Times.Once);
  226. }
  227. }
  228. }
  229. }