MySqlConnector vs MySql.Data
Compared to MySql.Data (i.e., Oracle’s MySQL Connector/NET), MySqlConnector has the following benefits:
- Higher performance (see benchmarks) and lower memory usage
- More reliable, particularly when using
Async
methods (see bugs fixed) - Leading edge: first MySQL driver to support .NET Core,
DbBatch
,DateOnly
,DbDataSource
and other new .NET features - Better compatibility with non-MySQL databases (MariaDB, Amazon Aurora, etc.)
- Better license (MIT, not GPL 2.0 with Universal FOSS Exception or commercial license)
- Faster pace of development (regular NuGet releases, not once per quarter)
- Development happens in the open on GitHub
Migrating from Connector/NET
Namespace
MySqlConnector supports the same core API as MySQL Connector/NET, but the classes are in a different
namespace. Change using MySql.Data.MySqlClient;
to using MySqlConnector;
.
DbProviderFactories
The MySqlClientFactory
type is named MySqlConnectorFactory
in MySqlConnector.
In a .NET Framework application, make the following app.config
change to register MySqlConnector instead of MySql.Data.
<system.data>
<DbProviderFactories>
<!-- REMOVE THIS -->
<!-- add name="MySQL Data Provider"
invariant="MySql.Data.MySqlClient"
description=".Net Framework Data Provider for MySQL"
type="MySql.Data.MySqlClient.MySqlClientFactory, MySql.Data, Version=9.0.0.0, Culture=neutral, PublicKeyToken=c5687fc88969c44d" / -->
<!-- ADD THIS -->
<add name="MySqlConnector"
invariant="MySqlConnector"
description="MySQL Connector for .NET"
type="MySqlConnector.MySqlConnectorFactory, MySqlConnector, Culture=neutral, PublicKeyToken=d33d3e53aa5f8c92" />
</DbProviderFactories>
</system.data>
Connection String Differences
MySqlConnector has some different default connection string options:
Option | MySqlConnector | Oracle’s Connector/NET | Notes |
---|---|---|---|
CharacterSet , CharSet | Ignored; utf8mb4 is always used | (server-defined) | MySqlConnector always uses utf8mb4 to send and receive strings from MySQL Server. This option may be specified (for backwards compatibility) but it will be ignored. |
ConnectionReset | Default is true | Default is false | MySqlConnector always resets pooled connections by default so that the connection is in a known state. This fixes MySQL Bug 77421. |
IgnoreCommandTransaction | Default is false | (not configurable, effective default is true ) | See remarks under MySqlCommand below. |
IgnorePrepare | Default is false | true for ≤ 8.0.22; false for ≥ 8.0.23 | This is a change if migrating from an older version of Connector/NET. |
LoadBalance | Default is RoundRobin | (not configurable, effective default is FailOver ) | Connector/NET versions prior to v8.0.19 have a bug that prevents multiple host names being used. |
ServerRSAPublicKeyFile | (no default) | (not configurable) | Specify a file containing the server’s RSA public key to allow sha256_password authentication over an insecure connection. |
Connector/NET uses CertificateFile
to specify the client’s private key, unless SslCert
and SslKey
are specified, in which case
it is used to specify the server’s CA certificate file; SslCa
is just an alias for this option. MySqlConnector always uses CertificateFile
for the client’s private key (in PFX format); SslCa
(aka CACertificateFile
) is a separate option to specify the server’s CA certificate.
Some connection string options that are supported in Connector/NET are not supported in MySqlConnector. For a full list of options that are supported in MySqlConnector, see the Connection Options.
Async
Connector/NET implements the standard ADO.NET async methods, and adds some new ones (e.g., MySqlConnection.BeginTransactionAsync
,
MySqlDataAdapter.FillAsync
) that don’t exist in ADO.NET. None of these methods have an asynchronous implementation,
but all execute synchronously then return a completed Task
. This longstanding known bug was fixed in
in Connector/NET v8.0.33.
Because the Connector/NET methods aren’t actually asynchronous, porting client code to MySqlConnector (which is asynchronous)
can expose bugs that only occur when an async method completes asynchronously and resumes the await
-ing code
on a background thread. To avoid deadlocks, make sure to never block on async code (e.g., with .Result
), use async all the way, use ConfigureAwait
correctly,
and follow the best practices in async programming.
Implicit Conversions
Prior to v8.3.0, Connector/NET allows MySqlDataReader.GetString()
to be called on many non-textual columns, and will implicitly
convert the value to a string
(using the current locale). This is a frequent source of locale-dependent bugs, so
MySqlConnector follows typical ADO.NET practice (e.g., SqlClient, npgsql) and disallows this (by throwing an InvalidCastException
).
To fix this, use the accessor method (e.g., GetInt32
, GetDouble
) that matches the column type, or perform an
explicit conversion to string
by calling GetValue(x).ToString()
(optionally supplying the right CultureInfo
to use
for formatting).
TransactionScope
MySqlConnector adds full distributed transaction support (for client code using System.Transactions.Transaction
),
while Connector/NET uses regular database transactions. As a result, code that uses TransactionScope
or MySqlConnection.EnlistTransaction
may execute differently with MySqlConnector. To get Connector/NET-compatible behavior, set
UseXaTransactions=false
in your connection string.
MySqlConnection
Connector/NET allows a MySqlConnection
object to be reused after it has been disposed. MySqlConnector requires a new MySqlConnection
object to be created. See #331 for more details.
The return value of MySqlConnection.BeginTransactionAsync
has changed from Task<MySqlTransaction>
to
ValueTask<MySqlTransaction>
to match the standard API in .NET Core 3.0.
(This method does always perform I/O, so ValueTask
is not an optimization for MySqlConnector.)
MySqlConnectionStringBuilder
All string
properties on MySqlConnectionStringBuilder
will return the empty string (instead of null
) if the property isn’t set.
MySqlCommand
Connector/NET allows a command to be executed even when MySqlCommand.Transaction
references a commited, rolled back, or
disposed MySqlTransaction
. MySqlConnector will throw an InvalidOperationException
if the MySqlCommand.Transaction
property doesn’t reference the active transaction. This fixes MySQL Bug 88611.
To disable this strict validation, set IgnoreCommandTransaction=true
in the connection string. See Transaction Usage for more details.
If MySqlCommand.CommandType
is CommandType.StoredProcedure
, the stored procedure name assigned to MySqlCommand.CommandText
must have any special characters escaped or quoted. Connector/NET will automatically quote some characters (such as spaces); MySqlConnector leaves this up to the developer.
MySqlDataAdapter
Connector/NET provides MySqlDataAdapter.FillAsync
, FillSchemaAsync
, and UpdateAsync
methods, but these methods
have a synchronous implementation. MySqlConnector only adds “Async” methods when they can be implemented asynchronously.
This functionality depends on dotnet/runtime#22109 being implemented first.
To migrate code, change it to call the synchronous methods instead.
MySqlGeometry
The Connector/NET MySqlGeometry
type assumes that the geometry can only be a simple point. MySqlConnector
removes most of the API that is based on those assumptions.
To avoid ambiguity, there are two different factory methods for constructing a MySqlGeometry
. Use the static factory method MySqlGeometry.FromMySql
(if you have a byte array containing MySQL’s internal format), or FromWkb
if you have
Well-known Binary bytes.
MySqlInfoMessageEventArgs
The MySqlError[] MySqlInfoMessageEventArgs.errors
property has changed to IReadOnlyList<MySqlError> MySqlInfoMessageEventArgs.Errors
.
MySqlParameter
Connector/NET will automatically convert unknown MySqlParameter.Value
values to a string
by calling ToString()
,
then convert that to bytes by calling Encoding.GetBytes()
using the packet’s encoding. This is error-prone and
can introduce culture-sensitive conversions.
MySqlConnector requires all parameter values to be of a known, supported type. See MySqlParameter Types for details.
MySqlParameterCollection
Connector/NET will assign the names @Parameter1
, @Parameter2
, etc. to unnamed MySqlParameter
objects that are
added to the MySqlCommand.Parameters
parameter collection. These generated names may be used in the SQL assigned to
MySqlCommand.CommandText
. MySqlConnector requires all MySqlParameter
objects to be explicitly given a name,
or used only as positional parameters if they’re unnamed.
Exceptions
For consistency with other ADO.NET providers, MySqlConnector will throw InvalidOperationException
(instead of MySqlException
)
for various precondition checks that indicate misuse of the API (and not a problem related to MySQL Server).
Fixed Bugs
The following bugs in Connector/NET are fixed by switching to MySqlConnector.
Open Bugs
- #14115: Compound statements are not supported by
MySqlCommand.Prepare
(Will never be supported) - #37283, #70587: Distributed transactions are not supported
- #50773: Can’t use multiple connections within one TransactionScope
- #61477:
ColumnOrdinal
in schema table is 1-based - #72494, #83330: EndOfStreamException inserting large blob with UseCompression=True
- #73788: Can’t use
DateTimeOffset
- #75917, #76597, #77691, #78650, #78919, #80921, #82136: “Reading from the stream has failed” when connecting to a server
- #77421: Connection is not reset when pulled from the connection pool
- #83229: “Unknown command” exception inserting large blob with UseCompression=True
- #85185:
ConnectionReset=True
does not preserve connection charset - #88124: CommandTimeout isn’t reset when calling Read/NextResult
- #88611:
MySqlCommand
can be executed even if it has “wrong” transaction - #89085:
MySqlConnection.Database
not updated afterUSE database;
- #89335:
MySqlCommandBuilder.DeriveParameters
fails forJSON
type - #91123: Database names are case-sensitive when calling a stored procedure
- #91199: Can’t insert
MySqlDateTime
values - #91753: Unnamed parameter not supported by
MySqlCommand.Prepare
- #91754: Inserting 16MiB
BLOB
shifts it by four bytes when prepared - #92367:
MySqlDataReader.GetDateTime
andGetValue
return inconsistent values - #93047:
MySqlDataAdapter
throws timeout exception when an error occurs - #93220: Can’t call FUNCTION when parameter name contains parentheses
- #93825:
MySqlException
loses data when serialized - #94760:
MySqlConnection.OpenAsync(CancellationToken)
doesn’t respect cancellation token - #95348: Inefficient query when executing stored procedures
- #95436: Client doesn’t authenticate with PEM certificate
- #96500:
MySqlDataReader.GetFieldValue<MySqlGeometry>
throwsInvalidCastException
- #96636:
MySqlConnection.Open()
slow under load when using SSL - #96717: Not compatible with MySQL Server 5.0
- #97067: Aggregate functions on BIT(n) columns return wrong result
- #97738: Cannot use PEM files when account uses
require subject
- #97872:
KeepAlive
in connection string throws exception on .NET Core - #103819: Can’t use
StringBuilder
containing non-BMP characters asMySqlParameter.Value
- #105965:
MySqlParameterCollection.Add(object)
has quadratic performance - #107259: Character Set ‘utf8mb3’ is not supported by .NET Framework
- #108970:
MySqlConnectionStringBuilder.ContainsKey
method gives wrong result - #109141: Insert of data into a table results in
System.ArgumentException
- #109331: MySQL Connector/NET is incompatible with MariaDB 10.10 and later
- #109390: Transaction lock held after connection timeout exception
- #109476:
TransactionScope.Dispose
throws “Connection must be valid and open to rollback” - #110789:
OpenAsync
throws unhandled exception from thread pool - #110790:
ExecuteReaderAsync
hangs instead of cancelling query afterCommandTimeout
- #110791:
OpenAsync(CancellationToken)
doesn’t throw for cancelled token - #111797: Certain sequence of special characters can break connection string validation
- #112088:
BIT(n)
parameter size incorrectly returns 0 - #114567: MySql.Data 8.0.33 is twice as slow as previous versions
- #115155: MySql.Data executes
SET NAMES
using an unsupported value - #115168:
MySqlDataReader.GetX
performs culture-sensitive conversions - #115222:
MySqlPool.ClearAsync
is not thread-safe
Fixed First by MySqlConnector
These bugs are fixed in the latest version of MySQL Connector/NET, but were fixed first in MySqlConnector.
- #66476, #106368: Connection pool uses queue instead of stack (fixed in v8.0.30)
- #70111:
Async
methods execute synchronously (fixed in v8.0.33) - #70686:
TIME(3)
andTIME(6)
fields serialize milliseconds incorrectly - #73610: Invalid password exception has wrong number (fixed in v8.0.32)
- #74392: Can’t use
Stream
to bulk load data - #75604: Crash after 29.4 days of uptime
- #78426: Unknown database exception has wrong number
- #78760: Error when using tabs and newlines in SQL statements
- #78917, #79196, #82292, #89040:
TINYINT(1)
values start being returned assbyte
afterNULL
- #80030: Slow to connect with pooling disabled
- #81650, #88962:
Server
connection string option may now contain multiple, comma separated hosts that will be tried in order until a connection succeeds - #83649: Connection cannot be made using IPv6
- #84220: Cannot call a stored procedure with
.
in its name - #84701: Can’t create a parameter using a 64-bit enum with a value greater than int.MaxValue
- #86263: Transaction isolation level affects all transactions in session
- #87307: NextResult hangs instead of timing out
- #87316: MySqlCommand.CommandTimeout can be set to a negative value
- #87868:
ColumnSize
in schema table is incorrect forCHAR(36)
andBLOB
columns - #87876:
IsLong
is schema table is incorrect forLONGTEXT
andLONGBLOB
columns - #88058:
decimal(n, 0)
has wrongNumericPrecision
- #88472:
TINYINT(1)
is not returned asbool
ifMySqlCommand.Prepare
is called - #88660:
MySqlClientFactory.Instance.CreateDataAdapter()
andCreateCommandBuilder
returnnull
- #89159, #97242:
MySqlDataReader
cannot outliveMySqlCommand
- #89639:
ReservedWords
schema contains incorrect data - #90086:
MySqlDataReader
is closed by an unrelated command disposal - #91751:
YEAR
column retrieved incorrectly with prepared command - #91752:
00:00:00
is converted toNULL
with prepared command - #91770:
TIME(n)
column loses microseconds with prepared command - #92465: “There is already an open DataReader”
MySqlException
thrown fromTransactionScope.Dispose
- #92734:
MySqlParameter.Clone
doesn’t copy all property values - #92789: Illegal connection attributes written for non-ASCII values
- #92912:
MySqlDbType.LongText
values encoded incorrectly with prepared statements - #92982, #93399:
FormatException
thrown when connecting to MySQL Server 8.0.13 - #93202: Connector runs
SHOW VARIABLES
when connection is made - #93370:
MySqlParameterCollection.Add
precondition check isn’t consistent - #93374:
MySqlDataReader.GetStream
throwsIndexOutOfRangeException
- #94075:
MySqlCommand.Cancel
throws exception - #95984: “Incorrect arguments to mysqld_stmt_execute” using prepared statement with
MySqlDbType.JSON
- #95986: “Incorrect integer value” using prepared statement with
MySqlDbType.Int24
- #96355, #96614:
Could not load file or assembly 'Renci.SshNet'
when opening connection - #96498:
WHERE
clause usingMySqlGeometry
as parameter finds no rows - #96499:
MySqlException
when inserting aMySqlGeometry
value - #97061:
MySqlCommand.LastInsertedId
returns 0 after executing multiple statements - #97300:
GetSchemaTable()
returns table for stored procedure with output parameters - #97448: Connecting fails if more than one IP is found in DNS for a named host
- #97473:
MySqlConnection.Clone
discloses connection password - #98322:
new MySqlConnection(null)
throwsNullReferenceException
- #99091: Unexpected return value getting integer for
TINYINT(1)
column - #99793: Prepared stored procedure command doesn’t verify parameter types
- #100159: SQL with DateTime parameter returns String value
- #100208:
GetSchema("Procedures")
returnsROUTINE_DEFINITION
of"System.Byte[]"
- #100218:
TIME(n)
microsecond values deserialized incorrectly with prepared command - #100306:
Command.Prepare
sends wrong statement to server - #100522:
MySqlCommand.Parameters.Insert(-1)
succeeds but should fail - #101252: Can’t query
CHAR(36)
column containingNULL
- #101253: Default value for
MySqlParameter.Value
changed from null to0
- #101302: Stored Procedure
BOOL
parameter can only be mapped toMySqlDbType.Byte
- #101485: Stored Procedure
JSON
parameter throws “Unhandled type encountered”MySqlException
- #101507:
MySqlCommand.Cancel
throwsNullReferenceException
for a closed connection - #101714: Extremely slow performance reading result sets
- #102303: Preparing command with incorrect parameter type puts connection in bad state
- #102593: Can’t use
MemoryStream
asMySqlParameter.Value
- #103390: Can’t query
CHAR(36)
column ifMySqlCommand
is prepared - #103430: Can’t connect using named pipe on Windows
- #103801:
TimeSpan
parameters lose microseconds with prepared statement - #104910:
MySqlConnectionStringBuilder.TryGetValue
always returnsfalse
- #104913: Cannot execute stored procedure with backtick in name
- #105209: Timespan value of zero can’t be read with prepared command
- #105516: Connector/Net doesn’t recognize utf8mb3
- #105728: Named command parameters override query attribute values
- #105730:
MySqlCommand.Clone
doesn’t clone attributes - #105768:
MySqlCommandBuilder
doesn’t support tables withBIGINT UNSIGNED
primary key - #106242:
MySqlConnection.Open
throwsAggregateException
instead ofMySqlException
- #106243:
CancellationToken
doesn’t cancelMySqlConnection.OpenAsync
- #106244:
MySqlDataReader.GetFieldValue<Stream>
throwsInvalidCastException
- #106247: Can’t use
MySqlDbType.Enum
orMySqlDbType.Set
with prepared command - #108756: Can’t insert negative number using prepared statement with
MySqlDbType.Int24
- #109670, #109752: Many MSB3246 errors when referencing MySql.Data 8.0.32
- #109682:
MySqlParameter.Clone
loses specificMySqlDbType
. - #109683:
MySqlCommand.LastInsertedId
is incorrect if multiple rows are inserted. - #109716, #109727: Failed to add reference to ‘comerr64’
- #110717, #111759: MySql.Data is not thread-safe
- #114527: Connector/NET hangs on second connection attempt to non-running database