SQL Server CLR Integration
Have you ever noticed that the ROUND
function in SQL Server returns different results than the Math.Round
method in the .NET Framework?
Try and execute the following code in SQL Server, and you will get 12.350.
select ROUND(12.345, 2)
Try and execute the following code in Visual Studio's C# Interactive, and you will get 12.34.
Math.Round(12.345, 2)
There must be an explanation, but I never cared to look for it.
Instead, I decided to leverage this fact and write a brief article that explains how to develop a .NET assembly, containing a user-defined function written in C#, and install the assembly in the Common Language Runtime hosted in Microsoft SQL Server (called CLR integration) in order to be able to invoke the user-defined function from SQL code.
Let us begin by creating a SQL Server Database Project in Visual Studio.
Ensure that, in the properties of the project, in the Project Settings section, you set the Target Platform to the correct version of SQL Server.
Should you want to use the features of the latest C# version, you may want to edit the project file and add the following XML element to it.
<PropertyGroup>
<LangVersion>preview</LangVersion>
</PropertyGroup>
Add a couple of references:
- System
- System.Data
Add a class similar to the following one.
public partial class UserDefinedFunctions
{
[SqlFunction]
public static SqlDouble MATH_ROUND(SqlDouble value, SqlInt32 digits)
{
SqlDouble result = new(0);
if (!value.IsNull)
result = new SqlDouble(Math.Round(value.Value, digits.Value));
return result;
}
}
Note that the class must be public and partial, and it must be named UserDefinedFunctions
, which means that you create a user-defined function by adding a method to an existing class with the same name.
Additionally, note that the method must be static and adorned with the SqlFunction
attribute.
Needless to say, the method above introduces a user-defined function called MATH_ROUND
that will round numbers in SQL Server just like the Math.Round
method in the .NET Framework.
It is now time to build the project and deploy the assembly to SQL Server.
The following SQL code will do the job for you, assuming the path to the assembly is "C:\Data\SqlUtils.dll".
sp_configure 'show advanced options', 1
go
reconfigure
go
sp_configure 'clr strict security', 0
go
reconfigure
go
sp_configure 'clr enabled', 1
go
reconfigure
go
drop function if exists MATH_ROUND
go
drop assembly if exists SqlUtils
go
create assembly SqlUtils from 'C:\Data\SqlUtils.dll'
go
create function MATH_ROUND(@value as float, @digits as int) returns float external name SqlUtils.UserDefinedFunctions.MATH_ROUND
go
Here is the description of each step performed by the code:
- Enable the advanced configuration options, which is a prerequisite for the next step.
- Disable the strict security policy that would prevent us from registering and executing our assembly -- for the sake of simplicity, I do not want to enter the Code Access Security realm in this article; however, disabling the strict security policy is not a good practice, especially in production environments.
- Enable the SQL Server CLR Integration, which is disabled by default.
- Delete the user-defined function entry, if it already exists.
- Delete the assembly, if it is already registered with the SQL Server CLR.
- Register the assembly with the SQL Server CLR.
- Create the user-defined function entry.
Now you can try and execute the following code in SQL Server, and you will get 12.34.
select dbo.MATH_ROUND(12.345, 2)
A sample project is available at https://github.com/fabioscagliola/SQL-Server-CLR-Integration