This is maybe more a question for code review rather than stack overflow.
I am using Dapper for a MicroORM to retrieve and Save Data to SQL Server 2014. I have got DTO classes in a DTO Proj that represent the Data retrieved from the DB or saved to the DB.
I am using the Repository Pattern so at my Service layer if a repository is required I am using constructor DI to inject that dependency and then call the method on the Repository to do the work.
so let say I have 2 services called CustomerService and CarService.
I then have 2 Repositories a CustomerRepository and a CarRepository.
I have an interface which defines all the methods in each Repository and then the concrete implementations.
An example method is shown below (calling a Stored Proc to do the DB INSERT (note the actual string variable for the stored proc is defined as a private string at the top of the class):
public void SaveCustomer(CustomerDTO custDTO)
{
using (IDbConnection db = new SqlConnection(ConfigurationManager.ConnectionStrings["myDB"].ConnectionString))
{
db.Execute(saveCustSp, custDTO, commandType: CommandType.StoredProcedure);
}
}
This all works fine but I am finding myself repeating the using block in every method in every repository. I have two real questions outlined below.
Is there a better approach which I could be using perhaps somehow using a BaseRepository class which every other Repository inherits from and the Base would implement the instantiation of the DB connection?
Would that still work ok for multiple concurrent Users on the system?
****UPDATE****
Based on Silas answer I have created the following
public interface IBaseRepository
{
void Execute(Action<IDbConnection> query);
}
public class BaseRepository: IBaseRepository
{
public void Execute(Action<IDbConnection> query)
{
using (IDbConnection db = new SqlConnection(ConfigurationManager.ConnectionStrings["myDB"].ConnectionString))
{
query.Invoke(db);
}
}
}
However, in my repositories, I have other methods such as the below:
public bool IsOnlyCarInStock(int carId, int year)
{
using (IDbConnection db = new SqlConnection(ConfigurationManager.ConnectionStrings["myDB"].ConnectionString))
{
var car = db.ExecuteScalar<int>(anotherStoredSp, new { CarID = carId, Year = year },
commandType: CommandType.StoredProcedure);
return car > 0 ? true : false;
}
}
and
public IEnumerable<EmployeeDTO> GetEmployeeDetails(int employeeId)
{
using (IDbConnection db = new SqlConnection(ConfigurationManager.ConnectionStrings["myDB"].ConnectionString))
{
return db.Query<EmployeeDTO>(anotherSp, new { EmployeeID = employeeId },
commandType: CommandType.StoredProcedure);
}
}
What is the correct way to add these to my Base repository using Generic Type T so I could return any type of DTO or any C# Native type
This is not directly relevant to your question. But I suggest you consider using DapperExtensions.
Initially, I did implemented Repository pattern using Dapper. The drawback was that, I have to write queries all over; it was very stringy. Due to hard coded queries it was near to impossible to write generic repository.
Recently, I upgraded my code to use DapperExtensions. This fixes lot many issues.
Following is the generic repository:
As you can see in above code, most of the methods are just wrapper over underlying
DapperExtensionsProxy
class.DapperExtensionsProxy
internally also manages UnitOfWork which you can see below. These two classes can be combined without any issue. I personally prefer to keep them separate.You can also notice that additional methods
Exists
,DeleteById
, andDeleteAll
are implemented those are not part ofDapperExtensionsProxy
.Method
poco.SetDbId
is defined in each POCO class to set its Identifier property. In my case, identifiers of POCOs may have different datatypes and names.Following is
DapperExtensionsProxy
:Following is the
BasePoco
used above:This also uses UnitOfWork which is explained here.
Sure, a function to create and dispose your Connection will work great.
And your simplified call site:
With Return Values:
In your call site, just write the logic you wish to use.