using System; using System.Collections.Generic; using System.Linq; 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 Socks5ConnectorTest_Connect_UserNamePasswordAuthentication_AuthenticationFailed : Socks5ConnectorTestBase { private ConnectionInfo _connectionInfo; private AsyncSocketListener _proxyServer; private Socket _clientSocket; private List _bytesReceivedByProxy; private bool _disconnected; private ProxyException _actualException; protected override void SetupData() { base.SetupData(); _connectionInfo = CreateConnectionInfo("aa", "bbbb"); _connectionInfo.Timeout = TimeSpan.FromMilliseconds(100); _bytesReceivedByProxy = new List(); _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); if (_bytesReceivedByProxy.Count == 4) { // We received the greeting _ = socket.Send(new byte[] { // SOCKS version 0x05, // Require username/password authentication 0x02 }); } else if (_bytesReceivedByProxy.Count == 4 + (1 + 1 + 2 + 1 + 4)) { // We received the username/password authentication request _ = socket.Send(new byte[] { // Authentication version 0x01, // Authentication failed 0x01 }); } }; _proxyServer.Start(); } protected override void SetupMocks() { _ = SocketFactoryMock.Setup(p => p.Create(SocketType.Stream, ProtocolType.Tcp)) .Returns(_clientSocket); } protected override void TearDown() { base.TearDown(); _proxyServer?.Dispose(); _clientSocket?.Dispose(); } protected override void Act() { try { _ = Connector.Connect(_connectionInfo); Assert.Fail(); } catch (ProxyException ex) { _actualException = ex; } // Give some time to process all messages Thread.Sleep(200); } [TestMethod] public void ConnectShouldHaveThrownProxyException() { Assert.IsNotNull(_actualException); Assert.IsNull(_actualException.InnerException); Assert.AreEqual("SOCKS5: Username/Password authentication failed.", _actualException.Message); } [TestMethod] public void ProxyShouldHaveReceivedExpectedSocksRequest() { var expectedSocksRequest = new List(); // // Client greeting // // SOCKS version expectedSocksRequest.Add(0x05); // Number of authentication methods supported expectedSocksRequest.Add(0x02); // No authentication expectedSocksRequest.Add(0x00); // Username/password expectedSocksRequest.Add(0x02); // // Username/password authentication request // // Version of the negotiation expectedSocksRequest.Add(0x01); // Length of the username expectedSocksRequest.Add((byte)_connectionInfo.ProxyUsername.Length); // Username expectedSocksRequest.AddRange(Encoding.ASCII.GetBytes(_connectionInfo.ProxyUsername)); // Length of the password expectedSocksRequest.Add((byte)_connectionInfo.ProxyPassword.Length); // Password expectedSocksRequest.AddRange(Encoding.ASCII.GetBytes(_connectionInfo.ProxyPassword)); var errorText = string.Format("Expected:{0}{1}{0}but was:{0}{2}", Environment.NewLine, PacketDump.Create(expectedSocksRequest, 2), PacketDump.Create(_bytesReceivedByProxy, 2)); Assert.IsTrue(expectedSocksRequest.SequenceEqual(_bytesReceivedByProxy), errorText); } [TestMethod] public void ConnectionToProxyShouldHaveBeenShutDown() { Assert.IsTrue(_disconnected); } [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()); } } }