R8.EntityFrameworkCore.AuditProvider 1.1.0

dotnet add package R8.EntityFrameworkCore.AuditProvider --version 1.1.0
NuGet\Install-Package R8.EntityFrameworkCore.AuditProvider -Version 1.1.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="R8.EntityFrameworkCore.AuditProvider" Version="1.1.0" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add R8.EntityFrameworkCore.AuditProvider --version 1.1.0
#r "nuget: R8.EntityFrameworkCore.AuditProvider, 1.1.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 R8.EntityFrameworkCore.AuditProvider as a Cake Addin
#addin nuget:?package=R8.EntityFrameworkCore.AuditProvider&version=1.1.0

// Install R8.EntityFrameworkCore.AuditProvider as a Cake Tool
#tool nuget:?package=R8.EntityFrameworkCore.AuditProvider&version=1.1.0

R8.EntityFrameworkCore.AuditProvider

A .NET package for Entity Framework, providing comprehensive change tracking with deep insights. Capture creation, updates, deletions, and restorations of entities, including property names, old and new values, stack traces, and user details, all neatly stored in an Audits column as JSON.

Seamless Entity Auditing: Easily integrate audit functionality into your Entity Framework applications, offering a complete audit trail enriched with stack traces and user information. Gain full visibility into entity lifecycle changes for compliance, debugging, and accountability.

Full Entity Lifecycle Visibility: Track and visualize the complete life cycle of your entities with detailed auditing. In addition to changes, this package records the stack trace of changes and user actions, enabling a deeper understanding of data evolution and robust audit trails.

Nuget Nuget Commit

Installation

dotnet add package R8.EntityFrameworkCore.AuditProvider

Usage

// ... other services

// Add AuditProvider
services.AddEntityFrameworkAuditProvider(options =>
{
    options.JsonOptions.WriteIndented = false;
    
    options.AuditFlagSupport.Created = AuditFlagState.ActionDate | AuditFlagState.Storage;
    options.AuditFlagSupport.Changed = AuditFlagState.ActionDate | AuditFlagState.Storage;
    options.AuditFlagSupport.Deleted = AuditFlagState.ActionDate | AuditFlagState.Storage;
    options.AuditFlagSupport.UnDeleted = AuditFlagState.ActionDate | AuditFlagState.Storage;
    
    options.MaxStoredAudits = 10;
    
    options.UserProvider = serviceProvider =>
    {
        var httpContextAccessor = serviceProvider.GetRequiredService<IHttpContextAccessor>();
        var user = httpContextAccessor.HttpContext?.User;
        if (user?.Identity?.IsAuthenticated == true)
        {
            var userId = user.FindFirstValue(ClaimTypes.NameIdentifier);
            var username = user.FindFirstValue(ClaimTypes.Name);
            return new AuditProviderUser(userId, new Dictionary<string, object>
            {
                { "Username", username }
            });
        }
        return null;
    };
});

services.AddDbContext<YourDbContext>((serviceProvider, optionsBuilder) =>
{
    // Your DbContext connection configuration here
    // ...
    optionsBuilder.AddEntityFrameworkAuditProviderInterceptor(serviceProvider);
});

Options

Option Type Description Default
JsonOptions System.Text.Json.JsonSerializerOptions Json serializer options to serialize and deserialize audits An optimal setting
AuditFlagSupport R8.EntityFrameworkCore.AuditProvider.AuditProviderFlagSupport Audit flags to include All flags are included
MaxStoredAudits* int? Maximum number of audits to store in Audits column null
DateTimeProvider Func<IServiceProvider, DateTime> DateTime provider to get current date time DateTime.UtcNow
UserProvider Func<IServiceProvider, EntityFrameworkAuditUser> User provider to get current user id null
  • If the number of audits exceeds this number, the earliest audits (except Created) will be removed from the column. If null, all audits will be stored.

Wiki

  • IAuditActivator interface: to start auditing entities.
  • IAuditStorage interface: to store audits in a column.
  • IAuditSoftDelete interface: to soft-delete entities.
  • IAuditCreateDate interface: to store creation date in a column.
  • IAuditUpdateDate interface: to store last update/restore date in a column.
  • IAuditDeleteDate interface: to store deletion date in a column.
  • [AuditIgnore] attribute: to ignore a property from audit.

Samples:

public record YourEntity : IAuditActivator, IAuditStorage, IAuditSoftDelete, IAuditCreateDate, IAuditUpdateDate, IAuditDeleteDate
{
    [Key]
    public int Id { get; set; }

    [Column(TypeName = "jsonb"), AuditIgnore]
    public JsonElement? Audits { get; set; }

    public bool IsDeleted { get; set; }
    
    [Column("CreatedAt", TypeName = "timestamp")]
    public DateTime? CreateDate { get; set; }
    
    [Column("UpdatedAt", TypeName = "timestamp")]
    public DateTime? UpdateDate { get; set; }
    
    [Column("DeletedAt", TypeName = "timestamp")]
    public DateTime? DeleteDate { get; set; }
    
    // ...
    // public string Name { get; set; }
    // public string Description { get; set; }
    // etc.
}

Migration

Highly recommended to test it on a test database first, to avoid any data loss.


Considerations

  • Since Microsoft Sql Server does not support json type, Audits column will be stored as nvarchar(max) and JsonElement will be serialized/deserialized to/from string. (See AggregateAuditable.cs)
  • The key to allow auditing entities is implementation of IAuditActivator to your entity.
    • the IAuditStorage, IAuditSoftDelete, IAuditCreateDate, IAuditUpdateDate, and IAuditDeleteDate interfaces takes effect only if IAuditActivator is implemented to entity. If not implemented, the entity will be updated with the proper SaveChanges/SaveChangesAsync functionality in Entity Framework Core.
  • Deleted and UnDeleted flag cannot be stored simultaneously with Created and Changed flags.
  • If IAuditStorage is implemented to your entity, Audits column will be stored in the specified table.
  • If any of IAuditCreateDate, IAuditUpdateDate or IAuditDeleteDate is implemented to entity, the corresponding date will be stored among the Audits column (of IAuditStorage interface) update.
  • Any support flag in AuditProviderOptions.AuditFlagSupport must be written as a flag: AuditFlagState.ActionDate | AuditFlagState.Storage
    • If any of AuditFlag enums are included/excluded from AuditFlagSupport, the corresponding flag will take action in Audits and/or {Action}Date column according to the its state in AuditFlagSupport. (For instance, if AuditFlagSupport.Created = AuditProviderFlagSupport.Excluded, IAuditCreateDate and IAuditStorage, also and Created flag will be ignored.)

Audit Collection

To take advantages of JsonElement Audits (as a property in IAuditStorage interface):

var entity = await dbContext.YourEntities.FindAsync(1);
var audits = entity.GetAuditCollection();

Audit[] deserializedAudits = audits.ToArray(); // Get audits as array
Audit creationAudit = audits.First(); // Get created audit
Audit lastAudit = audits.Last(false); // Get last audit. (false) means to exclude Deleted flag audit, if is the last one.
Audit[] changes = audit.Track(nameof(entity.Name)); // Get changes of a property

Output Example

Stored data in Audits column will be like this:

[
  {
    "f": 0, // Created
    "dt": "2023-09-25T12:00:00.0000000+03:30", // Date and time of the action
  },
  {
    "f": 1, // Changed
    "dt": "2023-09-25T12:00:00.0000000+03:30", // Date and time of the action
    "c": [ // Changes
      {
        "n": "Name", // Name of the property
        "_v": "OldName", // Old value
        "v": "NewName" // New value
      },
      {
        "n": "Age", // Name of the property
        "_v": 0, // Old value
        "v": 33 // New value
      }
    ],
    "u": { // User that made the change
      "id": "1", // The user id (if provided)
      "ad": { // The user additional info (if provided)
        "Username": "Foo"
      }
    }
  },
  {
    "f": 2, // Deleted
    "dt": "2023-09-25T12:00:00.0000000+03:30", // Date and time of the action
  },
  {
    "f": 3, // Restored/Undeleted
    "dt": "2023-09-25T12:00:00.0000000+03:30", // Date and time of the action
  }
]

🎆 Happy coding!

Product Compatible and additional computed target framework versions.
.NET net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 is compatible.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

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,794 2/3/2024
1.0.10 189 2/1/2024
1.0.9 72 2/1/2024
1.0.7 79 1/27/2024
1.0.6 72 1/26/2024
1.0.5 72 1/26/2024
1.0.4 63 1/26/2024
1.0.3 71 1/26/2024
1.0.2 74 1/26/2024
1.0.1 64 1/26/2024
1.0.0 75 1/26/2024