NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse
1.0.0
See the version list below for details.
dotnet add package NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse --version 1.0.0
NuGet\Install-Package NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse -Version 1.0.0
<PackageReference Include="NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse" Version="1.0.0" />
paket add NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse --version 1.0.0
#r "nuget: NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse, 1.0.0"
// Install NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse as a Cake Addin #addin nuget:?package=NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse&version=1.0.0 // Install NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse as a Cake Tool #tool nuget:?package=NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse&version=1.0.0
EF Core 3.x support for SQL Server's TRY_PARSE
function
Use
Install the NuGet package:
Install-Package NevaleeBusinessSolutions.EntityFrameworkCore.SqlServer.TryParse
Register the functions in your DbContext
's OnModelCreating
method:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
base.OnModelCreating(modelBuilder);
TryParse.Register(modelBuilder);
}
Then call the functions as part of a query:
var result = context.Set<SomeEntity>()
.Select(e => new { e.Id, e.Value, ValueInt32 = TryParse.Int32(e.Value) })
.ToList();
This will generate the expected SQL:
SELECT Id, Value, TRY_PARSE(Value As int) As ValueInt32 FROM SomeEntities
Background
TRY_PARSE was added in SQL Server 2012. However, EF Core 3.x does not support calling this function by default.
Whilst EF Core provides methods to map user-defined functions, mapping TRY_PARSE
is complicated by the way the arguments are passed. EF Core has great support for traditional functions, where the arguments are passed as a comma-separated list - eg:
dbo.SomeFunction(Foo.Bar, @b, 42)
But for TRY_PARSE
, the arguments are separated by spaces, not commas:
TRY_PARSE(Foo.Bar AS int)
To enable this, it was necessary to implement a custom SqlExpression
class to represent the parameter. This class needs to override both the Print
and Accept
methods in order to generate the correct SQL.
internal sealed class TryParseArgumentExpression : SqlExpression
{
private readonly SqlExpression _sourceExpression;
private readonly SqlFragmentExpression _asExpression;
public TryParseArgumentExpression(Type type, SqlExpression sourceExpression, string sqlTypeName)
: base(type, sourceExpression.TypeMapping)
{
_sourceExpression = sourceExpression ?? throw new ArgumentNullException(nameof(sourceExpression));
_asExpression = new SqlFragmentExpression($" AS {sqlTypeName}");
}
private TryParseArgumentExpression(Type type, SqlExpression sourceExpression, SqlFragmentExpression asExpression)
: base(type, sourceExpression.TypeMapping)
{
_sourceExpression = sourceExpression ?? throw new ArgumentNullException(nameof(sourceExpression));
_asExpression = asExpression ?? throw new ArgumentNullException(nameof(asExpression));
}
protected override Expression VisitChildren(ExpressionVisitor visitor)
{
var newSource = (SqlExpression?)visitor.Visit(_sourceExpression) ?? _sourceExpression;
var newAsExpression = (SqlFragmentExpression?)visitor.Visit(_asExpression) ?? _asExpression;
if (Equals(newSource, _sourceExpression) && Equals(newAsExpression, _asExpression)) return this;
return new TryParseArgumentExpression(Type, newSource, newAsExpression);
}
protected override Expression Accept(ExpressionVisitor visitor)
{
visitor.Visit(_sourceExpression);
visitor.Visit(_asExpression);
return this;
}
public override void Print(ExpressionPrinter expressionPrinter)
{
expressionPrinter.Visit(_sourceExpression);
expressionPrinter.Visit(_asExpression);
}
}
It was then possible to use this custom expression, along with an internal attribute which specifies the mapped SQL type name, to register the custom functions:
public static void Register(ModelBuilder modelBuilder)
{
foreach (var dbFunc in typeof(TryParse).GetMethods(BindingFlags.Public | BindingFlags.Static))
{
var attribute = dbFunc.GetCustomAttribute<SqlTypeNameAttribute>();
if (attribute is null) continue;
modelBuilder.HasDbFunction(dbFunc).HasTranslation(args =>
{
var newArgs = args.ToList();
newArgs[0] = new TryParseArgumentExpression(dbFunc.ReturnType, newArgs[0], attribute.SqlTypeName);
return SqlFunctionExpression.Create("TRY_PARSE", newArgs, dbFunc.ReturnType, null);
});
}
}
License
Copyright (c) 2021 Richard Deeming All rights reserved.
This code is free software: you can redistribute it and/or modify it under the terms of either
the Code Project Open License (CPOL) version 1 or later; or
the GNU General Public License as published by the Free Software Foundation, version 3 or later; or
the BSD 2-Clause License;
This code is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. 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. |
.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. |
-
.NETStandard 2.0
- Microsoft.EntityFrameworkCore.SqlServer (>= 3.1.18)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.