Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Issue with combo index search parameter when using date #6566

Merged
merged 14 commits into from
Jan 16, 2025
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
import ca.uhn.fhir.util.DateUtils;
import jakarta.annotation.Nonnull;
import org.apache.commons.lang3.ObjectUtils;
import org.apache.commons.lang3.Validate;
import org.hl7.fhir.instance.model.api.IPrimitiveType;

Expand Down Expand Up @@ -182,11 +183,9 @@ public DateRangeParam(String theLowerBound, String theUpperBound) {
}

private void addParam(DateParam theParsed) throws InvalidRequestException {
if (theParsed.getPrefix() == null) {
theParsed.setPrefix(EQUAL);
}
ParamPrefixEnum prefix = getPrefixOrDefault(theParsed);

switch (theParsed.getPrefix()) {
switch (prefix) {
case NOT_EQUAL:
case EQUAL:
if (myLowerBound != null || myUpperBound != null) {
Expand Down Expand Up @@ -312,24 +311,24 @@ public Integer getLowerBoundAsDateInteger() {
}
int retVal = DateUtils.convertDateToDayInteger(myLowerBound.getValue());

if (myLowerBound.getPrefix() != null) {
switch (myLowerBound.getPrefix()) {
case GREATERTHAN:
case STARTS_AFTER:
retVal += 1;
break;
case EQUAL:
case GREATERTHAN_OR_EQUALS:
case NOT_EQUAL:
break;
case LESSTHAN:
case LESSTHAN_OR_EQUALS:
case APPROXIMATE:
case ENDS_BEFORE:
throw new IllegalStateException(
Msg.code(1926) + "Invalid lower bound comparator: " + myLowerBound.getPrefix());
}
ParamPrefixEnum prefix = getPrefixOrDefault(myLowerBound);

switch (prefix) {
case GREATERTHAN:
case STARTS_AFTER:
retVal += 1;
break;
case EQUAL:
case GREATERTHAN_OR_EQUALS:
case NOT_EQUAL:
break;
case LESSTHAN:
case LESSTHAN_OR_EQUALS:
case APPROXIMATE:
case ENDS_BEFORE:
throw new IllegalStateException(Msg.code(1926) + "Invalid lower bound comparator: " + prefix);
}

return retVal;
}

Expand All @@ -343,24 +342,25 @@ public Integer getUpperBoundAsDateInteger() {
return null;
}
int retVal = DateUtils.convertDateToDayInteger(myUpperBound.getValue());
if (myUpperBound.getPrefix() != null) {
switch (myUpperBound.getPrefix()) {
case LESSTHAN:
case ENDS_BEFORE:
retVal -= 1;
break;
case EQUAL:
case LESSTHAN_OR_EQUALS:
case NOT_EQUAL:
break;
case GREATERTHAN_OR_EQUALS:
case GREATERTHAN:
case APPROXIMATE:
case STARTS_AFTER:
throw new IllegalStateException(
Msg.code(1927) + "Invalid upper bound comparator: " + myUpperBound.getPrefix());
}

ParamPrefixEnum prefix = getPrefixOrDefault(myUpperBound);

switch (prefix) {
case LESSTHAN:
case ENDS_BEFORE:
retVal -= 1;
break;
case EQUAL:
case LESSTHAN_OR_EQUALS:
case NOT_EQUAL:
break;
case GREATERTHAN_OR_EQUALS:
case GREATERTHAN:
case APPROXIMATE:
case STARTS_AFTER:
throw new IllegalStateException(Msg.code(1927) + "Invalid upper bound comparator: " + prefix);
}

return retVal;
}

Expand All @@ -374,28 +374,29 @@ public Date getLowerBoundAsInstant() {
@Nonnull
private static Date getLowerBoundAsInstant(@Nonnull DateParam theLowerBound) {
Date retVal = theLowerBound.getValue();

if (theLowerBound.getPrecision().ordinal() <= TemporalPrecisionEnum.DAY.ordinal()) {
retVal = DateUtils.getLowestInstantFromDate(retVal);
}

if (theLowerBound.getPrefix() != null) {
switch (theLowerBound.getPrefix()) {
case GREATERTHAN:
case STARTS_AFTER:
retVal = theLowerBound.getPrecision().add(retVal, 1);
break;
case EQUAL:
case NOT_EQUAL:
case GREATERTHAN_OR_EQUALS:
break;
case LESSTHAN_OR_EQUALS:
case LESSTHAN:
case APPROXIMATE:
case ENDS_BEFORE:
throw new IllegalStateException(
Msg.code(1928) + "Invalid lower bound comparator: " + theLowerBound.getPrefix());
}
ParamPrefixEnum prefix = getPrefixOrDefault(theLowerBound);

switch (prefix) {
case GREATERTHAN:
case STARTS_AFTER:
retVal = theLowerBound.getPrecision().add(retVal, 1);
break;
case EQUAL:
case NOT_EQUAL:
case GREATERTHAN_OR_EQUALS:
break;
case LESSTHAN_OR_EQUALS:
case LESSTHAN:
case APPROXIMATE:
case ENDS_BEFORE:
throw new IllegalStateException(Msg.code(1928) + "Invalid lower bound comparator: " + prefix);
}

return retVal;
}

Expand Down Expand Up @@ -446,26 +447,26 @@ private static Date getUpperBoundAsInstant(@Nonnull DateParam theUpperBound) {
retVal = DateUtils.getHighestInstantFromDate(retVal);
}

if (theUpperBound.getPrefix() != null) {
switch (theUpperBound.getPrefix()) {
case LESSTHAN:
case ENDS_BEFORE:
retVal = new Date(retVal.getTime() - 1L);
break;
case EQUAL:
case NOT_EQUAL:
case LESSTHAN_OR_EQUALS:
retVal = theUpperBound.getPrecision().add(retVal, 1);
retVal = new Date(retVal.getTime() - 1L);
break;
case GREATERTHAN_OR_EQUALS:
case GREATERTHAN:
case APPROXIMATE:
case STARTS_AFTER:
throw new IllegalStateException(
Msg.code(1929) + "Invalid upper bound comparator: " + theUpperBound.getPrefix());
}
ParamPrefixEnum prefix = getPrefixOrDefault(theUpperBound);

switch (prefix) {
case LESSTHAN:
case ENDS_BEFORE:
retVal = new Date(retVal.getTime() - 1L);
break;
case EQUAL:
case NOT_EQUAL:
case LESSTHAN_OR_EQUALS:
retVal = theUpperBound.getPrecision().add(retVal, 1);
retVal = new Date(retVal.getTime() - 1L);
break;
case GREATERTHAN_OR_EQUALS:
case GREATERTHAN:
case APPROXIMATE:
case STARTS_AFTER:
throw new IllegalStateException(Msg.code(1929) + "Invalid upper bound comparator: " + prefix);
}

return retVal;
}

Expand Down Expand Up @@ -586,6 +587,16 @@ public void setValuesAsQueryTokens(
FhirContext theContext, String theParamName, List<QualifiedParamList> theParameters)
throws InvalidRequestException {

// When we create and populate a DateRangeParam from a query parameter (?birthdate=2024-12-02 or
// ?birthdate=eq2024-12-02), we
// set the prefix only if it was specifically provided by the client as it is mandatory to retain the capability
// to make the differentiation. See {@link SearchBuilder#validateParamValuesAreValidForComboParam}.
//
// Since the FHIR specification says that "If no prefix is present, the prefix <code>eq</code> is assumed",
// we will do so by invoking method {@link DateRangeParam#getPrefixOrDefault} everytime computation is
// conditional on the
// prefix value.

boolean haveHadUnqualifiedParameter = false;
for (QualifiedParamList paramList : theParameters) {
if (paramList.size() == 0) {
Expand Down Expand Up @@ -701,4 +712,13 @@ private void validateAndSet(DateParam lowerBound, DateParam upperBound) {
myLowerBound = lowerBound;
myUpperBound = upperBound;
}

/**
*
* This method should be used when performing computation conditional on the prefix value to ensure that a dateParam
* without prefix is treated as if it has one set to 'eq'.
*/
private static ParamPrefixEnum getPrefixOrDefault(DateParam theDateParam) {
return ObjectUtils.defaultIfNull(theDateParam.getPrefix(), EQUAL);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
type: fix
issue: 6603
jira: SMILE-8048
title: "Previously, searches where processing was optimised with a combo index search parameters would skip searching
the optimised indexes if a date query string was prefixed with 'eq'. Since date=2025-01-01 and date=eq2025-01-01 are
equivalent search queries, we have harmonized the search behavior allowing optimised indexes inspection when the prefix
is present. This issue is fixed."
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,7 @@
import ca.uhn.fhir.rest.param.BaseParamWithPrefix;
import ca.uhn.fhir.rest.param.DateParam;
import ca.uhn.fhir.rest.param.DateRangeParam;
import ca.uhn.fhir.rest.param.ParamPrefixEnum;
import ca.uhn.fhir.rest.param.ParameterUtil;
import ca.uhn.fhir.rest.param.ReferenceParam;
import ca.uhn.fhir.rest.param.StringParam;
Expand Down Expand Up @@ -152,11 +153,13 @@
import static ca.uhn.fhir.jpa.search.builder.QueryStack.LOCATION_POSITION;
import static ca.uhn.fhir.jpa.search.builder.QueryStack.SearchForIdsParams.with;
import static ca.uhn.fhir.jpa.util.InClauseNormalizer.normalizeIdListForInClause;
import static ca.uhn.fhir.rest.param.ParamPrefixEnum.EQUAL;
import static java.util.Objects.requireNonNull;
import static org.apache.commons.collections4.CollectionUtils.isNotEmpty;
import static org.apache.commons.lang3.StringUtils.defaultString;
import static org.apache.commons.lang3.StringUtils.isBlank;
import static org.apache.commons.lang3.StringUtils.isNotBlank;
import static org.apache.commons.lang3.StringUtils.stripStart;

/**
* The SearchBuilder is responsible for actually forming the SQL query that handles
Expand Down Expand Up @@ -2354,7 +2357,9 @@ private void applyComboSearchParam(

String nextParamName = theComboParamNames.get(paramIndex);
IQueryParameterType nextOr = nextPermutation.get(paramIndex);
String nextOrValue = nextOr.getValueAsQueryToken(myContext);
// The only prefix accepted when combo searching is 'eq' (see validateParamValuesAreValidForComboParam).
// As a result, we strip the prefix if present.
String nextOrValue = stripStart(nextOr.getValueAsQueryToken(myContext), EQUAL.getValue());

RuntimeSearchParam nextParamDef = mySearchParamRegistry.getActiveSearchParam(
myResourceName, nextParamName, ISearchParamRegistry.SearchParamLookupContextEnum.SEARCH);
Expand Down Expand Up @@ -2453,7 +2458,10 @@ private boolean validateParamValuesAreValidForComboParam(
}
if (nextOrValue instanceof BaseParamWithPrefix) {
BaseParamWithPrefix<?> paramWithPrefix = (BaseParamWithPrefix<?>) nextOrValue;
if (paramWithPrefix.getPrefix() != null) {
ParamPrefixEnum prefix = paramWithPrefix.getPrefix();
// A parameter with the 'eq' prefix is the only accepted prefix when combo searching since
// birthdate=2025-01-01 and birthdate=eq2025-01-01 are equivalent searches.
if (prefix != null && prefix != EQUAL) {
String message = "Search with params " + theComboParamNames
+ " is not a candidate for combo searching - Parameter '" + nextParamName
+ "' has prefix: '"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -826,10 +826,7 @@ private boolean isNotEqualsComparator(DateRangeParam theDateRange) {
DateParam lb = theDateRange.getLowerBound();
DateParam ub = theDateRange.getUpperBound();

return lb != null
&& ub != null
&& lb.getPrefix().equals(NOT_EQUAL)
&& ub.getPrefix().equals(NOT_EQUAL);
return lb != null && ub != null && NOT_EQUAL.equals(lb.getPrefix()) && NOT_EQUAL.equals(ub.getPrefix());
}
return false;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -567,8 +567,8 @@ public String toNormalizedQueryString(FhirContext theCtx) {
private boolean isNotEqualsComparator(DateParam theLowerBound, DateParam theUpperBound) {
return theLowerBound != null
&& theUpperBound != null
&& theLowerBound.getPrefix().equals(NOT_EQUAL)
&& theUpperBound.getPrefix().equals(NOT_EQUAL);
&& NOT_EQUAL.equals(theLowerBound.getPrefix())
&& NOT_EQUAL.equals(theUpperBound.getPrefix());
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
import ca.uhn.fhir.jpa.search.DatabaseBackedPagingProvider;
import ca.uhn.fhir.jpa.search.reindex.ResourceReindexingSvcImpl;
import ca.uhn.fhir.jpa.test.BaseJpaR4Test;
import ca.uhn.fhir.jpa.test.util.ComboSearchParameterTestHelper;
import ca.uhn.fhir.jpa.util.SpringObjectCaster;
import ca.uhn.fhir.rest.server.util.ISearchParamRegistry;
import ca.uhn.fhir.test.utilities.MockInvoker;
Expand All @@ -30,6 +31,7 @@ public abstract class BaseComboParamsR4Test extends BaseJpaR4Test {
protected ISearchParamRegistry mySearchParamRegistry;
protected List<String> myMessages = new ArrayList<>();
private IInterceptorBroadcaster myInterceptorBroadcaster;
protected ComboSearchParameterTestHelper myComboSearchParameterTestHelper;

@Override
@BeforeEach
Expand Down Expand Up @@ -62,6 +64,7 @@ public void before() throws Exception {

// allow searches to use cached results
when(myInterceptorBroadcaster.getInvokersForPointcut(eq(Pointcut.STORAGE_PRECHECK_FOR_CACHED_SEARCH))).thenReturn(MockInvoker.list(params->true));
myComboSearchParameterTestHelper = new ComboSearchParameterTestHelper(mySearchParameterDao, mySearchParamRegistry);
}

@AfterEach
Expand All @@ -80,5 +83,9 @@ protected void logCapturedMessages() {
ourLog.info("Messages:\n {}", String.join("\n ", myMessages));
}

protected void createBirthdateAndGenderSps(boolean theUnique) {
myComboSearchParameterTestHelper.createBirthdateAndGenderSps(theUnique);
myMessages.clear();
}

}
Loading
Loading