AO.Dapper.Repository.SqlServer
1.4.1
dotnet add package AO.Dapper.Repository.SqlServer --version 1.4.1
NuGet\Install-Package AO.Dapper.Repository.SqlServer -Version 1.4.1
<PackageReference Include="AO.Dapper.Repository.SqlServer" Version="1.4.1" />
paket add AO.Dapper.Repository.SqlServer --version 1.4.1
#r "nuget: AO.Dapper.Repository.SqlServer, 1.4.1"
// Install AO.Dapper.Repository.SqlServer as a Cake Addin #addin nuget:?package=AO.Dapper.Repository.SqlServer&version=1.4.1 // Install AO.Dapper.Repository.SqlServer as a Cake Tool #tool nuget:?package=AO.Dapper.Repository.SqlServer&version=1.4.1
This library lets you write data access code that offers:
- an IoC-friendly single point of access to all your repository classes, keeping your constructors simple throughout your application
- a way to implement model-wide conventions along with table-specific business logic where needed
- connection extension methods for simple entity access
- efficient, typed user profile access
To implement, bear in mind:
- Your model classes must implement IModel from package AO.Models, installed automatically as a dependency.
- As a Dapper-based library, this uses direct database connections. As such, this works only on the backend -- such as in a Blazor Server app, API backend, or MVC/Razor Pages app.
Example, using a fictional MyContext
object:
public class SomeController : Controller
{
private readonly MyContext _context;
public SomeController(MyContext context)
{
_context = context;
}
public async Task Edit(int id)
{
var row = await _context.Employees.GetAsync(id);
return View(row);
}
public async Task Save(Employee employee)
{
await _context.Employees.SaveAsync(employee);
return Redirect("Edit", new { id = employee.Id });
}
}
The integration tests provide examples that give more context, but here's how to get started:
- Install NuGet package AO.Dapper.Repository.SqlServer
- Implement your model classes. My tests work with these examples. Your model classes must implement IModel from package AO.Models.
- Create a class based on
SqlServerContext<TUser>
that will provide the access point to all your repositories. Example: DataContext. You pass your database connection string, current user name, and anILogger
. My example uses a localdb connection string for test purposes. In a real application, it would typically come from your configuration in some way. Optionally, but most often, you'll need to override QueryUserInfo so that you can access properties of the current user in your crud operations. More on this below. - Create a
Repository
class that handles your common data access scenario. Example: BaseRepository. My example assumes anint
key type, and overrides the BeforeSaveAsync method to capture user and timestamp info during inserts and updates. - For models that require unique behavior, validation, or trigger-like behavior, create repository classes specifically for them. You would typically inherit from your own
BaseRepository
so as to preserve any existing conventional behavior. Example: WorkHoursRepository. Note, there are many overrides you can implement for various crud events, found here. - Add your repository classes as read-only properties of your
DataContext
, for example here. Note, I have more model classes than repositories because I'm lazy, and don't need them for a working demo. - Add your
DataContext
object to yourservices
collection in startup. A Blazor Server approach might look like this:
services.AddScoped((sp) =>
{
var authState = sp.GetRequiredService<AuthenticationStateProvider>();
var logger = sp.GetRequiredService<ILogger<DataContext>>();
var cache = sp.GetRequiredService<IDistributedCache>();
return new DataContext(connectionString, cache, authState, logger);
});
Since you create your own DataContext
object, you can decide what dependencies are useful to pass to it. The AuthenticationStateProvider
is used to get the current user name during QueryUserInfo
. The IDistributedCache
is used to avoid a database roundtrip to get user details, used here. The ILogger
is required by the low-level DbContext object. This is so SQL errors have a place to be logged consistently.
Working With TUser
Most applications will have authentication and need to track database operations by user in some way. When you create your DbContext
object, you must provide a TUser
that represents the current user. You also override the QueryUserAsync method, implementing any database query and/or cache access that makes sense in your application.
The test project uses DataContext where TUser
is UserInfoResult. Notice how the QueryUserAsync
override checks in a cache for the user, then queries the database if it's not found. This is how you achieve efficient, typed user profile access in your applications.
See the topic on Auditing next to see how this can be leveraged.
Audit Tracking
- For information on capturing the user name and timestamps of row inserts and updates, see the Wiki topic Shallow Auditing
- For information on capturing all row changes over time, see the Wiki topic Deep Auditing
Blazor Server Example
I don't have an open source Blazor Server app example now, but here's a code sample to show how to implement in Startup
:
public void ConfigureServices(IServiceCollection services)
{
var connectionString = Configuration.GetConnectionString("DefaultConnection");
// typical Identity stuff
services
.AddDbContext<ApplicationDbContext>(options => options.UseSqlServer(connectionString));
services
.AddDefaultIdentity<IdentityUser>(options => options.SignIn.RequireConfirmedAccount = true)
.AddEntityFrameworkStores<ApplicationDbContext>();
// Blazor boilerplate
services.AddRazorPages();
services.AddServerSideBlazor();
services.AddScoped<AuthenticationStateProvider, RevalidatingIdentityAuthenticationStateProvider<IdentityUser>>();
services.AddDatabaseDeveloperPageExceptionFilter();
// add a cache appropriate for dev/test scenarios. In production, consider something truly distributed
services.AddDistributedMemoryCache();
// Dapper.Repository specific
services.AddScoped((sp) =>
{
var authState = sp.GetRequiredService<AuthenticationStateProvider>();
var logger = sp.GetRequiredService<ILogger<DataContext>>();
var cache = sp.GetRequiredService<IDistributedCache>();
return new DataContext(connectionString, cache, authState, logger);
});
}
The DataContext
class referenced above:
public class DataContext : SqlServerContext<UserInfoResult>
{
private readonly AuthenticationStateProvider _authState;
private readonly IDistributedCache _cache;
public DataContext(string connectionString, IDistributedCache cache, AuthenticationStateProvider authState, ILogger logger) : base(connectionString, logger)
{
_authState = authState;
_cache = cache;
}
protected override async Task<UserInfoResult> QueryUserAsync(IDbConnection connection)
{
var authState = await _authState.GetAuthenticationStateAsync();
var userName = authState.User.Identity.Name;
if (string.IsNullOrEmpty(userName)) return null;
var key = $"userInfo.{userName}";
var result = await _cache.GetItemAsync<UserInfoResult>(key);
if (result == default(UserInfoResult))
{
result = await new UserInfo() { UserName = userName }.ExecuteSingleOrDefaultAsync(connection);
await _cache.SetItemAsync(key, result);
}
return result;
}
// repository classes follow...
}
A few points to note about the code above:
- The cache access methods you see
GetItemAsync
andSetItemAsync
are extensions you can find here that aren't part of the Dapper.Repository package proper. - The line
await new UserInfo() { UserName = userName }.ExecuteSingleOrDefaultAsync(connection)
executes a SQL query via a wrapper classUserInfo
. This functionality comes from my Dapper.QX library. The integration tests here use this also.
Customizing Error Messages
You can override default SQL Server messages by passing IEnumerable<IMessageErrorHandler>
to the SqlServerContext. There are three built-in message handlers for primary and foreign key errors, respectively. See the IErrorMessageHandler interface. Example usage:
internal static IEnumerable<IErrorMessageHandler> DefaultHandlers => new IErrorMessageHandler[]
{
new DeleteCascadeBlocked((info) => $"Can't delete '{info.ReferencedTable}' row because at least one '{info.ReferencingTable}' row is depending on it."),
new InvalidForeignKeyValue((info) => $"Can't save the '{info.ReferencingTable}' row because of a missing or unrecognized value in the '{string.Join(", ", info.Columns.Select(col => col.ReferencingName))}' field(s)."),
new DuplicateKeyError((value, tableName) => $"Can't save this row because the value '{value}' is already in use in table '{tableName}'.")
};
Then in the SqlServerContext
constructor, pass DefaultMessageHandlers
:
public class DataContext : SqlServerContext<UserInfoResult>
{
public DataContext(string connectionString, ILogger logger) : base(connectionString, logger, DefaultMessageHandlers)
{
}
// a lot omitted for clarity
}
If an error happens when executing a repository operation -- such as when saving or deleting -- your handlers will be searched for one that applies to the current exception, and its message used with the thrown exception.
Classic Extension Methods
If you need an easy way to perform CRUD operations on model types without any intermediary business logic, there are some "classic" extension methods for this. Most of these do not require IModel
except for SaveAsync
:
- Task<TModel> GetAsync<TKey> (this IDbConnection connection, TKey id, [ string identityColumn ], [ IDbTransaction txn ])
- Task<TModel> GetWhereAsync (this IDbConnection connection, object criteria, [ IDbTransaction txn ])
- Task<bool> ExistsWhereAsync (this IDbConnection connection, object criteria, [ IDbTransaction txn ])
- Task<TModel> InsertAsync<TModel> (this IDbConnection connection, TModel model, [ IEnumerable<string> columnNames ], [ string identityColumn ], [ Action<TModel, TKey> afterInsert ], [ IDbTransaction txn ])
- Task UpdateAsync<TModel> (this IDbConnection connection, TModel model, [ IEnumerable<string> columnNames ], [ string identityColumn ], [ IDbTransaction txn ])
- Task DeleteAsync<TKey> (this IDbConnection connection, TKey id, [ string identityColumn ], [ string tableName ], [ IDbTransaction txn ])
- Task<TModel> SaveAsync<TModel> (this IDbConnection connection, TModel model, [ IEnumerable<string> columnNames ], [ string identityColumn ], [ IDbTransaction txn ])
- Task<TModel> MergeAsync<TModel> (this IDbConnection connection, TModel model, [ Action<TModel> onExisting ], [ IDbTransaction txn ])
- Task<object> InsertAsync (this IDbConnection connection, string tableName, Dictionary<string, object> columnValues, [ IDbTransaction txn ])
- Task UpdateAsync (this IDbConnection connection, string tableName, Dictionary<string, object> columnValues, [ string identityColumn ], [ IDbTransaction txn ])
Background
I've been critical of the repository pattern in the past because I've seen it lead to verbosity and repeated code. But there's no inherent reason it has to be this way. I'm revisiting this now because my Dapper.CX project has been getting complicated. Once again I'm feeling the need to get back to basics, rethink the dependency footprint and my approach to business logic.
The main issue I'm having with Dapper.CX is that there's not a good place for business logic such as validation, tenant isolation, permission checks, navigation properties, and trigger-like behavior. My solution in the past has been to offer a bunch of interfaces from AO.Models that you would implement directly on your model classes. I like this opt-in approach in theory, but there are two problems.
- The logic for supporting all this custom behavior is embedded in the low-level CRUD provider itself. If you read through this class, you'll see a lot of business-logic-like things in many places for validation, trigger execution, auditing, change tracking, and so on. This is a lot of complexity and coupling where I don't think it belongs.
- Some of these interfaces like IGetRelated, ITrigger, IValidate have
IDbConnection
arguments. This forces a database dependency in your model layer. You can avoid this by playing an elaborate game with partial classes and linked source, but this is hard to manage.
Another issue worth mentioning is that the .NET Core integration approach I used in Dapper.CX was a little clumsy the way it queried the current user as part of the DI setup. I've moved that query to the async QueryUserAsync
virtual method. I've also removed the IUserBase
requirement from TUser
. IUserBase
was meant to ensure that you have access to the user's current local time. That's fine if you want that (and I use that in my test app), but there's no requirement for it going forward.
Product | Versions 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 was computed. 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 was computed. 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. |
-
net6.0
- AO.Models (>= 1.1.48)
- Dapper (>= 2.0.123)
- Microsoft.Data.SqlClient (>= 5.0.1)
- Microsoft.Extensions.Logging.Abstractions (>= 6.0.3)
NuGet packages (1)
Showing the top 1 NuGet packages that depend on AO.Dapper.Repository.SqlServer:
Package | Downloads |
---|---|
AO.UserVoice.RCL
Test cases, UAT features, issue reporting and feature voting for Blazor Server apps |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
1.4.1 | 297 | 1/9/2024 |
1.4.1-alpha | 244 | 4/26/2023 |
1.4.0 | 606 | 1/14/2023 |
1.4.0-alpha | 109 | 1/9/2024 |
1.3.2 | 365 | 12/23/2022 |
1.2.5 | 582 | 8/20/2022 |
1.2.4 | 617 | 5/8/2022 |
1.2.2 | 583 | 12/11/2021 |
1.2.1 | 1,009 | 11/20/2021 |
1.1.16 | 411 | 11/11/2021 |
1.1.15 | 370 | 10/17/2021 |
1.1.13 | 392 | 9/30/2021 |
1.1.12 | 530 | 9/25/2021 |
1.1.11 | 409 | 9/16/2021 |
1.1.10 | 445 | 9/6/2021 |
1.1.9 | 397 | 9/1/2021 |
1.1.8 | 417 | 8/29/2021 |
1.1.5 | 372 | 8/24/2021 |
1.1.1 | 453 | 8/21/2021 |
1.0.8 | 397 | 8/18/2021 |