MySQL.Data/src/Authentication/Sha256AuthenticationPlugin.cs (75 lines of code) (raw):

// Copyright © 2013, 2025, Oracle and/or its affiliates. // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License, version 2.0, as // published by the Free Software Foundation. // // This program is designed to work with certain software (including // but not limited to OpenSSL) that is licensed under separate terms, as // designated in a particular file or component or in included license // documentation. The authors of MySQL hereby grant you an additional // permission to link the program and your derivative works with the // separately licensed software that they have either included with // the program or referenced in the documentation. // // Without limiting anything contained in the foregoing, this file, // which is part of MySQL Connector/NET, is also subject to the // Universal FOSS Exception, version 1.0, a copy of which can be found at // http://oss.oracle.com/licenses/universal-foss-exception. // // This program is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU General Public License, version 2.0, for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software Foundation, Inc., // 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA using System; using System.Security.Cryptography; using System.Text; using System.Threading.Tasks; namespace MySql.Data.MySqlClient.Authentication { /// <summary> /// The implementation of the sha256_password authentication plugin. /// </summary> internal class Sha256AuthenticationPlugin : MySqlAuthenticationPlugin { /// <summary> /// The byte array representation of the public key provided by the server. /// </summary> protected byte[] rawPubkey; public override string PluginName => "sha256_password"; protected override Task<byte[]> MoreDataAsync(byte[] data, bool execAsync) { rawPubkey = data; byte[] buffer = GetNonLengthEncodedPassword(); return Task.FromResult<byte[]>(buffer); } public override object GetPassword() { if (Settings.SslMode != MySqlSslMode.Disabled) { // send as clear text, since the channel is already encrypted byte[] passBytes = Encoding.GetBytes(GetMFAPassword()); byte[] buffer = new byte[passBytes.Length + 2]; Array.Copy(passBytes, 0, buffer, 1, passBytes.Length); buffer[0] = (byte)(passBytes.Length + 1); buffer[buffer.Length - 1] = 0x00; return buffer; } else { if (GetMFAPassword().Length == 0) return new byte[1]; // send RSA encrypted, since the channel is not protected else if (rawPubkey == null) return new byte[] { 0x01 }; else if (!Settings.AllowPublicKeyRetrieval) throw new MySqlException(Resources.RSAPublicKeyRetrievalNotEnabled); else { byte[] bytes = GetRsaPassword(GetMFAPassword(), AuthenticationData, rawPubkey); if (bytes != null && bytes.Length == 1 && bytes[0] == 0) return null; return bytes; } } } private byte[] GetNonLengthEncodedPassword() { // Required for AuthChange requests. if (Settings.SslMode != MySqlSslMode.Disabled) { // Send as clear text, since the channel is already encrypted. byte[] passBytes = Encoding.GetBytes(GetMFAPassword()); byte[] buffer = new byte[passBytes.Length + 1]; Array.Copy(passBytes, 0, buffer, 0, passBytes.Length); buffer[passBytes.Length] = 0; return buffer; } else return GetPassword() as byte[]; } private byte[] GetRsaPassword(string password, byte[] seedBytes, byte[] rawPublicKey) { if (password.Length == 0) return new byte[1]; // Obfuscate the plain text password with the session scramble byte[] obfuscated = GetXor(Encoding.Default.GetBytes(password), seedBytes); // Encrypt the password and send it to the server RSACryptoServiceProvider rsa = MySqlPemReader.ConvertPemToRSAProvider(rawPublicKey); if (rsa == null) throw new MySqlException(Resources.UnableToReadRSAKey); return rsa.Encrypt(obfuscated, true); } /// <summary> /// Applies XOR to the byte arrays provided as input. /// </summary> /// <returns>A byte array that contains the results of the XOR operation.</returns> protected byte[] GetXor(byte[] src, byte[] pattern) { byte[] src2 = new byte[src.Length + 1]; Array.Copy(src, 0, src2, 0, src.Length); src2[src.Length] = 0; byte[] result = new byte[src2.Length]; for (int i = 0; i < src2.Length; i++) { result[i] = (byte)(src2[i] ^ (pattern[i % pattern.Length])); } return result; } } }