Syrx.Commanders.Databases.Oracle 2.4.0

dotnet add package Syrx.Commanders.Databases.Oracle --version 2.4.0                
NuGet\Install-Package Syrx.Commanders.Databases.Oracle -Version 2.4.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Syrx.Commanders.Databases.Oracle" Version="2.4.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Syrx.Commanders.Databases.Oracle --version 2.4.0                
#r "nuget: Syrx.Commanders.Databases.Oracle, 2.4.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Syrx.Commanders.Databases.Oracle as a Cake Addin
#addin nuget:?package=Syrx.Commanders.Databases.Oracle&version=2.4.0

// Install Syrx.Commanders.Databases.Oracle as a Cake Tool
#tool nuget:?package=Syrx.Commanders.Databases.Oracle&version=2.4.0                

Syrx.Oracle

Provides Syrx support for Oracle databases. The overall experience of using Syrx remains largely the same, however Oracle support of multiple result sets differs to that of many other RDBMS implementations.

[!CAUTION] As a result of this difference in the handling of multiple result sets, repositories written against Oracle instances may need code changes either when switching to Oracle or switching away from Oracle.

Installation

[!TIP] We recommend installing the Extensions package which includes extension methods for easier configuration.

Source Command
.NET CLI dotnet add package Syrx.Oracle.Extensions
Package Manager Install-Package Syrx.Oracle.Extensions
Package Reference <PackageReference Include="Syrx.Oracle.Extensions" Version="2.4.0" />
Paket CLI paket add Syrx.Oracle.Extensions --version 2.4.0

However, if you don't need the configuration options, you can install the standalone Oracle package.

Source Command
.NET CLI dotnet add package Syrx.Oracle
Package Manager Install-Package Syrx.Oracle
Package Reference <PackageReference Include="Syrx.Oracle" Version="2.4.0" />
Paket CLI paket add Syrx.Oracle --version 2.4.0

Known Issues

Multiple Result Sets

Oracle doesn't natively support multiple result sets in the same way that SQL Server does. However, it is possible to approximate this behaviour using cursors. For example, consider the PL/SQL below.

BEGIN
    OPEN :1 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < 2;
    OPEN :2 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < 3;
    OPEN :3 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < 4;
END;

Executing this SQL via Query<> or Query<> execute may return an OracleException with a similar stack trace to this:

Oracle.ManagedDataAccess.Client.OracleException : ORA-01008: not all variables bound
https://docs.oracle.com/error-help/db/ora-01008/
Stack Trace:
     at OracleInternal.ServiceObjects.OracleFailoverMgrImpl.OnError(OracleConnection connection, CallHistoryRecord chr, Object mi, Exception ex, Boolean bTopLevelCall, Boolean& bCanRecordNewCall)
     at Oracle.ManagedDataAccess.Client.OracleCommand.ExecuteDbDataReader(CommandBehavior behavior)
  /_/Dapper/SqlMapper.cs(1156,0): at Dapper.SqlMapper.ExecuteReaderWithFlagsFallback(IDbCommand cmd, Boolean wasClosed, CommandBehavior behavior)
  /_/Dapper/SqlMapper.cs(1123,0): at Dapper.SqlMapper.QueryMultipleImpl(IDbConnection cnn, CommandDefinition& command)
  /_/Dapper/SqlMapper.cs(1108,0): at Dapper.SqlMapper.QueryMultiple(IDbConnection cnn, CommandDefinition command)
   ....<your stack>....
     at System.RuntimeMethodHandle.InvokeMethod(Object target, Void** arguments, Signature sig, Boolean isConstructor)
     at System.Reflection.MethodBaseInvoker.InvokeDirectByRefWithFewArgs(Object obj, Span`1 copyOfArgs, BindingFlags invokeAttr)

Solution

The solution to this is simple, albeit less elegant than the authors would like.

  1. In your repository code, add a reference to Syrx.Commanders.Databases.Oracle
  2. Pass a static instance of OracleDynamicParameters.Cursors() method as part of your parameters.

We recommend leverage the using static language feature.

Examples

You can see many exmaples of this in the Syrx.Oracle.Tests.Integration project.

This solution was taken from https://stackoverflow.com/a/41110515 with many thanks to nw. and greyseal96.

Without parameters
using static Syrx.Commanders.Databases.Oracle.OracleDynamicParameters;

// assumping a Func<> delegate called 'map' on a method that would not normally need parameters 
var result = _commander.Query(map, Cursors());
With parameters

There are two flavours to cursors with parameters:

  • Named cursors: if you have some reaon to know the names of the cursors.
  • Numbered cursors: these are still named cursors, but Syrx provides a convenience method to leverage default values.
Numbered Cursors

This is the most common and simplest approach. Using the SQL below as an example, we can see that OPEN :1 is the first of the numbered cursors and :id1 as the corresponding parameter for that statement.

BEGIN
    OPEN :1 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < :id1;
    OPEN :2 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < :id2;
    OPEN :3 FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < :id3;
END;

To use this in C# we need to make use of the Cursors static method of the OracleDynamicParameters type. We recommend that that you make use of the using static directive to help keep your code neat and concise.

In this example all we're doing is passing the arguments we'd normally pass directly to the Query<> method to the NumberedCursors static method. We then take the instance of the OracleDynamicParameters and pass that to the Query<> method instead.

// at the top of your file, add this using static directive. 
using static Syrx.Commanders.Databases.Oracle.OracleDynamicParameters;

var arguments = new { id1 = 2, id2 = 3, id3 = 4 };  // the arguments we'd normally pass to the Query<> method. 
var parameters = Cursors(arguments);                // return instance of OracleDynamicParameters. 
var result = _commander.Query(map, parameters);     // pass OracleDynamicParameters to the Query<> method instead.

Named Cursors

Although it's unlikely to be that common, there may be cases where you need to pass the names of the cursors.

In this SQL we can see that the names of the cursors are by_id,by_name and by_value.S

BEGIN
    OPEN :by_id    FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where id < :id;
    OPEN :by_name  FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where name like :name and id < 3;
    OPEN :by_value FOR select cast(id as number(5)) as ""Id"", name as ""Name"", value as ""Value"", modified as ""Modified"" from poco where value < :value;
END;

To use this in C# we need to make use of the overload of the Cursors static method which accepts a mandatory string[] argument.

In this example initialize a string[] variable with the cursors names corresponding to our SQL above. We then pass this string array to the overloaded Cursors method. The parameters argument of this method is optional as it's entirely possible that you may find yourself with a query that leverages cursors but that has no parameters to limit the result sets within the query.

// at the top of your file, add this using static directive. 
using static Syrx.Commanders.Databases.Oracle.OracleDynamicParameters;


string[] cursors = { "by_id", "by_name", "by_value" };          // the string array holding our cursors 
var arguments = new { id = 2, name = "entry%", value = 40 };    // the arguments we'd normally pass to the Query<> method
var parameters = Cursors(cursors, arguments);                   // return instance of OracleDynamicParameters from overload. 
var result = _commander.Query(map, parameters);                 // pass OracleDynamicParameters to the Query<> method instead.

Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (2)

Showing the top 2 NuGet packages that depend on Syrx.Commanders.Databases.Oracle:

Package Downloads
Syrx.Oracle

This package provides Syrx support for Oracle databases.

Syrx.Oracle.Extensions

This package hosts extension methods to make wiring up support for Oracle databases simpler and easier.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
2.4.0 89 1/19/2025
2.2.0 112 11/23/2024
2.1.0 104 11/15/2024
2.0.0 133 7/5/2024

Last release on .NET8.0 exclusively. Next release will include .NET9.0.