diff --git a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SearchParameterQueryGenerator.cs b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SearchParameterQueryGenerator.cs index ec3c1b201e..446e3e7544 100644 --- a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SearchParameterQueryGenerator.cs +++ b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SearchParameterQueryGenerator.cs @@ -39,7 +39,7 @@ public override SearchParameterQueryGeneratorContext VisitSearchParameter(Search context.StringBuilder .Append(searchParamIdColumn, context.TableAlias) .Append(" = ") - .AppendLine(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true).ParameterName) + .Append(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true)).AppendLine() .Append("AND "); return expression.Expression.AcceptVisitor(this, context); @@ -53,7 +53,7 @@ public override SearchParameterQueryGeneratorContext VisitSortParameter(SortExpr context.StringBuilder .Append(searchParamIdColumn, context.TableAlias) .Append(" = ") - .Append(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true).ParameterName) + .Append(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true)) .Append(" "); return context; @@ -77,7 +77,7 @@ public override SearchParameterQueryGeneratorContext VisitMissingSearchParameter context.StringBuilder .Append(searchParamIdColumn, context.TableAlias) .Append(" = ") - .Append(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true).ParameterName) + .Append(context.Parameters.AddParameter(searchParamIdColumn, searchParamId, true)) .Append(" "); return context; @@ -187,7 +187,7 @@ protected static SearchParameterQueryGeneratorContext VisitSimpleBinary(BinaryOp throw new ArgumentOutOfRangeException(binaryOperator.ToString()); } - context.StringBuilder.Append(context.Parameters.AddParameter(column, value, includeInParameterHash).ParameterName); + context.StringBuilder.Append(context.Parameters.AddParameter(column, value, includeInParameterHash)); return context; } @@ -204,17 +204,14 @@ protected static SearchParameterQueryGeneratorContext VisitSimpleString(StringEx { case StringOperator.Contains: needsEscaping = TryEscapeValueForLike(ref value); - SqlParameter containsParameter = context.Parameters.AddParameter(column, $"%{value}%", true); - context.StringBuilder.Append(" LIKE ").Append(containsParameter.ParameterName); + context.StringBuilder.Append(" LIKE ").Append(context.Parameters.AddParameter(column, $"%{value}%", true)); break; case StringOperator.EndsWith: needsEscaping = TryEscapeValueForLike(ref value); - SqlParameter endWithParameter = context.Parameters.AddParameter(column, $"%{value}", true); - context.StringBuilder.Append(" LIKE ").Append(endWithParameter.ParameterName); + context.StringBuilder.Append(" LIKE ").Append(context.Parameters.AddParameter(column, $"%{value}", true)); break; case StringOperator.Equals: - SqlParameter equalsParameter = context.Parameters.AddParameter(column, value, true); - context.StringBuilder.Append(" = ").Append(equalsParameter.ParameterName); + context.StringBuilder.Append(" = ").Append(context.Parameters.AddParameter(column, value, true)); break; case StringOperator.NotContains: context.StringBuilder.Append(" NOT "); @@ -227,13 +224,11 @@ protected static SearchParameterQueryGeneratorContext VisitSimpleString(StringEx goto case StringOperator.StartsWith; case StringOperator.StartsWith: needsEscaping = TryEscapeValueForLike(ref value); - SqlParameter startsWithParameter = context.Parameters.AddParameter(column, $"{value}%", true); - context.StringBuilder.Append(" LIKE ").Append(startsWithParameter.ParameterName); + context.StringBuilder.Append(" LIKE ").Append(context.Parameters.AddParameter(column, $"{value}%", true)); break; case StringOperator.LeftSideStartsWith: needsEscaping = TryEscapeValueForLike(ref value); - SqlParameter leftParameter = context.Parameters.AddParameter(column, $"{value}", true); - context.StringBuilder.Append(leftParameter.ParameterName).Append(" LIKE "); + context.StringBuilder.Append(context.Parameters.AddParameter(column, $"{value}", true)).Append(" LIKE "); AppendColumnName(context, column, expression); context.StringBuilder.Append("+'%'"); break; @@ -259,8 +254,7 @@ protected static SearchParameterQueryGeneratorContext VisitSimpleString(StringEx context.StringBuilder.Append(" AND "); AppendColumnName(context, column, expression); - SqlParameter equalsParameter = context.Parameters.AddParameter(column, value, true); - context.StringBuilder.Append(" = ").Append(equalsParameter.ParameterName); + context.StringBuilder.Append(" = ").Append(context.Parameters.AddParameter(column, value, true)); } context.StringBuilder.Append(" COLLATE ").Append(expression.IgnoreCase ? DefaultCaseInsensitiveCollation : DefaultCaseSensitiveCollation); diff --git a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SqlQueryGenerator.cs b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SqlQueryGenerator.cs index f2b0e2bb5d..b612fea2e3 100644 --- a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SqlQueryGenerator.cs +++ b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/Expressions/Visitors/QueryGenerators/SqlQueryGenerator.cs @@ -337,6 +337,7 @@ private void AddHash() StringBuilder.Append("/* HASH "); Parameters.AppendHash(StringBuilder); + Parameters.AppendHashedParameterNames(StringBuilder); StringBuilder.AppendLine(" */"); } } diff --git a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/HashingSqlQueryParameterManager.cs b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/HashingSqlQueryParameterManager.cs index b09a8ce80d..46fbce6cc1 100644 --- a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/HashingSqlQueryParameterManager.cs +++ b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/HashingSqlQueryParameterManager.cs @@ -12,6 +12,7 @@ using System.Security.Cryptography; using EnsureThat; using Microsoft.Data.SqlClient; +using Microsoft.Health.Fhir.SqlServer.Features.Schema.Model; using Microsoft.Health.SqlServer; using Microsoft.Health.SqlServer.Features.Schema.Model; using Microsoft.Health.SqlServer.Features.Storage; @@ -35,7 +36,7 @@ public HashingSqlQueryParameterManager(SqlQueryParameterManager inner) public bool HasParametersToHash => _setToHash.Count > 0; /// - /// Add a parameter to the SQL command. + /// Add a parameter to the SQL command if it is not ResourceTypeId and not SearchParamId. /// /// The CLR column type /// The table column the parameter is bound to. @@ -44,14 +45,14 @@ public HashingSqlQueryParameterManager(SqlQueryParameterManager inner) /// Whether this parameter should be included in the hash of the overall parameters. /// If true, this parameter will prevent other identical queries with a different value for this parameter from re-using the query plan. /// - /// The SQL parameter. - public SqlParameter AddParameter(Column column, T value, bool includeInHash) + /// SQL parameter or input value depending on whether input was added to the list of parameters. + public object AddParameter(Column column, T value, bool includeInHash) { return AddParameter((Column)column, value, includeInHash); } /// - /// Add a parameter to the SQL command. + /// Add a parameter to the SQL command if it is not ResourceTypeId and not SearchParamId. /// /// The table column the parameter is bound to. /// The parameter value @@ -59,9 +60,16 @@ public SqlParameter AddParameter(Column column, T value, bool includeInHas /// Whether this parameter should be included in the hash of the overall parameters. /// If true, this parameter will prevent other identical queries with a different value for this parameter from re-using the query plan. /// - /// The SQL parameter. - public SqlParameter AddParameter(Column column, object value, bool includeInHash) + /// SQL parameter or input value depending on whether input was added to the list of parameters. + public object AddParameter(Column column, object value, bool includeInHash) { + if (column.Metadata.Name == VLatest.Resource.ResourceTypeId.Metadata.Name // logic uses "ResourceTypeId" string value. Resource table is chosen arbitrarily. + || column.Metadata.Name == VLatest.ReferenceSearchParam.ReferenceResourceTypeId.Metadata.Name + || column.Metadata.Name == VLatest.TokenSearchParam.SearchParamId.Metadata.Name) // logic uses "SearchParamId" string value. We don't have cross table column sharing concept yet, so to avoid hardcoding TokenSearchParam is arbitrarily chosen. + { + return value; + } + SqlParameter parameter = _inner.AddParameter(column, value); if (includeInHash) { @@ -182,6 +190,20 @@ public void AppendHash(IndentedStringBuilder stringBuilder) stringBuilder.Append(hashChars[..hashCharsLength]); } + /// + /// Appends comma delimited list of names of hashed parameters + /// + /// A string builder to append the list to. + public void AppendHashedParameterNames(IndentedStringBuilder stringBuilder) + { + var first = true; + foreach (var param in _setToHash) + { + stringBuilder.Append($"{(first ? " params=" : ",")}{param}"); + first = false; + } + } + private static void WriteAndAdvance(Span buffer, ref int currentIndex, ref IncrementalHash incrementalHash, T element) where T : struct { diff --git a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/SqlServerSearchService.cs b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/SqlServerSearchService.cs index 1d164fc31a..9535777703 100644 --- a/src/Microsoft.Health.Fhir.SqlServer/Features/Search/SqlServerSearchService.cs +++ b/src/Microsoft.Health.Fhir.SqlServer/Features/Search/SqlServerSearchService.cs @@ -882,6 +882,7 @@ private void LogSqlCommand(SqlCommand sqlCommand) sb.AppendLine(); } + sb.AppendLine("OPTION (RECOMPILE)"); // enables query compilation with provided parameter values in debugging sb.AppendLine($"-- execution timeout = {sqlCommandWrapper.CommandTimeout} sec."); _sqlRetryService.TryLogEvent("Search", "Start", sb.ToString(), null, CancellationToken.None); _logger.LogInformation("{SqlQuery}", sb.ToString()); diff --git a/test/Microsoft.Health.Fhir.Shared.Tests.Integration/Persistence/SqlCustomQueryTests.cs b/test/Microsoft.Health.Fhir.Shared.Tests.Integration/Persistence/SqlCustomQueryTests.cs index b87b0cb135..de22234693 100644 --- a/test/Microsoft.Health.Fhir.Shared.Tests.Integration/Persistence/SqlCustomQueryTests.cs +++ b/test/Microsoft.Health.Fhir.Shared.Tests.Integration/Persistence/SqlCustomQueryTests.cs @@ -129,15 +129,11 @@ private void AddSproc(string hash) { _fixture.SqlHelper.ExecuteSqlCmd(@$" CREATE OR ALTER PROCEDURE [dbo].[CustomQuery_{hash}] - @p0 smallint + @p0 datetime2 ,@p1 datetime2 - ,@p2 smallint - ,@p3 datetime2 - ,@p4 smallint - ,@p5 nvarchar(256) - ,@p6 smallint - ,@p7 nvarchar(256) - ,@p8 int + ,@p2 nvarchar(256) + ,@p3 nvarchar(256) + ,@p4 int AS set nocount on SELECT DISTINCT r.ResourceTypeId, r.ResourceId, r.Version, r.IsDeleted, r.ResourceSurrogateId, r.RequestMethod, CAST(1 AS bit) AS IsMatch, CAST(0 AS bit) AS IsPartial, r.IsRawResourceMetaSet, r.SearchParamHash, r.RawResource