Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/hnx8/dapperaid

SQL CRUD Query-Builder/Executor for Dapper
https://github.com/hnx8/dapperaid

c-sharp crud csharp dapper database mappings orm query-builder sql-generation

Last synced: 2 months ago
JSON representation

SQL CRUD Query-Builder/Executor for Dapper

Awesome Lists containing this project

README

        

# DapperAid
DapperAidは、[Dapper](https://github.com/StackExchange/Dapper)によるデータベースのCRUD操作を支援するSQL自動生成・実行ライブラリです。
- データベースのSelect, Insert, Update, Deleteの操作を、IDbConnection / IDbTransactionの拡張メソッドとして提供します。
- 実行SQLは、POCOオブジェクトに付与した属性に基づき、内蔵のクエリビルダが自動生成します。
- 実行SQLのWhere条件は、POCOオブジェクトのKey項目の値、または、ラムダ式(式木)の記述をもとに生成されます。
- 属性付与/メソッド引数指定により、生成実行されるSQLの内容をカスタマイズできます。
(必要な部分だけ手書きのSQLを混在させることもある程度可能です)
- Select時のfor update指定、orderby列指定、offset / limit条件、groupby要否、distinct指定など
- Select, Insert, Update対象とするカラムの限定
- Insert時 / Update時の設定値(設定せずDBデフォルト値に任せることも可)
- Insert時のIdentity/AutoIncrement自動採番値把握(各DBMS対応)
- その他オプション機能(使用任意):
- 簡易コードファースト(POCO定義内容からCreateTableのSQLを生成)
- SQL実行ログ取得(クエリビルダが生成したSQLの内容をトレース確認可能)
- 対応DBMS: Oracle, MySQL, Postgres, SQLite, SQLServer, MS-Access, DB2

DapperAid is a SQL automatic generation and execution library that assists database CRUD operation using [Dapper](https://github.com/StackExchange/Dapper).
- Provides Select, Insert, Update and Delete operations of the database as extension methods of IDbConnection / IDbTransaction.
- Execution SQL is automatically generated by the built-in query builder based on the attribute given to the POCO object.
- The execution SQL Where condition is generated based on the value of the key item of POCO object or the description of lambda expression (expression tree).
- You can customize the contents of generated SQL by specifying attribute assignment / method argument specification.
(It is also possible to mix handwritten SQL in specific places.)
- Row-Lock, Order-by, offset / limit conditions, need of group-by, specification of distinct and so on at the time of Select
- Select / Insert / Update only specific columns
- Setting value at Insert / Update (It is also possible to leave it to the DB default value without setting)
- Retrieve inserted Identity / AutoIncrement value (for each DBMS)
- Other extra features (use is optional) :
- A little code-first (Generate Create-Table SQL from POCO definition contents)
- SQL execution log acquisition (It is possible to view the SQL generated by the query builder)

# Installation
from NuGet https://www.nuget.org/packages/DapperAid
```
PM> Install-Package DapperAid
```
```
> dotnet add package DapperAid
```

# Examples
## Sample table
```cs
using System;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using DapperAid.DataAnnotations;

[Table("Members")]
[SelectSql(DefaultOtherClauses = "order by Id")]
class Member
{
[Key]
[InsertValue(false, RetrieveInsertedId = true)]
[DapperAid.Ddl.DDL("INTEGER")] // (for extra feature, generating Create-Table-SQL as SQLite Identity Column)
public int Id { get; set; }

public string Name { get; set; }

[Column("Phone_No")]
public string Tel { get; set; }

[InsertValue("CURRENT_TIMESTAMP"), UpdateValue(false)]
public DateTime? CreatedAt { get; set; }

[InsertValue("CURRENT_TIMESTAMP"), UpdateValue("CURRENT_TIMESTAMP")]
public DateTime? UpdatedAt { get; private set; }

[NotMapped]
public string TemporaryPassword { get; set; }
}
```
- Members declared as "Property" are subject to automatic SQL generation / execution.
- A Readonly-property can only be specified as a Where-clause-column or update value.
- A Writeonly-Property can only be specified as a Selection column.
- See [About Table Attributes](#attributes) for attribute details.

## Initializing
```cs
using DapperAid;

QueryBuilder queryBuilderInstance = new QueryBuilder.Sqlite(); // (example for SQLite)
```
Create an instance corresponding to your DBMS from below.

- new QueryBuilder.Oracle()
- new QueryBuilder.MySql()
- new QueryBuilder.Postgres()
- new QueryBuilder.SQLite()
- new QueryBuilder.SqlServer()
- new QueryBuilder.MsAccess()
- new QueryBuilder.DB2()

These instance generates appropriate SQL statement for your DBMS.
(You can also customize the QueryBuilder class as needed)

If you want to tie an instance only to a specific DB connection, write as follows.
```cs
// When linking with a DB connection object
connection.UseDapperAid(queryBuilderInstance);

// When linking with a DB connection string
queryBuilderInstance.MapDbConnectionString(yourDbDataSource.ConnectionString);
```

## Executing CRUD
```cs
using System.Collections.Generic;
using System.Data;

IDbConnection connection;
```
### `Select([ where[, targetColumns][, otherClauses]])` : returns list<T>
```cs
IReadOnlyList list1 = connection.Select();
// -> select (all columns) from Members order by Id

IReadOnlyList list2 = connection.Select(
r => r.Name == "TEST");
// -> select (all columns) from Members where "Name"=@Name(="TEST") order by Id

IReadOnlyList list3 = connection.Select(
r => r.Name != "TEST",
r => new { r.Id, r.Name });
// -> select "Id", "Name" from Members where "Name"<>@Name order by Id

IReadOnlyList list4 = connection.Select(
r => r.Tel != null,
$"ORDER BY {nameof(Member.Name)} LIMIT 5 OFFSET 10");
// -> select (all columns) from Members where Phone_No is not null
// ORDER BY Name LIMIT 5 OFFSET 10

IReadOnlyList list5 = connection.Select(
r => r.Tel != null,
r => new { r.Id, r.Name },
$"ORDER BY {nameof(Member.Name)} LIMIT 5 OFFSET 10");
// -> select "Id", "Name" from Members where Phone_No is not null
// ORDER BY Name LIMIT 5 OFFSET 10
```
### `SelectFirst([ where[, targetColumns][, otherClauses]])` : returns one row or exception
### `SelectFirstOrDefault([ where[, targetColumns][, otherClauses]])` : returns one row or null
```cs
Member first1 = connection.SelectFirst();
// -> Execute connection.QueryFirst(sql) instead of connection.Query(sql).

Member? firstOrDefault1 = connection.SelectFirstOrDefault();
// -> Execute connection.QueryFirstOrDefault(sql) instead of connection.Query(sql).

Member? selectForUpdate = connection.SelectFirst(
r => r.Id == 1,
otherClauses: "FOR UPDATE");
// -> select (all columns) from Members where "Id"=@Id FOR UPDATE
```
### `Select([ where[, otherClauses]])` : returns list<TColumns>
```cs
class SelColumns {
public string Name { get; private set; }
public string Tel { get; private set; }
[Column("CURRENT_TIMESTAMP")]
public DateTime Now { get; set; }
}

IReadOnlyList listS1 = connection.Select(
r => r.Tel != null
);
// -> select "Name", Phone_No as "Tel", CURRENT_TIMESTAMP as "Now"
// from Members where Phone_No is not null order by Id
```
### `SelectFirst([ where[, otherClauses]])` : returns one row or exception
### `SelectFirstOrDefault([ where[, otherClauses]])` : returns one row or null
```cs
SelColumns first2 = connection.SelectFirst(
r => r.Tel == null
);
// -> Execute connection.QueryFirst(sql) instead of connection.Query(sql).

SelColumns? firstOrDefault2 = connection.SelectFirstOrDefault(
r => r.Tel == null
);
// -> Execute connection.QueryFirstOrDefault(sql) instead of connection.Query(sql).
```
### `Select(by Key [, targetColumns[, otherClauses]])` : returns one row or null
```cs
Member? select1 = connection.Select(
() => new Member { Id = 1 });
// -> select "Id", "Name", Phone_No as "Tel", "CreatedAt", "UpdatedAt" from Members where "Id"=@Id(=1)

Member? select2 = connection.Select(
() => new Member { Id = 1 },
r => new { r.Id, r.Name });
// -> select "Id", "Name" from Members where "Id"=@Id

Member? selectForUpdate = connection.Select(
() => new Member { Id = 1 },
otherClauses: "FOR UPDATE");
// -> select (all columns) from Members where "Id"=@Id FOR UPDATE
```
### `Count([where])` : returns the number of rows
```cs
ulong count1 = connection.Count();
// -> select count(*) from Members

ulong count2 = connection.Count(
r => (r.Id >= 3 && r.Id <= 9));
// -> select count(*) from Members where "Id">=@Id(=3) and "Id"<=@P01(=9)
```
### `Insert(record[, targetColumns])` : returns 1(inserted row)
```cs
var rec1 = new Member { Name = "InsertTest", Tel = "177" };
int insert1 = connection.Insert(rec1);
// -> insert into Members("Name", Phone_No, "CreatedAt", "UpdatedAt")
// values (@Name, @Tel, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP)

var rec2 = new Member { Name = "ParticularColumnOnly1", CreatedAt = null };
int insert2 = connection.Insert(rec2,
r => new { r.Name, r.CreatedAt });
// -> insert into Members("Name", "CreatedAt") values (@Name, @CreatedAt(=null))
```
### `InsertAndRetrieveId(record[, targetColumns])` : returns 1(inserted row)
```cs
var rec3 = new Member { Name = "IdentityTest", Tel = "7777" };
int insert3 = connection.InsertAndRetrieveId(rec3);
// -> insert into Members("Name", Phone_No, "CreatedAt", "UpdatedAt")
// values (@Name, @Tel, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP) ; select LAST_INSERT_ROWID()
Trace.WriteLine("insertedID=" + rec3.Id); // The value assigned to the "Id" column is set
```
- Note: In these examples, the [[InsertValue](#InsertValueattribute)] attribute is specified that
the "Id" column is autoincrement and obtains the registered value.

### `Insert(specifiedColumnValue)` : returns 1(inserted row)
```cs
int insertX = connection.Insert(
() => new Member { Id = 888, Name = "ParticularColumnOnly2" });
// -> insert into Members("Id", "Name") values (@Id, @Name)
```

### `InsertRows(records[, targetColumns])` : returns the number of inserted rows
```cs
int insertMulti = connection.InsertRows(new[] {
new Member { Name = "MultiInsert1", Tel = null },
new Member { Name = "MultiInsert2", Tel = "999-999-9999" },
new Member { Name = "MultiInsert3", Tel = "88-8888-8888" },
});
// -> insert into Members("Name", Phone_No, "CreatedAt", "UpdatedAt") values
// ('MultiInsert1', null, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP),
// ('MultiInsert2', '999-999-9999', CURRENT_TIMESTAMP, CURRENT_TIMESTAMP),
// ('MultiInsert3', '88-8888-8888', CURRENT_TIMESTAMP, CURRENT_TIMESTAMP)
```
- Note: MultiRow-Insert is a SQL92 feature and is supported by DB2, MySQL, PostgreSQL, SQL Server, SQLite (3.7.11 or later), Oracle ("INSERT ALL" statement), and so on.
- Note: If there are many records (by default, more than 1000), the query will be executed in multiple statements. You can use the `queryBuilder.MultiInsertRowsPerQuery` property to change the number of records inserted by a single query.

### `InsertOrUpdate(record[, insertTargetColumns[, updateTargetColumns]])` : returns 1(inserted or updated row)
```cs
var upsertRow = new Member { Id = 1, Name = "UpsertTest", Tel = "7777" };
int upsertSingle = connection.InsertOrUpdate(upsertRow);
// -> insert into Members("Id", "Name", Phone_No, "CreatedAt", "UpdatedAt")
// values (@Id, @Name, @Tel, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP)
// on conflict("Id") do update set "Name"=excluded."Name", ...
```
- Note: This method generates the [UpSert(Merge)](https://en.wikipedia.org/wiki/Merge_(SQL)) statement. If the record already exists, the record is updated. Otherwise, a record is inserted. The generated SQL differs for each DBMS: "MERGE INTO" for SQLServer, Oracle, and DB2; "INSERT ... ON CONFLICT UPDATE" for Postgres and SQLite; "INSERT ... ON DUPLICATE KEY UPDATE " for MySQL.
- Note: If you set the `queryBuilder.SupportsUpsert` property to false, updates will be performed using simple Update and Insert statements instead of Upsert(Merge).
If you are using a DBMS that does not support UpSert such as SQLite less than 3.24(2018-06-04), PostgreSQL less than 9.5(2016-01-07), MS-Access, please set the property to false.

### `InsertOrUpdateRows(records[, insertTargetColumns[, updateTargetColumns]])` : returns the number of inserted (or updated) rows
```cs
var upsertData = new[] {
new Dept { Code = 110, Name = "Sales"},
new Dept { Code = 120, Name = "Marketing"},
new Dept { Code = 130, Name = "Publicity"},
};
int upsertMulti = connection.InsertOrUpdateRows(upsertData);
// -> insert into Dept("Code", "Name", ....)
// values (110, "Sales", ...), (120, "Marketing", ...), (130, "Publicity", ...)
// on conflict("Code") do update set "Name"=excluded."Name", .... ..
```
- Note: This method genetares and executes UpSert(Merge) statement. See also [`InsertOrUpdate()`](#insertorupdate).

### `Update(record[, targetColumns])` : returns the number of updated rows
```cs
var rec1 = new Member { Id = 555, ... };
int update1 = connection.Update(rec1);
// update Members set "Name"=@Name, Phone_No=@Tel, "UpdatedAt"=CURRENT_TIMESTAMP where "Id"=@Id

var rec2 = new Member { Id = 666, Tel = "123-456-7890" };
int update2 = connection.Update(rec2, r => new { r.Tel });
// -> update Members set Phone_No=@Tel where "Id"=@Id
```
### `Update(specifiedColumnValue, where)` : returns the number of updated rows
```cs
int update3 = connection.Update(
() => new Member { Name = "updateName" },
r => r.Tel == "55555-5-5555");
// -> update Members set "Name"=@Name where Phone_No=@Tel
```
### `Delete(record)` : returns the number of deleted rows
```cs
var delRec = new Member { Id = 999, ... };
int delete1 = connection.Delete(delRec);
// -> delete from Members where "Id"=@Id
```
### `Delete(where)` : returns the number of deleted rows
```cs
int delete2 = connection.Delete(
r => r.Name == null);
// -> delete from Members where "Name" is null
```
### `Truncate()`
```cs
connection.Truncate();
// -> truncate table Members
// (For DBMS without "truncate" syntax, execute delete instead)
```
### (Extra) `DDLAttribute.GenerateCreateSQL()` : returns the "create table" script
```cs
using DapperAid.Ddl;

var createTableSql = DDLAttribute.GenerateCreateSQL();
// -> create table Members
// (
// "Id" INTEGER,
// "Name",
// Phone_No,
// "CreatedAt",
// "UpdatedAt",
// primary key( "Id")
// )
connection.Execute(createTableSql);
```
- Note: If you use this feature, you should describe [[DDL](#ddlattribute)] attribute in each column
and specify database column types, constraints, default values, etc.
- Note: `DDLAttribute.GenerateTableDefTSV()` method is also provided, and returns tab-delimited text of table definition contents.

### (Extra) `LoggableDbConnection` class
```cs
using System.Data;
using System.Data.SQLite; // (example for SQLite)
using DapperAid.DbAccess;

IDbConnection GetYourDbConnection()
{
// Prepare a normal DB connection
var connectionSb = new SQLiteConnectionStringBuilder { DataSource = ":memory:" };
var conn = new SQLiteConnection(connectionSb.ToString());
conn.Open();

// Set into LoggableDbConnection object
return new LoggableDbConnection(conn,
errorLogger: (Exception ex, DbCommand cmd) =>
{ // Write Error Log
Trace.WriteLine(ex.ToString() + (cmd != null ? ":" + cmd.CommandText : null));
},
traceLogger: (string resultSummary, long mSec, DbCommand cmd) =>
{ // Write SQL Execution Trace Log
Trace.WriteLine(resultSummary + "(" + mSec + "ms)" + (cmd != null ? ":" + cmd.CommandText : null));
});
}
```
- The log method specified in the argument is called when SQL is executed / error occurs.
By using this, you can check the contents of the SQL generated by DapperAid.
Implement it to be logged.

# About Where Clause
Expression trees in LambdaExpression is converted to SQL search condition.
Condition values are bound to parameters.
## Comparison Operator
```cs
int? val1 = 100; // (bound to @IntCol)
.Select(t => t.IntCol == val1); // -> where "IntCol"=@IntCol
.Select(t => t.IntCol != val1); // -> where "IntCol"<>@IntCol
.Select(t => t.IntCol < val1); // -> where "IntCol"<@IntCol
.Select(t => t.IntCol > val1); // -> where "IntCol">@IntCol
.Select(t => t.IntCol <= val1); // -> where "IntCol"<=@IntCol
.Select(t => t.IntCol >= val1); // -> where "IntCol">=@IntCol

// If the value is null, SQL is also generated as "is"
int? val2 = null;
.Select(t => t.IntCol == val2); // -> where "IntCol" is null
.Select(t => t.IntCol != val2); // -> where "IntCol" is not null

// can also compare columns and columns.
.Select(t => t.IntCol == t.OtherCol); // -> where "IntCol"="OtherCol"
```
SQL-specific comparison operators `in`, `like`, and `between` are also supported.
```cs
using DapperAid; // uses "SqlExpr" static class

string[] inValues = {"111", "222", "333"}; // (bound to @TextCol)
.Select(t => t.TextCol == SqlExpr.In(inValues)); // -> where "TextCol" in @TextCol

string likeValue = "%test%"; // (bound to @TextCol)
.Select(t => t.TextCol == SqlExpr.Like(likeValue)); // -> where "TextCol" like @TextCol

int b1 = 1; // (bound to @IntCol)
int b2 = 99; // (bound to @P01)
.Select(t => t.IntCol == SqlExpr.Between(b1, b2)); // -> where "IntCol" between @IntCol and @P01

// when "!=" is used, SQL is also generated as "not"
.Select(t => t.TextCol != SqlExpr.In(inValues)); // -> where "TextCol" not in @TextCol
```
- Note: IN conditionals are further expanded by Dapper's List Support feature.

## Logical Operator
Supports And(`&&`), Or(`||`), Not(`!`).
```cs
.Select(t => t.TextCol == "111" && t.IntCol < 200);
// -> where "TextCol"=@TextCol and "IntCol"<@IntCol

.Select(t => t.TextCol == "111" || t.IntCol < 200);
// -> where ("TextCol"=@TextCol) or ("IntCol"<@IntCol)

.Select(t => !(t.TextCol == "111" || t.IntCol < 200));
// -> where not(("TextCol"=@TextCol) or ("IntCol"<@IntCol))
```
It can also be combined with the condition judgment not based on SQL.
```cs
// The part where the boolean value is found in advance is not converted to SQL, and is omitted
string text1 = "111";
.Select(t => text1 == null || t.TextCol == text1); // -> where "TextCol"=@TextCol
.Select(t => text1 != null && t.TextCol == text1); // -> where "TextCol"=@TextCol

// If the result is determined only by the left side, SQL is not generated
string text2 = null;
.Select(t => text2 == null || t.TextCol == text2); // -> where true
.Select(t => text2 != null && t.TextCol == text2); // -> where false
```
Ternary operators (cond ? trueCond : falseCond) are also supported.
```cs
int intVal = -1;
.Select(t.CondCol == 1 ? t.IntCol > intVal : t.IntCol < intVal) // -> where (("CondCol"=1 and "IntCol">@IntCol) or ("CondCol"<>1 and "IntCol"<@IntCol))`
.Select(intVal < 0 ? t.IntCol == null : t.IntCol > intVal) // -> where "IntCol">@IntCol
.Select(intVal > 0 ? t.IntCol == null : t.IntCol > intVal) // -> where "IntCol" is null`
```

## SQL direct description
You can describe conditional expressions and subqueries directly.
```cs
using DapperAid; // uses "SqlExpr" static class

.Select(t => t.TextCol == SqlExpr.In("select text from otherTable where..."));
// --> where "TextCol" in(select text from otherTable where...)

.Select(t => SqlExpr.Eval("ABS(IntCol) < 5"));
// --> where ABS(IntCol) < 5

.Select(t => SqlExpr.Eval("(exists(select * from otherTable where...))"));
// --> where (exists(select * from otherTable where...))
```
You can also bind parameter values by using `SqlExpr.In(...)` / `SqlExpr.Eval(...)`.
```cs
int intVal = 99; // (bound to @P00, @P01 or such name)

.Select(t => t.TextCol == SqlExpr.In("select text from otherTable where a=", intVal, " or b=", intVal))
// --> where "TextCol" in(select text from otherTable where a=@P00 or b=@P01)

.Select(t => SqlExpr.Eval("IntCol < ", intVal, " or IntCol2 > ", intVal));
// --> where IntCol < @P00 or IntCol2 > @P01
```
```cs
var idRegex = "userIdRegexPattern"; // (bound to @P00)
var pwText = "passswordText"; // (bound to @P01)

.Select(t => SqlExpr.Eval("id~", idRegex, " AND pw=CRYPT(", pwText, ", pw)"));
// --> where id~@P00 AND pw=CRYPT(@P01, pw) -- works only Postgres
```
If you want to descrive only the value expression, use `SqlExpr.Eval(...)`.
```cs
.Select(t => t.pw == SqlExpr.Eval("CRYPT('password', pw)"));
// --> select (columns) from TableX where "pw"=CRYPT('password', pw)

.Select(t => t.pw == SqlExpr.Eval("CRYPT(", pwText, ", pw)"));
// --> select (columns) from TableX where "pw"=CRYPT(@P00, pw)
```
- Note: `SqlExpr.Eval(...)` can also be used as the **Value** below.
- `Select(() => new Table { column = `**Value**`, ...})`
- `Insert(() => new Table { column = `**Value**`, ...})`
- `Update(() => new Table { column = `**Value**`, ...}[, where ])`
```cs
var pwText = "passswordText"; // (bound to @P00)
var salt = "hashsalt"; // (bound to @P01)

.Select(() => new TableX {
pw = SqlExpr.Eval("CRYPT(", pwText, ", pw)")
});
// --> select (columns) from TableX where "pw"=CRYPT(@P00, pw)

.Insert(() => new TableX {
pw = SqlExpr.Eval("CRYPT(", pwText, ",", salt, ")")
});
// --> insert into TableX("pw") values(CRYPT(@P00,@P01))

.Update(() => new TableX {
pw = SqlExpr.Eval("CRYPT(", pwText, ",", salt, ")")
}, r => { ... });
// --> update TableX set "pw"=CRYPT(@P00,@P01) where ...
```

# About Table Attributes
```cs
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
using DapperAid.DataAnnotations;
using DapperAid.Ddl; // (for extra feature)
```
## for Class
### `[Table]` : apply if tablename != classname or you want to customize the from clause
```cs
[Table("TABLE_NAME")] // specify table name
// -> select .... from TABLE_NAME

[Table("TABLE_NAME", Schema = "SCHEMA_NAME")] // specify schema
// -> select .... from SCHEMA_NAME.TABLE_NAME

[Table("TABLE1 T1 INNER JOIN TABLE2 T2 ON T1.ID=T2.ID")] // join
// -> select .... from TABLE1 T1 INNER JOIN TABLE2 T2 ON T1.ID=T2.ID
// Note: Also specify the acquisition source table in the column definition
```
### `[SelectSql]` : apply if you want to customize select statement
```cs
[SelectSql(Beginning = "SELECT DISTINCT")] // customize the beginning of select sql
// -> SELECT DISTINCT ... from ....

[SelectSql(BaseWhereClauses = "deleted_at IS NULL")] // append where condition of select sql
// -> select ... from .... where deleted_at IS NULL and .....

[SelectSql(GroupByKey = true)] // generate group-by clause
// -> select ... from ... where ... GROUP BY (colums with [Key] attributes)

[SelectSql(DefaultOtherClauses = "ORDER BY NAME NULLS LAST")] // append the end of select sql by default
// -> select ... from ... where ... ORDER BY NAME NULLS LAST
// (when {otherClauses} is not specified)
```
### (for extra feature) `[DDL]` : apply if you want to specify a table constraint of DDL
```cs
[DDL("FOREIGN KEY (C1,C2) REFERENCES MASTERTBL(C1,C2)")] // specify FK
// -> create table ...(
// ...,
// primary key ...,
// FOREIGN KEY (C1,C2) REFERENCES MASTERTBL(C1,C2)
// )
```
## for Properties
### `[Column]` : apply if columnname != propertyname or you want to customize the column values to retrieve
```cs
[Column("COLUMN_NAME")] // specify column name
public string ColumnName { get; set; }
// -> select ... COLUMN_NAME as "ColumnName", ...

[Column("T1.CODE")] // specify table alias and column name
public string T1Code { get; set; }
// -> select ... T1.CODE as "T1Code", ...

[Column("MONTH(DateOfBirth)")] // customize value
public int BirthMonth { get; set; }
// -> select ... MONTH(DateOfBirth) as "BirthMonth", ...

[Column("COUNT(*)")] // tally value
public int TotalCount { get; set; }
// -> select ... COUNT(*) as "TotalCount", ...
```
### `[Key]` : apply if you want to update/delete by record-object, or use [Select(GroupByKey = true)]
```cs
[Key]
// -> update/delete .... where (columns with [Key] attributes)=@(bindvalue)

// when [SelectSql(GroupByKey = true)] is applied to the class
// -> select .... where ... GROUP BY (colums with [Key] attributes)
```
- Note: You can also specify [Key] for multiple columns (as a composite key)

### `[InsertValue]` : apply if you want to modify the insert value
```cs
[InsertValue("CURRENT_TIMESTAMP")] // Specify the value to set with SQL instead of bind value
public DateTime CreatedAt { get; set; }
// -> insert into ...(..., "CreatedAt", ...) values(..., CURRENT_TIMESTAMP, ...)

[InsertValue("date(@DateOfBirth)")] // Edit bind value with SQL
public DateTime DateOfBirth
// -> insert into ...(..., "BirtyDay", ...) values(..., date(@DateOfBirth), ...)

// Do not set column (DB default value is set)
[InsertValue(false)]

// Default value(Identity etc.) is set, and obtain the value when InsertAndRetrieveId() is called
[InsertValue(false, RetrieveInsertedId = true)]

// set sequence value and obtain (works only PostgreSQL, Oracle)
[InsertValue("nextval(SEQUENCENAME)", RetrieveInsertedId = true)]
```
- Note: If you call Insert() with the target column explicitly specified,
The bind value is set instead of the value by this attribute.

### `[UpdateValue]` : apply if you want to modify the value on update
```cs
[UpdateValue("CURRENT_TIMESTAMP")] : // Specify the value to set with SQL instead of bind value
public DateTime UpdatedAt { get; set; }
// -> update ... set ..., "UpdatedAt"=CURRENT_TIMESTAMP, ....

[UpdateValue("COALESCE(@DCnt, 0)")] // Edit bind value with SQL
public Int? DCnt { get; set; }
// -> update ... set ..., "DCnt"=COALESCE(@DCnt, 0), ...

// Do not set column (not be updated)
[UpdateValue(false)]
```
- Note: If you call Update() with the target column explicitly specified,
The bind value is set instead of the value by this attribute.

### `[NotMapped]` : Denotes that a property should be excluded from database mapping
```cs
[NotMapped] // Do not select, insert, update
public Object NotMappedProperty { get; set; }
```
### (for extra feature) `[DDL]` : apply if you want to specify database column types, constraints, default values, etc.
```cs
[DDL("NUMERIC(5) DEFAULT 0 NOT NULL")]
public int Value { get; set; }
// -> create table ...(
// :
// Value NUMERIC(5) DEFAULT 0 NOT NULL,
// :
```
# Misc.
## when the error "The value of type (TypeName) cannot be represented as a sql literal." occured
- Call `QueryBuilder.AddSqlLiteralConverter()` to specify the function that converts the data value to an SQL representation.
```cs
using NetTopologySuite.Geometries;

var queryBuilderInstance = new QueryBuilder.Postgres();
// Here is an example of geometry type SQL.
queryBuilderInstance.AddSqlLiteralConverter(geom =>
{
var binaryHex = string.Concat(geom.AsBinary().Select(b => $"{b:X2}"));
return $"'{binaryHex}'::geometry";
});
```
## When you want to execute a query during transaction.
- use extension methods in `IDbTransaction`.
It provides the same method as the `IDbConnection` extension method.
## When you want to execute a asynchronus query.
- use ~~Async methods.
## When not using as an extension method.
- use `QueryRunner` class.
It Provides almost the same content as an extension method as an instance method.
## When you want to use only the SQL generation function.
- Use the [`QueryBuilder`](#querybuilders) class appropriate for your DBMS.

# License
[MIT License](http://opensource.org/licenses/MIT).

# About Author
hnx8(H.Takahashi) is a software developer in Japan.
(I wrote English sentences relying on Google translation. Please let me know if you find a strange expression)