Skip to content

Latest commit

 

History

History
80 lines (57 loc) · 2.51 KB

README.md

File metadata and controls

80 lines (57 loc) · 2.51 KB

How to use SQL scalar-valued functions in LINQ to Entities queries:

1.Create a custom function in your database

Example:

create function [dbo].[fn_UserFullName] (@userid uniqueidentifier) 
returns varchar(210)
as 
begin
	declare @fullname varchar(210)

	select	@fullname = case when (p.MiddleName is null) or (p.MiddleName = '') 
						then p.FirstName + ' ' + p.Surname 
						else p.FirstName + ' ' + p.MiddleName + ' ' + p.Surname 
						end
	from Users u
	inner join Persons p on u.Personid = p.Personid
	where	u.Userid = @userid	

	return (@fullname);
end;
  1. Install EntityFramework.CodeFirstStoreFunctions nuget package

    Nuget: https://www.nuget.org/packages/EntityFramework.CodeFirstStoreFunctions/ Source code and example: https://codefirstfunctions.codeplex.com/

  2. Create your mapped-to function using DbFunctionAttribute with the hard coded schema name CodeFirstDatabaseSchema

Example:

public class UserDbFunction
{
	[DbFunction("CodeFirstDatabaseSchema", "fn_UserFullName")]
	public static string FnUserFullName(Guid userId)
	{
		throw new NotSupportedException();
	}
}
  1. Override the OnModelCreating method and add the custom convention into Conventions configuration.

Example:

protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
	modelBuilder.Conventions.Add(new FunctionsConvention("dbo", typeof(UserDbFunction)));

	//...
}
  1. Call the custom function.

Example:

var dbContext = new SampleDataModel();

//EXAMPLE 1: use sql scalar user defined function in projection 
var userProfiles = dbContext.Users.Select(x => new {EmailAddress = x.EmailAddress, FullName = UserDbFunction.FnUserFullName(x.Userid)}).Take(3).ToList();

Console.WriteLine("Top 3 users:");
foreach (var userProfile in userProfiles)
{
	Console.WriteLine($"Email Address: {userProfile.EmailAddress} - Full Name: {userProfile.FullName}");
}

//EXAMPLE 2: use sql scalar user defined function in restriction (where clause)
//Find 'Barbara Santa' email address
var user = dbContext.Users.FirstOrDefault(x => UserDbFunction.FnUserFullName(x.Userid).Equals("Barbara Santa", StringComparison.OrdinalIgnoreCase));
Console.WriteLine(Environment.NewLine + $"'Barbara Santa' email address: {user?.EmailAddress}");

Output:

Top 3 users:
Email Address: [email protected] - Full Name: Sean Paul
Email Address: [email protected] - Full Name: Barbara Santa
Email Address: [email protected] - Full Name: Charmaine Sheh

'Barbara Santa' email address: [email protected]