.NET Core is great. But if you are looking to use an ORM (Object Relational Mapper) you are kind of out of luck unless you want to use Entity Framework and there happens to be a provider for it. Right now there is no EF Provider from Oracle, so to do true .NET core with Oracle, Entity Framework isn't going to work.
It may be possible to get NHibernate kind of working with the ODP.NET Core drivers, but it's not officially supported. So I wanted to look at what else is out there.
Let's look at Linq2DB. Which is available at Linq2DB on GitHub. What will this do for us?
LINQ to DB is the fastest LINQ database access library offering a simple, light, fast, and type-safe layer between your POCO objects and your database.
Architecturally it is one step above micro-ORMs like Dapper, Massive, or PetaPoco, in that you work with LINQ expressions, not with magic strings, while maintaining a thin abstraction layer between your code and the database. Your queries are checked by the C# compiler and allow for easy refactoring.
However, it's not as heavy as LINQ to SQL or Entity Framework. There is no change-tracking, so you have to manage that yourself, but on the positive side you get more control and faster access to your data.
In other words LINQ to DB is type-safe SQL.
Type safe and agnostic SQL! That means we can write the same LINQ queries and they are agnostic from the DB provider. So we could do stuff for Oracle, and maybe in some of our tests use an In Memory Database like SQL Lite! (This post only deals with Oracle, I will show you how to do the In Memory DB SQL tests in another one.)
So building off the last example using .NET core (Previous Example). We are going to want to do the following:
So we create a new project called "Northwind" and we create a folder called Models.
Then we use the nuget package manager to install the Linq2DB for core:
Install-Package linq2db.core
Now we need to create our models. Using SQL Wrangler which supports generation of models with mapping attributes for Linq2DB we execute "select * from northwind.products", click generate C#, and copy and paste the output into a new class called Products in the models folder. (Look at the comments in the SQL Wrangler output, it is basically a block of code snippets.
Next we need to create some way to tell Linq2DB where our database is! We want to create a class for our DbSettings.cs, to specify connection string and what provider to use. We also want a Startup.cs so we can start the "domain" up and start working with it.
using LinqToDB.Mapping;
namespace Northwind.Models
{
[Table("PRODUCTS", Schema = "NORTHWIND")]
public class Product
{
[PrimaryKey, Identity]
[Column(Name = "PRODUCT_ID"), NotNull]
public int Id { get; set; }
[Column(Name = "PRODUCT_NAME"), NotNull]
public string ProductName { get; set; }
[Column(Name = "SUPPLIER_ID"), NotNull]
public int SupplierId { get; set; }
[Column(Name = "CATEGORY_ID"), NotNull]
public int CategoryId { get; set; }
[Column(Name = "QUANTITY_PER_UNIT"), Nullable]
public string QuantityPerUnit { get; set; }
[Column(Name = "UNIT_PRICE"), NotNull]
public double UnitPrice { get; set; }
[Column(Name = "UNITS_IN_STOCK"), NotNull]
public int UnitsInStock { get; set; }
[Column(Name = "UNITS_ON_ORDER"), NotNull]
public int UnitsOnOrder { get; set; }
[Column(Name = "REORDER_LEVEL"), NotNull]
public int ReorderLevel { get; set; }
[Column(Name = "DISCONTINUED"), NotNull]
public string Discontinued { get; set; }
}
}
Our DbSettings.cs:
Our Startup
using System.Collections.Generic;
using LinqToDB.Configuration;
namespace Northwind
{
public class ConnectionStringSettings : IConnectionStringSettings
{
public string ConnectionString { get; set; }
public string Name { get; set; }
public string ProviderName { get; set; }
public bool IsGlobal => false;
}
public class NorthWindDbSettings : ILinqToDBSettings
{
public IEnumerable
So now we can just use Northwind.DbNorthwindStartup with a connection string, and we can start using it! How does this work from the Console Application?
using LinqToDB;
using LinqToDB.Data;
using Northwind.Models;
namespace Northwind
{
public static class DbNorthwindStartup
{
private static bool _started;
public static void Init(string connectionString)
{
if (!_started)
{
DataConnection.DefaultSettings = new NorthWindDbSettings(connectionString);
_started = true;
}
}
}
public class DbNorthwind : DataConnection
{
public DbNorthwind() : base("Northwind") { }
//Register our mappings
public ITable
From Program.cs:
using System;
using System.IO;
using System.Linq;
using Microsoft.Extensions.Configuration;
using Northwind;
namespace OdpNetCoreExample
{
class Program
{
public static IConfiguration Configuration { get; set; }
static void Main(string[] args)
{
var builder = new ConfigurationBuilder()
.SetBasePath(Directory.GetCurrentDirectory())
.AddJsonFile("appSettings.json");
Configuration = builder.Build();
DbNorthwindStartup.Init(Configuration.GetConnectionString("NorthwindDb"));
GetData();
}
static void GetData()
{
using (var db = new DbNorthwind())
{
var query = db.Product.Where(o => o.Id > 25).OrderByDescending(o=>o.ProductName)
.ToList();
foreach (var item in query.ToList())
{
Console.WriteLine(item.ProductName);
}
}
}
}
}
And running the console application now should yield:
Because instead of SQL you are writing LINQ queries and you are abstracted away from the DB layer. So instead of writing...
WRITE...
select * from northwind.product where product_id>25 order by product_name
There are benefits to this. I will highlight them in a future blog post. But the main one is, instead of pointing your tests at Oracle you can point them at something like In Memory SQL Lite.
var query = db.Product.Where(o => o.Id > 25).OrderByDescending(o=>o.ProductName)
.ToList();