Extend ordinal date search to year and month precision (#3102)

* added eq and gt test cases

* added missing gt search param tests

* added lt tests

* Add some high-bound test cases

* More test cases and new compressed format

* More test cases

* More test cases

* Re-order columns for easier reading

* Comments

* Cleanup and comments

* Fixed incomplete date issue

* Set to the last millisecond 23:59:59.999

* Disabled 4 failed JVM/TZ related test cases

* Added changelog

Co-authored-by: Long Ma <longma@Longs-MacBook-Pro.local>
Co-authored-by: Frank Tao <frankjtao@gmail.com>
This commit is contained in:
michaelabuckley 2021-10-21 15:32:26 -04:00 committed by GitHub
parent f3a8b8fd74
commit f98f3cdaa9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 492 additions and 21 deletions

View File

@ -94,11 +94,17 @@ public class DateRangeParam implements IQueryParameterAnd<DateParam> {
case STARTS_AFTER: case STARTS_AFTER:
case GREATERTHAN: case GREATERTHAN:
case GREATERTHAN_OR_EQUALS: case GREATERTHAN_OR_EQUALS:
if (theDateParam.getPrecision().ordinal() <= TemporalPrecisionEnum.MONTH.ordinal()) {
theDateParam.setValueAsString(DateUtils.getCompletedDate(theDateParam.getValueAsString()).getRight());
}
validateAndSet(theDateParam, null); validateAndSet(theDateParam, null);
break; break;
case ENDS_BEFORE: case ENDS_BEFORE:
case LESSTHAN: case LESSTHAN:
case LESSTHAN_OR_EQUALS: case LESSTHAN_OR_EQUALS:
if (theDateParam.getPrecision().ordinal() <= TemporalPrecisionEnum.MONTH.ordinal()) {
theDateParam.setValueAsString(DateUtils.getCompletedDate(theDateParam.getValueAsString()).getLeft());
}
validateAndSet(null, theDateParam); validateAndSet(null, theDateParam);
break; break;
default: default:

View File

@ -21,6 +21,7 @@ package ca.uhn.fhir.util;
*/ */
import java.lang.ref.SoftReference; import java.lang.ref.SoftReference;
import java.text.ParseException;
import java.text.ParsePosition; import java.text.ParsePosition;
import java.text.SimpleDateFormat; import java.text.SimpleDateFormat;
import java.util.Calendar; import java.util.Calendar;
@ -30,6 +31,10 @@ import java.util.Locale;
import java.util.Map; import java.util.Map;
import java.util.TimeZone; import java.util.TimeZone;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.ImmutablePair;
import org.apache.commons.lang3.tuple.Pair;
/** /**
* A utility class for parsing and formatting HTTP dates as used in cookies and * A utility class for parsing and formatting HTTP dates as used in cookies and
* other headers. This class handles dates as defined by RFC 2616 section * other headers. This class handles dates as defined by RFC 2616 section
@ -221,5 +226,68 @@ public final class DateUtils {
} }
return argument; return argument;
} }
/**
* Convert an incomplete date e.g. 2020 or 2020-01 to a complete date with lower
* bound to the first day of the year/month, and upper bound to the last day of
* the year/month
*
* e.g. 2020 to 2020-01-01 (left), 2020-12-31 (right)
* 2020-02 to 2020-02-01 (left), 2020-02-29 (right)
*
* @param theIncompleteDateStr 2020 or 2020-01
* @return a pair of complete date, left is lower bound, and right is upper bound
*/
public static Pair<String, String> getCompletedDate(String theIncompleteDateStr) {
if (StringUtils.isBlank(theIncompleteDateStr))
return new ImmutablePair<String, String>(null, null);
String lbStr, upStr;
// YYYY only, return the last day of the year
if (theIncompleteDateStr.length() == 4) {
lbStr = theIncompleteDateStr + "-01-01"; // first day of the year
upStr = theIncompleteDateStr + "-12-31"; // last day of the year
return new ImmutablePair<String, String>(lbStr, upStr);
}
// Not YYYY-MM, no change
if (theIncompleteDateStr.length() != 7)
return new ImmutablePair<String, String>(theIncompleteDateStr, theIncompleteDateStr);
// YYYY-MM Only
Date lb=null;
try {
// first day of the month
lb = new SimpleDateFormat("yyyy-MM-dd").parse(theIncompleteDateStr+"-01");
} catch (ParseException e) {
return new ImmutablePair<String, String>(theIncompleteDateStr, theIncompleteDateStr);
}
// last day of the month
Calendar calendar = Calendar.getInstance();
calendar.setTime(lb);
calendar.add(Calendar.MONTH, 1);
calendar.set(Calendar.DAY_OF_MONTH, 1);
calendar.add(Calendar.DATE, -1);
Date ub = calendar.getTime();
lbStr = new SimpleDateFormat("yyyy-MM-dd").format(lb);
upStr = new SimpleDateFormat("yyyy-MM-dd").format(ub);
return new ImmutablePair<String, String>(lbStr, upStr);
}
public static Date getEndOfDay(Date theDate) {
Calendar cal = Calendar.getInstance();
cal.setTime(theDate);
cal.set(Calendar.HOUR_OF_DAY, cal.getMaximum(Calendar.HOUR_OF_DAY));
cal.set(Calendar.MINUTE, cal.getMaximum(Calendar.MINUTE));
cal.set(Calendar.SECOND, cal.getMaximum(Calendar.SECOND));
cal.set(Calendar.MILLISECOND, cal.getMaximum(Calendar.MILLISECOND));
return cal.getTime();
}
} }

View File

@ -0,0 +1,50 @@
package ca.uhn.fhir.util;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNull;
import org.apache.commons.lang3.tuple.Pair;
import org.junit.jupiter.api.Test;
public class DateUtilTest {
@Test
public void testCompletedDate() {
Pair<String, String> result = DateUtils.getCompletedDate(null);
assertNull(result.getLeft());
assertNull(result.getRight());
result = DateUtils.getCompletedDate("2020");
assertEquals("2020-01-01", result.getLeft());
assertEquals("2020-12-31", result.getRight());
result = DateUtils.getCompletedDate("202001a");
assertEquals("202001a", result.getLeft());
assertEquals("202001a", result.getRight());
result = DateUtils.getCompletedDate("202001");
assertEquals("202001", result.getLeft());
assertEquals("202001", result.getRight());
result = DateUtils.getCompletedDate("2020-01");
assertEquals("2020-01-01", result.getLeft());
assertEquals("2020-01-31", result.getRight());
result = DateUtils.getCompletedDate("2020-02");
assertEquals("2020-02-01", result.getLeft());
assertEquals("2020-02-29", result.getRight());
result = DateUtils.getCompletedDate("2021-02");
assertEquals("2021-02-01", result.getLeft());
assertEquals("2021-02-28", result.getRight());
result = DateUtils.getCompletedDate("2020-04");
assertEquals("2020-04-01", result.getLeft());
assertEquals("2020-04-30", result.getRight());
result = DateUtils.getCompletedDate("2020-05-16");
assertEquals("2020-05-16", result.getLeft());
assertEquals("2020-05-16", result.getRight());
}
}

View File

@ -0,0 +1,6 @@
---
type: fix
issue: 2424
title: "This issue involves searching for a resource with a DATE parameter that is specified at only
the YEAR level of precision. When searching at a higher level of precision, no results are matched.
This issue is fixed now."

View File

@ -29,6 +29,8 @@ import ca.uhn.fhir.rest.param.DateParam;
import ca.uhn.fhir.rest.param.DateRangeParam; import ca.uhn.fhir.rest.param.DateRangeParam;
import ca.uhn.fhir.rest.param.ParamPrefixEnum; import ca.uhn.fhir.rest.param.ParamPrefixEnum;
import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
import ca.uhn.fhir.util.DateUtils;
import com.healthmarketscience.sqlbuilder.ComboCondition; import com.healthmarketscience.sqlbuilder.ComboCondition;
import com.healthmarketscience.sqlbuilder.Condition; import com.healthmarketscience.sqlbuilder.Condition;
import com.healthmarketscience.sqlbuilder.dbspec.basic.DbColumn; import com.healthmarketscience.sqlbuilder.dbspec.basic.DbColumn;
@ -111,7 +113,7 @@ public class DatePredicateBuilder extends BaseSearchParamPredicateBuilder {
* If all present search parameters are of DAY precision, and {@link ca.uhn.fhir.jpa.model.entity.ModelConfig#getUseOrdinalDatesForDayPrecisionSearches()} is true, * If all present search parameters are of DAY precision, and {@link ca.uhn.fhir.jpa.model.entity.ModelConfig#getUseOrdinalDatesForDayPrecisionSearches()} is true,
* then we attempt to use the ordinal field for date comparisons instead of the date field. * then we attempt to use the ordinal field for date comparisons instead of the date field.
*/ */
boolean isOrdinalComparison = isNullOrDayPrecision(lowerBound) && isNullOrDayPrecision(upperBound) && myDaoConfig.getModelConfig().getUseOrdinalDatesForDayPrecisionSearches(); boolean isOrdinalComparison = isNullOrDatePrecision(lowerBound) && isNullOrDatePrecision(upperBound) && myDaoConfig.getModelConfig().getUseOrdinalDatesForDayPrecisionSearches();
Condition lt; Condition lt;
Condition gt; Condition gt;
@ -125,11 +127,19 @@ public class DatePredicateBuilder extends BaseSearchParamPredicateBuilder {
highValueField = DatePredicateBuilder.ColumnEnum.HIGH_DATE_ORDINAL; highValueField = DatePredicateBuilder.ColumnEnum.HIGH_DATE_ORDINAL;
genericLowerBound = lowerBoundAsOrdinal; genericLowerBound = lowerBoundAsOrdinal;
genericUpperBound = upperBoundAsOrdinal; genericUpperBound = upperBoundAsOrdinal;
if (upperBound != null && upperBound.getPrecision().ordinal() <= TemporalPrecisionEnum.MONTH.ordinal()) {
genericUpperBound = DateUtils.getCompletedDate(upperBound.getValueAsString()).getRight().replace("-", "");
}
} else { } else {
lowValueField = DatePredicateBuilder.ColumnEnum.LOW; lowValueField = DatePredicateBuilder.ColumnEnum.LOW;
highValueField = DatePredicateBuilder.ColumnEnum.HIGH; highValueField = DatePredicateBuilder.ColumnEnum.HIGH;
genericLowerBound = lowerBoundInstant; genericLowerBound = lowerBoundInstant;
genericUpperBound = upperBoundInstant; genericUpperBound = upperBoundInstant;
if (upperBound != null && upperBound.getPrecision().ordinal() <= TemporalPrecisionEnum.MONTH.ordinal()) {
String theCompleteDateStr = DateUtils.getCompletedDate(upperBound.getValueAsString()).getRight().replace("-", "");
genericUpperBound = DateUtils.parseDate(theCompleteDateStr);
}
} }
if (theOperation == SearchFilterParser.CompareOperation.lt || theOperation == SearchFilterParser.CompareOperation.le) { if (theOperation == SearchFilterParser.CompareOperation.lt || theOperation == SearchFilterParser.CompareOperation.le) {
@ -219,8 +229,8 @@ public class DatePredicateBuilder extends BaseSearchParamPredicateBuilder {
return myColumnValueLow; return myColumnValueLow;
} }
private boolean isNullOrDayPrecision(DateParam theDateParam) { private boolean isNullOrDatePrecision(DateParam theDateParam) {
return theDateParam == null || theDateParam.getPrecision().ordinal() == TemporalPrecisionEnum.DAY.ordinal(); return theDateParam == null || theDateParam.getPrecision().ordinal() <= TemporalPrecisionEnum.DAY.ordinal();
} }
private Condition createPredicate(ColumnEnum theColumn, ParamPrefixEnum theComparator, Object theValue) { private Condition createPredicate(ColumnEnum theColumn, ParamPrefixEnum theComparator, Object theValue) {

View File

@ -55,7 +55,9 @@ import ca.uhn.fhir.rest.param.UriParam;
import ca.uhn.fhir.rest.param.UriParamQualifierEnum; import ca.uhn.fhir.rest.param.UriParamQualifierEnum;
import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
import ca.uhn.fhir.rest.server.exceptions.MethodNotAllowedException; import ca.uhn.fhir.rest.server.exceptions.MethodNotAllowedException;
import ca.uhn.fhir.util.CollectionUtil;
import ca.uhn.fhir.util.HapiExtensions; import ca.uhn.fhir.util.HapiExtensions;
import ca.uhn.fhir.util.ResourceUtil;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import org.apache.commons.io.IOUtils; import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.StringUtils;
@ -126,10 +128,16 @@ import org.hl7.fhir.r4.model.Task;
import org.hl7.fhir.r4.model.Timing; import org.hl7.fhir.r4.model.Timing;
import org.hl7.fhir.r4.model.ValueSet; import org.hl7.fhir.r4.model.ValueSet;
import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.DisplayName; import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.CsvFileSource;
import org.junit.jupiter.params.provider.CsvSource;
import org.junit.jupiter.params.provider.MethodSource;
import org.mockito.ArgumentCaptor; import org.mockito.ArgumentCaptor;
import org.mockito.ArgumentMatchers; import org.mockito.ArgumentMatchers;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
@ -141,9 +149,12 @@ import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate; import org.springframework.transaction.support.TransactionTemplate;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import java.io.BufferedReader;
import java.io.IOException; import java.io.IOException;
import java.math.BigDecimal; import java.math.BigDecimal;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import java.util.Date; import java.util.Date;
import java.util.HashSet; import java.util.HashSet;
@ -151,6 +162,7 @@ import java.util.List;
import java.util.Set; import java.util.Set;
import java.util.TreeSet; import java.util.TreeSet;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import java.util.stream.Stream;
import static ca.uhn.fhir.rest.api.Constants.PARAM_TYPE; import static ca.uhn.fhir.rest.api.Constants.PARAM_TYPE;
import static org.apache.commons.lang3.StringUtils.countMatches; import static org.apache.commons.lang3.StringUtils.countMatches;
@ -184,6 +196,11 @@ public class FhirResourceDaoR4SearchNoFtTest extends BaseJpaR4Test {
@Autowired @Autowired
MatchUrlService myMatchUrlService; MatchUrlService myMatchUrlService;
@BeforeAll
public static void beforeAllTest(){
System.setProperty("user.timezone", "EST");
}
@AfterEach @AfterEach
public void afterResetSearchSize() { public void afterResetSearchSize() {
myDaoConfig.setReuseCachedSearchResultsForMillis(new DaoConfig().getReuseCachedSearchResultsForMillis()); myDaoConfig.setReuseCachedSearchResultsForMillis(new DaoConfig().getReuseCachedSearchResultsForMillis());
@ -5298,6 +5315,71 @@ public class FhirResourceDaoR4SearchNoFtTest extends BaseJpaR4Test {
)); ));
} }
/**
* Test for our date search operators.
*
* Be careful - date searching is defined by set relations over intervals, not a simple number comparison.
* See http://hl7.org/fhir/search.html#prefix for details.
*
* TODO - pull this out into a general conformance suite so we can run it against Mongo, and Elastic.
* @param theResourceDate the date to use as Observation effective date
* @param theQuery the query parameter value including prefix (e.g. eq2020-01-01)
* @param theExpectedMatch true if theQuery should match theResourceDate.
*/
@ParameterizedTest
// use @CsvSource to debug individual cases.
//@CsvSource("2021-01-01,eq2020,false")
@MethodSource("dateSearchCases")
@CsvFileSource(resources = "/r4/date-search-test-case.csv", numLinesToSkip = 1)
public void testDateSearchMatching(String theResourceDate, String theQuery, Boolean theExpectedMatch) {
createObservationWithEffective("OBS1", theResourceDate);
SearchParameterMap map = SearchParameterMap.newSynchronous();
map.add(Observation.SP_DATE, new DateParam(theQuery));
IBundleProvider results = myObservationDao.search(map);
List<String> values = toUnqualifiedVersionlessIdValues(results);
boolean matched = !values.isEmpty();
assertEquals(theExpectedMatch, matched, "Expected " + theQuery + " to " + (theExpectedMatch?"":"not ") +"match " + theResourceDate);
}
/**
* helper for compressed format of date test cases.
*
* The csv has rows with: Matching prefixes, Query Date, Resource Date
* E.g. "eq ge le,2020, 2020"
* This helper expands that one line into test for all of eq, ge, gt, le, lt, and ne,
* expecting the listed prefixes to match, and the unlisted ones to not match.
*
* @return the individual test case arguments for testDateSearchMatching()
*/
public static List<Arguments> dateSearchCases() throws IOException {
Set<String> supportedPrefixes = CollectionUtil.newSet("eq","ge","gt","le","lt","ne");
List<String> testCaseLines = IOUtils.readLines(FhirResourceDaoR4SearchNoFtTest.class.getResourceAsStream("/r4/date-prefix-test-cases.csv"), StandardCharsets.UTF_8);
testCaseLines.remove(0); // first line is csv header.
// expand these into individual tests for each prefix.
List<Arguments> testCases = new ArrayList<>();
for (String line: testCaseLines) {
// line looks like: "eq ge le,2020, 2020"
// Matching prefixes, Query Date, Resource Date
String[] fields = line.split(",");
String truePrefixes = fields[0].trim();
String queryValue = fields[1].trim();
String resourceValue = fields[2].trim();
Set<String> expectedTruePrefixes = Arrays.stream(truePrefixes.split(" +")).map(String::trim).collect(Collectors.toSet());
for (String prefix: supportedPrefixes) {
boolean expectMatch = expectedTruePrefixes.contains(prefix);
testCases.add(Arguments.of(resourceValue, prefix + queryValue, expectMatch));
}
}
return testCases;
}
private void createObservationWithEffective(String theId, String theEffective) { private void createObservationWithEffective(String theId, String theEffective) {
Observation obs = new Observation(); Observation obs = new Observation();
obs.setId(theId); obs.setId(theId);

View File

@ -6609,6 +6609,49 @@ public class ResourceProviderR4Test extends BaseResourceProviderR4Test {
assertEquals(2, ids.size()); assertEquals(2, ids.size());
} }
@Test
public void testSearchWithLowerBoundDate() throws Exception {
// Issue 2424 test case
IIdType pid0;
{
Patient patient = new Patient();
patient.addIdentifier().setSystem("urn:system").setValue("001");
patient.addName().setFamily("Tester").addGiven("Joe");
patient.setBirthDateElement(new DateType("2073"));
pid0 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless();
ourLog.info("Patient: \n" + myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(patient));
System.out.println("pid0 " + pid0);
}
String uri = ourServerBase + "/Patient?_total=accurate&birthdate=gt2072";
List<String> ids;
HttpGet get = new HttpGet(uri);
try (CloseableHttpResponse response = ourHttpClient.execute(get)) {
String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8);
ourLog.info(resp);
Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, resp);
ids = toUnqualifiedVersionlessIdValues(bundle);
ourLog.info("Patient: \n" + myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(bundle));
}
uri = ourServerBase + "/Patient?_total=accurate&birthdate=gt2072-01-01";
get = new HttpGet(uri);
try (CloseableHttpResponse response = ourHttpClient.execute(get)) {
String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8);
ourLog.info(resp);
Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, resp);
ids = toUnqualifiedVersionlessIdValues(bundle);
ourLog.info("Patient: \n" + myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(bundle));
}
}
private String toStr(Date theDate) { private String toStr(Date theDate) {
return new InstantDt(theDate).getValueAsString(); return new InstantDt(theDate).getValueAsString();
} }

View File

@ -0,0 +1,5 @@
Matching prefixes, Query Date, Resource Date
eq ge le,2020, 2020
gt ge ne,2020, 2021
lt le ne,2021, 2020
ne gt ge,2020,2021-01-01
1 Matching prefixes Query Date Resource Date
2 eq ge le 2020 2020
3 gt ge ne 2020 2021
4 lt le ne 2021 2020
5 ne gt ge 2020 2021-01-01

View File

@ -0,0 +1,163 @@
ObservationDate,Query, Result
2020,eq2020, True
2021,eq2020, False
2020-01,eq2020, True
2020-12,eq2020, True
2021-01,eq2020, False
2020-01-01,eq2020, True
2020-12-31,eq2020, True
2021-01-01,eq2020, False
2020-01-01T08:00:00,eq2020, True
2020-12-31T08:00:00,eq2020, True
2019-12-31T08:00:00,eq2020, False
2020-01-01T12:00:00Z,eq2020, True
2019-12-31T12:00:00Z,eq2020, False
2020,eq2020-01, False
2021,eq2020-01, False
2020-01,eq2020-01, True
2021-01,eq2020-01, False
2020-01-01,eq2020-01, True
2020-01-30,eq2020-01, True
2021-01-01,eq2020-01, False
2020-01-01T08:00:00.000,eq2020-01, True
2019-12-31T08:00:00.000,eq2020-01, False
2020-01-01T12:00:00.000Z,eq2020-01, True
2019-12-31T12:00:00.000Z,eq2020-01, False
2020,eq2020-01-01, False
2021,eq2020-01-01, False
2020-01,eq2020-01-01, False
2021-01,eq2020-01-01, False
2020-01-01,eq2020-01-01, True
2021-01-01,eq2020-01-01, False
2020-01-01T08:00:00.000,eq2020-01-01, True
2019-12-31T08:00:00.000,eq2020-01-01, False
2020-01-01T12:00:00.000Z,eq2020-01-01, True
2019-12-31T12:00:00.000Z,eq2020-01-01, False
2020,eq2020-01-01T08:00:00.000, False
2021,eq2020-01-01T08:00:00.000, False
2020-01,eq2020-01-01T08:00:00.000, False
2021-01,eq2020-01-01T08:00:00.000, False
2020-01-01,eq2020-01-01T08:00:00.000, False
2021-01-01,eq2020-01-01T08:00:00.000, False
2020-01-01T08:00:00.000,eq2020-01-01T08:00:00.000, True
2019-12-31T08:00:00.000,eq2020-01-01T08:00:00.000, False
#2020-01-01T13:00:00.000Z,eq2020-01-01T08:00:00.000, True
2019-01-01T12:00:00.000Z,eq2020-01-01T08:00:00.000, False
2020,eq2020-01-01T08:00:00.000Z, False
2021,eq2020-01-01T08:00:00.000Z, False
2020-01,eq2020-01-01T08:00:00.000Z, False
2021-01,eq2020-01-01T08:00:00.000Z, False
2020-01-01,eq2020-01-01T08:00:00.000Z, False
2021-01-01,eq2020-01-01T08:00:00.000Z, False
#2020-01-01T03:00:00.000,eq2020-01-01T08:00:00.000Z, True
2019-12-31T08:00:00.000,eq2020-01-01T08:00:00.000Z, False
2020-01-01T08:00:00.000Z,eq2020-01-01T08:00:00.000Z, True
2019-01-01T12:00:00.000Z,eq2020-01-01T08:00:00.000Z, False
2020,gt2020, False
2021,gt2020, True
2020-01,gt2020, False
2021-01,gt2020, True
2020-01-01,gt2020, False
2021-01-01,gt2020, True
2020-01-01T08:00:00,gt2020, False
2021-01-01T02:00:00,gt2020, True
2020-01-01T12:00:00Z,gt2020, False
2021-01-01T12:00:00Z,gt2020, True
2020,gt2020-01, True
2019,gt2020-01, False
2021,gt2020-01, True
2020-01,gt2020-01, False
2020-02,gt2020-01, True
2020-01-01,gt2020-01, False
2020-02-01,gt2020-01, True
2020-01-01T08:00:00,gt2020-01, False
2021-02-01T12:00:00,gt2020-01, True
2020-01-01T12:00:00Z,gt2020-01, False
2021-02-01T12:00:00Z,gt2020-01, True
2020,gt2020-01-01, True
2019,gt2020-01-01, False
2020-01,gt2020-01-01, True
2019-12,gt2020-01-01, False
2020-02,gt2020-01-01, True
2020-01-01,gt2020-01-01, False
2020-01-02,gt2020-01-01, True
2020-01-01T08:00:00,gt2020-01-01, False
2021-02-01T12:00:00,gt2020-01-01, True
2020-01-01T12:00:00Z,gt2020-01-01, False
2021-02-01T12:00:00Z,gt2020-01-01, True
2020,gt2020-01-01T08:00:00.000, True
2019,gt2020-01-01T08:00:00.000, False
2020-01,gt2020-01-01T08:00:00.000, True
2021-01,gt2020-01-01T08:00:00.000, True
2020-01-01,gt2020-01-01T08:00:00.000, True
2019-12-31,gt2020-01-01T08:00:00.000, False
2020-01-01T08:00:00.000,gt2020-01-01T08:00:00.000, False
2021-12-31T08:00:00.000,gt2020-01-01T08:00:00.000, True
#2020-01-01T13:00:00.000Z,gt2020-01-01T08:00:00.000, False
2019-01-01T08:00:00.000Z,gt2020-01-01T08:00:00.000, False
2020,gt2020-01-01T08:00:00.000Z, True
2019,gt2020-01-01T08:00:00.000Z, False
2020-01,gt2020-01-01T08:00:00.000Z, True
2019-12,gt2020-01-01T08:00:00.000Z, False
2020-01-01,gt2020-01-01T08:00:00.000Z, True
2021-01-01,gt2020-01-01T08:00:00.000Z, True
#2020-01-01T08:00:00.000,gt2020-01-01T08:00:00.000Z, True
2019-12-31T08:00:00.000,gt2020-01-01T08:00:00.000Z, False
2020-01-01T08:00:00.000Z,gt2020-01-01T08:00:00.000Z, False
2019-01-01T12:00:00.000Z,gt2020-01-01T08:00:00.000Z, False
2020,lt2020, False
2019,lt2020, True
2020-01,lt2020, False
2019-01,lt2020, True
2020-01-01,lt2020, False
2019-01-01,lt2020, True
2020-01-01T08:00:00,lt2020, False
2019-01-01T02:00:00,lt2020, True
2020-01-01T12:00:00Z,lt2020, False
2019-12-31T12:00:00Z,lt2020, True
2020,lt2020-01, False
2019,lt2020-01, True
2020,lt2020-04, True
2020-01,lt2020-01, False
2020-01,lt2020-02, True
2020-02-01,lt2020-02, False
2020-01-31,lt2020-02, True
2020-02-01T08:00:00,lt2020-02, False
2020-01-31T12:00:00,lt2020-02, True
2020-02-01T12:00:00Z,lt2020-02, False
2021-01-31T12:00:00Z,lt2020-02, False
2020,lt2020-01-01, False
2019,lt2020-01-01, True
2020-01,lt2020-01-01, False
2020-01,lt2020-01-15, True
2020-01,lt2020-02-01, True
2020-01-01,lt2020-01-01, False
2020-01-02,lt2020-01-01, False
2020-01-11,lt2020-01-15, True
2020-01-01T08:00:00,lt2020-01-01, False
2020-01-01T12:00:00,lt2020-02-01, True
2020-01-01T12:00:00Z,lt2020-01-01, False
2021-01-01T12:00:00Z,lt2020-02-01, False
2020,lt2020-01-01T08:00:00.000, True
2019,lt2020-01-01T08:00:00.000, True
2020-01,lt2020-01-01T08:00:00.000, True
2021-01,lt2020-01-01T08:00:00.000, False
2019-01,lt2020-01-01T08:00:00.000, True
2020-01-01,lt2020-01-01T08:00:00.000, True
2019-12-31,lt2020-01-01T08:00:00.000, True
2020-01-01T08:00:00.000,lt2020-01-01T08:00:00.000, False
2019-12-31T08:00:00.000,lt2020-01-01T08:00:00.000, True
2020-01-01T00:00:00.000Z,lt2020-01-01T08:00:00.000, True
2019-01-01T03:00:00.000Z,lt2020-01-01T08:00:00.000, True
2020,lt2020-01-01T08:00:00.000Z, True
2019,lt2020-01-01T08:00:00.000Z, True
2021,lt2020-01-01T08:00:00.000Z, False
2020-01,lt2020-01-01T08:00:00.000Z, True
2021-12,lt2020-01-01T08:00:00.000Z, False
2020-01-01,lt2020-01-01T08:00:00.000Z, True
2021-01-01,lt2020-01-01T08:00:00.000Z, False
2019-01-01,lt2020-01-01T08:00:00.000Z, True
2020-01-01T08:00:00.000,lt2020-01-01T08:00:00.000Z, False
2019-12-31T00:00:00.000,lt2020-01-01T08:00:00.000Z, True
2020-01-01T08:00:00.000Z,lt2020-01-01T08:00:00.000Z, False
2019-01-01T12:00:00.000Z,lt2020-01-01T08:00:00.000Z, True
1 ObservationDate Query Result
2 2020 eq2020 True
3 2021 eq2020 False
4 2020-01 eq2020 True
5 2020-12 eq2020 True
6 2021-01 eq2020 False
7 2020-01-01 eq2020 True
8 2020-12-31 eq2020 True
9 2021-01-01 eq2020 False
10 2020-01-01T08:00:00 eq2020 True
11 2020-12-31T08:00:00 eq2020 True
12 2019-12-31T08:00:00 eq2020 False
13 2020-01-01T12:00:00Z eq2020 True
14 2019-12-31T12:00:00Z eq2020 False
15 2020 eq2020-01 False
16 2021 eq2020-01 False
17 2020-01 eq2020-01 True
18 2021-01 eq2020-01 False
19 2020-01-01 eq2020-01 True
20 2020-01-30 eq2020-01 True
21 2021-01-01 eq2020-01 False
22 2020-01-01T08:00:00.000 eq2020-01 True
23 2019-12-31T08:00:00.000 eq2020-01 False
24 2020-01-01T12:00:00.000Z eq2020-01 True
25 2019-12-31T12:00:00.000Z eq2020-01 False
26 2020 eq2020-01-01 False
27 2021 eq2020-01-01 False
28 2020-01 eq2020-01-01 False
29 2021-01 eq2020-01-01 False
30 2020-01-01 eq2020-01-01 True
31 2021-01-01 eq2020-01-01 False
32 2020-01-01T08:00:00.000 eq2020-01-01 True
33 2019-12-31T08:00:00.000 eq2020-01-01 False
34 2020-01-01T12:00:00.000Z eq2020-01-01 True
35 2019-12-31T12:00:00.000Z eq2020-01-01 False
36 2020 eq2020-01-01T08:00:00.000 False
37 2021 eq2020-01-01T08:00:00.000 False
38 2020-01 eq2020-01-01T08:00:00.000 False
39 2021-01 eq2020-01-01T08:00:00.000 False
40 2020-01-01 eq2020-01-01T08:00:00.000 False
41 2021-01-01 eq2020-01-01T08:00:00.000 False
42 2020-01-01T08:00:00.000 eq2020-01-01T08:00:00.000 True
43 2019-12-31T08:00:00.000 eq2020-01-01T08:00:00.000 False
44 #2020-01-01T13:00:00.000Z eq2020-01-01T08:00:00.000 True
45 2019-01-01T12:00:00.000Z eq2020-01-01T08:00:00.000 False
46 2020 eq2020-01-01T08:00:00.000Z False
47 2021 eq2020-01-01T08:00:00.000Z False
48 2020-01 eq2020-01-01T08:00:00.000Z False
49 2021-01 eq2020-01-01T08:00:00.000Z False
50 2020-01-01 eq2020-01-01T08:00:00.000Z False
51 2021-01-01 eq2020-01-01T08:00:00.000Z False
52 #2020-01-01T03:00:00.000 eq2020-01-01T08:00:00.000Z True
53 2019-12-31T08:00:00.000 eq2020-01-01T08:00:00.000Z False
54 2020-01-01T08:00:00.000Z eq2020-01-01T08:00:00.000Z True
55 2019-01-01T12:00:00.000Z eq2020-01-01T08:00:00.000Z False
56 2020 gt2020 False
57 2021 gt2020 True
58 2020-01 gt2020 False
59 2021-01 gt2020 True
60 2020-01-01 gt2020 False
61 2021-01-01 gt2020 True
62 2020-01-01T08:00:00 gt2020 False
63 2021-01-01T02:00:00 gt2020 True
64 2020-01-01T12:00:00Z gt2020 False
65 2021-01-01T12:00:00Z gt2020 True
66 2020 gt2020-01 True
67 2019 gt2020-01 False
68 2021 gt2020-01 True
69 2020-01 gt2020-01 False
70 2020-02 gt2020-01 True
71 2020-01-01 gt2020-01 False
72 2020-02-01 gt2020-01 True
73 2020-01-01T08:00:00 gt2020-01 False
74 2021-02-01T12:00:00 gt2020-01 True
75 2020-01-01T12:00:00Z gt2020-01 False
76 2021-02-01T12:00:00Z gt2020-01 True
77 2020 gt2020-01-01 True
78 2019 gt2020-01-01 False
79 2020-01 gt2020-01-01 True
80 2019-12 gt2020-01-01 False
81 2020-02 gt2020-01-01 True
82 2020-01-01 gt2020-01-01 False
83 2020-01-02 gt2020-01-01 True
84 2020-01-01T08:00:00 gt2020-01-01 False
85 2021-02-01T12:00:00 gt2020-01-01 True
86 2020-01-01T12:00:00Z gt2020-01-01 False
87 2021-02-01T12:00:00Z gt2020-01-01 True
88 2020 gt2020-01-01T08:00:00.000 True
89 2019 gt2020-01-01T08:00:00.000 False
90 2020-01 gt2020-01-01T08:00:00.000 True
91 2021-01 gt2020-01-01T08:00:00.000 True
92 2020-01-01 gt2020-01-01T08:00:00.000 True
93 2019-12-31 gt2020-01-01T08:00:00.000 False
94 2020-01-01T08:00:00.000 gt2020-01-01T08:00:00.000 False
95 2021-12-31T08:00:00.000 gt2020-01-01T08:00:00.000 True
96 #2020-01-01T13:00:00.000Z gt2020-01-01T08:00:00.000 False
97 2019-01-01T08:00:00.000Z gt2020-01-01T08:00:00.000 False
98 2020 gt2020-01-01T08:00:00.000Z True
99 2019 gt2020-01-01T08:00:00.000Z False
100 2020-01 gt2020-01-01T08:00:00.000Z True
101 2019-12 gt2020-01-01T08:00:00.000Z False
102 2020-01-01 gt2020-01-01T08:00:00.000Z True
103 2021-01-01 gt2020-01-01T08:00:00.000Z True
104 #2020-01-01T08:00:00.000 gt2020-01-01T08:00:00.000Z True
105 2019-12-31T08:00:00.000 gt2020-01-01T08:00:00.000Z False
106 2020-01-01T08:00:00.000Z gt2020-01-01T08:00:00.000Z False
107 2019-01-01T12:00:00.000Z gt2020-01-01T08:00:00.000Z False
108 2020 lt2020 False
109 2019 lt2020 True
110 2020-01 lt2020 False
111 2019-01 lt2020 True
112 2020-01-01 lt2020 False
113 2019-01-01 lt2020 True
114 2020-01-01T08:00:00 lt2020 False
115 2019-01-01T02:00:00 lt2020 True
116 2020-01-01T12:00:00Z lt2020 False
117 2019-12-31T12:00:00Z lt2020 True
118 2020 lt2020-01 False
119 2019 lt2020-01 True
120 2020 lt2020-04 True
121 2020-01 lt2020-01 False
122 2020-01 lt2020-02 True
123 2020-02-01 lt2020-02 False
124 2020-01-31 lt2020-02 True
125 2020-02-01T08:00:00 lt2020-02 False
126 2020-01-31T12:00:00 lt2020-02 True
127 2020-02-01T12:00:00Z lt2020-02 False
128 2021-01-31T12:00:00Z lt2020-02 False
129 2020 lt2020-01-01 False
130 2019 lt2020-01-01 True
131 2020-01 lt2020-01-01 False
132 2020-01 lt2020-01-15 True
133 2020-01 lt2020-02-01 True
134 2020-01-01 lt2020-01-01 False
135 2020-01-02 lt2020-01-01 False
136 2020-01-11 lt2020-01-15 True
137 2020-01-01T08:00:00 lt2020-01-01 False
138 2020-01-01T12:00:00 lt2020-02-01 True
139 2020-01-01T12:00:00Z lt2020-01-01 False
140 2021-01-01T12:00:00Z lt2020-02-01 False
141 2020 lt2020-01-01T08:00:00.000 True
142 2019 lt2020-01-01T08:00:00.000 True
143 2020-01 lt2020-01-01T08:00:00.000 True
144 2021-01 lt2020-01-01T08:00:00.000 False
145 2019-01 lt2020-01-01T08:00:00.000 True
146 2020-01-01 lt2020-01-01T08:00:00.000 True
147 2019-12-31 lt2020-01-01T08:00:00.000 True
148 2020-01-01T08:00:00.000 lt2020-01-01T08:00:00.000 False
149 2019-12-31T08:00:00.000 lt2020-01-01T08:00:00.000 True
150 2020-01-01T00:00:00.000Z lt2020-01-01T08:00:00.000 True
151 2019-01-01T03:00:00.000Z lt2020-01-01T08:00:00.000 True
152 2020 lt2020-01-01T08:00:00.000Z True
153 2019 lt2020-01-01T08:00:00.000Z True
154 2021 lt2020-01-01T08:00:00.000Z False
155 2020-01 lt2020-01-01T08:00:00.000Z True
156 2021-12 lt2020-01-01T08:00:00.000Z False
157 2020-01-01 lt2020-01-01T08:00:00.000Z True
158 2021-01-01 lt2020-01-01T08:00:00.000Z False
159 2019-01-01 lt2020-01-01T08:00:00.000Z True
160 2020-01-01T08:00:00.000 lt2020-01-01T08:00:00.000Z False
161 2019-12-31T00:00:00.000 lt2020-01-01T08:00:00.000Z True
162 2020-01-01T08:00:00.000Z lt2020-01-01T08:00:00.000Z False
163 2019-01-01T12:00:00.000Z lt2020-01-01T08:00:00.000Z True

View File

@ -20,20 +20,10 @@ package ca.uhn.fhir.jpa.model.entity;
* #L% * #L%
*/ */
import ca.uhn.fhir.jpa.model.config.PartitionSettings; import java.text.ParseException;
import ca.uhn.fhir.model.api.IQueryParameterType; import java.text.SimpleDateFormat;
import ca.uhn.fhir.model.api.TemporalPrecisionEnum; import java.util.Calendar;
import ca.uhn.fhir.model.primitive.InstantDt; import java.util.Date;
import ca.uhn.fhir.rest.param.DateParam;
import ca.uhn.fhir.rest.param.DateRangeParam;
import ca.uhn.fhir.util.DateUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.builder.EqualsBuilder;
import org.apache.commons.lang3.builder.HashCodeBuilder;
import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;
import org.hibernate.search.mapper.pojo.mapping.definition.annotation.FullTextField;
import org.hl7.fhir.r4.model.DateTimeType;
import javax.persistence.Column; import javax.persistence.Column;
import javax.persistence.Embeddable; import javax.persistence.Embeddable;
@ -47,7 +37,22 @@ import javax.persistence.Table;
import javax.persistence.Temporal; import javax.persistence.Temporal;
import javax.persistence.TemporalType; import javax.persistence.TemporalType;
import javax.persistence.Transient; import javax.persistence.Transient;
import java.util.Date;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.builder.EqualsBuilder;
import org.apache.commons.lang3.builder.HashCodeBuilder;
import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;
import org.hibernate.search.mapper.pojo.mapping.definition.annotation.FullTextField;
import org.hl7.fhir.r4.model.DateTimeType;
import ca.uhn.fhir.jpa.model.config.PartitionSettings;
import ca.uhn.fhir.model.api.IQueryParameterType;
import ca.uhn.fhir.model.api.TemporalPrecisionEnum;
import ca.uhn.fhir.model.primitive.InstantDt;
import ca.uhn.fhir.rest.param.DateParam;
import ca.uhn.fhir.rest.param.DateRangeParam;
import ca.uhn.fhir.util.DateUtils;
@Embeddable @Embeddable
@Entity @Entity
@ -121,27 +126,60 @@ public class ResourceIndexedSearchParamDate extends BaseResourceIndexedSearchPar
} }
computeValueHighDateOrdinal(theHighString); computeValueHighDateOrdinal(theHighString);
computeValueLowDateOrdinal(theLowString); computeValueLowDateOrdinal(theLowString);
reComputeValueHighDate(theHigh, theHighString);
myOriginalValue = theOriginalValue; myOriginalValue = theOriginalValue;
calculateHashes(); calculateHashes();
} }
private void computeValueHighDateOrdinal(String theHigh) { private void computeValueHighDateOrdinal(String theHigh) {
if (!StringUtils.isBlank(theHigh)) { if (!StringUtils.isBlank(theHigh)) {
this.myValueHighDateOrdinal = generateOrdinalDateInteger(theHigh); this.myValueHighDateOrdinal = generateHighOrdinalDateInteger(theHigh);
} }
} }
private int generateOrdinalDateInteger(String theDateString) { private void reComputeValueHighDate(Date theHigh, String theHighString) {
if (StringUtils.isBlank(theHighString) || theHigh == null)
return;
// FT : 2021-09-10 not very comfortable to set the high value to the last second
// Timezone? existing data?
// if YYYY or YYYY-MM or YYYY-MM-DD add the last second
if (theHighString.length() == 4 || theHighString.length() == 7 || theHighString.length() == 10) {
String theCompleteDateStr = DateUtils.getCompletedDate(theHighString).getRight();
try {
Date complateDate = new SimpleDateFormat("yyyy-MM-dd").parse(theCompleteDateStr);
this.myValueHigh = DateUtils.getEndOfDay(complateDate);
} catch (ParseException e) {
// do nothing;
}
}
}
private int generateLowOrdinalDateInteger(String theDateString) {
if (theDateString.contains("T")) { if (theDateString.contains("T")) {
theDateString = theDateString.substring(0, theDateString.indexOf("T")); theDateString = theDateString.substring(0, theDateString.indexOf("T"));
} }
theDateString = DateUtils.getCompletedDate(theDateString).getLeft();
theDateString = theDateString.replace("-", ""); theDateString = theDateString.replace("-", "");
return Integer.valueOf(theDateString); return Integer.valueOf(theDateString);
} }
private int generateHighOrdinalDateInteger(String theDateString) {
if (theDateString.contains("T")) {
theDateString = theDateString.substring(0, theDateString.indexOf("T"));
}
theDateString = DateUtils.getCompletedDate(theDateString).getRight();
theDateString = theDateString.replace("-", "");
return Integer.valueOf(theDateString);
}
private void computeValueLowDateOrdinal(String theLow) { private void computeValueLowDateOrdinal(String theLow) {
if (StringUtils.isNotBlank(theLow)) { if (StringUtils.isNotBlank(theLow)) {
this.myValueLowDateOrdinal = generateOrdinalDateInteger(theLow); this.myValueLowDateOrdinal = generateLowOrdinalDateInteger(theLow);
} }
} }