SftpClientTest_AsyncExceptions.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. using System;
  2. using System.IO;
  3. using System.Text;
  4. using System.Threading;
  5. using System.Threading.Tasks;
  6. using Microsoft.Extensions.Logging;
  7. using Microsoft.Extensions.Logging.Abstractions;
  8. using Microsoft.VisualStudio.TestTools.UnitTesting;
  9. using Moq;
  10. #if !NET
  11. using Renci.SshNet.Abstractions;
  12. #endif
  13. using Renci.SshNet.Channels;
  14. using Renci.SshNet.Common;
  15. using Renci.SshNet.Connection;
  16. using Renci.SshNet.Messages;
  17. using Renci.SshNet.Messages.Authentication;
  18. using Renci.SshNet.Messages.Connection;
  19. using Renci.SshNet.Sftp;
  20. using Renci.SshNet.Sftp.Responses;
  21. namespace Renci.SshNet.Tests.Classes
  22. {
  23. [TestClass]
  24. public class SftpClientTest_AsyncExceptions
  25. {
  26. private MySession _session;
  27. private SftpClient _client;
  28. [TestInitialize]
  29. public void Init()
  30. {
  31. var socketFactoryMock = new Mock<ISocketFactory>(MockBehavior.Strict);
  32. var serviceFactoryMock = new Mock<IServiceFactory>(MockBehavior.Strict);
  33. var connInfo = new PasswordConnectionInfo("host", "user", "pwd");
  34. _session = new MySession(connInfo);
  35. var concreteServiceFactory = new ServiceFactory();
  36. serviceFactoryMock
  37. .Setup(p => p.CreateSocketFactory())
  38. .Returns(socketFactoryMock.Object);
  39. serviceFactoryMock
  40. .Setup(p => p.CreateSession(It.IsAny<ConnectionInfo>(), socketFactoryMock.Object))
  41. .Returns(_session);
  42. serviceFactoryMock
  43. .Setup(p => p.CreateSftpResponseFactory())
  44. .Returns(concreteServiceFactory.CreateSftpResponseFactory);
  45. serviceFactoryMock
  46. .Setup(p => p.CreateSftpSession(_session, It.IsAny<int>(), It.IsAny<Encoding>(), It.IsAny<ISftpResponseFactory>()))
  47. .Returns(concreteServiceFactory.CreateSftpSession);
  48. _client = new SftpClient(connInfo, false, serviceFactoryMock.Object);
  49. _client.Connect();
  50. }
  51. [TestMethod]
  52. public async Task Async_ObservesSessionDisconnected()
  53. {
  54. Task<SftpFileStream> openTask = _client.OpenAsync("path", FileMode.Create, FileAccess.Write, CancellationToken.None);
  55. Assert.IsFalse(openTask.IsCompleted);
  56. _session.InvokeDisconnected();
  57. var ex = await Assert.ThrowsExceptionAsync<SshException>(() => openTask);
  58. Assert.AreEqual("Connection was closed by the server.", ex.Message);
  59. }
  60. [TestMethod]
  61. public async Task Async_ObservesChannelClosed()
  62. {
  63. Task<SftpFileStream> openTask = _client.OpenAsync("path", FileMode.Create, FileAccess.Write, CancellationToken.None);
  64. Assert.IsFalse(openTask.IsCompleted);
  65. _session.InvokeChannelCloseReceived();
  66. var ex = await Assert.ThrowsExceptionAsync<SshException>(() => openTask);
  67. Assert.AreEqual("Channel was closed.", ex.Message);
  68. }
  69. [TestMethod]
  70. public async Task Async_ObservesCancellationToken()
  71. {
  72. using CancellationTokenSource cts = new();
  73. Task<SftpFileStream> openTask = _client.OpenAsync("path", FileMode.Create, FileAccess.Write, cts.Token);
  74. Assert.IsFalse(openTask.IsCompleted);
  75. await cts.CancelAsync();
  76. var ex = await Assert.ThrowsExceptionAsync<TaskCanceledException>(() => openTask);
  77. Assert.AreEqual(cts.Token, ex.CancellationToken);
  78. }
  79. [TestMethod]
  80. public async Task Async_ObservesOperationTimeout()
  81. {
  82. _client.OperationTimeout = TimeSpan.FromMilliseconds(250);
  83. Task<SftpFileStream> openTask = _client.OpenAsync("path", FileMode.Create, FileAccess.Write, CancellationToken.None);
  84. var ex = await Assert.ThrowsExceptionAsync<SshOperationTimeoutException>(() => openTask);
  85. }
  86. [TestMethod]
  87. public async Task Async_ObservesErrorOccurred()
  88. {
  89. Task<SftpFileStream> openTask = _client.OpenAsync("path", FileMode.Create, FileAccess.Write, CancellationToken.None);
  90. Assert.IsFalse(openTask.IsCompleted);
  91. MyException ex = new("my exception");
  92. _session.InvokeErrorOccurred(ex);
  93. var ex2 = await Assert.ThrowsExceptionAsync<MyException>(() => openTask);
  94. Assert.AreEqual(ex.Message, ex2.Message);
  95. }
  96. #pragma warning disable IDE0022 // Use block body for method
  97. #pragma warning disable IDE0025 // Use block body for property
  98. #pragma warning disable CS0067 // event is unused
  99. private class MySession(ConnectionInfo connectionInfo) : ISession
  100. {
  101. public IConnectionInfo ConnectionInfo => connectionInfo;
  102. public event EventHandler<MessageEventArgs<ChannelCloseMessage>> ChannelCloseReceived;
  103. public event EventHandler<MessageEventArgs<ChannelDataMessage>> ChannelDataReceived;
  104. public event EventHandler<MessageEventArgs<ChannelEofMessage>> ChannelEofReceived;
  105. public event EventHandler<MessageEventArgs<ChannelExtendedDataMessage>> ChannelExtendedDataReceived;
  106. public event EventHandler<MessageEventArgs<ChannelFailureMessage>> ChannelFailureReceived;
  107. public event EventHandler<MessageEventArgs<ChannelOpenConfirmationMessage>> ChannelOpenConfirmationReceived;
  108. public event EventHandler<MessageEventArgs<ChannelOpenFailureMessage>> ChannelOpenFailureReceived;
  109. public event EventHandler<MessageEventArgs<ChannelOpenMessage>> ChannelOpenReceived;
  110. public event EventHandler<MessageEventArgs<ChannelRequestMessage>> ChannelRequestReceived;
  111. public event EventHandler<MessageEventArgs<ChannelSuccessMessage>> ChannelSuccessReceived;
  112. public event EventHandler<MessageEventArgs<ChannelWindowAdjustMessage>> ChannelWindowAdjustReceived;
  113. public event EventHandler<EventArgs> Disconnected;
  114. public event EventHandler<ExceptionEventArgs> ErrorOccured;
  115. public event EventHandler<SshIdentificationEventArgs> ServerIdentificationReceived;
  116. public event EventHandler<HostKeyEventArgs> HostKeyReceived;
  117. public event EventHandler<MessageEventArgs<RequestSuccessMessage>> RequestSuccessReceived;
  118. public event EventHandler<MessageEventArgs<RequestFailureMessage>> RequestFailureReceived;
  119. public event EventHandler<MessageEventArgs<BannerMessage>> UserAuthenticationBannerReceived;
  120. public void InvokeDisconnected()
  121. {
  122. Disconnected?.Invoke(this, new EventArgs());
  123. }
  124. public void InvokeChannelCloseReceived()
  125. {
  126. ChannelCloseReceived?.Invoke(
  127. this,
  128. new MessageEventArgs<ChannelCloseMessage>(new ChannelCloseMessage(0)));
  129. }
  130. public void InvokeErrorOccurred(Exception ex)
  131. {
  132. ErrorOccured?.Invoke(this, new ExceptionEventArgs(ex));
  133. }
  134. public void SendMessage(Message message)
  135. {
  136. if (message is ChannelOpenMessage)
  137. {
  138. ChannelOpenConfirmationReceived?.Invoke(
  139. this,
  140. new MessageEventArgs<ChannelOpenConfirmationMessage>(
  141. new ChannelOpenConfirmationMessage(0, int.MaxValue, int.MaxValue, 0)));
  142. }
  143. else if (message is ChannelRequestMessage)
  144. {
  145. ChannelSuccessReceived?.Invoke(
  146. this,
  147. new MessageEventArgs<ChannelSuccessMessage>(new ChannelSuccessMessage(0)));
  148. }
  149. else if (message is ChannelDataMessage dataMsg)
  150. {
  151. if (dataMsg.Data[sizeof(uint)] == (byte)SftpMessageTypes.Init)
  152. {
  153. ChannelDataReceived?.Invoke(
  154. this,
  155. new MessageEventArgs<ChannelDataMessage>(
  156. new ChannelDataMessage(0, new SftpVersionResponse() { Version = 3 }.GetBytes())));
  157. }
  158. else if (dataMsg.Data[sizeof(uint)] == (byte)SftpMessageTypes.RealPath)
  159. {
  160. ChannelDataReceived?.Invoke(
  161. this,
  162. new MessageEventArgs<ChannelDataMessage>(
  163. new ChannelDataMessage(0,
  164. new SftpNameResponse(3, Encoding.UTF8)
  165. {
  166. ResponseId = 1,
  167. Files = [new("thepath", new SftpFileAttributes(default, default, default, default, default, default, default))]
  168. }.GetBytes())));
  169. }
  170. }
  171. }
  172. public bool IsConnected => false;
  173. public SemaphoreSlim SessionSemaphore { get; } = new(1);
  174. public IChannelSession CreateChannelSession() => new ChannelSession(this, 0, int.MaxValue, int.MaxValue);
  175. public WaitHandle MessageListenerCompleted => throw new NotImplementedException();
  176. public ILoggerFactory SessionLoggerFactory => NullLoggerFactory.Instance;
  177. public void Connect()
  178. {
  179. }
  180. public Task ConnectAsync(CancellationToken cancellationToken) => throw new NotImplementedException();
  181. public IChannelDirectTcpip CreateChannelDirectTcpip() => throw new NotImplementedException();
  182. public IChannelForwardedTcpip CreateChannelForwardedTcpip(uint remoteChannelNumber, uint remoteWindowSize, uint remoteChannelDataPacketSize)
  183. => throw new NotImplementedException();
  184. public void Dispose()
  185. {
  186. }
  187. public void OnDisconnecting()
  188. {
  189. }
  190. public void Disconnect() => throw new NotImplementedException();
  191. public void RegisterMessage(string messageName) => throw new NotImplementedException();
  192. public bool TrySendMessage(Message message) => throw new NotImplementedException();
  193. public WaitResult TryWait(WaitHandle waitHandle, TimeSpan timeout, out Exception exception) => throw new NotImplementedException();
  194. public WaitResult TryWait(WaitHandle waitHandle, TimeSpan timeout) => throw new NotImplementedException();
  195. public void UnRegisterMessage(string messageName) => throw new NotImplementedException();
  196. public void WaitOnHandle(WaitHandle waitHandle)
  197. {
  198. }
  199. public void WaitOnHandle(WaitHandle waitHandle, TimeSpan timeout) => throw new NotImplementedException();
  200. }
  201. [TestCleanup]
  202. public void Cleanup() => _client?.Dispose();
  203. #pragma warning disable
  204. private class MyException : Exception
  205. {
  206. public MyException(string message) : base(message)
  207. {
  208. }
  209. }
  210. }
  211. }