| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319 | using System;using System.Collections.Generic;using System.Diagnostics;using System.Globalization;using System.Net;using System.Net.Sockets;using System.Text;using Renci.SshNet.Abstractions;using Renci.SshNet.Messages;namespace Renci.SshNet.Common{    /// <summary>    /// Collection of different extension methods.    /// </summary>    internal static partial class Extensions    {        internal static byte[] ToArray(this ServiceName serviceName)        {            switch (serviceName)            {                case ServiceName.UserAuthentication:                    return SshData.Ascii.GetBytes("ssh-userauth");                case ServiceName.Connection:                    return SshData.Ascii.GetBytes("ssh-connection");                default:                    throw new NotSupportedException(string.Format("Service name '{0}' is not supported.", serviceName));            }        }        internal static ServiceName ToServiceName(this byte[] data)        {            var sshServiceName = SshData.Ascii.GetString(data, 0, data.Length);            switch (sshServiceName)            {                case "ssh-userauth":                    return ServiceName.UserAuthentication;                case "ssh-connection":                    return ServiceName.Connection;                default:                    throw new NotSupportedException(string.Format("Service name '{0}' is not supported.", sshServiceName));            }        }        internal static BigInteger ToBigInteger(this byte[] data)        {            var reversed = new byte[data.Length];            Buffer.BlockCopy(data, 0, reversed, 0, data.Length);            return new BigInteger(reversed.Reverse());        }        /// <summary>        /// Initializes a new instance of the <see cref="BigInteger"/> structure using the SSH BigNum2 Format.        /// </summary>        public static BigInteger ToBigInteger2(this byte[] data)        {            if ((data[0] & (1 << 7)) != 0)            {                var buf = new byte[data.Length + 1];                Buffer.BlockCopy(data, 0, buf, 1, data.Length);                data = buf;            }            return data.ToBigInteger();        }        /// <summary>        /// Reverses the sequence of the elements in the entire one-dimensional <see cref="Array"/>.        /// </summary>        /// <param name="array">The one-dimensional <see cref="Array"/> to reverse.</param>        /// <returns>        /// The <see cref="Array"/> with its elements reversed.        /// </returns>        internal static T[] Reverse<T>(this T[] array)        {            Array.Reverse(array);            return array;        }        /// <summary>        /// Prints out the specified bytes.        /// </summary>        /// <param name="bytes">The bytes.</param>        internal static void DebugPrint(this IEnumerable<byte> bytes)        {            var sb = new StringBuilder();            foreach (var b in bytes)            {                _ = sb.AppendFormat(CultureInfo.CurrentCulture, "0x{0:x2}, ", b);            }            Debug.WriteLine(sb.ToString());        }        internal static void ValidatePort(this uint value, string argument)        {            if (value > IPEndPoint.MaxPort)            {                throw new ArgumentOutOfRangeException(argument,                                                      string.Format(CultureInfo.InvariantCulture, "Specified value cannot be greater than {0}.", IPEndPoint.MaxPort));            }        }        internal static void ValidatePort(this int value, string argument)        {            if (value < IPEndPoint.MinPort)            {                throw new ArgumentOutOfRangeException(argument, string.Format(CultureInfo.InvariantCulture, "Specified value cannot be less than {0}.", IPEndPoint.MinPort));            }            if (value > IPEndPoint.MaxPort)            {                throw new ArgumentOutOfRangeException(argument, string.Format(CultureInfo.InvariantCulture, "Specified value cannot be greater than {0}.", IPEndPoint.MaxPort));            }        }        /// <summary>        /// Returns a specified number of contiguous bytes from a given offset.        /// </summary>        /// <param name="value">The array to return a number of bytes from.</param>        /// <param name="offset">The zero-based offset in <paramref name="value"/> at which to begin taking bytes.</param>        /// <param name="count">The number of bytes to take from <paramref name="value"/>.</param>        /// <returns>        /// A <see cref="byte"/> array that contains the specified number of bytes at the specified offset        /// of the input array.        /// </returns>        /// <exception cref="ArgumentNullException"><paramref name="value"/> is <see langword="null"/>.</exception>        /// <remarks>        /// When <paramref name="offset"/> is zero and <paramref name="count"/> equals the length of <paramref name="value"/>,        /// then <paramref name="value"/> is returned.        /// </remarks>        public static byte[] Take(this byte[] value, int offset, int count)        {            if (value is null)            {                throw new ArgumentNullException(nameof(value));            }            if (count == 0)            {                return Array.Empty<byte>();            }            if (offset == 0 && value.Length == count)            {                return value;            }            var taken = new byte[count];            Buffer.BlockCopy(value, offset, taken, 0, count);            return taken;        }        /// <summary>        /// Returns a specified number of contiguous bytes from the start of the specified byte array.        /// </summary>        /// <param name="value">The array to return a number of bytes from.</param>        /// <param name="count">The number of bytes to take from <paramref name="value"/>.</param>        /// <returns>        /// A <see cref="byte"/> array that contains the specified number of bytes at the start of the input array.        /// </returns>        /// <exception cref="ArgumentNullException"><paramref name="value"/> is <see langword="null"/>.</exception>        /// <remarks>        /// When <paramref name="count"/> equals the length of <paramref name="value"/>, then <paramref name="value"/>        /// is returned.        /// </remarks>        public static byte[] Take(this byte[] value, int count)        {            if (value is null)            {                throw new ArgumentNullException(nameof(value));            }            if (count == 0)            {                return Array.Empty<byte>();            }            if (value.Length == count)            {                return value;            }            var taken = new byte[count];            Buffer.BlockCopy(value, 0, taken, 0, count);            return taken;        }        public static bool IsEqualTo(this byte[] left, byte[] right)        {            if (left is null)            {                throw new ArgumentNullException(nameof(left));            }            if (right is null)            {                throw new ArgumentNullException(nameof(right));            }            if (left == right)            {                return true;            }#if NETSTANDARD2_1_OR_GREATER || NET6_0_OR_GREATER            return left.AsSpan().SequenceEqual(right);#else            if (left.Length != right.Length)            {                return false;            }            for (var i = 0; i < left.Length; i++)            {                if (left[i] != right[i])                {                    return false;                }            }            return true;#endif        }        /// <summary>        /// Trims the leading zero from a byte array.        /// </summary>        /// <param name="value">The value.</param>        /// <returns>        /// <paramref name="value"/> without leading zeros.        /// </returns>        public static byte[] TrimLeadingZeros(this byte[] value)        {            if (value is null)            {                throw new ArgumentNullException(nameof(value));            }            for (var i = 0; i < value.Length; i++)            {                if (value[i] == 0)                {                    continue;                }                // if the first byte is non-zero, then we return the byte array as is                if (i == 0)                {                    return value;                }                var remainingBytes = value.Length - i;                var cleaned = new byte[remainingBytes];                Buffer.BlockCopy(value, i, cleaned, 0, remainingBytes);                return cleaned;            }            return value;        }        /// <summary>        /// Pads with leading zeros if needed.        /// </summary>        /// <param name="data">The data.</param>        /// <param name="length">The length to pad to.</param>        public static byte[] Pad(this byte[] data, int length)        {            if (length <= data.Length)            {                return data;            }            var newData = new byte[length];            Buffer.BlockCopy(data, 0, newData, newData.Length - data.Length, data.Length);            return newData;        }        public static byte[] Concat(this byte[] first, byte[] second)        {            if (first is null || first.Length == 0)            {                return second;            }            if (second is null || second.Length == 0)            {                return first;            }            var concat = new byte[first.Length + second.Length];            Buffer.BlockCopy(first, 0, concat, 0, first.Length);            Buffer.BlockCopy(second, 0, concat, first.Length, second.Length);            return concat;        }        internal static bool CanRead(this Socket socket)        {            return SocketAbstraction.CanRead(socket);        }        internal static bool CanWrite(this Socket socket)        {            return SocketAbstraction.CanWrite(socket);        }        internal static bool IsConnected(this Socket socket)        {            if (socket is null)            {                return false;            }            return socket.Connected;        }    }}
 |