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