RSCG – Dapper.AOT
RSCG – Dapper.AOT
name | Dapper.AOT |
nuget | https://www.nuget.org/packages/Dapper.AOT/ |
link | https://aot.dapperlib.dev/ |
author | Marc Gravell |
Generating AOT code for Dapper -hydrating classes from SQL queries.
This is how you can use Dapper.AOT .
The code that you start with is
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 | < Project Sdk = "Microsoft.NET.Sdk" > < PropertyGroup > < OutputType >Exe</ OutputType > < TargetFramework >net9.0</ TargetFramework > < ImplicitUsings >enable</ ImplicitUsings > < Nullable >enable</ Nullable > </ PropertyGroup > < ItemGroup > < PackageReference Include = "Dapper" Version = "2.1.35" /> < PackageReference Include = "Dapper.AOT" Version = "1.0.31" /> < PackageReference Include = "Microsoft.Data.SqlClient" Version = "5.2.2" /> </ ItemGroup > < PropertyGroup > < InterceptorsPreviewNamespaces >$(InterceptorsPreviewNamespaces);Dapper.AOT</ InterceptorsPreviewNamespaces > </ PropertyGroup > < PropertyGroup > < EmitCompilerGeneratedFiles >true</ EmitCompilerGeneratedFiles > < CompilerGeneratedFilesOutputPath >$(BaseIntermediateOutputPath)\GX</ CompilerGeneratedFilesOutputPath > </ PropertyGroup > </ Project > |
The code that you will use is
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 | <Project Sdk= "Microsoft.NET.Sdk" > <PropertyGroup> <OutputType>Exe</OutputType> <TargetFramework>net9.0</TargetFramework> <ImplicitUsings>enable</ImplicitUsings> <Nullable>enable</Nullable> </PropertyGroup> <ItemGroup> <PackageReference Include= "Dapper" Version= "2.1.35" /> <PackageReference Include= "Dapper.AOT" Version= "1.0.31" /> <PackageReference Include= "Microsoft.Data.SqlClient" Version= "5.2.2" /> </ItemGroup> <PropertyGroup> <InterceptorsPreviewNamespaces>$(InterceptorsPreviewNamespaces);Dapper.AOT</InterceptorsPreviewNamespaces> </PropertyGroup> <PropertyGroup> <EmitCompilerGeneratedFiles> true </EmitCompilerGeneratedFiles> <CompilerGeneratedFilesOutputPath>$(BaseIntermediateOutputPath)\GX</CompilerGeneratedFilesOutputPath> </PropertyGroup> </Project> |
1 2 3 4 | // See https://aka.ms/new-console-template for more information Console.WriteLine( "Hello, World!" ); var p= Product.GetProduct( new SqlConnection( "Server=localhost;Database=AdventureWorks2019;Trusted_Connection=True;" ), 1); |
1 2 3 4 5 6 7 8 9 | namespace DapperDemo; internal partial class Product { public int ID { get ; set ; } public string Name { get ; set ; } = "" ; public string ProductId { get ; set ; } = "" ; public static Product GetProduct(SqlConnection connection, int productId) => connection.QueryFirst<Product>( "select ID, Name, ProductId from Production.Product where ProductId=@productId" , new { productId }); } |
1 2 3 4 5 6 | global using Dapper; global using Microsoft.Data.SqlClient; global using DapperDemo; [module: DapperAot] |
The code that is generated is
001 002 003 004 005 006 007 008 009 010 011 012 013 014 015 016 017 018 019 020 021 022 023 024 025 026 027 028 029 030 031 032 033 034 035 036 037 038 039 040 041 042 043 044 045 046 047 048 049 050 051 052 053 054 055 056 057 058 059 060 061 062 063 064 065 066 067 068 069 070 071 072 073 074 075 076 077 078 079 080 081 082 083 084 085 086 087 088 089 090 091 092 093 094 095 096 097 098 099 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 | #nullable enable namespace Dapper.AOT // interceptors must be in a known namespace { file static class DapperGeneratedInterceptors { [global::System.Runtime.CompilerServices.InterceptsLocationAttribute( "D:\\gth\\RSCG_Examples\\v2\\rscg_examples\\Dapper.AOT\\src\\DapperDemo\\DapperDemo\\Product.cs" , 8, 93)] internal static global::DapperDemo.Product QueryFirst0( this global::System.Data.IDbConnection cnn, string sql, object ? param, global::System.Data.IDbTransaction? transaction, int ? commandTimeout, global::System.Data.CommandType? commandType) { // Query, TypedResult, HasParameters, SingleRow, Text, AtLeastOne, BindResultsByName, KnownParameters // takes parameter: <anonymous type: int productId> // parameter map: productId // returns data: global::DapperDemo.Product global::System.Diagnostics.Debug.Assert(! string .IsNullOrWhiteSpace(sql)); global::System.Diagnostics.Debug.Assert((commandType ?? global::Dapper.DapperAotExtensions.GetCommandType(sql)) == global::System.Data.CommandType.Text); global::System.Diagnostics.Debug.Assert(param is not null ); return global::Dapper.DapperAotExtensions.Command(cnn, transaction, sql, global::System.Data.CommandType.Text, commandTimeout.GetValueOrDefault(), CommandFactory0.Instance).QueryFirst(param, RowFactory0.Instance); } private static global::Dapper.CommandFactory< object ?> DefaultCommandFactory => global::Dapper.CommandFactory.Simple; private sealed class RowFactory0 : global::Dapper.RowFactory<global::DapperDemo.Product> { internal static readonly RowFactory0 Instance = new (); private RowFactory0() {} public override object ? Tokenize(global::System.Data.Common.DbDataReader reader, global::System.Span< int > tokens, int columnOffset) { for ( int i = 0; i < tokens.Length; i++) { int token = -1; var name = reader.GetName(columnOffset); var type = reader.GetFieldType(columnOffset); switch (NormalizedHash(name)) { case 926444256U when NormalizedEquals(name, "id" ): token = type == typeof ( int ) ? 0 : 3; // two tokens for right-typed and type-flexible break ; case 2369371622U when NormalizedEquals(name, "name" ): token = type == typeof ( string ) ? 1 : 4; break ; case 2521315361U when NormalizedEquals(name, "productid" ): token = type == typeof ( string ) ? 2 : 5; break ; } tokens[i] = token; columnOffset++; } return null ; } public override global::DapperDemo.Product Read(global::System.Data.Common.DbDataReader reader, global::System.ReadOnlySpan< int > tokens, int columnOffset, object ? state) { global::DapperDemo.Product result = new (); foreach ( var token in tokens) { switch (token) { case 0: result.ID = reader.GetInt32(columnOffset); break ; case 3: result.ID = GetValue< int >(reader, columnOffset); break ; case 1: result.Name = reader.GetString(columnOffset); break ; case 4: result.Name = GetValue< string >(reader, columnOffset); break ; case 2: result.ProductId = reader.GetString(columnOffset); break ; case 5: result.ProductId = GetValue< string >(reader, columnOffset); break ; } columnOffset++; } return result; } } private sealed class CommandFactory0 : global::Dapper.CommandFactory< object ?> // <anonymous type: int productId> { internal static readonly CommandFactory0 Instance = new (); public override void AddParameters( in global::Dapper.UnifiedCommand cmd, object ? args) { var typed = Cast(args, static () => new { productId = default ( int ) }); // expected shape var ps = cmd.Parameters; global::System.Data.Common.DbParameter p; p = cmd.CreateParameter(); p.ParameterName = "productId" ; p.DbType = global::System.Data.DbType.Int32; p.Direction = global::System.Data.ParameterDirection.Input; p.Value = AsValue(typed.productId); ps.Add(p); } public override void UpdateParameters( in global::Dapper.UnifiedCommand cmd, object ? args) { var typed = Cast(args, static () => new { productId = default ( int ) }); // expected shape var ps = cmd.Parameters; ps[0].Value = AsValue(typed.productId); } public override bool CanPrepare => true ; } } } namespace System.Runtime.CompilerServices { // this type is needed by the compiler to implement interceptors - it doesn't need to // come from the runtime itself, though [global::System.Diagnostics.Conditional( "DEBUG" )] // not needed post-build, so: evaporate [global::System.AttributeUsage(global::System.AttributeTargets.Method, AllowMultiple = true )] sealed file class InterceptsLocationAttribute : global::System.Attribute { public InterceptsLocationAttribute( string path, int lineNumber, int columnNumber) { _ = path; _ = lineNumber; _ = columnNumber; } } } |
Code and pdf at
https://ignatandrei.github.io/RSCG_Examples/v2/docs/Dapper.AOT
Leave a Reply