| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734 | using System;using System.Collections.Generic;using System.Diagnostics;using System.Globalization;using System.IO;using System.Security.Cryptography;using System.Text;using System.Text.RegularExpressions;using Renci.SshNet.Abstractions;using Renci.SshNet.Common;using Renci.SshNet.Security;using Renci.SshNet.Security.Cryptography;using Renci.SshNet.Security.Cryptography.Ciphers;using Renci.SshNet.Security.Cryptography.Ciphers.Modes;using Renci.SshNet.Security.Cryptography.Ciphers.Paddings;namespace Renci.SshNet{    /// <summary>    /// Represents private key information.    /// </summary>    /// <example>    ///     <code source="..\..\src\Renci.SshNet.Tests\Data\Key.RSA.txt" language="Text" title="Private RSA key example" />    /// </example>    /// <remarks>    /// <para>    /// The following private keys are supported:    /// <list type="bullet">    ///     <item>    ///         <description>RSA in OpenSSL PEM, ssh.com and OpenSSH key format</description>    ///     </item>    ///     <item>    ///         <description>DSA in OpenSSL PEM and ssh.com format</description>    ///     </item>    ///     <item>    ///         <description>ECDSA 256/384/521 in OpenSSL PEM and OpenSSH key format</description>    ///     </item>    ///     <item>    ///         <description>ED25519 in OpenSSH key format</description>    ///     </item>    /// </list>    /// </para>    /// <para>    /// The following encryption algorithms are supported:    /// <list type="bullet">    ///     <item>    ///         <description>DES-EDE3-CBC</description>    ///     </item>    ///     <item>    ///         <description>DES-EDE3-CFB</description>    ///     </item>    ///     <item>    ///         <description>DES-CBC</description>    ///     </item>    ///     <item>    ///         <description>AES-128-CBC</description>    ///     </item>    ///     <item>    ///         <description>AES-192-CBC</description>    ///     </item>    ///     <item>    ///         <description>AES-256-CBC</description>    ///     </item>    /// </list>    /// </para>    /// </remarks>    public class PrivateKeyFile : IPrivateKeySource, IDisposable    {        private static readonly Regex PrivateKeyRegex = new Regex(@"^-+ *BEGIN (?<keyName>\w+( \w+)*) PRIVATE KEY *-+\r?\n((Proc-Type: 4,ENCRYPTED\r?\nDEK-Info: (?<cipherName>[A-Z0-9-]+),(?<salt>[A-F0-9]+)\r?\n\r?\n)|(Comment: ""?[^\r\n]*""?\r?\n))?(?<data>([a-zA-Z0-9/+=]{1,80}\r?\n)+)-+ *END \k<keyName> PRIVATE KEY *-+",                                                                  RegexOptions.Compiled | RegexOptions.Multiline | RegexOptions.ExplicitCapture);        private readonly List<HostAlgorithm> _hostAlgorithms = new List<HostAlgorithm>();        private Key _key;        private bool _isDisposed;        /// <summary>        /// Gets the supported host algorithms for this key file.        /// </summary>        public IReadOnlyCollection<HostAlgorithm> HostKeyAlgorithms        {            get            {                return _hostAlgorithms;            }        }        /// <summary>        /// Gets the key.        /// </summary>        public Key Key        {            get            {                return _key;            }        }        /// <summary>        /// Initializes a new instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        /// <param name="key">The key.</param>        public PrivateKeyFile(Key key)        {            _key = key;            _hostAlgorithms.Add(new KeyHostAlgorithm(key.ToString(), key));        }        /// <summary>        /// Initializes a new instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        /// <param name="privateKey">The private key.</param>        public PrivateKeyFile(Stream privateKey)        {            Open(privateKey, passPhrase: null);            Debug.Assert(_hostAlgorithms.Count > 0, $"{nameof(HostKeyAlgorithms)} is not set.");        }        /// <summary>        /// Initializes a new instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        /// <param name="fileName">Name of the file.</param>        /// <exception cref="ArgumentNullException"><paramref name="fileName"/> is <see langword="null"/> or empty.</exception>        /// <remarks>        /// This method calls <see cref="File.Open(string, FileMode)"/> internally, this method does not catch exceptions from <see cref="File.Open(string, FileMode)"/>.        /// </remarks>        public PrivateKeyFile(string fileName)            : this(fileName, passPhrase: null)        {        }        /// <summary>        /// Initializes a new instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        /// <param name="fileName">Name of the file.</param>        /// <param name="passPhrase">The pass phrase.</param>        /// <exception cref="ArgumentNullException"><paramref name="fileName"/> is <see langword="null"/> or empty, or <paramref name="passPhrase"/> is <see langword="null"/>.</exception>        /// <remarks>        /// This method calls <see cref="File.Open(string, FileMode)"/> internally, this method does not catch exceptions from <see cref="File.Open(string, FileMode)"/>.        /// </remarks>        public PrivateKeyFile(string fileName, string passPhrase)        {            if (string.IsNullOrEmpty(fileName))            {                throw new ArgumentNullException(nameof(fileName));            }            using (var keyFile = File.Open(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))            {                Open(keyFile, passPhrase);            }            Debug.Assert(_hostAlgorithms.Count > 0, $"{nameof(HostKeyAlgorithms)} is not set.");        }        /// <summary>        /// Initializes a new instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        /// <param name="privateKey">The private key.</param>        /// <param name="passPhrase">The pass phrase.</param>        /// <exception cref="ArgumentNullException"><paramref name="privateKey"/> or <paramref name="passPhrase"/> is <see langword="null"/>.</exception>        public PrivateKeyFile(Stream privateKey, string passPhrase)        {            Open(privateKey, passPhrase);            Debug.Assert(_hostAlgorithms.Count > 0, $"{nameof(HostKeyAlgorithms)} is not set.");        }        /// <summary>        /// Opens the specified private key.        /// </summary>        /// <param name="privateKey">The private key.</param>        /// <param name="passPhrase">The pass phrase.</param>        private void Open(Stream privateKey, string passPhrase)        {            if (privateKey is null)            {                throw new ArgumentNullException(nameof(privateKey));            }            Match privateKeyMatch;            using (var sr = new StreamReader(privateKey))            {                var text = sr.ReadToEnd();                privateKeyMatch = PrivateKeyRegex.Match(text);            }            if (!privateKeyMatch.Success)            {                throw new SshException("Invalid private key file.");            }            var keyName = privateKeyMatch.Result("${keyName}");            var cipherName = privateKeyMatch.Result("${cipherName}");            var salt = privateKeyMatch.Result("${salt}");            var data = privateKeyMatch.Result("${data}");            var binaryData = Convert.FromBase64String(data);            byte[] decryptedData;            if (!string.IsNullOrEmpty(cipherName) && !string.IsNullOrEmpty(salt))            {                if (string.IsNullOrEmpty(passPhrase))                {                    throw new SshPassPhraseNullOrEmptyException("Private key is encrypted but passphrase is empty.");                }                var binarySalt = new byte[salt.Length / 2];                for (var i = 0; i < binarySalt.Length; i++)                {                    binarySalt[i] = Convert.ToByte(salt.Substring(i * 2, 2), 16);                }                CipherInfo cipher;                switch (cipherName)                {                    case "DES-EDE3-CBC":                        cipher = new CipherInfo(192, (key, iv) => new TripleDesCipher(key, new CbcCipherMode(iv), new PKCS7Padding()));                        break;                    case "DES-EDE3-CFB":                        cipher = new CipherInfo(192, (key, iv) => new TripleDesCipher(key, new CfbCipherMode(iv), new PKCS7Padding()));                        break;                    case "DES-CBC":                        cipher = new CipherInfo(64, (key, iv) => new DesCipher(key, new CbcCipherMode(iv), new PKCS7Padding()));                        break;                    case "AES-128-CBC":                        cipher = new CipherInfo(128, (key, iv) => new AesCipher(key, iv, AesCipherMode.CBC, pkcs7Padding: true));                        break;                    case "AES-192-CBC":                        cipher = new CipherInfo(192, (key, iv) => new AesCipher(key, iv, AesCipherMode.CBC, pkcs7Padding: true));                        break;                    case "AES-256-CBC":                        cipher = new CipherInfo(256, (key, iv) => new AesCipher(key, iv, AesCipherMode.CBC, pkcs7Padding: true));                        break;                    default:                        throw new SshException(string.Format(CultureInfo.InvariantCulture, "Private key cipher \"{0}\" is not supported.", cipherName));                }                decryptedData = DecryptKey(cipher, binaryData, passPhrase, binarySalt);            }            else            {                decryptedData = binaryData;            }            switch (keyName)            {                case "RSA":                    var rsaKey = new RsaKey(decryptedData);                    _key = rsaKey;#pragma warning disable CA2000 // Dispose objects before losing scope                    _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-512", _key, new RsaDigitalSignature(rsaKey, HashAlgorithmName.SHA512)));                    _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-256", _key, new RsaDigitalSignature(rsaKey, HashAlgorithmName.SHA256)));#pragma warning restore CA2000 // Dispose objects before losing scope                    _hostAlgorithms.Add(new KeyHostAlgorithm("ssh-rsa", _key));                    break;                case "DSA":                    _key = new DsaKey(decryptedData);                    _hostAlgorithms.Add(new KeyHostAlgorithm("ssh-dss", _key));                    break;                case "EC":                    _key = new EcdsaKey(decryptedData);                    _hostAlgorithms.Add(new KeyHostAlgorithm(_key.ToString(), _key));                    break;                case "OPENSSH":                    _key = ParseOpenSshV1Key(decryptedData, passPhrase);                    if (_key is RsaKey parsedRsaKey)                    {#pragma warning disable CA2000 // Dispose objects before losing scope                        _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-512", _key, new RsaDigitalSignature(parsedRsaKey, HashAlgorithmName.SHA512)));                        _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-256", _key, new RsaDigitalSignature(parsedRsaKey, HashAlgorithmName.SHA256)));#pragma warning restore CA2000 // Dispose objects before losing scope                        _hostAlgorithms.Add(new KeyHostAlgorithm("ssh-rsa", _key));                    }                    else                    {                        _hostAlgorithms.Add(new KeyHostAlgorithm(_key.ToString(), _key));                    }                    break;                case "SSH2 ENCRYPTED":                    var reader = new SshDataReader(decryptedData);                    var magicNumber = reader.ReadUInt32();                    if (magicNumber != 0x3f6ff9eb)                    {                        throw new SshException("Invalid SSH2 private key.");                    }                    _ = reader.ReadUInt32(); // Read total bytes length including magic number                    var keyType = reader.ReadString(SshData.Ascii);                    var ssh2CipherName = reader.ReadString(SshData.Ascii);                    var blobSize = (int)reader.ReadUInt32();                    byte[] keyData;                    if (ssh2CipherName == "none")                    {                        keyData = reader.ReadBytes(blobSize);                    }                    else if (ssh2CipherName == "3des-cbc")                    {                        if (string.IsNullOrEmpty(passPhrase))                        {                            throw new SshPassPhraseNullOrEmptyException("Private key is encrypted but passphrase is empty.");                        }                        var key = GetCipherKey(passPhrase, 192 / 8);                        var ssh2Сipher = new TripleDesCipher(key, new CbcCipherMode(new byte[8]), new PKCS7Padding());                        keyData = ssh2Сipher.Decrypt(reader.ReadBytes(blobSize));                    }                    else                    {                        throw new SshException(string.Format("Cipher method '{0}' is not supported.", cipherName));                    }                    /*                     * TODO: Create two specific data types to avoid using SshDataReader class.                     */                    reader = new SshDataReader(keyData);                    var decryptedLength = reader.ReadUInt32();                    if (decryptedLength > blobSize - 4)                    {                        throw new SshException("Invalid passphrase.");                    }                    if (keyType == "if-modn{sign{rsa-pkcs1-sha1},encrypt{rsa-pkcs1v2-oaep}}")                    {                        var exponent = reader.ReadBigIntWithBits(); // e                        var d = reader.ReadBigIntWithBits(); // d                        var modulus = reader.ReadBigIntWithBits(); // n                        var inverseQ = reader.ReadBigIntWithBits(); // u                        var q = reader.ReadBigIntWithBits(); // p                        var p = reader.ReadBigIntWithBits(); // q                        var decryptedRsaKey = new RsaKey(modulus, exponent, d, p, q, inverseQ);                        _key = decryptedRsaKey;#pragma warning disable CA2000 // Dispose objects before losing scope                        _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-512", _key, new RsaDigitalSignature(decryptedRsaKey, HashAlgorithmName.SHA512)));                        _hostAlgorithms.Add(new KeyHostAlgorithm("rsa-sha2-256", _key, new RsaDigitalSignature(decryptedRsaKey, HashAlgorithmName.SHA256)));#pragma warning restore CA2000 // Dispose objects before losing scope                        _hostAlgorithms.Add(new KeyHostAlgorithm("ssh-rsa", _key));                    }                    else if (keyType == "dl-modp{sign{dsa-nist-sha1},dh{plain}}")                    {                        var zero = reader.ReadUInt32();                        if (zero != 0)                        {                            throw new SshException("Invalid private key");                        }                        var p = reader.ReadBigIntWithBits();                        var g = reader.ReadBigIntWithBits();                        var q = reader.ReadBigIntWithBits();                        var y = reader.ReadBigIntWithBits();                        var x = reader.ReadBigIntWithBits();                        _key = new DsaKey(p, q, g, y, x);                        _hostAlgorithms.Add(new KeyHostAlgorithm("ssh-dss", _key));                    }                    else                    {                        throw new NotSupportedException(string.Format("Key type '{0}' is not supported.", keyType));                    }                    break;                default:                    throw new NotSupportedException(string.Format(CultureInfo.CurrentCulture, "Key '{0}' is not supported.", keyName));            }        }        private static byte[] GetCipherKey(string passphrase, int length)        {            var cipherKey = new List<byte>();            using (var md5 = CryptoAbstraction.CreateMD5())            {                var passwordBytes = Encoding.UTF8.GetBytes(passphrase);                var hash = md5.ComputeHash(passwordBytes);                cipherKey.AddRange(hash);                while (cipherKey.Count < length)                {                    hash = passwordBytes.Concat(hash);                    hash = md5.ComputeHash(hash);                    cipherKey.AddRange(hash);                }            }            return cipherKey.ToArray().Take(length);        }        /// <summary>        /// Decrypts encrypted private key file data.        /// </summary>        /// <param name="cipherInfo">The cipher info.</param>        /// <param name="cipherData">Encrypted data.</param>        /// <param name="passPhrase">Decryption pass phrase.</param>        /// <param name="binarySalt">Decryption binary salt.</param>        /// <returns>Decrypted byte array.</returns>        /// <exception cref="ArgumentNullException"><paramref name="cipherInfo" />, <paramref name="cipherData" />, <paramref name="passPhrase" /> or <paramref name="binarySalt" /> is <see langword="null"/>.</exception>        private static byte[] DecryptKey(CipherInfo cipherInfo, byte[] cipherData, string passPhrase, byte[] binarySalt)        {            if (cipherInfo is null)            {                throw new ArgumentNullException(nameof(cipherInfo));            }            if (cipherData is null)            {                throw new ArgumentNullException(nameof(cipherData));            }            if (binarySalt is null)            {                throw new ArgumentNullException(nameof(binarySalt));            }            var cipherKey = new List<byte>();            using (var md5 = CryptoAbstraction.CreateMD5())            {                var passwordBytes = Encoding.UTF8.GetBytes(passPhrase);                // Use 8 bytes binary salt                var initVector = passwordBytes.Concat(binarySalt.Take(8));                var hash = md5.ComputeHash(initVector);                cipherKey.AddRange(hash);                while (cipherKey.Count < cipherInfo.KeySize / 8)                {                    hash = hash.Concat(initVector);                    hash = md5.ComputeHash(hash);                    cipherKey.AddRange(hash);                }            }            var cipher = cipherInfo.Cipher(cipherKey.ToArray(), binarySalt);            return cipher.Decrypt(cipherData);        }        /// <summary>        /// Parses an OpenSSH V1 key file (i.e. ED25519 key) according to the the key spec:        /// https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.key.        /// </summary>        /// <param name="keyFileData">The key file data (i.e. base64 encoded data between the header/footer).</param>        /// <param name="passPhrase">Passphrase or <see langword="null"/> if there isn't one.</param>        /// <returns>        /// The OpenSSH V1 key.        /// </returns>        private static Key ParseOpenSshV1Key(byte[] keyFileData, string passPhrase)        {            var keyReader = new SshDataReader(keyFileData);            // check magic header            var authMagic = Encoding.UTF8.GetBytes("openssh-key-v1\0");            var keyHeaderBytes = keyReader.ReadBytes(authMagic.Length);            if (!authMagic.IsEqualTo(keyHeaderBytes))            {                throw new SshException("This openssh key does not contain the 'openssh-key-v1' format magic header");            }            // cipher will be "aes256-cbc" if using a passphrase, "none" otherwise            var cipherName = keyReader.ReadString(Encoding.UTF8);            // key derivation function (kdf): bcrypt or nothing            var kdfName = keyReader.ReadString(Encoding.UTF8);            // kdf options length: 24 if passphrase, 0 if no passphrase            var kdfOptionsLen = (int)keyReader.ReadUInt32();            byte[] salt = null;            var rounds = 0;            if (kdfOptionsLen > 0)            {                var saltLength = (int) keyReader.ReadUInt32();                salt = keyReader.ReadBytes(saltLength);                rounds = (int) keyReader.ReadUInt32();            }            // number of public keys, only supporting 1 for now            var numberOfPublicKeys = (int)keyReader.ReadUInt32();            if (numberOfPublicKeys != 1)            {                throw new SshException("At this time only one public key in the openssh key is supported.");            }            // read public key in ssh-format, but we dont need it            _ = keyReader.ReadString(Encoding.UTF8);            // possibly encrypted private key            var privateKeyLength = (int) keyReader.ReadUInt32();            var privateKeyBytes = keyReader.ReadBytes(privateKeyLength);            // decrypt private key if necessary            if (cipherName != "none")            {                if (string.IsNullOrEmpty(passPhrase))                {                    throw new SshPassPhraseNullOrEmptyException("Private key is encrypted but passphrase is empty.");                }                if (string.IsNullOrEmpty(kdfName) || kdfName != "bcrypt")                {                    throw new SshException("kdf " + kdfName + " is not supported for openssh key file");                }                // inspired by the SSHj library (https://github.com/hierynomus/sshj)                // apply the kdf to derive a key and iv from the passphrase                var passPhraseBytes = Encoding.UTF8.GetBytes(passPhrase);                var keyiv = new byte[48];                new BCrypt().Pbkdf(passPhraseBytes, salt, rounds, keyiv);                var key = new byte[32];                Array.Copy(keyiv, 0, key, 0, 32);                var iv = new byte[16];                Array.Copy(keyiv, 32, iv, 0, 16);                AesCipher cipher;                switch (cipherName)                {                    case "aes256-cbc":                        cipher = new AesCipher(key, iv, AesCipherMode.CBC, pkcs7Padding: false);                        break;                    case "aes256-ctr":                        cipher = new AesCipher(key, iv, AesCipherMode.CTR, pkcs7Padding: false);                        break;                    default:                        throw new SshException("Cipher '" + cipherName + "' is not supported for an OpenSSH key.");                }                try                {                    privateKeyBytes = cipher.Decrypt(privateKeyBytes);                }                finally                {                    cipher.Dispose();                }            }            // validate private key length            privateKeyLength = privateKeyBytes.Length;            if (privateKeyLength % 8 != 0)            {                throw new SshException("The private key section must be a multiple of the block size (8)");            }            // now parse the data we called the private key, it actually contains the public key again            // so we need to parse through it to get the private key bytes, plus there's some            // validation we need to do.            var privateKeyReader = new SshDataReader(privateKeyBytes);            // check ints should match, they wouldn't match for example if the wrong passphrase was supplied            var checkInt1 = (int) privateKeyReader.ReadUInt32();            var checkInt2 = (int) privateKeyReader.ReadUInt32();            if (checkInt1 != checkInt2)            {                throw new SshException(string.Format(CultureInfo.InvariantCulture,                                                     "The random check bytes of the OpenSSH key do not match ({0} <-> {1}).",                                                     checkInt1.ToString(CultureInfo.InvariantCulture),                                                     checkInt2.ToString(CultureInfo.InvariantCulture)));            }            // key type            var keyType = privateKeyReader.ReadString(Encoding.UTF8);            Key parsedKey;            byte[] publicKey;            byte[] unencryptedPrivateKey;            switch (keyType)            {                case "ssh-ed25519":                    // https://datatracker.ietf.org/doc/html/draft-miller-ssh-agent-11#section-3.2.3                    // ENC(A)                    _ = privateKeyReader.ReadBignum2();                    // k || ENC(A)                    unencryptedPrivateKey = privateKeyReader.ReadBignum2();                    parsedKey = new ED25519Key(unencryptedPrivateKey);                    break;                case "ecdsa-sha2-nistp256":                case "ecdsa-sha2-nistp384":                case "ecdsa-sha2-nistp521":                    // curve                    var len = (int) privateKeyReader.ReadUInt32();                    var curve = Encoding.ASCII.GetString(privateKeyReader.ReadBytes(len));                    // public key                    publicKey = privateKeyReader.ReadBignum2();                    // private key                    unencryptedPrivateKey = privateKeyReader.ReadBignum2();                    parsedKey = new EcdsaKey(curve, publicKey, unencryptedPrivateKey.TrimLeadingZeros());                    break;                case "ssh-rsa":                    var modulus = privateKeyReader.ReadBignum(); // n                    var exponent = privateKeyReader.ReadBignum(); // e                    var d = privateKeyReader.ReadBignum(); // d                    var inverseQ = privateKeyReader.ReadBignum(); // iqmp                    var p = privateKeyReader.ReadBignum(); // p                    var q = privateKeyReader.ReadBignum(); // q                    parsedKey = new RsaKey(modulus, exponent, d, p, q, inverseQ);                    break;                default:                    throw new SshException("OpenSSH key type '" + keyType + "' is not supported.");            }            parsedKey.Comment = privateKeyReader.ReadString(Encoding.UTF8);            // The list of privatekey/comment pairs is padded with the bytes 1, 2, 3, ...            // until the total length is a multiple of the cipher block size.            var padding = privateKeyReader.ReadBytes();            for (var i = 0; i < padding.Length; i++)            {                if ((int) padding[i] != i + 1)                {                    throw new SshException("Padding of openssh key format contained wrong byte at position: " +                                           i.ToString(CultureInfo.InvariantCulture));                }            }            return parsedKey;        }        /// <summary>        /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.        /// </summary>        public void Dispose()        {            Dispose(disposing: true);            GC.SuppressFinalize(this);        }        /// <summary>        /// Releases unmanaged and - optionally - managed resources.        /// </summary>        /// <param name="disposing"><see langword="true"/> to release both managed and unmanaged resources; <see langword="false"/> to release only unmanaged resources.</param>        protected virtual void Dispose(bool disposing)        {            if (_isDisposed)            {                return;            }            if (disposing)            {                var key = _key;                if (key != null)                {                    ((IDisposable) key).Dispose();                    _key = null;                }                _isDisposed = true;            }        }        /// <summary>        /// Finalizes an instance of the <see cref="PrivateKeyFile"/> class.        /// </summary>        ~PrivateKeyFile()        {            Dispose(disposing: false);        }        private sealed class SshDataReader : SshData        {            public SshDataReader(byte[] data)            {                Load(data);            }            public new uint ReadUInt32()            {                return base.ReadUInt32();            }            public new string ReadString(Encoding encoding)            {                return base.ReadString(encoding);            }            public new byte[] ReadBytes(int length)            {                return base.ReadBytes(length);            }            public new byte[] ReadBytes()            {                return base.ReadBytes();            }            /// <summary>            /// Reads next mpint data type from internal buffer where length specified in bits.            /// </summary>            /// <returns>mpint read.</returns>            public BigInteger ReadBigIntWithBits()            {                var length = (int) base.ReadUInt32();                length = (length + 7) / 8;                var data = base.ReadBytes(length);                var bytesArray = new byte[data.Length + 1];                Buffer.BlockCopy(data, 0, bytesArray, 1, data.Length);                return new BigInteger(bytesArray.Reverse());            }            public BigInteger ReadBignum()            {                return new BigInteger(ReadBignum2().Reverse());            }            public byte[] ReadBignum2()            {                var length = (int)base.ReadUInt32();                return base.ReadBytes(length);            }            protected override void LoadData()            {            }            protected override void SaveData()            {            }        }    }}
 |