Skip to content

Dapper Extension similar to Dapper.SimpleCRUD that supports Bulk operations, AutoSync, MatchUpdate, MatchDelete, Distinct, Top/Limit, Upsert, and Insert If Not Exists.

Notifications You must be signed in to change notification settings

ffhighwind/DapperExtraCRUD

Repository files navigation

Introduction

Nuget: Dapper.ExtraCRUD ExtraCRUD

A thread-safe Dapper extension that was inspired by Dapper.SimpleCRUD, Dapper-Plus, and more. Unique additions include Bulk operations, AutoSync, MatchUpdate, MatchDelete, Distinct, Top/Limit, Upsert, and Insert If Not Exists. It also automatically sets auto-increment keys on insert, supports Enum/string/byte[] based primary keys, and exposes most of the underlying metadata to allow customization and improved performance.

Extension Methods

Bulk operations are only supported for Microsoft SQL Server and do not work for keys with null values.

Extension SQL Command
RecordCount Returns the number of rows that match a condition.
Truncate Truncates a table.
Update Updates a row or a subset of columns.
Insert Inserts a row.
InsertIfNotExists Inserts a row if it doesn't exist.
Upsert Updates a row if it exists, otherwise it inserts the row.
Delete Deletes a row.
DeleteList Deletes rows that match a condition.
Get Returns a row if it exists.
GetList Returns the rows that match a condition.
GetLimit Returns a limited number of rows that match a condition with an optional column filter.
GetKeys Returns the primary keys that match a condition.
GetDistinct Returns the unique rows that match a condition with an optional column filter.
GetDistinctLimit Returns a limited number of unique rows that match a condition with an optional column filter.
BulkUpdate Updates a list of rows using a bulk method and a temporary table.
BulkInsert Inserts a list of rows using a bulk method.
BulkInsertIfNotExists Inserts rows that don't exist using a bulk method and a temporary table.
BulkUpsert Upserts rows using a bulk method and a temporary table.
BulkDelete Deletes rows using a bulk method and a temporary table.
BulkGet Returns rows using a bulk method and a temporary table.

Mapping Objects

Annotations can be used to map classes to a table and properties to columns.

Annotation Description
Table Map a class to a table.
Column Map a property to a column.
NotMapped Prevent a property from being mapped.
Key Map a property to a primary key.
AutoSync Automatically selects a column/class after an insert/update. Does not occur in bulk operations. Synchronization is a delayed action, which means that the object may contain inaccurate data without a transaction.
IgnoreDelete Prevents deletes.
IgnoreSelect Ignores a column for selects. This does not work for primary keys.
IgnoreInsert Ignores a column for insertions. A raw SQL string can replace the value.
IgnoreUpdate Ignores a column for updates. A raw SQL string can replace the value.
MatchUpdate Treats a column as a primary key for updates. A raw SQL string can replace the value.
MatchDelete Treats a column as a primary key for deletes.

Priority

NotMapped > Key > ...
IgnoreSelect > AutoSync
IgnoreInsert > MatchInsert
IgnoreUpdate > MatchUpdate

Alternatives

Some annotations from System.ComponentModel are supported as replacements for Dapper.Extra.Annotations.

System.ComponentModel Dapper.Extra.Annotations
[Table] [Table]
[Key][Required] [Key(false)]
[Key] [Key(true)]
[Column] [Column]
[Editable(false)] [IgnoreInsert][IgnoreUpdate]
[ReadOnly(true)] (property) [IgnoreInsert][IgnoreUpdate]
[ReadOnly(true)] (class) [IgnoreInsert][IgnoreUpdate][IgnoreDelete]
public int Property { set; private get; } [IgnoreInsert][IgnoreUpdate][IgnoreDelete]
[NotMapped] [NotMapped]

Example

This example shows how to define a "Users" table and perform some operations on it. The syntax is similar to other Dapper extensions.

[IgnoreDelete]
[Table(name: "Users", declaredOnly: true, inheritAttrs: true)]
public class User
{
	[Key]
	public int UserID { get; private set; } // private is OKAY

	public string FirstName { get; set; }

	public string LastName { get; set; }

	[Column("Account Name")]
	public string UserName { get; set; }

	public UserPermissions Permissions { get; set; }

	[IgnoreInsert(value: null, autoSync: true)]
	[MatchUpdate(value: "getdate()", autoSync: true)]
	[MatchDelete]
	public DateTime Modified { get; protected set; }

	[IgnoreUpdate]
	[IgnoreInsert("getdate()")]
	public DateTime Created { get; private set; }
}

public static class Program {
	private const string ConnString = "Server=myServerAddress;Database=myDataBase;User Id=myUsername;Password=myPassword;";
	
	public static void Main()
	{
		using (SqlConnection conn = new SqlConnection(ConnString)) {
			DateTime minHireDate = DateTime.Today.AddDays(-30);

			IEnumerable<User> users = conn.GetList<User>("where HireDate >= @minHireDate ", new { minHireDate });

			User user = new User() {
				FirstName = "Jason",
				LastName = "Borne",
				UserName = "jborne",
				Permissions = UserPermissions.Admin,
			};
			conn.Insert(user);
		}
	}
}

Where Expressions

Expressions are supported as type-safe alternative to raw SQL where conditions. This acts similarly to LINQ to SQL and is implemented using a custom expression parser WhereConditionGenerator. Compiling these expressions can be slow, so a small expression cache is used for each type.

// Input:
string lastName = "Borne";
IEnumerable<User> result = conn.GetList((u) => u.FirstName == "Jason" || u.LastName == lastName);

// Output:
string whereCondition = "(Table.[FirstName] = 'jason' || Table.[LastName] = @P0)";
IDictionary<string, object> param = new { "P0" => lastName };

Not all expressions are supported. Please post an issue if you need something added.

// Supported
(x) => new []{ "Jason", "Steven" }.Contains(x.FirstName) // Table.[FirstName] in @P0
(x) => x.LastName.Equals("Borne") // Table.[LastName] = 'Borne'

// Not supported, but could be added
string methods: StartsWith, EndsWith, CompareTo, Contains, IndexOf, Trim, TrimStart, TrimEnd

Utilities

AutoAccessObject / DataAccessObject

These include the same the functionality as the extension methods but require fewer parameters per call because they store an SqlConnection or SqlTransaction. They also perform a slightly better than the extension methods because they store a reference to the ISqlQueries.

WhereConditionGenerator

This generates SQL WHERE conditions from a Linq.Expression<Predicate>. It can be somewhat expensive to generate, so I recommend caching the results when possible. See SqlQueries for an example on how caching is done internally using a circular buffer.

ExtraUtil

This static class contains a few helper functions such as IsQuotedSqlType, IsSqlIdentifier, SqlValue, etc.

Accessing Metadata

public static void Main(string[] args)
{
	// Changes the default dialect
	ExtraCrud.Dialect = SqlDialect.PostgreSQL;

	// Contains class/property information such as attributes that are used internally
	SqlTypeInfo typeInfo = ExtraCrud.TypeInfo<User>();

	// Contains delegates and other metadata for accessing this table/type
	SqlBuilder<User> builder = ExtraCrud.Builder<User>(); 

	// Contains delegates for SQL commands
	SqlQueries<User> queries = ExtraCrud.Queries<User>();

	// Clear metadata to allow garbage collection. This will increase memory usage if you keep a reference 
	// to purged metadata. 
	ExtraCrud.Purge<User>();
	ExtraCrud.Purge();

	// Compares User objects for equality based on the primary keys
	IEqualityComparer<User> comparer = ExtraCrud.EqualityComparer<User>();
	Dictionary<User, User> map = new Dictionary<User, User>(comparer);
}

Tips

  • MatchUpdate and MatchDelete do not always work on DateTime columns. This is due to a difference in precision for databases and C# DateTime. You need to limit precision of the datetime in the table such as using datetime2(2) in MS SQL Server. If you cannot do this then you must to use another type as a versioning key.
  • Fields are not supported due to the fact that Dapper only supports them for select queries.
  • All property setters are supported including private and internal. However, only public getters are supported.

Performance

The extension methods in DapperExtraExtensions perform lookups on a ConcurrentDictionary and a cast the results every time they are called. This is negligible, but it can prevented by storing the ISqlQueries object and accessing the delegates directly (e.g. AutoAccessObject/DataAccessObject). Also, less frequently used delegates such as bulk operations have lazy initialization. There is a small synchronization cost every time these are accessed. This can be prevented by storing a reference to each delegate outside of the ISqlQueries object.

Customization

Changing the SqlAdapter using ExtraCrud.SetAdapter() will change the default behavior of the delegates. This is mainly used to support multiple RDBMS syntaxes such as MySQL and PostgreSQL. However, it could also be used for other purposes. For example, you could use BulkOptionsAdapter in order to prevent bulk insert from firing triggers or checking constraints. Note that this will only work for bulk insert as other bulk operations use a temporary table. You can also use CollateAdapter to change the case sensitivity for primary key comparisons.

Future Plans

  • ITypeHandler tests
  • Other RDBMS tests
  • Better async support and tests. This is done in a very lazy way instead of using Dapper's async calls, so it will not perform quite as well.

License

MIT License

Copyright (c) 2018 Wesley Hamilton

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

About

Dapper Extension similar to Dapper.SimpleCRUD that supports Bulk operations, AutoSync, MatchUpdate, MatchDelete, Distinct, Top/Limit, Upsert, and Insert If Not Exists.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages