SessionTest_Connected_ServerAndClientDisconnectRace.cs 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Net;
  5. using System.Net.Sockets;
  6. using System.Security.Cryptography;
  7. using System.Text;
  8. using Microsoft.VisualStudio.TestTools.UnitTesting;
  9. using Moq;
  10. using Renci.SshNet.Common;
  11. using Renci.SshNet.Compression;
  12. using Renci.SshNet.Messages;
  13. using Renci.SshNet.Messages.Transport;
  14. using Renci.SshNet.Security;
  15. using Renci.SshNet.Security.Cryptography;
  16. using Renci.SshNet.Tests.Common;
  17. namespace Renci.SshNet.Tests.Classes
  18. {
  19. [TestClass]
  20. public class SessionTest_Connected_ServerAndClientDisconnectRace
  21. {
  22. private Mock<IServiceFactory> _serviceFactoryMock;
  23. private Mock<IKeyExchange> _keyExchangeMock;
  24. private Mock<IClientAuthentication> _clientAuthenticationMock;
  25. private IPEndPoint _serverEndPoint;
  26. private string _keyExchangeAlgorithm;
  27. private DisconnectMessage _disconnectMessage;
  28. protected Random Random { get; private set; }
  29. protected byte[] SessionId { get; private set; }
  30. protected ConnectionInfo ConnectionInfo { get; private set; }
  31. protected IList<EventArgs> DisconnectedRegister { get; private set; }
  32. protected IList<MessageEventArgs<DisconnectMessage>> DisconnectReceivedRegister { get; private set; }
  33. protected IList<ExceptionEventArgs> ErrorOccurredRegister { get; private set; }
  34. protected AsyncSocketListener ServerListener { get; private set; }
  35. protected IList<byte[]> ServerBytesReceivedRegister { get; private set; }
  36. protected Session Session { get; private set; }
  37. protected Socket ServerSocket { get; private set; }
  38. private void TearDown()
  39. {
  40. if (ServerListener != null)
  41. {
  42. ServerListener.Dispose();
  43. }
  44. if (Session != null)
  45. {
  46. Session.Dispose();
  47. }
  48. }
  49. protected virtual void SetupData()
  50. {
  51. Random = new Random();
  52. _serverEndPoint = new IPEndPoint(IPAddress.Loopback, 8122);
  53. ConnectionInfo = new ConnectionInfo(
  54. _serverEndPoint.Address.ToString(),
  55. _serverEndPoint.Port,
  56. "user",
  57. new PasswordAuthenticationMethod("user", "password"))
  58. { Timeout = TimeSpan.FromSeconds(20) };
  59. _keyExchangeAlgorithm = Random.Next().ToString(CultureInfo.InvariantCulture);
  60. SessionId = new byte[10];
  61. Random.NextBytes(SessionId);
  62. DisconnectedRegister = new List<EventArgs>();
  63. DisconnectReceivedRegister = new List<MessageEventArgs<DisconnectMessage>>();
  64. ErrorOccurredRegister = new List<ExceptionEventArgs>();
  65. ServerBytesReceivedRegister = new List<byte[]>();
  66. _disconnectMessage = new DisconnectMessage(DisconnectReason.ServiceNotAvailable, "Not today!");
  67. Session = new Session(ConnectionInfo, _serviceFactoryMock.Object);
  68. Session.Disconnected += (sender, args) => DisconnectedRegister.Add(args);
  69. Session.DisconnectReceived += (sender, args) => DisconnectReceivedRegister.Add(args);
  70. Session.ErrorOccured += (sender, args) => ErrorOccurredRegister.Add(args);
  71. Session.KeyExchangeInitReceived += (sender, args) =>
  72. {
  73. var newKeysMessage = new NewKeysMessage();
  74. var newKeys = newKeysMessage.GetPacket(8, null);
  75. ServerSocket.Send(newKeys, 4, newKeys.Length - 4, SocketFlags.None);
  76. };
  77. ServerListener = new AsyncSocketListener(_serverEndPoint);
  78. ServerListener.Connected += socket =>
  79. {
  80. ServerSocket = socket;
  81. socket.Send(Encoding.ASCII.GetBytes("\r\n"));
  82. socket.Send(Encoding.ASCII.GetBytes("WELCOME banner\r\n"));
  83. socket.Send(Encoding.ASCII.GetBytes("SSH-2.0-SshStub\r\n"));
  84. };
  85. var counter = 0;
  86. ServerListener.BytesReceived += (received, socket) =>
  87. {
  88. ServerBytesReceivedRegister.Add(received);
  89. switch (counter++)
  90. {
  91. case 0:
  92. var keyExchangeInitMessage = new KeyExchangeInitMessage
  93. {
  94. CompressionAlgorithmsClientToServer = new string[0],
  95. CompressionAlgorithmsServerToClient = new string[0],
  96. EncryptionAlgorithmsClientToServer = new string[0],
  97. EncryptionAlgorithmsServerToClient = new string[0],
  98. KeyExchangeAlgorithms = new[] { _keyExchangeAlgorithm },
  99. LanguagesClientToServer = new string[0],
  100. LanguagesServerToClient = new string[0],
  101. MacAlgorithmsClientToServer = new string[0],
  102. MacAlgorithmsServerToClient = new string[0],
  103. ServerHostKeyAlgorithms = new string[0]
  104. };
  105. var keyExchangeInit = keyExchangeInitMessage.GetPacket(8, null);
  106. ServerSocket.Send(keyExchangeInit, 4, keyExchangeInit.Length - 4, SocketFlags.None);
  107. break;
  108. case 1:
  109. var serviceAcceptMessage =ServiceAcceptMessageBuilder.Create(ServiceName.UserAuthentication).Build();
  110. ServerSocket.Send(serviceAcceptMessage, 0, serviceAcceptMessage.Length, SocketFlags.None);
  111. break;
  112. }
  113. };
  114. }
  115. private void CreateMocks()
  116. {
  117. _serviceFactoryMock = new Mock<IServiceFactory>(MockBehavior.Strict);
  118. _keyExchangeMock = new Mock<IKeyExchange>(MockBehavior.Strict);
  119. _clientAuthenticationMock = new Mock<IClientAuthentication>(MockBehavior.Strict);
  120. }
  121. private void SetupMocks()
  122. {
  123. _serviceFactoryMock.Setup(
  124. p =>
  125. p.CreateKeyExchange(ConnectionInfo.KeyExchangeAlgorithms, new[] { _keyExchangeAlgorithm })).Returns(_keyExchangeMock.Object);
  126. _keyExchangeMock.Setup(p => p.Name).Returns(_keyExchangeAlgorithm);
  127. _keyExchangeMock.Setup(p => p.Start(Session, It.IsAny<KeyExchangeInitMessage>()));
  128. _keyExchangeMock.Setup(p => p.ExchangeHash).Returns(SessionId);
  129. _keyExchangeMock.Setup(p => p.CreateServerCipher()).Returns((Cipher)null);
  130. _keyExchangeMock.Setup(p => p.CreateClientCipher()).Returns((Cipher)null);
  131. _keyExchangeMock.Setup(p => p.CreateServerHash()).Returns((HashAlgorithm)null);
  132. _keyExchangeMock.Setup(p => p.CreateClientHash()).Returns((HashAlgorithm)null);
  133. _keyExchangeMock.Setup(p => p.CreateCompressor()).Returns((Compressor)null);
  134. _keyExchangeMock.Setup(p => p.CreateDecompressor()).Returns((Compressor)null);
  135. _keyExchangeMock.Setup(p => p.Dispose());
  136. _serviceFactoryMock.Setup(p => p.CreateClientAuthentication()).Returns(_clientAuthenticationMock.Object);
  137. _clientAuthenticationMock.Setup(p => p.Authenticate(ConnectionInfo, Session));
  138. }
  139. protected virtual void Arrange()
  140. {
  141. CreateMocks();
  142. SetupData();
  143. SetupMocks();
  144. ServerListener.Start();
  145. Session.Connect();
  146. }
  147. [TestMethod]
  148. public void Act()
  149. {
  150. for (var i = 0; i < 50; i++)
  151. {
  152. Arrange();
  153. try
  154. {
  155. var disconnect = _disconnectMessage.GetPacket(8, null);
  156. ServerSocket.Send(disconnect, 4, disconnect.Length - 4, SocketFlags.None);
  157. Session.Disconnect();
  158. }
  159. finally
  160. {
  161. TearDown();
  162. }
  163. }
  164. }
  165. private class ServiceAcceptMessageBuilder
  166. {
  167. private readonly ServiceName _serviceName;
  168. private ServiceAcceptMessageBuilder(ServiceName serviceName)
  169. {
  170. _serviceName = serviceName;
  171. }
  172. public static ServiceAcceptMessageBuilder Create(ServiceName serviceName)
  173. {
  174. return new ServiceAcceptMessageBuilder(serviceName);
  175. }
  176. public byte[] Build()
  177. {
  178. var serviceName = _serviceName.ToArray();
  179. var sshDataStream = new SshDataStream(4 + 1 + 1 + 4 + serviceName.Length);
  180. sshDataStream.Write((uint)(sshDataStream.Capacity - 4)); // packet length
  181. sshDataStream.WriteByte(0); // padding length
  182. sshDataStream.WriteByte(ServiceAcceptMessage.MessageNumber);
  183. sshDataStream.WriteBinary(serviceName);
  184. return sshDataStream.ToArray();
  185. }
  186. }
  187. }
  188. }