-
Notifications
You must be signed in to change notification settings - Fork 145
Auto Interface Implementation
(Why would you do this? Read: Insight.Database: The Anti-Anti-ORM for .NET)
A best practice for coding is to create interface boundaries between parts of your systems. One place to put an interface boundary is between your business logic and the database. We encourage you to use stored procedures to achieve this, but up until now, you would still have to write the code to call into the database.
Not any more! Insight will now generate the code to call the database for you. First, write your procedures:
CREATE PROC InsertBeer @type varchar(128), @description varchar(128) AS
INSERT INTO Beer (Type, Description)
OUTPUT inserted.ID
VALUES (@type, @description)
GO
CREATE PROC GetBeerByType @type [varchar] AS
SELECT * FROM Beer WHERE Type = @type
GO
Create an interface to match your stored procedures:
public interface IBeerRepository
{
void InsertBeer(Beer beer);
IList<Beer> GetBeerByType(string type);
}
Then you convert your DbConnection, DbConnectionStringBuilder or ConnectionStringSettings to your interface with the As extension methods:
DbConnection c = new SqlConnection(connectionString);
IBeerRepository i = c.As<IBeerRepository>();
Insight implements the interface for you automatically, and you can just call the method, with full type-safety and full performance of IL-generated code:
i.InsertBeer(b);
var results = i.GetBeerByType("ipa");
Starting with Insight 6.0 + C#7 + .NET 4.5, you can use named tuples for return values of your interfaces.
public interface IHandleTuples
{
[Sql("SELECT one=1, two=2")]
List<(int one, int two)> GetTuples();
[Sql("SELECT one=1, two=2")]
(int one, int two) GetTuple();
}
When generating the implementation of the interface, Insight uses the DbConnection extension methods. It maps the interface methods by the return type using the following rules:
-
void
, method name starts with "Insert/Upsert", first param is IEnumerable => InsertList -
void
, method name starts with "Insert/Upsert", first param is updatable -> Insert -
void
, otherwise => Execute -
IList<T>
=> Query -
Results<>
=> QueryResults - other type, primitive type => ExecuteScalar
- other type, otherwise => Single
IMPORTANT NOTE: If your Insert/Upsert method does not return void, any IDs that you return will not be mapped onto your objects, since the IDs are returned as the result of the method
Example:
CREATE PROC InsertBeer(...) AS INSERT INTO Beer () OUTPUT Inserted.ID ...
void InsertBeer(Beer b); // this returns void, so the inserted id is merged onto your object
int InsertBeer(Beer b); // this returns int, so the inserted id is returned from the function
For methods that return Task, Insight uses the same rules as above, but the methods are called asynchronously:
-
Task
=> same asvoid
-
Task<IList<T>>
=> same asIList<T>
-
Task<Results<>>
=> same asResults<T>
-
Task<other type>
=> same as other type
For asynchronous methods, Insight will also automatically remove the word "Async" from the end of your method name when mapping to the name of the stored procedure.
Note that when using Insert/Upsert, the result set will automatically be merged into the object that is the first parameter in the method call. This lets you do:
i.InsertBeer(b);
and have the ID sent back into the object b
.
Parameters are mapped the same way that they would be sent to the appropriate extension method. In general, object members are mapped to parameters by name, and lists of objects are sent to the database as Table-Valued parameters.
Here's one way to map an interface method to a procedure:
CREATE PROC MyProc(@a int, @b varchar(50), @c varchar(50)) AS SELECT * FROM Beer...
IList<Beer> MyProc(int a, string b, string c);
When binding parameters, Insight will automatically peek inside any class you pass to it:
CREATE PROC MyProc(@a int, @b varchar(50), @c varchar(50)) AS SELECT * FROM Beer...
class Selector {
public int a;
public string b;
}
// here, s.a and s.b are mapped to @a and @b, respectively, and c is mapped to @c
IList<Beer> MyProc(Selector s, string c);
You can disable this feature with the BindChildrenFor
attribute.
// @a and @b are not bound!
[BindChildrenFor(BindFor.None)]
IList<Beer> MyProc(Selector s, string c);
See Query Parameter Mapping for more details on parameter mapping.
You can get output parameters back from your Stored Procedure. Just use a ref or out parameter for your method:
public interface IBeerRepository
{
void UpdateBeer(string type, decimal price, out int recordsAffected);
}
By default, Insight can infer the structure of your recordset by the signature of your method.
- T - returns a Single
- IList, etc. - returns a List
- Results<T1, T2> - returns two recordsets, containing T1, T2
However, if your results contain one-to-one or one-to-many mappings, you have to give Insight a hint. You do this with the RecordsetAttribute
.
Here, we say that Recordset 0 (the first one) has a OneToOne<Beer, Glass> relationship:
[Recordset(0, typeof(Beer), typeof(Glass))]
List<Beer> GetBeer();
Here, we say that the second recordset has a OneToOne<Beer, Glass> relationship. Note that we don't specify the format of Recordset 0, so Insight knows that it just contains wine.
[Recordset(1, typeof(Beer), typeof(Glass))]
Results<Wine, Beer> GetBeer();
Here, we say that the second recordset has a one-to-many relationship. Insight will automatically read in the list of glasses and attempt to add them to the proper beer.
[Recordset(1, typeof(Glass), IsChild=true)]
List<Beer> GetBeer();
If you need to override the ID or Into fields, you can do that on the attribute:
[Recordset(1, typeof(Glass), IsChild=true, ID="BeerNumber", Into="Container")]
List<Beer> GetBeer();
In this case, the Recordset
attribute corresponds to adding the proper Query.Returns
statement to your call.
For composite keys, use the ID parameter for the parent keys, and GroupBy for the child keys. In many cases, this can also be auto-detected.
[Recordset(1, typeof(Glass), IsChild=true, ID="BeerNumber,PourSize", GroupBy="BeerNumber,PourSize" Into="Container")]
List<Beer> GetBeer();
See Specifying Result Structures for details on how results structures are specified.
The class that Insight generates also supports IDbConnection and IDbTransaction, so you can add those interfaces to your interface like this:
public interface IBeerRepository : IDbConnection, IDbTransaction
{
// blah
}
Then you can use the object in your using statements like this:
// this opens the connection and starts a transaction
// the using statement will dispose the connection automatically for you
using (var repo = connectionString.OpenWithTransactionAs<IBeerRepository>())
{
repo.InsertBeer(b1);
repo.InsertBeer(b2);
// this is called on IDbTransaction
repo.Commit();
}
I don't think you can make transactional calls to the database with less code than this.
By default, As<T>
will generate a single-threaded implementation. If you make calls on the interface from multiple threads, crazy things could happen (You would likely see things where the connection is closed unexpectedly or complaints that the connection is busy.)
Example:
public async Task Multithreaded()
{
var ifoo = connection.As<IFoo>();
var t1 = ifoo.FooAsync();
var t2 = ifoo.FooAsync();
Task.WaitAll(t1, t2);
}
Here, we are starting two tasks calling methods on the same connection. There is nothing to prevent the system from trying to make both calls at the same time. This code will go boom at some point.
To fix this, tell Insight to create a multi-threaded interface implementation by calling AsParallel<T>
instead of As<T>
:
public async Task Multithreaded()
{
// get a parallel-enabled interface
var ifoo = connection.AsParallel<IFoo>();
var t1 = ifoo.FooAsync();
var t2 = ifoo.FooAsync();
Task.WaitAll(t1, t2);
}
A parallel interface will use a new connection for each method call. You don't have to worry about method calls conflicting anymore. The only difference is that parallel connections don't let you manage the connection lifetime with Open
or use transactions with OpenWithTransactionAs
.
If you want to use parallel connections in a transaction, you would need to use System.Transactions
to create a lightweight distributed transaction. But in that case, you probably should stick to single-threaded.
IMPORTANT NOTE - if you are using Password-based security with AsParallel see below:
When using password-based security, .NET will erase the password from your connection string when the connection is opened. Insight implements AsParallel by cloning the database connection from the connection string. If you pass an connection to AsParallel, then open it outside of the AsParallel method, the connection will not have a password to connect to the database. To correct this, do one of the following:
- Use
AsParallel
only directly with connection strings, not with DbConnections. - Add
Persist Security Info=true;
to your connection string. - Use
Integrated Security=true;
for your connections.
If you want your interfaces and objects to be private, you will have to expose the types to the dynamic assembly that Insight creates. You can do that with the InternalsVisibleTo attribute. Add this to your assembly:
[assembly: System.Runtime.CompilerServices.InternalsVisibleTo("Insight.Database")]
Note that this doesn't work for private interfaces nested inside of a class. It only exposes top-level classes to Insight.
If you need to change the mapping between your interface method and the stored procedure, you can add the SqlAttribute to the method:
public interface IBeerRepository
{
// since this is one word, Insight assumes this is a stored procedure
[Sql("MyOtherInsertProc")]
void InsertBeer(Beer beer);
}
And I guess technically if you don't want to bother with stored procedures at all, you can call SQL text directly:
public interface IBeerRepository
{
// since this is more than one word, Insight assumes this is a text call
[Sql("SELECT * FROM Beer WHERE type = @type")]
IList<Beer> GetBeerByType(string type);
}
Because, underneath it all, Insight is creating a class like this:
class Anonymous : DbConnectionWrapper, IBeerRepository
{
public IList<Beer> GetBeerByType(string type)
{
return InnerConnection.QuerySql("SELECT * FROM Beer WHERE type = @type", new { type = type });
}
}
But seriously, why should you have to write that code at all?
If you are using SQL Schemas, you can also apply the Schema to the class:
[Sql(Schema="MySchema")]
public interface IBeerRepository
{
// this evaluates to MySchema.MyOtherInsertProc
[Sql("MyOtherInsertProc")]
void InsertBeer(Beer beer);
}
For certain scenarios, you might need Insight to ignore variables declared inline, consider the following scenario:
public interface IBeerRepository
{
//This will not work on Insight
[Sql(@"DECLARE @ibu float = 40.0
SELECT * FROM Beer WHERE type = @type AND Ibu <= @ibu")]
IList<Beer> GetBeerByTypeWithLowIbu(string type);
}
But this will error, since Insight expects to map both variables @ibu and @type. The solution is to add /NOBIND/ on the declaration of the variable:
public interface IBeerRepository
{
[Sql(@"DECLARE /*NOBIND*/ @ibu float = 40.0
SELECT * FROM Beer WHERE type = @type AND Ibu <= /*NOBIND*/ @ibu")]
IList<Beer> GetBeerByTypeWithLowIbu(string type);
}
In some cases, you may need to control some of the other Common Method Parameters like commandTimeout
dynamically. If you add a parameter that matches the common parameter, Insight will pass that along to the framework rather than to the stored procedure.
NOTE: both the name and the type must match:
public interface IBeerRepository
{
// commandTimeout is passed to the framework
void InsertBeer(Beer beer, int? commandTimeout);
// cancellationToken is passed to the framework
Task GetBeerByType(string type, CancellationToken? cancellationToken);
}
Insight can also implement an abstract class for you. That way, you can create a repository object with some of your code, and still leave the boilerplate for Insight. If you pass an abstract class as the type for .As
or .AsParallel
, Insight will implement any unimplemented methods.
public abstract class BeerRepository
{
// this is implemented by insight
public abstract void InsertBeer(Beer beer);
public Beer MakeBeer(string name)
{
var beer = new Beer(name);
InsertBeer(beer);
return beer;
}
}
var repo = connection.AsParallel<BeerRepository>();
If your code needs access to the connection, simply define a GetConnection method. Insight will fill it in:
public abstract class BeerRepository
{
// this is implemented by insight
public abstract IDbConnection GetConnection();
public Beer MakeBeer(string name)
{
return GetConnection().Query<Beer>("beerify", name);
}
}
var repo = connection.AsParallel<BeerRepository>();
Note that in single-threaded mode, GetConnection will always return the same connection. In multi-threaded mode (AsParallel), it will return a new connection for each invocation. If you need to make multiple or transactional calls in your repository method, it's best to get the connection once:
public abstract class BeerRepository
{
// this is implemented by insight
public abstract IDbConnection GetConnection();
public Beer MakeBeer(string name)
{
var connection = GetConnection();
using (var tx = connection.BeginTransaction())
{
connection.Execute("somethingelse");
var beer = connection.Query<Beer>("beerify", name);
tx.Commit();
return beer;
}
}
}
var repo = connection.AsParallel<BeerRepository>();
You may also want to derive your class from DbConnectionWrapper
. Then you get the benefit of all of the DbConnection methods, but you lose support for AsParallel
, and you might just be messing up encapsulation.
- Home
- About
- Getting Started
- Connections
- Execute
- Getting Results
- Advanced Results & Mapping
- Insert/Update Considerations
- Data Types
- Write Even Less Code
- Performance & Speed
- Other Topics
- Supported Databases
- Working with the Code