| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398 | #pragma warning disable IDE0005 // Using directive is unnecessary; IntegrationTests use implicit usingsusing System;using System.Collections.Generic;using System.Net;using System.Net.Sockets;using System.Threading;#pragma warning restore IDE0005namespace Renci.SshNet.Tests.Common{    public class AsyncSocketListener : IDisposable    {        private readonly IPEndPoint _endPoint;        private readonly ManualResetEvent _acceptCallbackDone;        private readonly List<Socket> _connectedClients;        private readonly object _syncLock;        private Socket _listener;        private Thread _receiveThread;        private bool _started;        private string _stackTrace;        public delegate void BytesReceivedHandler(byte[] bytesReceived, Socket socket);        public delegate void ConnectedHandler(Socket socket);        public event BytesReceivedHandler BytesReceived;        public event ConnectedHandler Connected;        public event ConnectedHandler Disconnected;        public AsyncSocketListener(IPEndPoint endPoint)        {            _endPoint = endPoint;            _acceptCallbackDone = new ManualResetEvent(false);            _connectedClients = new List<Socket>();            _syncLock = new object();            ShutdownRemoteCommunicationSocket = true;        }        /// <summary>        /// Gets a value indicating whether the <see cref="Socket.Shutdown(SocketShutdown)"/> is invoked on the <see cref="Socket"/>        /// that is used to handle the communication with the remote host, when the remote host has closed the connection.        /// </summary>        /// <value>        /// <see langword="true"/> to invoke <see cref="Socket.Shutdown(SocketShutdown)"/> on the <see cref="Socket"/> that is used        /// to handle the communication with the remote host, when the remote host has closed the connection; otherwise,        /// <see langword="false"/>. The default is <see langword="true"/>.        /// </value>        public bool ShutdownRemoteCommunicationSocket { get; set; }        public void Start()        {            _listener = new Socket(_endPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);            _listener.Bind(_endPoint);            _listener.Listen(1);            _started = true;            _receiveThread = new Thread(StartListener);            _receiveThread.Start(_listener);            _stackTrace = Environment.StackTrace;        }        public void Stop()        {            _started = false;            lock (_syncLock)            {                foreach (var connectedClient in _connectedClients)                {                    try                    {                        connectedClient.Shutdown(SocketShutdown.Send);                    }                    catch (Exception ex)                    {                        Console.Error.WriteLine("[{0}] Failure shutting down socket: {1}",                                                typeof(AsyncSocketListener).FullName,                                                ex);                    }                    DrainSocket(connectedClient);                    connectedClient.Dispose();                }                _connectedClients.Clear();            }            _listener?.Dispose();            if (_receiveThread != null)            {                _receiveThread.Join();                _receiveThread = null;            }        }        public void Dispose()        {            Stop();            GC.SuppressFinalize(this);        }        private void StartListener(object state)        {            try            {                var listener = (Socket)state;                while (_started)                {                    _ = _acceptCallbackDone.Reset();                    _ = listener.BeginAccept(AcceptCallback, listener);                    _ = _acceptCallbackDone.WaitOne();                }            }            catch (Exception ex)            {                // On .NET framework when Thread throws an exception then unit tests                // were executed without any problem.                // On new .NET exceptions from Thread breaks unit tests session.                Console.Error.WriteLine("[{0}] Failure in StartListener: {1}",                    typeof(AsyncSocketListener).FullName,                    ex);            }        }        private void AcceptCallback(IAsyncResult ar)        {            // Signal the main thread to continue            _ = _acceptCallbackDone.Set();            // Get the socket that listens for inbound connections            var listener = (Socket)ar.AsyncState;            // Get the socket that handles the client request            Socket handler;            try            {                handler = listener.EndAccept(ar);            }            catch (SocketException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.EndAccept(...) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure accepting new connection: {1}",                        typeof(AsyncSocketListener).FullName,                        ex);                }                return;            }            catch (ObjectDisposedException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.EndAccept(IAsyncResult) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure accepting new connection: {1}",                        typeof(AsyncSocketListener).FullName,                        ex);                }                return;            }            // Signal new connection            SignalConnected(handler);            lock (_syncLock)            {                // Register client socket                _connectedClients.Add(handler);            }            var state = new SocketStateObject(handler);            try            {                _ = handler.BeginReceive(state.Buffer, 0, state.Buffer.Length, 0, ReadCallback, state);            }            catch (SocketException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.BeginReceive(...) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure receiving new data: {1}",                        typeof(AsyncSocketListener).FullName,                        ex);                }            }            catch (ObjectDisposedException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.BeginReceive(...) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure receiving new data: {1}",                                            typeof(AsyncSocketListener).FullName,                                            ex);                }            }        }        private void ReadCallback(IAsyncResult ar)        {            // Retrieve the state object and the handler socket            // from the asynchronous state object            var state = (SocketStateObject)ar.AsyncState;            var handler = state.Socket;            int bytesRead;            try            {                // Read data from the client socket.                bytesRead = handler.EndReceive(ar);            }            catch (SocketException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.EndReceive(...) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure receiving new data: {1}",                                            typeof(AsyncSocketListener).FullName,                                            ex);                }                return;            }            catch (ObjectDisposedException ex)            {                // The listener is stopped through a Dispose() call, which in turn causes                // Socket.EndReceive(...) to throw a SocketException or                // ObjectDisposedException                //                // Since we consider such an exception normal when the listener is being                // stopped, we only write a message to stderr if the listener is considered                // to be up and running                if (_started)                {                    Console.Error.WriteLine("[{0}] Failure receiving new data: {1}",                                            typeof(AsyncSocketListener).FullName,                                            ex);                }                return;            }            void ConnectionDisconnected()            {                SignalDisconnected(handler);                if (ShutdownRemoteCommunicationSocket)                {                    lock (_syncLock)                    {                        if (!_started)                        {                            return;                        }                        try                        {                            if (handler.Connected)                            {                                handler.Shutdown(SocketShutdown.Send);                            }                            handler.Close();                        }                        catch (SocketException ex) when (ex.SocketErrorCode == SocketError.ConnectionReset)                        {                            // On .NET 7 we got Socker Exception with ConnectionReset from Shutdown method                            // when the socket is disposed                        }                        catch (SocketException ex)                        {                            throw new Exception("Exception in ReadCallback: " + ex.SocketErrorCode + " " + _stackTrace, ex);                        }                        catch (Exception ex)                        {                            throw new Exception("Exception in ReadCallback: " + _stackTrace, ex);                        }                        _ = _connectedClients.Remove(handler);                    }                }            }            if (bytesRead > 0)            {                var bytesReceived = new byte[bytesRead];                Array.Copy(state.Buffer, bytesReceived, bytesRead);                SignalBytesReceived(bytesReceived, handler);                try                {                    _ = handler.BeginReceive(state.Buffer, 0, state.Buffer.Length, 0, ReadCallback, state);                }                catch (ObjectDisposedException)                {                    // TODO On .NET 7, sometimes we get ObjectDisposedException when _started but only on appveyor, locally it works                    ConnectionDisconnected();                }                catch (SocketException ex)                {                    if (!_started)                    {                        throw new Exception("BeginReceive while stopping!", ex);                    }                    throw new Exception("BeginReceive while started!: " + ex.SocketErrorCode + " " + _stackTrace, ex);                }            }            else            {                ConnectionDisconnected();            }        }        private void SignalBytesReceived(byte[] bytesReceived, Socket client)        {            BytesReceived?.Invoke(bytesReceived, client);        }        private void SignalConnected(Socket client)        {            Connected?.Invoke(client);        }        private void SignalDisconnected(Socket client)        {            Disconnected?.Invoke(client);        }        private static void DrainSocket(Socket socket)        {            var buffer = new byte[128];            try            {                while (true && socket.Connected)                {                    var bytesRead = socket.Receive(buffer);                    if (bytesRead == 0)                    {                        break;                    }                }            }            catch (SocketException ex)            {                Console.Error.WriteLine("[{0}] Failure draining socket ({1}): {2}",                                        typeof(AsyncSocketListener).FullName,                                        ex.SocketErrorCode.ToString("G"),                                        ex);            }        }        private class SocketStateObject        {            public Socket Socket { get; private set; }            public readonly byte[] Buffer = new byte[2048];            public SocketStateObject(Socket handler)            {                Socket = handler;            }        }    }}
 |