Working With Transactions In EF Core

Working With Transactions In EF Core

4 min read ·

Thank you to our sponsors who keep this newsletter free to the reader:

Ultimate ASP.NET Core Web API - Second Edition is a step-by-step and easy-to-digest guide to creating scalable, robust, and high-performance Web APIs.

QuadSpinner Highlighter is an open-source Visual Studio extension that lets you highlight important objects and arbitrary texts to help you navigate your code more easily.

Every software engineer working with SQL databases needs to know about transactions. And since most of the time the SQL database will be abstracted by an ORM like EF Core, it's important to understand how you can work with transactions using the available abstractions.

So today, I'll show you how to work with transactions in EF Core.

Here's what we will cover:

  • Default transaction behavior
  • Creating transactions
  • Using existing transactions

Let's dive in.

Default Transaction Behavior

What is the default EF Core transaction behavior?

By default, all changes made in a single call to SaveChanges are applied in a transaction. If any of the changes fail, the entire transaction is rolled back and no changes are applied to the database. Only if all changes are successfully persisted to the database, the call to SaveChanges can complete.

This is a wonderful feature of SQL databases and it saves us many headaches. We don't have to think about the databases remaining in an inconsistent state, because database transactions can do the work for us.

Let's take a look at an example.

using var context = new ShoppingContext();

context.LineItems.Add(new LineItem
{
    ProductId = productId,
    Quantity = quantity
});

var stock = context.Stock.FirstOrDefault(s => s.ProductId == productId);

stock.Quantity -= quantity;

context.SaveChanges();

Because we are adding a LineItem, and in the same scope reducing the Stock quantity, the call to SaveChanges will apply both changes inside of a transaction. We can guarantee that the database will remain in a consistent state.

Creating Transactions With EF Core

What if you want to have more control over transactions when working with EF Core?

You can manually create a transaction by accessing the Database facade available on a DbContext instance and calling BeginTransaction.

Here's an example where we have multiple calls to SaveChanges. In the default scenario, both calls would run in their own transaction. This leaves the possibility of the second call to SaveChanges failing, and leaving the database in an inconsistent state.

using var context = new ShoppingContext();
using var transaction  = context.Database.BeginTransaction();

try
{
    context.LineItems.Add(new LineItem
    {
        ProductId = productId,
        Quantity = quantity
    });

    context.SaveChanges();

    var stock = context.Stock.FirstOrDefault(s => s.ProductId == productId);

    stock.Quantity -= quantity;

    context.SaveChanges();

    // When we commit the changes, they will be applied to the databases.
    // The transaction will auto-rollback when it is disposed,
    // if any command fails.
    transaction.Commit();
}
catch (Exception)
{
    transaction.Rollback();
}

We call BeginTransaction to manually start a new database transaction. This will create a new transaction and return it, so that we can Commit the transaction when we want to complete the operation. You also want to add a try-catch block around your code, so that you can Rollback the transaction if there are any exceptions.

Using Existing Transactions With EF Core

Creating a transaction using the EF Core DbContext isn't the only option. You can create a SqlTransaction instance and pass it to EF Core, so that the changes applied with EF Core can be committed inside the same transaction.

Here's what I mean:

using var sqlConnection = new SqlTransaction(connectionString);
sqlConnection.Open();

using var transaction = sqlConnection.BeginTransaction();

try
{
    using var context = new ShoppingContext();

    // Tell EF Core to use an existing transaction.
    context.UseTransaction(transaction);

    context.LineItems.Add(new LineItem
    {
        ProductId = productId,
        Quantity = quantity
    });

    context.SaveChanges();

    var stock = context.Stock.FirstOrDefault(s => s.ProductId == productId);

    stock.Quantity -= quantity;

    context.SaveChanges();

    transaction.Commit();
}
catch (Exception)
{
    transaction.Rollback();
}

In Summary

EF Core has excellent support for transactions and it's very easy to work with.

You have three options available:

  • Rely on the default transaction behavior
  • Create a new transaction
  • Use an existing transaction

Most of the time, you want to rely on the default behavior and not have to think about it.

As soon as you need to perform multiple SaveChanges calls, you should manually create a transaction, and manage the transaction yourself.

See you next week, and have an excellent Saturday.


Whenever you're ready, there are 4 ways I can help you:

  1. Modular Monolith Architecture (COMING SOON): This in-depth course will transform the way you build monolith systems. You will learn the best practices for applying the Modular Monolith architecture in a real-world scenario. Join the waitlist here.
  2. Pragmatic Clean Architecture: This comprehensive course will teach you the system I use to ship production-ready applications using Clean Architecture. Learn how to apply the best practices of modern software architecture. Join 2,600+ students here.
  3. Patreon Community: Join a community of 1,050+ engineers and gain access to the source code I use in my YouTube videos, early access to future videos, and exclusive discounts for my courses. Join 1,050+ engineers here.
  4. Promote yourself to 46,000+ subscribers by sponsoring this newsletter.

Become a Better .NET Software Engineer

Join 46,000+ engineers who are improving their skills every Saturday morning.