| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991 | #nullable enableusing System;using System.Collections.Generic;using System.Diagnostics;using System.Globalization;using System.IO;using System.Text;using System.Text.RegularExpressions;using System.Threading;using System.Threading.Tasks;using Renci.SshNet.Abstractions;using Renci.SshNet.Channels;using Renci.SshNet.Common;namespace Renci.SshNet{    /// <summary>    /// Contains operation for working with SSH Shell.    /// </summary>    public class ShellStream : Stream    {        private const int DefaultBufferSize = 1024;        private readonly ISession _session;        private readonly Encoding _encoding;        private readonly IChannelSession _channel;        private readonly byte[] _carriageReturnBytes;        private readonly byte[] _lineFeedBytes;        private readonly object _sync = new object();        private readonly byte[] _writeBuffer;        private readonly bool _noTerminal;        private int _writeLength; // The length of the data in _writeBuffer.        private byte[] _readBuffer;        private int _readHead; // The index from which the data starts in _readBuffer.        private int _readTail; // The index at which to add new data into _readBuffer.        private bool _disposed;        /// <summary>        /// Occurs when data was received.        /// </summary>        public event EventHandler<ShellDataEventArgs>? DataReceived;        /// <summary>        /// Occurs when an error occurred.        /// </summary>        public event EventHandler<ExceptionEventArgs>? ErrorOccurred;        /// <summary>        /// Occurs when the channel was closed.        /// </summary>        public event EventHandler<EventArgs>? Closed;        /// <summary>        /// Gets a value indicating whether data is available on the <see cref="ShellStream"/> to be read.        /// </summary>        /// <value>        /// <see langword="true"/> if data is available to be read; otherwise, <see langword="false"/>.        /// </value>        public bool DataAvailable        {            get            {                lock (_sync)                {                    AssertValid();                    return _readTail != _readHead;                }            }        }        [Conditional("DEBUG")]        private void AssertValid()        {            Debug.Assert(Monitor.IsEntered(_sync), $"Should be in lock on {nameof(_sync)}");            Debug.Assert(_readHead >= 0, $"{nameof(_readHead)} should be non-negative but is {_readHead.ToString(CultureInfo.InvariantCulture)}");            Debug.Assert(_readTail >= 0, $"{nameof(_readTail)} should be non-negative but is {_readTail.ToString(CultureInfo.InvariantCulture)}");            Debug.Assert(_readHead <= _readBuffer.Length, $"{nameof(_readHead)} should be <= {nameof(_readBuffer)}.Length but is {_readHead.ToString(CultureInfo.InvariantCulture)}");            Debug.Assert(_readTail <= _readBuffer.Length, $"{nameof(_readTail)} should be <= {nameof(_readBuffer)}.Length but is {_readTail.ToString(CultureInfo.InvariantCulture)}");            Debug.Assert(_readHead <= _readTail, $"Should have {nameof(_readHead)} <= {nameof(_readTail)} but have {_readHead.ToString(CultureInfo.InvariantCulture)} <= {_readTail.ToString(CultureInfo.InvariantCulture)}");        }        /// <summary>        /// Initializes a new instance of the <see cref="ShellStream"/> class.        /// </summary>        /// <param name="session">The SSH session.</param>        /// <param name="terminalName">The <c>TERM</c> environment variable.</param>        /// <param name="columns">The terminal width in columns.</param>        /// <param name="rows">The terminal width in rows.</param>        /// <param name="width">The terminal width in pixels.</param>        /// <param name="height">The terminal height in pixels.</param>        /// <param name="terminalModeValues">The terminal mode values.</param>        /// <param name="bufferSize">The size of the buffer.</param>        /// <exception cref="SshException">The channel could not be opened.</exception>        /// <exception cref="SshException">The pseudo-terminal request was not accepted by the server.</exception>        /// <exception cref="SshException">The request to start a shell was not accepted by the server.</exception>        internal ShellStream(ISession session, string terminalName, uint columns, uint rows, uint width, uint height, IDictionary<TerminalModes, uint> terminalModeValues, int bufferSize)               : this(session, bufferSize, noTerminal: false)        {            try            {                _channel.Open();                if (!_channel.SendPseudoTerminalRequest(terminalName, columns, rows, width, height, terminalModeValues))                {                    throw new SshException("The pseudo-terminal request was not accepted by the server. Consult the server log for more information.");                }                if (!_channel.SendShellRequest())                {                    throw new SshException("The request to start a shell was not accepted by the server. Consult the server log for more information.");                }            }            catch            {                Dispose();                throw;            }        }        /// <summary>        /// Initializes a new instance of the <see cref="ShellStream"/> class.        /// </summary>        /// <param name="session">The SSH session.</param>        /// <param name="bufferSize">The size of the buffer.</param>        /// <exception cref="SshException">The channel could not be opened.</exception>        /// <exception cref="SshException">The request to start a shell was not accepted by the server.</exception>        internal ShellStream(ISession session, int bufferSize)            : this(session, bufferSize, noTerminal: true)        {            try            {                _channel.Open();                if (!_channel.SendShellRequest())                {                    throw new SshException("The request to start a shell was not accepted by the server. Consult the server log for more information.");                }            }            catch            {                Dispose();                throw;            }        }        /// <summary>        /// Initializes a new instance of the <see cref="ShellStream"/> class.        /// </summary>        /// <param name="session">The SSH session.</param>        /// <param name="bufferSize">The size of the buffer.</param>        /// <param name="noTerminal">Disables pseudo terminal allocation or not.</param>        /// <exception cref="SshException">The channel could not be opened.</exception>        private ShellStream(ISession session, int bufferSize, bool noTerminal)        {            if (bufferSize == -1)            {                bufferSize = DefaultBufferSize;            }#if NET8_0_OR_GREATER            ArgumentOutOfRangeException.ThrowIfNegativeOrZero(bufferSize);#else            if (bufferSize <= 0)            {                throw new ArgumentOutOfRangeException(nameof(bufferSize));            }#endif            _encoding = session.ConnectionInfo.Encoding;            _session = session;            _carriageReturnBytes = _encoding.GetBytes("\r");            _lineFeedBytes = _encoding.GetBytes("\n");            _channel = _session.CreateChannelSession();            _channel.DataReceived += Channel_DataReceived;            _channel.Closed += Channel_Closed;            _session.Disconnected += Session_Disconnected;            _session.ErrorOccured += Session_ErrorOccured;            _readBuffer = new byte[bufferSize];            _writeBuffer = new byte[bufferSize];            _noTerminal = noTerminal;        }        /// <summary>        /// Gets a value indicating whether the current stream supports reading.        /// </summary>        /// <value>        /// <see langword="true"/>.        /// </value>        /// <remarks>        /// It is safe to read from <see cref="ShellStream"/> even after disposal.        /// </remarks>        public override bool CanRead        {            get { return true; }        }        /// <summary>        /// Gets a value indicating whether the current stream supports seeking.        /// </summary>        /// <value>        /// <see langword="false"/>.        /// </value>        public override bool CanSeek        {            get { return false; }        }        /// <summary>        /// Gets a value indicating whether the current stream supports writing.        /// </summary>        /// <value>        /// <see langword="true"/> if this stream has not been disposed and the underlying channel        /// is still open, otherwise <see langword="false"/>.        /// </value>        /// <remarks>        /// A value of <see langword="true"/> does not necessarily mean a write will succeed. It is possible        /// that the channel is closed and/or the stream is disposed by another thread between a call to        /// <see cref="CanWrite"/> and the call to write.        /// </remarks>        public override bool CanWrite        {            get { return !_disposed; }        }        /// <inheritdoc/>        public override void Flush()        {            ThrowIfDisposed();            Debug.Assert(_writeLength >= 0 && _writeLength <= _writeBuffer.Length);            if (_writeLength > 0)            {                _channel.SendData(_writeBuffer, 0, _writeLength);                _writeLength = 0;            }        }        /// <summary>        /// Gets the number of bytes currently available for reading.        /// </summary>        /// <value>A value representing the length of the stream in bytes.</value>        public override long Length        {            get            {                lock (_sync)                {                    AssertValid();                    return _readTail - _readHead;                }            }        }        /// <summary>        /// This property always returns 0, and throws <see cref="NotSupportedException"/>        /// when calling the setter.        /// </summary>        /// <value>        /// 0.        /// </value>        /// <exception cref="NotSupportedException">The setter is called.</exception>#pragma warning disable SA1623 // The property's documentation should begin with 'Gets or sets'        public override long Position#pragma warning restore SA1623 // The property's documentation should begin with 'Gets or sets'        {            get { return 0; }            set { throw new NotSupportedException(); }        }        /// <summary>        /// This method always throws <see cref="NotSupportedException"/>.        /// </summary>        /// <param name="offset">A byte offset relative to the <paramref name="origin"/> parameter.</param>        /// <param name="origin">A value of type <see cref="SeekOrigin"/> indicating the reference point used to obtain the new position.</param>        /// <returns>Never.</returns>        /// <exception cref="NotSupportedException">Always.</exception>        public override long Seek(long offset, SeekOrigin origin)        {            throw new NotSupportedException();        }        /// <summary>        /// This method always throws <see cref="NotSupportedException"/>.        /// </summary>        /// <param name="value">The desired length of the current stream in bytes.</param>        /// <exception cref="NotSupportedException">Always.</exception>        public override void SetLength(long value)        {            throw new NotSupportedException();        }        /// <summary>        /// Expects the specified expression and performs action when one is found.        /// </summary>        /// <param name="expectActions">The expected expressions and actions to perform.</param>        public void Expect(params ExpectAction[] expectActions)        {            Expect(Timeout.InfiniteTimeSpan, expectActions);        }        /// <summary>        /// Expects the specified expression and performs action when one is found.        /// </summary>        /// <param name="timeout">Time to wait for input. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="expectActions">The expected expressions and actions to perform, if the specified time elapsed and expected condition have not met, that method will exit without executing any action.</param>        /// <remarks>        /// If a TimeSpan representing -1 millisecond is specified for the <paramref name="timeout"/> parameter,        /// this method blocks indefinitely until either the regex matches the data in the buffer, or the stream        /// is closed (via disposal or via the underlying channel closing).        /// </remarks>        public void Expect(TimeSpan timeout, params ExpectAction[] expectActions)        {            _ = ExpectRegex(timeout, lookback: -1, expectActions);        }        /// <summary>        /// Expects the specified expression and performs action when one is found.        /// </summary>        /// <param name="timeout">Time to wait for input. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="lookback">The amount of data to search through from the most recent data in the buffer, or -1 to always search the entire buffer.</param>        /// <param name="expectActions">The expected expressions and actions to perform, if the specified time elapsed and expected condition have not met, that method will exit without executing any action.</param>        /// <remarks>        /// <para>        /// If a TimeSpan representing -1 millisecond is specified for the <paramref name="timeout"/> parameter,        /// this method blocks indefinitely until either the regex matches the data in the buffer, or the stream        /// is closed (via disposal or via the underlying channel closing).        /// </para>        /// <para>        /// Use the <paramref name="lookback"/> parameter to constrain the search space to a fixed-size rolling window at the end of the buffer.        /// This can reduce the amount of work done in cases where lots of output from the shell is expected to be received before the matching expression is found.        /// </para>        /// <para>        /// Note: in situations with high volumes of data and a small value for <paramref name="lookback"/>, some data may not be searched through.        /// It is recommended to set <paramref name="lookback"/> to a large enough value to be able to search all data as it comes in,        /// but which still places a limit on the amount of work needed.        /// </para>        /// </remarks>        public void Expect(TimeSpan timeout, int lookback, params ExpectAction[] expectActions)        {            _ = ExpectRegex(timeout, lookback, expectActions);        }        /// <summary>        /// Expects the expression specified by text.        /// </summary>        /// <param name="text">The text to expect.</param>        /// <returns>        /// The text available in the shell up to and including the expected text,        /// or <see langword="null"/> if the the stream is closed without a match.        /// </returns>        public string? Expect(string text)        {            return Expect(text, Timeout.InfiniteTimeSpan);        }        /// <summary>        /// Expects the expression specified by text.        /// </summary>        /// <param name="text">The text to expect.</param>        /// <param name="timeout">Time to wait for input. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="lookback">The amount of data to search through from the most recent data in the buffer, or -1 to always search the entire buffer.</param>        /// <returns>        /// The text available in the shell up to and including the expected expression,        /// or <see langword="null"/> if the specified time has elapsed or the stream is closed        /// without a match.        /// </returns>        /// <remarks><inheritdoc cref="Expect(TimeSpan, int, ExpectAction[])"/></remarks>        public string? Expect(string text, TimeSpan timeout, int lookback = -1)        {            ValidateTimeout(timeout);            ValidateLookback(lookback);            var timeoutTime = DateTime.Now.Add(timeout);            var expectBytes = _encoding.GetBytes(text);            lock (_sync)            {                while (true)                {                    AssertValid();                    var searchHead = lookback == -1                        ? _readHead                        : Math.Max(_readTail - lookback, _readHead);                    Debug.Assert(_readHead <= searchHead && searchHead <= _readTail);                    var indexOfMatch = _readBuffer.AsSpan(searchHead, _readTail - searchHead).IndexOf(expectBytes);                    if (indexOfMatch >= 0)                    {                        var returnText = _encoding.GetString(_readBuffer, _readHead, searchHead - _readHead + indexOfMatch + expectBytes.Length);                        _readHead = searchHead + indexOfMatch + expectBytes.Length;                        AssertValid();                        return returnText;                    }                    if (_disposed)                    {                        return null;                    }                    if (timeout == Timeout.InfiniteTimeSpan)                    {                        _ = Monitor.Wait(_sync);                    }                    else                    {                        var waitTimeout = timeoutTime - DateTime.Now;                        if (waitTimeout < TimeSpan.Zero || !Monitor.Wait(_sync, waitTimeout))                        {                            return null;                        }                    }                }            }        }        /// <summary>        /// Expects the expression specified by regular expression.        /// </summary>        /// <param name="regex">The regular expression to expect.</param>        /// <returns>        /// The text available in the shell up to and including the expected expression,        /// or <see langword="null"/> if the stream is closed without a match.        /// </returns>        public string? Expect(Regex regex)        {            return Expect(regex, Timeout.InfiniteTimeSpan);        }        /// <summary>        /// Expects the expression specified by regular expression.        /// </summary>        /// <param name="regex">The regular expression to expect.</param>        /// <param name="timeout">Time to wait for input. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="lookback">The amount of data to search through from the most recent data in the buffer, or -1 to always search the entire buffer.</param>        /// <returns>        /// The text available in the shell up to and including the expected expression,        /// or <see langword="null"/> if the specified timeout has elapsed or the stream        /// is closed without a match.        /// </returns>        /// <remarks>        /// <inheritdoc cref="Expect(TimeSpan, int, ExpectAction[])"/>        /// </remarks>        public string? Expect(Regex regex, TimeSpan timeout, int lookback = -1)        {            return ExpectRegex(timeout, lookback, [new ExpectAction(regex, s => { })]);        }        private string? ExpectRegex(TimeSpan timeout, int lookback, ExpectAction[] expectActions)        {            ValidateTimeout(timeout);            ValidateLookback(lookback);            var timeoutTime = DateTime.Now.Add(timeout);            lock (_sync)            {                while (true)                {                    AssertValid();                    var bufferText = _encoding.GetString(_readBuffer, _readHead, _readTail - _readHead);                    var searchStart = lookback == -1                        ? 0                        : Math.Max(bufferText.Length - lookback, 0);                    foreach (var expectAction in expectActions)                    {#if NET7_0_OR_GREATER                        var matchEnumerator = expectAction.Expect.EnumerateMatches(bufferText.AsSpan(searchStart));                        if (matchEnumerator.MoveNext())                        {                            var match = matchEnumerator.Current;                            var returnText = bufferText.Substring(0, searchStart + match.Index + match.Length);#else                        var match = expectAction.Expect.Match(bufferText, searchStart);                        if (match.Success)                        {                            var returnText = bufferText.Substring(0, match.Index + match.Length);#endif                            _readHead += _encoding.GetByteCount(returnText);                            AssertValid();                            expectAction.Action(returnText);                            return returnText;                        }                    }                    if (_disposed)                    {                        return null;                    }                    if (timeout == Timeout.InfiniteTimeSpan)                    {                        Monitor.Wait(_sync);                    }                    else                    {                        var waitTimeout = timeoutTime - DateTime.Now;                        if (waitTimeout < TimeSpan.Zero || !Monitor.Wait(_sync, waitTimeout))                        {                            return null;                        }                    }                }            }        }        /// <summary>        /// Begins the expect.        /// </summary>        /// <param name="expectActions">The expect actions.</param>        /// <returns>        /// An <see cref="IAsyncResult" /> that references the asynchronous operation.        /// </returns>        public IAsyncResult BeginExpect(params ExpectAction[] expectActions)        {            return BeginExpect(Timeout.InfiniteTimeSpan, callback: null, state: null, expectActions);        }        /// <summary>        /// Begins the expect.        /// </summary>        /// <param name="callback">The callback.</param>        /// <param name="expectActions">The expect actions.</param>        /// <returns>        /// An <see cref="IAsyncResult" /> that references the asynchronous operation.        /// </returns>        public IAsyncResult BeginExpect(AsyncCallback? callback, params ExpectAction[] expectActions)        {            return BeginExpect(Timeout.InfiniteTimeSpan, callback, state: null, expectActions);        }        /// <summary>        /// Begins the expect.        /// </summary>        /// <param name="callback">The callback.</param>        /// <param name="state">The state.</param>        /// <param name="expectActions">The expect actions.</param>        /// <returns>        /// An <see cref="IAsyncResult" /> that references the asynchronous operation.        /// </returns>        public IAsyncResult BeginExpect(AsyncCallback? callback, object? state, params ExpectAction[] expectActions)        {            return BeginExpect(Timeout.InfiniteTimeSpan, callback, state, expectActions);        }        /// <summary>        /// Begins the expect.        /// </summary>        /// <param name="timeout">The timeout. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="callback">The callback.</param>        /// <param name="state">The state.</param>        /// <param name="expectActions">The expect actions.</param>        /// <returns>        /// An <see cref="IAsyncResult" /> that references the asynchronous operation.        /// </returns>        public IAsyncResult BeginExpect(TimeSpan timeout, AsyncCallback? callback, object? state, params ExpectAction[] expectActions)        {            return BeginExpect(timeout, lookback: -1, callback, state, expectActions);        }        /// <summary>        /// Begins the expect.        /// </summary>        /// <param name="timeout">The timeout. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <param name="lookback">The amount of data to search through from the most recent data in the buffer, or -1 to always search the entire buffer.</param>        /// <param name="callback">The callback.</param>        /// <param name="state">The state.</param>        /// <param name="expectActions">The expect actions.</param>        /// <returns>        /// An <see cref="IAsyncResult" /> that references the asynchronous operation.        /// </returns>        public IAsyncResult BeginExpect(TimeSpan timeout, int lookback, AsyncCallback? callback, object? state, params ExpectAction[] expectActions)        {            return TaskToAsyncResult.Begin(Task.Run(() => ExpectRegex(timeout, lookback, expectActions)), callback, state);        }        /// <summary>        /// Ends the execute.        /// </summary>        /// <param name="asyncResult">The async result.</param>        /// <returns>        /// The text available in the shell up to and including the expected expression.        /// </returns>        public string? EndExpect(IAsyncResult asyncResult)        {            return TaskToAsyncResult.End<string?>(asyncResult);        }        /// <summary>        /// Reads the next line from the shell. If a line is not available it will block and wait for a new line.        /// </summary>        /// <returns>        /// The line read from the shell.        /// </returns>        /// <remarks>        /// <para>        /// This method blocks indefinitely until either a line is available in the buffer, or the stream is closed        /// (via disposal or via the underlying channel closing).        /// </para>        /// <para>        /// When the stream is closed and there are no more newlines in the buffer, this method returns the remaining data        /// (if any) and then <see langword="null"/> indicating that no more data is in the buffer.        /// </para>        /// </remarks>        public string? ReadLine()        {            return ReadLine(Timeout.InfiniteTimeSpan);        }        /// <summary>        /// Reads a line from the shell. If line is not available it will block the execution and will wait for new line.        /// </summary>        /// <param name="timeout">Time to wait for input. Must non-negative or equal to -1 millisecond (for infinite timeout).</param>        /// <returns>        /// The line read from the shell, or <see langword="null"/> when no input is received for the specified timeout.        /// </returns>        /// <remarks>        /// <para>        /// If a TimeSpan representing -1 millisecond is specified for the <paramref name="timeout"/> parameter, this method        /// blocks indefinitely until either a line is available in the buffer, or the stream is closed (via disposal or via        /// the underlying channel closing).        /// </para>        /// <para>        /// When the stream is closed and there are no more newlines in the buffer, this method returns the remaining data        /// (if any) and then <see langword="null"/> indicating that no more data is in the buffer.        /// </para>        /// </remarks>        public string? ReadLine(TimeSpan timeout)        {            ValidateTimeout(timeout);            var timeoutTime = DateTime.Now.Add(timeout);            lock (_sync)            {                while (true)                {                    AssertValid();                    var indexOfCr = _readBuffer.AsSpan(_readHead, _readTail - _readHead).IndexOf(_carriageReturnBytes);                    if (indexOfCr >= 0)                    {                        // We have found \r. We only need to search for \n up to and just after the \r                        // (in order to consume \r\n if we can).                        var indexOfLf = indexOfCr + _carriageReturnBytes.Length + _lineFeedBytes.Length <= _readTail - _readHead                            ? _readBuffer.AsSpan(_readHead, indexOfCr + _carriageReturnBytes.Length + _lineFeedBytes.Length).IndexOf(_lineFeedBytes)                            : _readBuffer.AsSpan(_readHead, indexOfCr).IndexOf(_lineFeedBytes);                        if (indexOfLf >= 0 && indexOfLf < indexOfCr)                        {                            // If there is \n before the \r, then return up to the \n                            var returnText = _encoding.GetString(_readBuffer, _readHead, indexOfLf);                            _readHead += indexOfLf + _lineFeedBytes.Length;                            AssertValid();                            return returnText;                        }                        else if (indexOfLf == indexOfCr + _carriageReturnBytes.Length)                        {                            // If we have \r\n, then consume both                            var returnText = _encoding.GetString(_readBuffer, _readHead, indexOfCr);                            _readHead += indexOfCr + _carriageReturnBytes.Length + _lineFeedBytes.Length;                            AssertValid();                            return returnText;                        }                        else                        {                            // Return up to the \r                            var returnText = _encoding.GetString(_readBuffer, _readHead, indexOfCr);                            _readHead += indexOfCr + _carriageReturnBytes.Length;                            AssertValid();                            return returnText;                        }                    }                    else                    {                        // There is no \r. What about \n?                        var indexOfLf = _readBuffer.AsSpan(_readHead, _readTail - _readHead).IndexOf(_lineFeedBytes);                        if (indexOfLf >= 0)                        {                            var returnText = _encoding.GetString(_readBuffer, _readHead, indexOfLf);                            _readHead += indexOfLf + _lineFeedBytes.Length;                            AssertValid();                            return returnText;                        }                    }                    if (_disposed)                    {                        var lastLine = _readHead == _readTail                            ? null                            : _encoding.GetString(_readBuffer, _readHead, _readTail - _readHead);                        _readHead = _readTail = 0;                        return lastLine;                    }                    if (timeout == Timeout.InfiniteTimeSpan)                    {                        _ = Monitor.Wait(_sync);                    }                    else                    {                        var waitTimeout = timeoutTime - DateTime.Now;                        if (waitTimeout < TimeSpan.Zero || !Monitor.Wait(_sync, waitTimeout))                        {                            return null;                        }                    }                }            }        }        private static void ValidateTimeout(TimeSpan timeout)        {            if (timeout < TimeSpan.Zero && timeout != Timeout.InfiniteTimeSpan)            {                throw new ArgumentOutOfRangeException(nameof(timeout), "Value must be non-negative or equal to -1 millisecond (for infinite timeout)");            }        }        private static void ValidateLookback(int lookback)        {            if (lookback is <= 0 and not -1)            {                throw new ArgumentOutOfRangeException(nameof(lookback), "Value must be positive or equal to -1 (for no window)");            }        }        private void ThrowIfDisposed()        {#if NET7_0_OR_GREATER            ObjectDisposedException.ThrowIf(_disposed, this);#else            if (_disposed)            {                throw new ObjectDisposedException(GetType().FullName);            }#endif // NET7_0_OR_GREATER        }        /// <summary>        /// Reads all of the text currently available in the shell.        /// </summary>        /// <returns>        /// The text available in the shell.        /// </returns>        public string Read()        {            lock (_sync)            {                AssertValid();                var text = _encoding.GetString(_readBuffer, _readHead, _readTail - _readHead);                _readHead = _readTail = 0;                return text;            }        }        /// <inheritdoc/>        public override int Read(byte[] buffer, int offset, int count)        {            lock (_sync)            {                while (_readHead == _readTail && !_disposed)                {                    _ = Monitor.Wait(_sync);                }                AssertValid();                var bytesRead = Math.Min(count, _readTail - _readHead);                Buffer.BlockCopy(_readBuffer, _readHead, buffer, offset, bytesRead);                _readHead += bytesRead;                AssertValid();                return bytesRead;            }        }        /// <summary>        /// Writes the specified text to the shell.        /// </summary>        /// <param name="text">The text to be written to the shell.</param>        /// <remarks>        /// If <paramref name="text"/> is <see langword="null"/>, nothing is written.        /// Otherwise, <see cref="Flush"/> is called after writing the data to the buffer.        /// </remarks>        /// <exception cref="ObjectDisposedException">The stream is closed.</exception>        public void Write(string? text)        {            if (text is null)            {                return;            }            var data = _encoding.GetBytes(text);            Write(data, 0, data.Length);            Flush();        }        /// <inheritdoc/>        public override void Write(byte[] buffer, int offset, int count)        {            ThrowIfDisposed();            while (count > 0)            {                if (_writeLength == _writeBuffer.Length)                {                    Flush();                }                var bytesToCopy = Math.Min(count, _writeBuffer.Length - _writeLength);                Buffer.BlockCopy(buffer, offset, _writeBuffer, _writeLength, bytesToCopy);                offset += bytesToCopy;                count -= bytesToCopy;                _writeLength += bytesToCopy;                Debug.Assert(_writeLength >= 0 && _writeLength <= _writeBuffer.Length);            }        }        /// <summary>        /// Writes the line to the shell.        /// </summary>        /// <param name="line">The line to be written to the shell.</param>        /// <remarks>        /// If <paramref name="line"/> is <see langword="null"/>, only the line terminator is written.        /// <see cref="Flush"/> is called once the data is written.        /// </remarks>        /// <exception cref="ObjectDisposedException">The stream is closed.</exception>        public void WriteLine(string line)        {            // By default, the terminal driver translates carriage return to line feed on input.            // See option ICRLF at https://www.man7.org/linux/man-pages/man3/termios.3.html.            Write(line + (_noTerminal ? "\n" : "\r"));        }        /// <inheritdoc/>        protected override void Dispose(bool disposing)        {            if (!disposing)            {                base.Dispose(disposing);                return;            }            lock (_sync)            {                if (_disposed)                {                    return;                }                _disposed = true;                // Do not dispose _session (we don't own it)                _session.Disconnected -= Session_Disconnected;                _session.ErrorOccured -= Session_ErrorOccured;                // But we do own _channel                _channel.DataReceived -= Channel_DataReceived;                _channel.Closed -= Channel_Closed;                _channel.Dispose();                Monitor.PulseAll(_sync);            }            base.Dispose(disposing);        }        private void Session_ErrorOccured(object? sender, ExceptionEventArgs e)        {            ErrorOccurred?.Invoke(this, e);        }        private void Session_Disconnected(object? sender, EventArgs e)        {            Dispose();        }        private void Channel_Closed(object? sender, ChannelEventArgs e)        {            Dispose();            if (Closed != null)            {                // Handle event on different thread                ThreadAbstraction.ExecuteThread(() => Closed?.Invoke(this, EventArgs.Empty));            }        }        private void Channel_DataReceived(object? sender, ChannelDataEventArgs e)        {            lock (_sync)            {                AssertValid();                // Ensure sufficient buffer space and copy the new data in.                if (_readBuffer.Length - _readTail >= e.Data.Length)                {                    // If there is enough space after _tail for the new data,                    // then copy the data there.                    Buffer.BlockCopy(e.Data, 0, _readBuffer, _readTail, e.Data.Length);                    _readTail += e.Data.Length;                }                else                {                    // We can't fit the new data after _tail.                    var newLength = _readTail - _readHead + e.Data.Length;                    if (newLength <= _readBuffer.Length)                    {                        // If there is sufficient space at the start of the buffer,                        // then move the current data to the start of the buffer.                        Buffer.BlockCopy(_readBuffer, _readHead, _readBuffer, 0, _readTail - _readHead);                    }                    else                    {                        // Otherwise, we're gonna need a bigger buffer.                        var newBuffer = new byte[Math.Max(newLength, _readBuffer.Length * 2)];                        Buffer.BlockCopy(_readBuffer, _readHead, newBuffer, 0, _readTail - _readHead);                        _readBuffer = newBuffer;                    }                    // Copy the new data into the freed-up space.                    Buffer.BlockCopy(e.Data, 0, _readBuffer, _readTail - _readHead, e.Data.Length);                    _readHead = 0;                    _readTail = newLength;                }                AssertValid();                Monitor.PulseAll(_sync);            }            DataReceived?.Invoke(this, new ShellDataEventArgs(e.Data));        }    }}
 |