From b6cbcd60cf0682e0ad84648984da10d6178ea503 Mon Sep 17 00:00:00 2001 From: BillRob Date: Wed, 30 Oct 2024 20:43:18 -0500 Subject: [PATCH] Custom CreateCommand method. Resolves #2127 This allows the user to provide their own IDbCommand to be used by Dapper internally. --- Dapper/CommandDefinition.cs | 32 +++++++++++++++++++++++++++++++- Dapper/PublicAPI.Shipped.txt | 1 + tests/Dapper.Tests/MiscTests.cs | 17 +++++++++++++++++ 3 files changed, 49 insertions(+), 1 deletion(-) diff --git a/Dapper/CommandDefinition.cs b/Dapper/CommandDefinition.cs index 19963ba67..c9c6dae2c 100644 --- a/Dapper/CommandDefinition.cs +++ b/Dapper/CommandDefinition.cs @@ -66,6 +66,11 @@ internal void OnCompleted() /// public CommandFlags Flags { get; } + /// + /// The create Func that will be used to create the IDbCommand. When null the connection.CreateCommand() will be used. + /// + internal Func? CreateCommand { get; } + /// /// Can async queries be pipelined? /// @@ -95,6 +100,26 @@ public CommandDefinition(string commandText, object? parameters = null, IDbTrans CancellationToken = cancellationToken; } + /// + /// Initialize the command definition + /// + /// The text for this command. + /// The a custom way to create a IDbCommand object. This is to override the default connection.CreateCommand() behavior. + /// The parameters for this command. + /// The transaction for this command to participate in. + /// The timeout (in seconds) for this command. + /// The for this command. + /// The behavior flags for this command. + /// The cancellation token for this command. + public CommandDefinition(string commandText, Func createCommand, object? parameters = null, IDbTransaction? transaction = null, + int? commandTimeout = null, CommandType? commandType = null, CommandFlags flags = CommandFlags.Buffered + , CancellationToken cancellationToken = default + ) + : this(commandText, parameters, transaction, commandTimeout, commandType, flags, cancellationToken) + { + CreateCommand = createCommand; + } + internal static CommandType InferCommandType(string sql) { // if the sql contains any whitespace character (space/tab/cr/lf/etc - via unicode), @@ -120,7 +145,12 @@ private CommandDefinition(object? parameters, CommandFlags flags) : this() internal IDbCommand SetupCommand(IDbConnection cnn, Action? paramReader) { - var cmd = cnn.CreateCommand(); + IDbCommand cmd; + if (CreateCommand == null) + cmd = cnn.CreateCommand(); + else + cmd = CreateCommand(cnn); + var init = GetInit(cmd.GetType()); init?.Invoke(cmd); if (Transaction is not null) diff --git a/Dapper/PublicAPI.Shipped.txt b/Dapper/PublicAPI.Shipped.txt index c1b08ea99..669c24d8d 100644 --- a/Dapper/PublicAPI.Shipped.txt +++ b/Dapper/PublicAPI.Shipped.txt @@ -9,6 +9,7 @@ Dapper.CommandDefinition.Buffered.get -> bool Dapper.CommandDefinition.CancellationToken.get -> System.Threading.CancellationToken Dapper.CommandDefinition.CommandDefinition() -> void Dapper.CommandDefinition.CommandDefinition(string! commandText, object? parameters = null, System.Data.IDbTransaction? transaction = null, int? commandTimeout = null, System.Data.CommandType? commandType = null, Dapper.CommandFlags flags = Dapper.CommandFlags.Buffered, System.Threading.CancellationToken cancellationToken = default(System.Threading.CancellationToken)) -> void +Dapper.CommandDefinition.CommandDefinition(string! commandText, System.Func! createCommand, object? parameters = null, System.Data.IDbTransaction? transaction = null, int? commandTimeout = null, System.Data.CommandType? commandType = null, Dapper.CommandFlags flags = Dapper.CommandFlags.Buffered, System.Threading.CancellationToken cancellationToken = default(System.Threading.CancellationToken)) -> void Dapper.CommandDefinition.CommandText.get -> string! Dapper.CommandDefinition.CommandTimeout.get -> int? Dapper.CommandDefinition.CommandType.get -> System.Data.CommandType? diff --git a/tests/Dapper.Tests/MiscTests.cs b/tests/Dapper.Tests/MiscTests.cs index 9323be671..23bf9b7ae 100644 --- a/tests/Dapper.Tests/MiscTests.cs +++ b/tests/Dapper.Tests/MiscTests.cs @@ -1394,5 +1394,22 @@ public void SetDynamicProperty_WithValueType_Succeeds() obj.NewProperty = true; Assert.True(obj.NewProperty); } + + [Fact] + public void TestCreateCommandCalled() + { + var customCreateCommandCalled = false; + var createCommand = new Func(conn => + { + customCreateCommandCalled = true; + return conn.CreateCommand(); + }); + + var definition = new CommandDefinition("select 1", createCommand); + + connection.Query(definition); + + Assert.True(customCreateCommandCalled); + } } }