| 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(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(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(SocketType.Stream, ProtocolType.Tcp),                                     Times.Once());        }    }}
 |