ASP.NET MVC, Javscript, JQuery, Angular JS, HTML 5, CSS3
Web Services, Web Api, SQL Server, C#.NET, Azure
20 May 2011
16 May 2011
Dynamic Stored Procedures
Have you every felt the need to write a stored procedure that would be modified at runtime?
Well then let me show you how you can achieve this in SQL Server 2005 or upper version.
First of all let me tell you about the scenario where i had to write such a stored procedure:
Recently i was dealing with a large database (having more than 5 million records), and i had to provide the client with an option to search and filter data from a table with various search parameters in an asp.net web page.
Now few of the parameters when combined together in the where clause of the search query worked fine without much delay (mostly integer columns), but some the text columns took really long time especially when combined together in the where clause. For instance:
Query: select * from table 1 where
personame ='' or
cityid= '' or
state='' .....
In such a scenario i had to write an dynamic stored procedure which would accept parameters and form query according to the number of paramters passed.
Here is the stored procedure that i had wrote for fetching some patients information from an insurance database:
ALTER procedure [dbo].[sp_GetEligibility]
(@Insured nvarchar(50)=null,@GroupKey nvarchar(50)=null,MAX_RESULTS nvarchar(50)=null )
AS
BEGIN
if @MAX_RESULTS is null
begin
set @MAX_RESULTS=250
end
DECLARE @SQL nvarchar(MAX)
DECLARE @SQLWhere nvarchar(MAX)
set @SQLWhere = ''
set @Insured = LTRIM(RTRIM(@Insured))
set @GroupKey = LTRIM(RTRIM(@GroupKey))
if (@Insured is not null And @Insured <> '')
begin
--set @SQLWhere = @SQLWhere + ' and (I.ssn LIKE ''%' + @Insured + '%'' ) '
set @SQLWhere = @SQLWhere + ' and (I.ssn LIKE ''%'' + @Insured + ''%'' ) '
end
if (@GroupKey is not null And @GroupKey = '')
begin
set @GroupKey = null
end
Else if @GroupKey <> '0'
begin
--set @SQLWhere = @SQLWhere + ' and (E.fk_group = '+ @GroupKey + ')'
set @SQLWhere = @SQLWhere + ' and (E.fk_group = @GroupKey )'
end
IF not (@Insured is null and @GroupKey is null)
BEGIN
Select @SQL ='Select TOP(' + @MAX_RESULTS + ') * from (' +
'select TOP(' + @MAX_RESULTS + ') I.first_name,' +
'R.Relationship,' +
'I.last_name, ' +
'I.SSN,' +
'E.subscriber_id,' +
'I.first_name + '' ''+ I.last_name as Names,' +
'e.eff_date,' +
'e.term_date,' +
'G.group_name,' +
'B.benefitPlan_name, ' +
'convert(varchar,D.eff_date ,101) as dependEffDate,' +
'convert(varchar,d.Term_date ,101) as dependTermdate,' +
'I.pk_insured,' +
'(SELECT top 1 ' +
'(' +
' SELECT distinct(C.benefit_name + '', '')' +
' FROM EligibilityBenefitCodes B ' +
' left join BenefitPlanCodes C on c.pk_benefitplancodes = B.fk_benefitplancodes' +
' left join BenefitPlan BP on BP.pk_benefitplan= C.fk_benefitplan ' +
' WHERE C.pk_benefitplancodes = B.fk_benefitplancodes' +
' and B.fk_eligibility = E.pk_eligibility FOR XML PATH('''')' +
' )' +
')AS TypeofCoverage ' +
'from Eligibility E With(NOLOCK) ' +
'Inner Join Dependents D With(NOLOCK) ON e.fk_insured=D.fk_ insuredprimary ' +
'Inner Join Insured I With(NOLOCK) ON I.pk_insured=D.fk_ insuredDependent and E.fk_insured = I.pk_insured ' +
'Inner Join BenefitPlan B With(NOLOCK) ON E.fk_benefitplan=B.pk_ benefitplan ' +
'Inner Join Relation R With(NOLOCK) ON I.fk_relation=R.pk_relation ' +
'Inner Join [Group] G With(NOLOCK) ON E.fk_group = G.pk_group ' +
'where e.eff_date <= getDate() and E.IsArchived=0 ' +
@SQLWhere +
' UNION ' +
'select TOP(' + @MAX_RESULTS + ') I.first_name,R.Relationship,I. last_name, I.SSN, E.subscriber_id, ' +
'I.first_name + '' ''+ I.last_name as Names, e.eff_date, ' +
'e.term_date, G.group_name, B.benefitPlan_name, ' +
'convert(varchar,D.eff_date ,101) as dependEffDate, ' +
'convert(varchar,d.Term_date ,101) as dependTermdate, ' +
'I.pk_insured , ' +
'(SELECT top 1' +
' ( ' +
' SELECT distinct(C.benefit_name + '', '')' +
'FROM EligibilityBenefitCodes B ' +
'left join BenefitPlanCodes C on c.pk_benefitplancodes = B.fk_benefitplancodes ' +
'left join BenefitPlan BP on BP.pk_benefitplan= C.fk_benefitplan ' +
'WHERE C.pk_benefitplancodes = B.fk_benefitplancodes and B.fk_eligibility = E.pk_eligibility ' +
' FOR XML PATH('''') ' +
' )) AS BenefitCode ' +
'from Eligibility E With(NOLOCK) ' +
'Inner Join Dependents D With(NOLOCK) ON e.fk_insured=D.fk_ insuredprimary ' +
'Inner Join Insured I With(NOLOCK) ON I.pk_insured=D.fk_ insuredDependent ' +
'Inner Join BenefitPlan B With(NOLOCK) ON E.fk_benefitplan=B.pk_ benefitplan ' +
'Inner Join Relation R With(NOLOCK) ON I.fk_relation=R.pk_relation ' +
'Inner Join [Group] G With(NOLOCK) ON E.fk_group = G.pk_group ' +
'where e.eff_date <= getDate() and E.IsArchived=0 ' +
@SQLWhere +
')A ' +
' order by A.Relationship,A.first_name,A. last_name desc'
--EXEC(@SQL)
--For preventing SQL Injection use following method
EXEC sp_executesql @SQL, N'@Insured nvarchar(50),@GroupKey nvarchar(50)',@Insured,@ GroupKey
END
END
As you can see in this example, the stored procedure generates the sql query as string and at the end the parameters are passed to the generated final query.
This is pretty much similar to the if else conditions that we used to write in code behind to generate a query.
Well then let me show you how you can achieve this in SQL Server 2005 or upper version.
First of all let me tell you about the scenario where i had to write such a stored procedure:
Recently i was dealing with a large database (having more than 5 million records), and i had to provide the client with an option to search and filter data from a table with various search parameters in an asp.net web page.
Now few of the parameters when combined together in the where clause of the search query worked fine without much delay (mostly integer columns), but some the text columns took really long time especially when combined together in the where clause. For instance:
Query: select * from table 1 where
personame ='' or
cityid= '' or
state='' .....
In such a scenario i had to write an dynamic stored procedure which would accept parameters and form query according to the number of paramters passed.
Here is the stored procedure that i had wrote for fetching some patients information from an insurance database:
ALTER procedure [dbo].[sp_GetEligibility]
(@Insured nvarchar(50)=null,@GroupKey nvarchar(50)=null,MAX_RESULTS nvarchar(50)=null )
AS
BEGIN
if @MAX_RESULTS is null
begin
set @MAX_RESULTS=250
end
DECLARE @SQL nvarchar(MAX)
DECLARE @SQLWhere nvarchar(MAX)
set @SQLWhere = ''
set @Insured = LTRIM(RTRIM(@Insured))
set @GroupKey = LTRIM(RTRIM(@GroupKey))
if (@Insured is not null And @Insured <> '')
begin
--set @SQLWhere = @SQLWhere + ' and (I.ssn LIKE ''%' + @Insured + '%'' ) '
set @SQLWhere = @SQLWhere + ' and (I.ssn LIKE ''%'' + @Insured + ''%'' ) '
end
if (@GroupKey is not null And @GroupKey = '')
begin
set @GroupKey = null
end
Else if @GroupKey <> '0'
begin
--set @SQLWhere = @SQLWhere + ' and (E.fk_group = '+ @GroupKey + ')'
set @SQLWhere = @SQLWhere + ' and (E.fk_group = @GroupKey )'
end
IF not (@Insured is null and @GroupKey is null)
BEGIN
Select @SQL ='Select TOP(' + @MAX_RESULTS + ') * from (' +
'select TOP(' + @MAX_RESULTS + ') I.first_name,' +
'R.Relationship,' +
'I.last_name, ' +
'I.SSN,' +
'E.subscriber_id,' +
'I.first_name + '' ''+ I.last_name as Names,' +
'e.eff_date,' +
'e.term_date,' +
'G.group_name,' +
'B.benefitPlan_name, ' +
'convert(varchar,D.eff_date ,101) as dependEffDate,' +
'convert(varchar,d.Term_date ,101) as dependTermdate,' +
'I.pk_insured,' +
'(SELECT top 1 ' +
'(' +
' SELECT distinct(C.benefit_name + '', '')' +
' FROM EligibilityBenefitCodes B ' +
' left join BenefitPlanCodes C on c.pk_benefitplancodes = B.fk_benefitplancodes' +
' left join BenefitPlan BP on BP.pk_benefitplan= C.fk_benefitplan ' +
' WHERE C.pk_benefitplancodes = B.fk_benefitplancodes' +
' and B.fk_eligibility = E.pk_eligibility FOR XML PATH('''')' +
' )' +
')AS TypeofCoverage ' +
'from Eligibility E With(NOLOCK) ' +
'Inner Join Dependents D With(NOLOCK) ON e.fk_insured=D.fk_
'Inner Join Insured I With(NOLOCK) ON I.pk_insured=D.fk_
'Inner Join BenefitPlan B With(NOLOCK) ON E.fk_benefitplan=B.pk_
'Inner Join Relation R With(NOLOCK) ON I.fk_relation=R.pk_relation ' +
'Inner Join [Group] G With(NOLOCK) ON E.fk_group = G.pk_group ' +
'where e.eff_date <= getDate() and E.IsArchived=0 ' +
@SQLWhere +
' UNION ' +
'select TOP(' + @MAX_RESULTS + ') I.first_name,R.Relationship,I.
'I.first_name + '' ''+ I.last_name as Names, e.eff_date, ' +
'e.term_date, G.group_name, B.benefitPlan_name, ' +
'convert(varchar,D.eff_date ,101) as dependEffDate, ' +
'convert(varchar,d.Term_date ,101) as dependTermdate, ' +
'I.pk_insured , ' +
'(SELECT top 1' +
' ( ' +
' SELECT distinct(C.benefit_name + '', '')' +
'FROM EligibilityBenefitCodes B ' +
'left join BenefitPlanCodes C on c.pk_benefitplancodes = B.fk_benefitplancodes ' +
'left join BenefitPlan BP on BP.pk_benefitplan= C.fk_benefitplan ' +
'WHERE C.pk_benefitplancodes = B.fk_benefitplancodes and B.fk_eligibility = E.pk_eligibility ' +
' FOR XML PATH('''') ' +
' )) AS BenefitCode ' +
'from Eligibility E With(NOLOCK) ' +
'Inner Join Dependents D With(NOLOCK) ON e.fk_insured=D.fk_
'Inner Join Insured I With(NOLOCK) ON I.pk_insured=D.fk_
'Inner Join BenefitPlan B With(NOLOCK) ON E.fk_benefitplan=B.pk_
'Inner Join Relation R With(NOLOCK) ON I.fk_relation=R.pk_relation ' +
'Inner Join [Group] G With(NOLOCK) ON E.fk_group = G.pk_group ' +
'where e.eff_date <= getDate() and E.IsArchived=0 ' +
@SQLWhere +
')A ' +
' order by A.Relationship,A.first_name,A.
--EXEC(@SQL)
--For preventing SQL Injection use following method
EXEC sp_executesql @SQL, N'@Insured nvarchar(50),@GroupKey nvarchar(50)',@Insured,@
END
END
As you can see in this example, the stored procedure generates the sql query as string and at the end the parameters are passed to the generated final query.
This is pretty much similar to the if else conditions that we used to write in code behind to generate a query.
Subscribe to:
Comments (Atom)