Welcome to the first chapter of my “Mastering EF Core” series!

This series will dive into real-life scenarios and lessons I’ve learned while writing clean, object-oriented code using Entity Framework (EF) Core. My goal is to share with you practical insights that can help you unlock the full potential of EF Core in your applications.

Introduction

Entity Framework Core is a powerful ORM that facilitates data access in .NET applications. In this article, we’ll explore how to manage entity relationships without relying on the Entity Framework DbContext within your domain logic. I will emphasize the importance of understanding OnDelete behaviors and how they can impact your application’s behavior.

The Scenario: Devices and Permissions

Imagine we’re building an application that manages devices and their permissions. While our example uses “Device” and “Permission,” the concepts apply broadly to any domain involving complex relationships.

Tech Stack

This example is designed for applications using .NET 8, EF Core, and either Azure SQL/SQL Server as the database.

Entity Definitions

Here’s how our entities look:

public sealed class Device
{
    public int Id { get; private set; }
    public string Name { get; private set; }
    public List<Permission> Permissions { get; private set; }

    public Device(int id, string name)
    {
        Id = id;
        Name = name;
        Permissions = new List<Permission>();
    }
    
    public void RemovePermission(int permissionId)
    {
        Permissions.RemoveAll(p => p.Id == permissionId);
    }
}

public sealed class Permission
{
    public int Id { get; private set; }
    public DateOnly? From { get; private set; }
    public DateOnly? To { get; private set; }

    public Permission(int id, DateOnly? from, DateOnly? to)
    {
        Id = id;
        From = from;
        To = to;
    }
}

In this setup:

  • Device has a collection of Permissions.
  • We aim to remove a permission using object-oriented principles without direct manipulation our DbContext implementation.

The Challenge: Constraints and Delete Behaviors

When attempting to remove a permission from a device using device.RemovePermission(permissionId), we encounter a foreign key constraint error upon calling _dbContext.SaveChangesAsync(). This error arises due to the way we’ve configured the relationship in EF Core and how our database handles delete behaviors.

Initial Relationship Configuration

public class DeviceEntityTypeConfiguration : IEntityTypeConfiguration<Device>
{
    public void Configure(EntityTypeBuilder<Device> builder)
    {
        builder.HasKey(d => d.Id);
        builder.Property(d => d.Name).IsRequired();
        
        builder.HasMany(d => d.Permissions)
            .WithOne()
            .HasForeignKey(p => p.Id)
            .OnDelete(DeleteBehavior.Restrict);
    }
}

Using DeleteBehavior.Restrict means that EF will prevent the deletion of a Permission if it would violate a foreign key constraint with Device. This setup requires us to manage deletions explicitly, often leading to less clean code that directly interacts with DbContext.

The Problem in Code

Before Refactoring

public sealed class RemovePermissionsHandlerBefore
{
    private readonly DataContext _dbContext;

    public RemovePermissionsHandlerBefore(DataContext dataContext)
    {
        _dbContext = dataContext;
    }

    public async Task Handle(int deviceId, int permissionId, CancellationToken cancellationToken)
    {
        Device device = await _dbContext.Devices
            .Include(d => d.Permissions)
            .FirstOrDefaultAsync(d => d.Id == deviceId, cancellationToken);
        
        if (device is null) return;
        
        Permission permission = device.Permissions.FirstOrDefault(p => p.Id == permissionId);
        
        if (permission is null) return;
        
        // We do not want to manually remove related Permission entities from the database
        _dbContext.Permissions.Remove(permission);
        
        await _dbContext.SaveChangesAsync(cancellationToken);
    }
}

This code directly manipulates the DbContext to remove the Permission, coupling our domain logic with the data access layer.

The Solution: Understanding and using ClientCascade behavior

To achieve a cleaner, more object-oriented approach, we need to adjust our understanding of delete behaviors and how our database handles them.

Evaluating Database Support

Before making changes, it’s important to check how your database handles NoAction and Restrict. Not all databases differentiate between these settings. Azure SQL/SQL Server treats Restrict and NoAction identically. Even if we configure it as Restrict in EF Core, on the database level - is creating non-cascading same foreign key as for NoAction.

Knowing your database’s capabilities is key, as it affects whether your delete behaviors will work as expected. If the database doesn’t support a specific behavior, you could face unexpected constraint errors or data inconsistencies on the application side.

Understanding DeleteBehavior.ClientCascade

DeleteBehavior.ClientCascade is an EF Core setting that:

  • Enables EF to violate non-cascading FK constraints on application level.
  • Creates non-cascading FK on the database level.
  • Allows us to remove entities from collections and have EF Core handle the deletions upon SaveChangesAsync().

By using ClientCascade, we can remove a permission from a device’s collection, and EF Core will understand that it needs to delete the permission from the database.

Updated Relationship Configuration

public class DeviceEntityTypeConfiguration : IEntityTypeConfiguration<Device>
{
    public void Configure(EntityTypeBuilder<Device> builder)
    {
        builder.HasKey(d => d.Id);
        builder.Property(d => d.Name).IsRequired();
        
        builder.HasMany(d => d.Permissions)
            .WithOne()
            .HasForeignKey(p => p.Id)
            .OnDelete(DeleteBehavior.ClientCascade);
    }
}

Refactored Code

After Applying ClientCascade

public sealed class RemovePermissionsHandlerAfter
{
    private readonly DataContext _dbContext;

    public RemovePermissionsHandlerAfter(DataContext dataContext)
    {
        _dbContext = dataContext;
    }

    public async Task Handle(int deviceId, int permissionId, CancellationToken cancellationToken)
    {
        Device device = await _dbContext.Devices
            .Include(d => d.Permissions)
            .FirstOrDefaultAsync(d => d.Id == deviceId, cancellationToken);
        
        if (device is null) return;
        
        // Now, we can remove the permission directly on Device without manipulating the DbContext
        device.RemovePermission(permissionId);
        
        await _dbContext.SaveChangesAsync(cancellationToken);
    }
}

Now, we’re removing the permission directly from the device’s collection, adhering to object-oriented principles. EF Core changer tracker updates the removed Permission state as Deleted and removes it from the database upon calling _dbContext.SaveChangesAsync().

Important Considerations

  • Database Behavior: Always verify how your database interprets delete behaviors. Misalignment between EF Core configurations and database capabilities can lead to unexpected errors.
  • Foreign Key Recreation: Be aware that EF Core might drop and recreate foreign keys during migrations, even if they appear identical. This can have implications for database deployments and should be managed carefully.

Conclusion

By understanding and correctly applying DeleteBehavior.ClientCascade, we can write cleaner, more maintainable code that aligns with object-oriented principles. This approach allows us to manage entity relationships without tightly coupling our domain logic to the data access layer.

Stay Tuned for More!

This is just the beginning of my “Mastering EF Core” series. In future articles, I’ll dig into real-world challenges and scenarios I’ve tackled, sharing insights and solutions that have broadened my understanding of solid object-oriented programming with EF Core.

Join the Conversation

Have you faced similar issues with EF Core or entity relationships? Share your thoughts or experiences in the comments below!

Comments