| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171 |
- using System;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Globalization;
- using System.Net;
- using System.Net.Sockets;
- using System.Text;
- using System.Threading;
- using Microsoft.VisualStudio.TestTools.UnitTesting;
- using Moq;
- using Renci.SshNet.Common;
- using Renci.SshNet.Tests.Common;
- namespace Renci.SshNet.Tests.Classes.Connection
- {
- [TestClass]
- public class HttpConnectorTest_Connect_TimeoutReadingHttpContent : HttpConnectorTestBase
- {
- private ConnectionInfo _connectionInfo;
- private SshOperationTimeoutException _actualException;
- private Socket _clientSocket;
- private AsyncSocketListener _proxyServer;
- private List<byte> _bytesReceivedByProxy;
- private string _expectedHttpRequest;
- private Stopwatch _stopWatch;
- private AsyncSocketListener _server;
- private bool _disconnected;
- protected override void SetupData()
- {
- base.SetupData();
- var random = new Random();
- _connectionInfo = new ConnectionInfo(IPAddress.Loopback.ToString(),
- 1026,
- "user",
- ProxyTypes.Http,
- IPAddress.Loopback.ToString(),
- 8122,
- "proxyUser",
- "proxyPwd",
- new KeyboardInteractiveAuthenticationMethod("user"))
- {
- Timeout = TimeSpan.FromMilliseconds(random.Next(50, 200))
- };
- _expectedHttpRequest = string.Format("CONNECT {0}:{1} HTTP/1.0{2}" +
- "Proxy-Authorization: Basic cHJveHlVc2VyOnByb3h5UHdk{2}{2}",
- _connectionInfo.Host,
- _connectionInfo.Port.ToString(CultureInfo.InvariantCulture),
- "\r\n");
- _bytesReceivedByProxy = new List<byte>();
- _stopWatch = new Stopwatch();
- _actualException = null;
- _clientSocket = SocketFactory.Create(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
- _proxyServer = new AsyncSocketListener(new IPEndPoint(IPAddress.Loopback, _connectionInfo.ProxyPort));
- _proxyServer.Disconnected += (socket) => _disconnected = true;
- _proxyServer.BytesReceived += (bytesReceived, socket) =>
- {
- _bytesReceivedByProxy.AddRange(bytesReceived);
- // Force a timeout by sending less content than indicated by Content-Length header
- if (_bytesReceivedByProxy.Count == _expectedHttpRequest.Length)
- {
- _ = socket.Send(Encoding.ASCII.GetBytes("HTTP/1.0 200 OK\r\n"));
- _ = socket.Send(Encoding.ASCII.GetBytes("Content-Length: 10\r\n"));
- _ = socket.Send(Encoding.ASCII.GetBytes("Content-Type: application/octet-stream\r\n"));
- _ = socket.Send(Encoding.ASCII.GetBytes("\r\n"));
- _ = socket.Send(Encoding.ASCII.GetBytes("TOO_FEW"));
- }
- };
- _proxyServer.Start();
- _server = new AsyncSocketListener(new IPEndPoint(IPAddress.Loopback, _connectionInfo.Port));
- _server.Start();
- }
- protected override void SetupMocks()
- {
- _ = SocketFactoryMock.Setup(p => p.Create(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp))
- .Returns(_clientSocket);
- }
- protected override void TearDown()
- {
- base.TearDown();
- _server?.Dispose();
- _proxyServer?.Dispose();
- }
- protected override void Act()
- {
- _stopWatch.Start();
- try
- {
- _ = Connector.Connect(_connectionInfo);
- Assert.Fail();
- }
- catch (SshOperationTimeoutException ex)
- {
- _actualException = ex;
- }
- finally
- {
- _stopWatch.Stop();
- }
- // Give some time to process all messages
- Thread.Sleep(200);
- }
- [TestMethod]
- public void ConnectShouldHaveThrownSshOperationTimeoutException()
- {
- Assert.IsNull(_actualException.InnerException);
- Assert.AreEqual(string.Format(CultureInfo.InvariantCulture, "Socket read operation has timed out after {0:F0} milliseconds.", _connectionInfo.Timeout.TotalMilliseconds), _actualException.Message);
- }
- [TestMethod]
- public void ProxyShouldHaveReceivedExpectedHttpRequest()
- {
- Assert.AreEqual(_expectedHttpRequest, Encoding.ASCII.GetString(_bytesReceivedByProxy.ToArray()));
- }
- [TestMethod]
- public void ConnectShouldHaveRespectedTimeout()
- {
- var errorText = string.Format("Elapsed: {0}, Timeout: {1}",
- _stopWatch.ElapsedMilliseconds,
- _connectionInfo.Timeout.TotalMilliseconds);
- // Compare elapsed time with configured timeout, allowing for a margin of error
- Assert.IsTrue(_stopWatch.ElapsedMilliseconds >= _connectionInfo.Timeout.TotalMilliseconds - 10, errorText);
- Assert.IsTrue(_stopWatch.ElapsedMilliseconds < _connectionInfo.Timeout.TotalMilliseconds + 100, errorText);
- }
- [TestMethod]
- public void ClientSocketShouldNotBeConnected()
- {
- Assert.IsTrue(_disconnected);
- Assert.IsFalse(_clientSocket.Connected);
- }
- [TestMethod]
- public void ClientSocketShouldHaveBeenDisposed()
- {
- try
- {
- _ = _clientSocket.Receive(new byte[0]);
- Assert.Fail();
- }
- catch (ObjectDisposedException)
- {
- }
- }
- [TestMethod]
- public void CreateOnSocketFactoryShouldHaveBeenInvokedOnce()
- {
- SocketFactoryMock.Verify(p => p.Create(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp),
- Times.Once());
- }
- }
- }
|