SDE 0.20.0

There is a newer version of this package available.
See the version list below for details.
dotnet add package SDE --version 0.20.0                
NuGet\Install-Package SDE -Version 0.20.0                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="SDE" Version="0.20.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add SDE --version 0.20.0                
#r "nuget: SDE, 0.20.0"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install SDE as a Cake Addin
#addin nuget:?package=SDE&version=0.20.0

// Install SDE as a Cake Tool
#tool nuget:?package=SDE&version=0.20.0                

(SDE) System.Data.Extensions

Methods:

  • CreateCommand
  • CreateStoredProcCommand
  • CreateParameter
  • AddInParameter
  • AddOutParameter
  • AddParameter
  • ExecuteNonQuery
  • ExecuteScalar
  • Query
  • QueryOne
  • QueryMultiple
  • CleanSql: remove "Go" delimiter from queries. Avoid fail on execution
  • Open: Shortcut for fluent code

Async

  • ExecuteNonQueryAsync
  • ExecuteScalarAsync
  • QueryAsync
  • QueryOneAsync
  • QueryMultipleAsync
  • OpenAsync

Db supported:

  • Sql Server
  • Sqlite
  • MySql
  • PostgreSql
  • And more

Samples

Query

using (var connection = new SqlConnection(ConnectionString))
{
    var products = connection.Query<Product>("SELECT * FROM [Products]").ToList();
}

Query Many

using (var connection = new SqlConnection(ConnectionString))
{
    var products = connection.Query<Product>("SELECT * FROM [Products] WHERE [CategoryID]=@CategoryID",
        new[]
        {
            new { CategoryID = 1 },
            new { CategoryID = 2 }
        }).ToList();
}

Multiple queries

using (var connection = new SqlConnection(ConnectionString))
{
    var products = connection.Query<Product>("SELECT * FROM [Products] WHERE [ProductID]=@ProductID;SELECT * FROM [Products] WHERE [ProductID]=@ProductID2",
        new dynamic[]
        {
            new { ProductID = 1 },
            new { ProductID2 = 2 }
        }).ToList();
}

Tip: single parameter for multiple queries

var products = connection.Query<Product>("SELECT * FROM [Products] WHERE [ProductID]=@ProductID;SELECT * FROM [Products] WHERE [ProductID]=@ProductID2",
    new { ProductID = 1, ProductID2 = 2 }).ToList();

QueryMultiple (and QueryMultipleAsync)

using (var connection = new SqlConnection(ConnectionString))
{
    var results = connection.QueryMultiple("SELECT * FROM [Categories];SELECT * FROM [Products]");

    var categories = results.Read<Category>().ToList(); // or ReadOne
    var products = results.Read<Product>().ToList();
}

QueryOne with parameter

using (var connection = new SqlConnection(ConnectionString))
{
    var product = connection.QueryOne<Product>(@"SELECT * FROM [Products] WHERE [ProductID]=@ProductID", new { ProductID = 1 }); // anonymous object
}

ExecuteNonQuery

using (var connection = new SqlConnection(ConnectionString))
{
    connection.ExecuteNonQuery("INSERT INTO [Posts]([Title],[Content]) VALUES (@Title,@Content)", new Post { Title = "First Article", Content = "First Content" }); // entity
}

Insert Many

using (var connection = new SqlConnection(ConnectionString))
{
    // rows affected = 2
    var rowsAffected = connection.ExecuteNonQuery(@"INSERT INTO [Posts]([Title],[Content]) VALUES (@Title,@Content)",
            new[]
            {
                new { Title = "Article A", Content = "Content A" },
                new { Title = "Article B", Content = "Content B" }
            });
}

ExecuteScalar

using (var connection = new SqlConnection(ConnectionString))
{
    var result = (int)connection.ExecuteScalar("SELECT COUNT(*) FROM [Posts]"); 
}

Transactions with TransactionScope

try
{
    using (var scope = new TransactionScope())
    {
        using (var connection = new SqlConnection(ConnectionString))
        {
            connection.ExecuteNonQuery("CREATE TABLE [Posts]([PostId] INT NOT NULL PRIMARY KEY IDENTITY(1,1),[Title] NVARCHAR(MAX),Content NTEXT)");
            connection.ExecuteNonQuery("INSERT INTO [Posts]([Title],[Content]) VALUES (@Title,@Content)", new Post { Title = "First Article", Content = "First Content" }); // entity

            scope.Complete();
        }
    }
}
catch (Exception ex)
{
    throw;
}

With Async

using (var scope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
// ect.

Relations

  • Foreign Key
using (var connection = new SqlConnection(ConnectionString))
{
    var products = connection.Query<Product>("SELECT * FROM [Products]").ToList();
    foreach (var product in products)
    {
        var category = connection.QueryOne<Category>("SELECT * FROM [Categories] WHERE [CategoryId]=@CategoryId", new { CategoryId = product.CategoryId });
        product.Category = category;
    }
}

Or only 1 query

var connection = new SqlConnection(ConnectionString);
var products = connection.Query<Product, Category, Product>(@"select * from Products p inner join Categories c on p.CategoryId = c.CategoryId", (product, category) =>
{
    product.Category = category; // navigation property
    return product;
}).ToList();
  • Many relation
var connection = new SqlConnection(ConnectionString);
var rows = connection.Query<Category, Product, Category>(@"select * from Categories c inner join Products p on p.CategoryId = c.CategoryId", (category, product) =>
{
    category.Products.Add(product); // navigation property

    return category;
}).ToList();

// group by CategoryId
var categories = new List<Category>();
foreach (var row in rows)
{
    // each row has one product
    var category = categories.FirstOrDefault(x => x.CategoryId == row.CategoryId);
    if(category != null)
    {
        // append row's product to category
        category.Products.AddRange(row.Products);
    }
    else
        categories.Add(row);                
}

Async

using (var connection = new SqlConnection(ConnectionString))
{
    var posts = await connection.QueryAsync<Post>("SELECT * FROM [Posts]");
    var post = await connection.QueryOneAsync<Post>("SELECT * FROM [Posts] WHERE PostId=@PostId", new { PostId = 1 });
}

And more ...

Get the new id with Query

var sql = "INSERT INTO [Posts]([Title],[Content]) VALUES (@Title,@Content);SELECT CAST(SCOPE_IDENTITY() as int)";
var id = connection.Query<int>(sql, new Post { Title = "First Article", Content = "First Content" }).Single();

Or with ExecuteScalar

var sql = "INSERT INTO [Posts]([Title],[Content]) output inserted.id VALUES (@Title,@Content);SELECT CAST(SCOPE_IDENTITY() as int)";
int id = (int)connection.ExecuteScalar(sql, new Post { Title = "First Article", Content = "First Content" });

For unspecified object use Row

var rows = connection.Query<Row>("select * from Posts; select * from Categories").ToList();

Output parameter with Stored procedure

var connection = new SqlConnection(ConnectionString);

connection.ExecuteNonQuery(@"
    CREATE PROC usp_AddUser
    (
        @UserName nvarchar(150),
        @UserId int OUTPUT
    )
    AS
    BEGIN
        INSERT INTO Users(UserName) VALUES(@UserName);
        SET @UserId = SCOPE_IDENTITY();
    END
");

var command = connection.CreateStoredProcCommand("usp_AddUser")
    .AddInParameter("UserName", "Brad")
    .AddOutParameter("UserId", dbType: DbType.Int32);

command.Open().ExecuteNonQuery();

var userId = ((IDbDataParameter)command.Parameters["UserId"]).Value;

TableAttribute and ColumnAttribute (used by DefaultDataReaderMapper)

  • Table Attribute allows to define the table name. Also the mapper tries to find a match to the type name (pluralized and singularized + ignore case)
  • Column attribute allows to define the column name. Also the mapper tries to find a match to the property name.

Note: NpgsqlDbParameterBuilder (PostgreSql) uses the Column attribute to set NpgsqlDbType to Json and Xml. Avoid conversion errors with Npgsql library.

public class MyClass
{
    [Column(TypeName = "json")]
    public string MyJson { get; set; }

    [Column(TypeName = "xml")]
    public string MyXml { get; set; }
}

ColumnAnnotations

Avoids conflicts for tables with same column names and allows to resolve columns with aliases.

var results = connection.Query<Row>(@"Select ProductId, t1.Name, CategoryId, t2.Name from Products t1, Categories t2", columnAnnotations: new ColumnAnnotations
{
    new ColumnAnnotation("Name","Products",1),
    new ColumnAnnotation("Name","Categories",3),
}).ToList();

Alias

var results = connection.Query<Category>(@"Select Name as MyAlias from Categories", columnAnnotations: new ColumnAnnotations
{
    new ColumnAnnotation("MyAlias","Name","Categories",0),
}).ToList();

DbParameterBuilder

Allows to set up the DbParameter created for a param. A param is an anonymous object, an entity, etc.

Anonymous object :

var product = connection.QueryOne<Product>(@"SELECT * FROM [Products] WHERE [ProductID]=@ProductID", new { ProductID = 1 });

Entity :

connection.ExecuteNonQuery("INSERT INTO [Posts]([Title],[Content]) VALUES (@Title,@Content)", new Post { Title = "First Article", Content = "First Content" }); 

2 types of DbParameterBuilders:

  • By Provider
  • By Type

Providers:

  • "System.Data.SqlClient": Sql Server
  • "Microsoft.Data.Sqlite": Sqlite
  • "MySql.Data.MySqlClient" or "MySqlConnector": MySql
  • "Npgsql": PostgreSql
  • DefaultDbParameterBuilder is used if no factory found for a provider.

Sample replace the DbParameterBuilder for the provider "Npgsql"

public class MyDbParameterBuilder : DbParameterBuilderBase
{
    protected override void SetUpDbParameterInternal(IDbDataParameter parameter, string parameterName, object value, PropertyInfo sourceProperty)
    {
        parameter.ParameterName = parameterName;
        parameter.Value = value; // caution: conversion required. For example: PostgreSql doesn't support uint
            
        // DbType, NpgsqlDbType, etc.
    }
}

Register

DbParameterBuilderRegistry.Default.Register("Npgsql", () => new MyDbParameterBuilder());

Create a DbParameterBuilder for a Type

Usefull for :

  • Custom parameter configuration (example set the NpgsqlDbType to Json)
  • Parameter nameless (OleDb)

Sample

public class SelectCustomerByCountryAndCityDbParameterBuilder : DbParameterBuilder<CustomerFilter>
{
    protected internal override void SetUpDbParameter(
        IDbDataParameter parameter, 
        string parameterName, 
        string originalParameterName, 
        int parameterIndex, 
        CustomerFilter customerFilter)
    {
        parameter.ParameterName = parameterName;

        // use parameter index
        if (parameterIndex == 0)
            parameter.Value = customerFilter.Country;
        else if (parameterIndex == 1)
            parameter.Value = customerFilter.City;
    }
}
public class CustomerFilter
{
    public string Country { get; set; }
    public string City { get; set; }
}

Register and use

DbParameterBuilderRegistry.Default.Register(typeof(CustomerFilter), () => new SelectCustomerByCountryAndCityDbParameterBuilder());

using (var connection = new OleDbConnection(ConnectionString))
{
    var customers = connection.Query<Customer>("select * from Customers where Country=? and City=?", new CustomerFilter { Country = "UK", City = "London" }).ToList();
}

DataReader Mapper

Uses the data reader to convert values to entity.

public class Post
{
    public int PostId { get; set; }
    public string Title { get; set; }
    // etc.
}

public class PostMapper : IDataReaderMapper<Post>
{
    public Post Map(IDataReader reader)
    {
        var post = new Post();
        for (int i = 0; i < reader.FieldCount; i++)
        {
            var name = reader.GetName(i);
            if (name == "PostId")
                post.PostId = reader.GetInt32(i);
            else if (name == "Title")
                post.Title = reader.IsDBNull(i) ? null : reader.GetString(i);
        }
        return post;
    }
}

Register

DataReaderMapperRegistry.Default.Register<Post, PostMapper>();

Or directly

connection.Query<Post>("Select * from Posts", map: (reader) =>
{
    var post = new Post();
    for (int i = 0; i < reader.FieldCount; i++)
    {
        var name = reader.GetName(i);
        if (name == "PostId")
            post.PostId = reader.GetInt32(i);
        else if (name == "Title")
            post.Title = reader.IsDBNull(i) ? null : reader.GetString(i);
    }
    return post;
});

Sql Server Spatial

With NetTopologySuite and NetTopologySuite.IO.SqlServerBytes

Sample :

public class City
{
    public string CityName { get; set; }
    public Point Location { get; set; }
}

Create a custom DbParameterBuilder

public class CityDbParameterBuilder : DbParameterBuilder<City>
{
    protected override void SetUpDbParameter(
        IDbDataParameter parameter,
        string parameterName,
        string originalParameterName,
        int parameterIndex,
        City currentParam)
    {
        parameter.ParameterName = parameterName;
        if (originalParameterName == "CityName")
        {
            parameter.Value = currentParam.CityName;
        }
        else if (originalParameterName == "Location")
        {
            var geometryWriter = new SqlServerBytesWriter { IsGeography = true };
            var bytes = geometryWriter.Write(currentParam.Location);
            // with sql server (optional)
            //if (parameter is SqlParameter sqlParameter)
            //{
            //    sqlParameter.SqlDbType = SqlDbType.Udt;
            //    sqlParameter.UdtTypeName = "geography";
            //}
            parameter.Value = new SqlBytes(bytes);
        }
    }
}

Create a DataReaderMapper

public class CityMap : IDataReaderMapper<City>
{
    public City Map(IDataReader reader)
    {
        var city = new City();
        for (int i = 0; i < reader.FieldCount; i++)
        {
            var columnName = reader.GetName(i);
            if (columnName == "CityName")
            {
                city.CityName = reader.GetString(i);
            }
            else if (columnName == "Location")
            {
                if (reader is SqlDataReader sqlDataReader)
                {
                    var bytes = sqlDataReader.GetSqlBytes(i).Value;
                    var geoReader = new SqlServerBytesReader();
                    var geometry = geoReader.Read(bytes);
                    city.Location = (Point)geometry;
                }

                // or with GetBytes
                //byte[] bytes = Utils.GetBytes(reader, i); // utility method
                //var geoReader = new SqlServerBytesReader();
                //var geometry = geoReader.Read(bytes);
                //city.Location = (Point)geometry;
            }
        }
        return city;
    }
}

Insert

DbParameterBuilderRegistry.Default.Register(typeof(City), () => new CityDbParameterBuilder());

using (var connection = new SqlConnection(SampleConnectionString))
{
    connection.ExecuteNonQuery("INSERT INTO Cities(CityName, Location)VALUES(@CityName,@Location);", new City
    {
        CityName = "Santo Domingo",
        Location = new Point(new Coordinate(-69.9388777, 18.4839233)) { SRID = 4326 }
    });
}

Read

DataReaderMapperRegistry.Default.Register<City, CityMap>();

using (var connection = new SqlConnection(SampleConnectionString))
{
    var cities = connection.Query<City>("select CityName,Location from Cities").ToList();
    return cities;

}

Dependencies

Replace a dependency

Dependencies.Default.Replace<IPluralizer, MyPluralizer>();
  • IPluralizer: HumanizerPluralizer by default. Allows to pluralize or singularize Entity Names to find table names.
  • IValueToPropertyConverter: ValueToPropertyConverter by default. Allows to convert DataReader values to property types.
  • IProviderResolver: ProviderResolver by default. Allows to resolve Providers.
  • IQueryParameterFinder: QueryParameterFinder by default. Allows to find parameters in sql
  • ISqlQuerySplitter: SqlQuerySplitter by default. Allows to split sql into queries (split on ";" and "GO") and clean queries.
  • IInstanceFactory: Service used by DefaultDateReaderMapper to create instances ... availables: ReflectionInstanceFactory (default).
  • IPropertySetProvider: Service used by DefaultDateReaderMapper to set property value ... availables: ReflectionPropertySetProvider (default)

Interception

SDEProxy. Intercept IDbConnection, IDbCommand, etc.

public class Sample
{
    private const string ConnectionString = "Server=(localdb)\\MSSQLLocalDB;Database=Northwind;Trusted_Connection=True;MultipleActiveResultSets=true;";

    public void Run()
    {
        using (var connection = new SqlConnection(ConnectionString))
        {
            var command = connection.CreateCommand();
            command.CommandText = "SELECT COUNT(*) FROM [Customers] WHERE [Country]=@Country";
            command.Parameters.Add(new SqlParameter("Country", "France"));

            var proxy = SDEProxy<IDbCommand>.CreateProxy(command, BeforeInvoke, AfterInvoke, OnFailed);

            connection.Open();

            int result = (int)command.ExecuteScalar();
            Console.WriteLine("result: " + result);

            int proxyResult = (int)proxy.ExecuteScalar();
            Console.WriteLine("proxy result: " + proxyResult);
        }
    }

    private void BeforeInvoke(MethodInfo method, object[] parameters, IDbCommand command)
    {
        // update command parameter value
        int index = command.Parameters.IndexOf("Country");
        if (index != -1)
            ((IDataParameter)command.Parameters[index]).Value = "UK";
    }

    private void AfterInvoke(MethodInfo method, object[] parameters, IDbCommand command, object result) { }
    private void OnFailed(MethodInfo method, object[] parameters, IDbCommand command, Exception ex) { }
}

Or use a library like PostSharp, Fody, NIntercept, Castle, etc.

Or use Visitor pattern to create a custom command for example.

Product Compatible and additional computed target framework versions.
.NET net5.0 is compatible.  net5.0-windows was computed.  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. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.1.3 514 3/14/2022
1.1.0 440 3/13/2022
1.0.2 450 2/27/2022
0.20.0 479 2/25/2022
0.19.0 468 2/24/2022
0.18.2 477 2/24/2022
0.17.2 451 2/22/2022
0.17.1 444 2/22/2022
0.15.0 441 2/19/2022

MySqlConnector, CommandTimeout, OpenAsync