linq2db.EntityFrameworkCore 6.0.0-preview.2

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

// Install linq2db.EntityFrameworkCore as a Cake Tool
#tool nuget:?package=linq2db.EntityFrameworkCore&version=6.0.0-preview.2&prerelease                

linq2db.EntityFrameworkCore

linq2db.EntityFrameworkCore is an integration of LINQ To DB with existing EntityFrameworkCore projects. It was inspired by this issue in EF.Core repository.

Unique features

  • Fast eager loading (incomparable faster on massive Include query)
  • Global query filters optimization
  • Better SQL optimization
  • CTE support
  • MERGE support
  • Table hints
  • Analytic/Window functions support
  • Fast BulkCopy of millions records
  • Native SQL operations for updating, deleting, inserting records via LINQ query
  • Temporary tables support
  • Cross database/linked server queries.
  • Full-Text search extensions
  • A lot of extensions to cover ANSI SQL

How to use

In your code you need to initialize integration using following call:

LinqToDBForEFTools.Initialize();

After that you can just call DbContext and IQueryable extension methods, provided by LINQ To DB.

You can also register additional options like interceptors for LinqToDB during EF context registration, here is an example:

var optionsBuilder = new DbContextOptionsBuilder<MyDbContext>();
optionsBuilder.UseSqlite();
optionsBuilder.UseLinqToDB(builder =>
{
    // add custom command interceptor
    builder.AddInterceptor(new MyCommandInterceptor());
    // add additional mappings
    builder.AddMappingSchema(myCustomMappings);
    // configure SQL Server dialect explicitly
    builder.AddCustomOptions(o => o.UseSqlServer(SqlServerVersion.v2022));
});

There are many extensions for CRUD Operations missing in vanilla EF (watch our video):

// fast insert of big recordsets
ctx.BulkCopy(new BulkCopyOptions {...}, items);

// query for retrieving products that do not have duplicates by Name
var query =
    from p in ctx.Products
    from op in ctx.Products.LeftJoin(op => op.ProductID != p.ProductID && op.Name == p.Name)
    where Sql.ToNullable(op.ProductID) == null
    select p;

// insert these records into the same or another table
query.Insert(ctx.Products.ToLinqToDBTable(), s => new Product { Name = s.Name ... });

// update these records by changing name based on previous value
query.Update(prev => new Product { Name = "U_" + prev.Name ... });

// delete records that matched by query
query.Delete();

Some extensions require LINQ To DB ITable<T> interface, which could be acquired from DbSet<T> using ToLinqToDBTable() extension method.

For ITable<T> interface LINQ To DB provides several extensions that may be useful for complex databases and custom queries:

table = table.TableName("NewTableName");     // change table name in query
table = table.DatabaseName("OtherDatabase"); // change database name, useful for cross database queries.
table = table.OwnerName("OtherOwner");       // change owner.

// inserting into other existing table Products2
query.Insert(ctx.Products.ToLinqToDBTable().TableName("Products2"), s => new Product { Name = s.Name ... });

It is not required to work directly with LINQ To DB's DataConnection class but there are several ways to do that. LINQ To DB will try to reuse your configuration and select appropriate data provider:

// uing DbContext
using (var dc = ctx.CreateLinqToDBConnection())
{
   // linq queries using linq2db extensions
}

// using DbContextOptions
using (var dc = options.CreateLinqToDBConnection())
{
   // linq queries using linq2db extensions
}

You can use all LINQ To DB extension functions in your EF linq queries. Just ensure you have called ToLinqToDB() function before materializing objects for synchronous methods.

Since EF Core have defined it's own asynchronous methods, we have to duplicate them to resolve naming collisions. Async methods have the same name but with LinqToDB suffix. E.g. ToListAsyncLinqToDB(), SumAsyncLinqToDB(), ect. The same methods are added whe you need EF Core query processing but there is collision with LINQ To DB and they have extensions with EF suffix - ToListAsyncEF(), SumAsyncEF(), ect.

using (var ctx = CreateAdventureWorksContext())
{
    var productsWithModelCount =
        from p in ctx.Products
        select new
        {
            // Window Function
            Count = Sql.Ext.Count().Over().PartitionBy(p.ProductModelID).ToValue(),
            Product = p
        };

    var neededRecords =
        from p in productsWithModelCount
        where p.Count.Between(2, 4) // LINQ To DB extension
        select new
        {
            p.Product.Name,
            p.Product.Color,
            p.Product.Size,
            // retrieving value from column dynamically
            PhotoFileName = Sql.Property<string>(p.Product, "ThumbnailPhotoFileName")
        };

    // ensure we have replaced EF context
    var items1 = neededRecords.ToLinqToDB().ToArray();       
    
    // async version
    var items2 = await neededRecords.ToLinqToDB().ToArrayAsync(); 
    
    // and simple bonus - how to generate SQL
    var command = neededRecords.ToLinqToDB().ToSqlQuery();
}

Also check existing tests in test project for some examples.

Why should I want to use it?

There are many reasons. Some of them:

  • you want to use advanced SQL functionality, not supported or poorly supported by EntityFrameworkCore like BulkCopy support, SQL MERGE operations, convinient DML (Insert/Delete/Update) operations and many-many-many other features LINQ To DB provides, but you need change tracking functionality that EntityFramework provides.
  • you want to migrate to LINQ To DB, but need to do it step-by-step.
  • just because LINQ To DB is cool.

Current status

Below is a list of providers, that should work right now:

  • SQL Server
  • MySQL (including Devart and Pomelo providers)
  • PostgreSQL (Both npgsql and Devart providers)
  • SQLite (including Devart provider)
  • Firebird
  • DB2 LUW
  • Oracle
  • SQL Server CE

Known limitations

  • No lazy loading support
  • No way to work with in-memory database
  • No TPT (table per type) support
  • No direct many-to-many support

Help! It doesn't work

If you encounter any issue with this library, first check issues to see if it was already reported and if not, feel free to report new issue.

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 (18)

Showing the top 5 NuGet packages that depend on linq2db.EntityFrameworkCore:

Package Downloads
SimpleIdServer.Scim.Persistence.EF

EntityFramework persistence layer for SCIM2.0.

DotNetBrightener.DataAccess.EF

Package Description

fbognini.Infrastructure

Package Description

DotNetBrightener.Core.Logging.DbStorage

Package Description

DotNetBrightener.DataAccess.DataMigration

Package Description

GitHub repositories (5)

Showing the top 5 popular GitHub repositories that depend on linq2db.EntityFrameworkCore:

Repository Stars
bitwarden/server
Bitwarden infrastructure/backend (API, database, Docker, etc).
simpleidserver/SimpleIdServer
OpenID, OAuth 2.0, SCIM2.0, UMA2.0, FAPI, CIBA & OPENBANKING Framework for ASP.NET Core
HandyOrg/HandyWinGet
GUI for installing apps through WinGet and Creating Yaml file
win7user10/Laraue.EfCoreTriggers
Library to write triggers in C# with EF.Core
SapiensAnatis/Dawnshard
Server emulator for Dragalia Lost
Version Downloads Last updated
9.1.0-preview.2 0 1/22/2025
9.0.0 22,631 11/13/2024
9.0.0-rc.2 171 10/22/2024
8.1.0 490,796 3/4/2024
8.0.0 105,872 12/20/2023
8.0.0-rc.2 7,629 10/16/2023
7.7.0 39,324 3/4/2024
7.6.0 206,229 10/16/2023
7.5.0 220,812 5/18/2023
7.3.0 40,362 3/21/2023
7.2.0 32,284 2/26/2023
7.1.0 87,176 12/15/2022
7.0.0 23,609 11/21/2022
6.18.0-preview.2 0 1/22/2025
6.17.0 10,913 3/4/2024
6.16.0 23,716 10/16/2023
6.15.0 123,458 5/18/2023
6.13.0 13,634 3/21/2023
6.12.0 10,352 2/26/2023
6.11.0 144,663 12/15/2022
6.10.0 75,665 10/11/2022
6.9.0 43,986 9/6/2022
6.8.0 83,531 5/30/2022
6.7.1 109,392 4/8/2022
6.7.0 1,145 4/8/2022
6.6.1 126,884 11/15/2021
6.6.0 1,470 11/11/2021
6.5.0 21,169 11/8/2021
6.4.0 495 10/15/2021
6.3.0 318 10/6/2021
6.2.0 419 9/15/2021
6.1.0 772 8/13/2021
6.0.0 4,367 8/4/2021
6.0.0-preview.2 0 1/22/2025
5.19.0 4,149 3/4/2024
5.18.0 965 10/16/2023
5.17.0 8,729 5/18/2023
5.15.0 4,546 2/26/2023
5.14.0 1,400 12/15/2022
5.13.0 5,099 10/11/2022
5.12.0 1,453 9/6/2022
5.11.0 20,673 5/30/2022
5.10.0 13,000 4/8/2022
5.9.0 18,539 11/11/2021
5.8.0 54,844 10/15/2021
5.7.0 4,366 10/6/2021
5.6.0 23,173 9/15/2021
5.5.0 10,976 8/13/2021
5.4.0 17,806 7/12/2021
5.3.1 19,635 6/10/2021
5.3.0 4,054 6/3/2021
5.2.1 60,090 4/30/2021
5.2.0 14,038 3/25/2021
5.1.6 7,770 3/9/2021
5.1.5 3,313 2/18/2021
5.1.4 3,455 2/15/2021
5.1.3 4,323 1/29/2021
5.1.2 1,571 1/26/2021
5.1.1 953 1/25/2021
5.1.0 16,316 1/14/2021
5.0.2 16,898 11/14/2020
5.0.0 3,820 11/13/2020
3.28.0-preview.2 0 1/22/2025
3.27.0 8,001 3/4/2024
3.26.0 945 10/16/2023
3.25.0 1,174 5/18/2023
3.22.0 925 12/15/2022
3.21.0 1,075 10/11/2022
3.20.0 1,128 9/6/2022
3.19.0 9,336 5/30/2022
3.18.0 3,229 4/8/2022
3.17.0 35,352 11/11/2021
3.16.0 20,030 10/15/2021
3.15.0 1,037 10/6/2021
3.14.0 995 9/15/2021
3.13.0 1,372 8/13/2021
3.12.0 3,818 7/12/2021
3.11.1 2,268 6/10/2021
3.11.0 4,154 6/3/2021
3.10.1 31,340 4/30/2021
3.10.0 23,305 3/25/2021
3.9.6 9,748 3/9/2021
3.9.5 2,805 2/18/2021
3.9.4 949 2/15/2021
3.9.3 3,718 1/29/2021
3.9.2 1,028 1/26/2021
3.9.1 974 1/25/2021
3.9.0 1,501 1/14/2021
3.8.0 3,736 11/13/2020
3.7.0 159,255 10/20/2020
3.6.0 12,046 9/3/2020
3.5.0 18,408 7/19/2020
3.4.0 5,972 7/10/2020
3.3.0 10,919 6/19/2020
3.2.0 5,826 5/30/2020
3.1.0 37,179 5/21/2020
3.0.0 4,068 4/2/2020
2.14.0 266 3/4/2024
2.13.0 449 10/16/2023
2.12.0 709 5/18/2023
2.10.0 790 2/26/2023
2.9.0 804 12/15/2022
2.8.0 907 10/11/2022
2.7.0 898 9/6/2022
2.6.0 1,458 5/30/2022
2.5.0 1,384 4/8/2022
2.4.0 6,832 11/11/2021
2.3.0 969 10/15/2021
2.2.0 974 10/6/2021
2.1.0 911 9/15/2021
2.0.5 21,718 2/27/2020
2.0.4 4,018 2/11/2020
2.0.3 1,075 2/10/2020
2.0.2 2,085 2/1/2020
2.0.1 1,138 1/30/2020
2.0.0 11,376 10/9/2019
1.7.1 4,470 1/30/2020
1.7.0 37,914 7/25/2019
1.6.0 88,826 4/26/2019
1.5.2 1,593 3/4/2019
1.5.1 1,305 3/4/2019
1.5.0 5,670 2/4/2019
1.4.0 1,541 12/14/2018
1.3.0 1,329 12/7/2018
1.2.0 2,930 9/3/2018
1.0.1 3,570 7/19/2018
1.0.0 2,759 7/16/2018