OracleUdtClassGenerator 0.2.0

There is a newer version of this package available.
See the version list below for details.
dotnet add package OracleUdtClassGenerator --version 0.2.0                
NuGet\Install-Package OracleUdtClassGenerator -Version 0.2.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="OracleUdtClassGenerator" Version="0.2.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add OracleUdtClassGenerator --version 0.2.0                
#r "nuget: OracleUdtClassGenerator, 0.2.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 OracleUdtClassGenerator as a Cake Addin
#addin nuget:?package=OracleUdtClassGenerator&version=0.2.0

// Install OracleUdtClassGenerator as a Cake Tool
#tool nuget:?package=OracleUdtClassGenerator&version=0.2.0                

Oracle UDT Class Generator

NuGet Badge

This NuGet package contains a source generator that emits C# source code for mapping to and from Oracle User-Defined Data Types. This allows your project to easily send records and arrays of records to and from an Oracle database.

Simply install the NuGet package into the project in which you wish to generate mapping classes. The generator runs entirely at compile time and does not add any runtime dependencies to your project.

The emitted code has no dependencies other than Oracle.ManagedDataAccess.Core which you must manually add to the client project.

For examples of using UDTs please refer to the 'OracleExamples' in my GitHub solution BassUtils and its corresponding NuGet Package

'oraudt' Input Format

The C# code necessary to map UDTs is quite long winded and error-prone, and this package greatly simplifies the process by using a domain-specific language which allows very concise specification of the Oracle UDT.

  1. Create a file ending in .oraudt, for example Person.oraudt.
  2. Set the file's build action to C# analyzer additional file.
  3. Write a DSL specification for your UDT.

The DSL consists of a series of sections. First, the class specification:

class PersonRecord MYSCHEMA.objPerson MYSCHEMA.tblPerson
  • The first identifier, PersonRecord is the name of the C# class to be generated. It's mandatory.
  • The second identifier, MYSCHEMA.objPerson is the name of the Oracle UDT type that you created in the database using "CREATE TYPE objPerson AS OBJECT". This is also mandatory.
  • The third identifier, "MYSCHEMA.tblPerson" is the name of the Oracle UDT nested table type that you created, for example "CREATE TYPE tblPerson AS TABLE OF objPerson". This field is optional. If it is not present, then C# classes for mapping arrays of records will not be generated.

Then come 3 lines which are all optional:

class PersonRecord MYSCHEMA.objPerson MYSCHEMA.tblPerson
    Namespace People
    DebuggerDisplay "{Sku}/{Price}"
    ToString "{Sku}/{Color}/{Size}/{Price}"

Namespace specifies the exact namespace to generate the code in. If absent, the generator tries to guess the namespace based on the folder that the .oraudt file is in. The string .OracleUDTs will always be appended to that namespace.

DebuggerDisplay and ToString can be used to automatically generate a DebuggerDisplayAttribute and a ToString method, respectively. This is only intended for simple usages, for more complex requirements, leave them blank and write your own implementations - this is easy because all the code generated is in partial classes.

Finally we come to the field list specification. This section is mandatory:

    Fields [
          FirstName,
          LastName
          int Age
          string EducationLevel EDUCLEVEL
    ]

It consists of the word "Fields" and then a list of fields inside "[]" brackets, and separated either by commas or newlines.

An individual field specification can have 1 to 3 parts. A 3 part name is the most flexible; if you use a two part name or a 1 part name then conventions are used to fill in this missing parts.

An example of a 3 part name is

decimal? Width WDTH

Here decimal? is the C# type of the property, Width is the C# name of the property, and WDTH is the name of the field in the Oracle OBJECT type.

A 2 part name omits the last component and defaults it to the name of the field - in the following example, it will be WEIGHT (it must always be uppercase).

decimal? Weight

Finally, 1 part name allows you to omit the C# type name which causes it to be defaulted to string

FirstName

So this last field is equivalent to

string FirstName FIRSTNAME

This can allow for very succinct object definitions as string fields are very common.

Full Examples

A simple Person class:

class PersonRecord MYSCHEMA.objPerson MYSCHEMA.tblPerson
    Namespace People
    Fields [int Age,FirstName,LastName]

A more expansive example:

class MyClass MYSCHEMA.objArticleRecord MYSCHEMA.tblArticleRecord
    Namespace MyNamespace
    DebuggerDisplay "{Sku}"
    ToString "Hello world"
    Fields [
        int? MyNullableProp SOMENULLABLEPROP
        JustSomeString
        int AnIntegerProperty
        decimal? ThreeParter SOMEORACLENAME
        SingleParter
    ]

Notes

A single .oraudt file can contain multiple class stanzas. Each will be generated into a separate C# file.

Bool fields do not map well to Oracle types since Oracle SQL has no concept of a boolean type. Typically a NUMBER(1) or a CHAR(1) is used. Therefore I suggest using the partial class facility to create a boolean property which proxies the mapped field.

All the generated files will be in a single virtual folder, therefore all your class names must be distinct across your entire project.

Output Format

Two classes will be generated for each .oraudt file, or four if the Oracle table type is specified. Given a class name of Person, a class Person will be created containing all the fields, and a class PersonFactory which is required by Oracle Data Acccess to instantiate the Person.

If you specify the table type then a PersonArray type will also be generated, plus a factory for that, called PersonArrayFactory. The names of all these classes are derived from the class name and cannot be overridden. They are all partial classes.

The generated code can be seen under the Analyzers node in Visual Studio.

n.b. Visual Studio can be temperamental about showing these files. Sometimes it is necessary to close and reopen Visual Studio.

If all goes well you will also have a log file that looks something like this:

/*
DateTime.UtcNow: 2022-01-01T22:33:03.5397481Z

Found file C:\repos\OracleUdtClassGenerator\OracleUdtClassGenerator.ConsoleTestHarness\ArticleMaster.oraudt
  Found spec for ArticleMasterRecord with 17 fields
  Generated file ArticleMasterRecord.g.cs in namespace OracleUdtClassGenerator.ConsoleTestHarness.OracleUdts
Found file C:\repos\OracleUdtClassGenerator\OracleUdtClassGenerator.ConsoleTestHarness\MyClass.oraudt
  Found spec for MyClass with 5 fields
  Generated file MyClass.g.cs in namespace MyNamespace
Found file C:\repos\OracleUdtClassGenerator\OracleUdtClassGenerator.ConsoleTestHarness\SubFolder\ArticleMaster2.oraudt
  Found spec for ArticleMasterRecord2 with 17 fields
  Generated file ArticleMasterRecord2.g.cs in namespace OracleUdtClassGenerator.ConsoleTestHarness.SubFolder.OracleUdts
Found file C:\repos\OracleUdtClassGenerator\OracleUdtClassGenerator.ConsoleTestHarness\Person.oraudt
  Found spec for PersonRecord with 3 fields
  Generated file PersonRecord.g.cs in namespace People
*/

Further Reading

An article describing the implementation of this source generator is available on my blog.

I have another NuGet package available called BassUtils.Oracle which can help with calling procs with UDTs, for example it will create OracleParameters correctly for you. It is also described in an article on my blog.

There are no supported framework assets in this package.

Learn more about Target Frameworks and .NET Standard.

  • .NETStandard 2.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.1.0 1,741 4/7/2022
1.0.0 557 1/12/2022
0.2.0 330 1/2/2022
0.1.0 323 1/1/2022

2021-01-02 Initial release