diff --git a/Dapper/CommandDefinition.cs b/Dapper/CommandDefinition.cs index 5860cd461..87d0d7119 100644 --- a/Dapper/CommandDefinition.cs +++ b/Dapper/CommandDefinition.cs @@ -131,6 +131,9 @@ internal IDbCommand SetupCommand(IDbConnection cnn, Action p private static SqlMapper.Link> commandInitCache; + internal static void ResetCommandInitCache() + => SqlMapper.Link>.Clear(ref commandInitCache); + private static Action GetInit(Type commandType) { if (commandType == null) @@ -141,14 +144,15 @@ private static Action GetInit(Type commandType) } var bindByName = GetBasicPropertySetter(commandType, "BindByName", typeof(bool)); var initialLongFetchSize = GetBasicPropertySetter(commandType, "InitialLONGFetchSize", typeof(int)); + var fetchSize = GetBasicPropertySetter(commandType, "FetchSize", typeof(long)); action = null; - if (bindByName != null || initialLongFetchSize != null) + if (bindByName is not null || initialLongFetchSize is not null || fetchSize is not null) { var method = new DynamicMethod(commandType.Name + "_init", null, new Type[] { typeof(IDbCommand) }); var il = method.GetILGenerator(); - if (bindByName != null) + if (bindByName is not null) { // .BindByName = true il.Emit(OpCodes.Ldarg_0); @@ -156,7 +160,7 @@ private static Action GetInit(Type commandType) il.Emit(OpCodes.Ldc_I4_1); il.EmitCall(OpCodes.Callvirt, bindByName, null); } - if (initialLongFetchSize != null) + if (initialLongFetchSize is not null) { // .InitialLONGFetchSize = -1 il.Emit(OpCodes.Ldarg_0); @@ -164,6 +168,18 @@ private static Action GetInit(Type commandType) il.Emit(OpCodes.Ldc_I4_M1); il.EmitCall(OpCodes.Callvirt, initialLongFetchSize, null); } + if (fetchSize is not null) + { + var snapshot = SqlMapper.Settings.FetchSize; + if (snapshot >= 0) + { + // .FetchSize = {withValue} + il.Emit(OpCodes.Ldarg_0); + il.Emit(OpCodes.Castclass, commandType); + il.Emit(OpCodes.Ldc_I8, snapshot); // bake it as a constant + il.EmitCall(OpCodes.Callvirt, fetchSize, null); + } + } il.Emit(OpCodes.Ret); action = (Action)method.CreateDelegate(typeof(Action)); } diff --git a/Dapper/SqlMapper.Link.cs b/Dapper/SqlMapper.Link.cs index 277f2a9e9..fe6de9226 100644 --- a/Dapper/SqlMapper.Link.cs +++ b/Dapper/SqlMapper.Link.cs @@ -13,6 +13,8 @@ public static partial class SqlMapper /// The value type of the cache. internal class Link where TKey : class { + public static void Clear(ref Link head) => Interlocked.Exchange(ref head, null); + public static bool TryGet(Link link, TKey key, out TValue value) { while (link != null) diff --git a/Dapper/SqlMapper.Settings.cs b/Dapper/SqlMapper.Settings.cs index cc71238b2..192cd893d 100644 --- a/Dapper/SqlMapper.Settings.cs +++ b/Dapper/SqlMapper.Settings.cs @@ -1,5 +1,6 @@ using System; using System.Data; +using System.Threading; namespace Dapper { @@ -63,7 +64,9 @@ static Settings() public static void SetDefaults() { CommandTimeout = null; - ApplyNullValues = false; + ApplyNullValues = PadListExpansions = UseIncrementalPseudoPositionalParameterNames = false; + AllowedCommandBehaviors = DefaultAllowedCommandBehaviors; + FetchSize = InListStringSplitCount = -1; } /// @@ -99,6 +102,26 @@ public static void SetDefaults() /// instead of the original name; for most scenarios, this is ignored since the name is redundant, but "snowflake" requires this. /// public static bool UseIncrementalPseudoPositionalParameterNames { get; set; } + + /// + /// If assigned a non-negative value, then that value is applied to any commands FetchSize property, if it exists; + /// see https://docs.oracle.com/en/database/oracle/oracle-database/18/odpnt/CommandFetchSize.html; note that this value + /// can only be set globally - it is not intended for frequent/contextual changing. + /// + public static long FetchSize + { + get => Volatile.Read(ref s_FetchSize); + set + { + if (Volatile.Read(ref s_FetchSize) != value) + { + Volatile.Write(ref s_FetchSize, value); + CommandDefinition.ResetCommandInitCache(); // if this setting is useful: we've invalidated things + } + } + } + + private static long s_FetchSize = -1; } } } diff --git a/docs/index.md b/docs/index.md index 326c30cc8..2d7a2bffe 100644 --- a/docs/index.md +++ b/docs/index.md @@ -22,6 +22,8 @@ Note: to get the latest pre-release build, add ` -Pre` to the end of the command ### unreleased + +- add global `FetchSize` setting for use with Oracle (#1946 via mgravell, fixes #1945) (also add some missing logic in `Settings.Reset()`) - add underscore handling with constructors (#1786 via @jo-goro, fixes #818; also #1947 via mgravell) (note: new PRs will not be merged until they add release note wording here)