diff --git a/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/3992-elastic-search-mapper-exception-on-loinc-value-sets-pre-expansion.yaml b/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/3992-elastic-search-mapper-exception-on-loinc-value-sets-pre-expansion.yaml new file mode 100644 index 00000000000..4c7724f5948 --- /dev/null +++ b/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/3992-elastic-search-mapper-exception-on-loinc-value-sets-pre-expansion.yaml @@ -0,0 +1,7 @@ +--- +type: fix +issue: 3992 +title: "Previously when ValueSets were pre-expanded after loinc terminology upload, expansion was failing with an exception for each ValueSet + with more than 10,000 properties. This problem has been fixed. +This fix changed some freetext mappings (definitions about how resources are freetext indexed) for terminology resources, which requires + reindexing those resources. To do this use the `reindex-terminology` command." diff --git a/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/upgrade.md b/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/upgrade.md new file mode 100644 index 00000000000..79590186e65 --- /dev/null +++ b/hapi-fhir-docs/src/main/resources/ca/uhn/hapi/fhir/changelog/6_2_0/upgrade.md @@ -0,0 +1,4 @@ +This release has breaking changes. +* Hibernate Search mappings for Terminology entities have been upgraded, which requires terminology freetext reindexing. + +To recreate Terminology freetext indexes use CLI command: `reindex-terminology` diff --git a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConcept.java b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConcept.java index 663831a6a6d..342c1b6eb9b 100644 --- a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConcept.java +++ b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConcept.java @@ -30,14 +30,14 @@ 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.engine.backend.types.ObjectStructure; import org.hibernate.search.engine.backend.types.Projectable; import org.hibernate.search.engine.backend.types.Searchable; +import org.hibernate.search.mapper.pojo.bridge.mapping.annotation.PropertyBinderRef; import org.hibernate.search.mapper.pojo.bridge.mapping.annotation.RoutingBinderRef; import org.hibernate.search.mapper.pojo.mapping.definition.annotation.FullTextField; import org.hibernate.search.mapper.pojo.mapping.definition.annotation.GenericField; import org.hibernate.search.mapper.pojo.mapping.definition.annotation.Indexed; -import org.hibernate.search.mapper.pojo.mapping.definition.annotation.IndexedEmbedded; +import org.hibernate.search.mapper.pojo.mapping.definition.annotation.PropertyBinding; import org.hl7.fhir.r4.model.Coding; import javax.annotation.Nonnull; @@ -114,12 +114,8 @@ public class TermConcept implements Serializable { @FullTextField(name = "myDisplayPhonetic", searchable= Searchable.YES, projectable= Projectable.NO, analyzer = "autocompletePhoneticAnalyzer") private String myDisplay; - /** - * IndexedEmbedded uses ObjectStructure.NESTED to be able to use hibernate search nested predicate queries. - * @see "https://docs.jboss.org/hibernate/search/6.0/reference/en-US/html_single/#search-dsl-predicate-nested" - */ @OneToMany(mappedBy = "myConcept", orphanRemoval = false, fetch = FetchType.LAZY) - @IndexedEmbedded(structure = ObjectStructure.NESTED) + @PropertyBinding(binder = @PropertyBinderRef(type = TermConceptPropertyBinder.class)) private Collection myProperties; @OneToMany(mappedBy = "myConcept", orphanRemoval = false, fetch = FetchType.LAZY) @@ -458,7 +454,7 @@ public class TermConcept implements Serializable { * Returns a view of {@link #getChildren()} but containing the actual child codes */ public List getChildCodes() { - return getChildren().stream().map(t -> t.getChild()).collect(Collectors.toList()); + return getChildren().stream().map(TermConceptParentChildLink::getChild).collect(Collectors.toList()); } } diff --git a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConceptPropertyBinder.java b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConceptPropertyBinder.java new file mode 100644 index 00000000000..5ef3f2a7ff0 --- /dev/null +++ b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/entity/TermConceptPropertyBinder.java @@ -0,0 +1,83 @@ +package ca.uhn.fhir.jpa.entity; + +/*- + * #%L + * HAPI FHIR JPA Server + * %% + * Copyright (C) 2014 - 2022 Smile CDR, Inc. + * %% + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * #L% + */ + +import org.hibernate.search.engine.backend.document.DocumentElement; +import org.hibernate.search.engine.backend.document.model.dsl.IndexSchemaElement; +import org.hibernate.search.engine.backend.types.dsl.IndexFieldTypeFactory; +import org.hibernate.search.mapper.pojo.bridge.PropertyBridge; +import org.hibernate.search.mapper.pojo.bridge.binding.PropertyBindingContext; +import org.hibernate.search.mapper.pojo.bridge.mapping.programmatic.PropertyBinder; +import org.hibernate.search.mapper.pojo.bridge.runtime.PropertyBridgeWriteContext; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.Collection; + +import static org.apache.commons.lang3.StringUtils.isNotBlank; + +/** + * Allows hibernate search to index individual concepts' properties + */ +public class TermConceptPropertyBinder implements PropertyBinder { + + + public static final String CONCEPT_PROPERTY_PREFIX_NAME = "P:"; + + private static final Logger ourLog = LoggerFactory.getLogger(TermConceptPropertyBinder.class); + + @Override + public void bind(PropertyBindingContext thePropertyBindingContext) { + thePropertyBindingContext.dependencies().use("myKey").use("myValue"); + IndexSchemaElement indexSchemaElement = thePropertyBindingContext.indexSchemaElement(); + + //In order to support dynamic fields, we have to use field templates. We _must_ define the template at bootstrap time and cannot + //create them adhoc. https://docs.jboss.org/hibernate/search/6.0/reference/en-US/html_single/#mapper-orm-bridge-index-field-dsl-dynamic + indexSchemaElement.fieldTemplate("propTemplate", IndexFieldTypeFactory::asString) + .matchingPathGlob(CONCEPT_PROPERTY_PREFIX_NAME + "*") + .multiValued(); + + + thePropertyBindingContext.bridge(Collection.class, new TermConceptPropertyBridge()); + } + + @SuppressWarnings("rawtypes") + private static class TermConceptPropertyBridge implements PropertyBridge { + + @Override + public void write(DocumentElement theDocument, Collection theObject, PropertyBridgeWriteContext thePropertyBridgeWriteContext) { + + @SuppressWarnings("unchecked") + Collection properties = (Collection) theObject; + + if (properties != null) { + for (TermConceptProperty next : properties) { + theDocument.addValue(CONCEPT_PROPERTY_PREFIX_NAME + next.getKey(), next.getValue()); + ourLog.trace("Adding Prop: {}{} -- {}", CONCEPT_PROPERTY_PREFIX_NAME, next.getKey(), next.getValue()); + if (next.getType() == TermConceptPropertyTypeEnum.CODING && isNotBlank(next.getDisplay())) { + theDocument.addValue(CONCEPT_PROPERTY_PREFIX_NAME + next.getKey(), next.getDisplay()); + ourLog.trace("Adding multivalue Prop: {}{} -- {}", CONCEPT_PROPERTY_PREFIX_NAME, next.getKey(), next.getDisplay()); + } + } + } + } + } +} diff --git a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/search/HapiHSearchAnalysisConfigurers.java b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/search/HapiHSearchAnalysisConfigurers.java index 74d37f99fda..36eef1aafb9 100644 --- a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/search/HapiHSearchAnalysisConfigurers.java +++ b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/search/HapiHSearchAnalysisConfigurers.java @@ -104,9 +104,6 @@ public class HapiHSearchAnalysisConfigurers { theLuceneCtx.analyzer("conceptParentPidsAnalyzer").custom() .tokenizer(WhitespaceTokenizerFactory.class); - theLuceneCtx.analyzer("termConceptPropertyAnalyzer").custom() - .tokenizer(WhitespaceTokenizerFactory.class); - theLuceneCtx.normalizer(LOWERCASE_ASCIIFOLDING_NORMALIZER).custom() .tokenFilter(LowerCaseFilterFactory.class) .tokenFilter(ASCIIFoldingFilterFactory.class); @@ -178,9 +175,6 @@ public class HapiHSearchAnalysisConfigurers { theConfigCtx.analyzer("conceptParentPidsAnalyzer").custom() .tokenizer("whitespace"); - theConfigCtx.analyzer("termConceptPropertyAnalyzer").custom() - .tokenizer("whitespace"); - theConfigCtx.normalizer( LOWERCASE_ASCIIFOLDING_NORMALIZER ).custom() .tokenFilters( "lowercase", "asciifolding" ); diff --git a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/term/TermReadSvcImpl.java b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/term/TermReadSvcImpl.java index 0db16fb0d33..619ccc9ce8f 100644 --- a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/term/TermReadSvcImpl.java +++ b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/term/TermReadSvcImpl.java @@ -65,7 +65,6 @@ import ca.uhn.fhir.jpa.model.sched.HapiJob; import ca.uhn.fhir.jpa.model.sched.ISchedulerService; import ca.uhn.fhir.jpa.model.sched.ScheduledJobDefinition; import ca.uhn.fhir.jpa.model.util.JpaConstants; -import ca.uhn.fhir.jpa.search.ElasticsearchNestedQueryBuilderUtil; import ca.uhn.fhir.jpa.search.builder.SearchBuilder; import ca.uhn.fhir.jpa.term.api.ITermDeferredStorageSvc; import ca.uhn.fhir.jpa.term.api.ITermReadSvc; @@ -91,7 +90,6 @@ import com.github.benmanes.caffeine.cache.Caffeine; import com.google.common.annotations.VisibleForTesting; import com.google.common.base.Stopwatch; import com.google.common.collect.ArrayListMultimap; -import com.google.gson.JsonObject; import org.apache.commons.collections4.ListUtils; import org.apache.commons.lang3.ObjectUtils; import org.apache.commons.lang3.StringUtils; @@ -99,12 +97,7 @@ import org.apache.commons.lang3.Validate; import org.apache.commons.lang3.time.DateUtils; import org.apache.lucene.index.Term; import org.apache.lucene.search.BooleanQuery; -import org.apache.lucene.search.Query; -import org.apache.lucene.search.RegexpQuery; -import org.apache.lucene.search.TermQuery; import org.hibernate.CacheMode; -import org.hibernate.search.backend.elasticsearch.ElasticsearchExtension; -import org.hibernate.search.backend.lucene.LuceneExtension; import org.hibernate.search.engine.search.predicate.dsl.BooleanPredicateClausesStep; import org.hibernate.search.engine.search.predicate.dsl.PredicateFinalStep; import org.hibernate.search.engine.search.predicate.dsl.SearchPredicateFactory; @@ -193,6 +186,7 @@ import java.util.concurrent.TimeUnit; import java.util.function.Consumer; import java.util.stream.Collectors; +import static ca.uhn.fhir.jpa.entity.TermConceptPropertyBinder.CONCEPT_PROPERTY_PREFIX_NAME; import static ca.uhn.fhir.jpa.term.api.ITermLoaderSvc.LOINC_URI; import static java.lang.String.join; import static java.util.stream.Collectors.joining; @@ -284,10 +278,6 @@ public class TermReadSvcImpl implements ITermReadSvc { private HibernatePropertiesProvider myHibernatePropertiesProvider; - private boolean isFullTextSetToUseElastic() { - return "elasticsearch".equalsIgnoreCase(myHibernatePropertiesProvider.getHibernateSearchBackend()); - } - @Override public boolean isCodeSystemSupported(ValidationSupportContext theValidationSupportContext, String theSystem) { TermCodeSystemVersion cs = getCurrentCodeSystemVersion(theSystem); @@ -509,7 +499,7 @@ public class TermReadSvcImpl implements ITermReadSvc { /* * ValueSet doesn't exist in pre-expansion database, so perform in-memory expansion */ - if (!optionalTermValueSet.isPresent()) { + if (optionalTermValueSet.isEmpty()) { ourLog.debug("ValueSet is not present in terminology tables. Will perform in-memory expansion without parameters. {}", getValueSetInfo(theValueSetToExpand)); String msg = myContext.getLocalizer().getMessage(TermReadSvcImpl.class, "valueSetExpandedUsingInMemoryExpansion", getValueSetInfo(theValueSetToExpand)); theAccumulator.addMessage(msg); @@ -815,7 +805,7 @@ public class TermReadSvcImpl implements ITermReadSvc { } /** - * @return Returns true if there are potentially more results to process. + * Returns true if there are potentially more results to process. */ private void expandValueSetHandleIncludeOrExclude(@Nullable ValueSetExpansionOptions theExpansionOptions, IValueSetConceptAccumulator theValueSetCodeAccumulator, @@ -853,9 +843,8 @@ public class TermReadSvcImpl implements ITermReadSvc { } } - Consumer consumer = c -> { + Consumer consumer = c -> addOrRemoveCode(theValueSetCodeAccumulator, theAddedCodes, theAdd, system, c.getCode(), c.getDisplay(), c.getSystemVersion()); - }; try { ConversionContext40_50.INSTANCE.init(new VersionConvertor_40_50(new BaseAdvisor_40_50()), "ValueSet"); @@ -901,7 +890,6 @@ public class TermReadSvcImpl implements ITermReadSvc { return myFulltextSearchSvc != null && !ourForceDisableHibernateSearchForUnitTest; } - @Nonnull private void expandValueSetHandleIncludeOrExcludeUsingDatabase( ValueSetExpansionOptions theExpansionOptions, IValueSetConceptAccumulator theValueSetCodeAccumulator, @@ -1141,11 +1129,7 @@ public class TermReadSvcImpl implements ITermReadSvc { private @Nonnull ValueSetExpansionOptions provideExpansionOptions(@Nullable ValueSetExpansionOptions theExpansionOptions) { - if (theExpansionOptions != null) { - return theExpansionOptions; - } else { - return DEFAULT_EXPANSION_OPTIONS; - } + return Objects.requireNonNullElse(theExpansionOptions, DEFAULT_EXPANSION_OPTIONS); } private void addOrRemoveCode(IValueSetConceptAccumulator theValueSetCodeAccumulator, Set theAddedCodes, boolean theAdd, String theSystem, String theCode, String theDisplay, String theSystemVersion) { @@ -1205,7 +1189,9 @@ public class TermReadSvcImpl implements ITermReadSvc { private void handleFilterPropertyDefault(SearchPredicateFactory theF, BooleanPredicateClausesStep theB, ValueSet.ConceptSetFilterComponent theFilter) { - theB.must(getPropertyNameValueNestedPredicate(theF, theFilter.getProperty(), theFilter.getValue())); + String value = theFilter.getValue(); + Term term = new Term(CONCEPT_PROPERTY_PREFIX_NAME + theFilter.getProperty(), value); + theB.must(theF.match().field(term.field()).matching(term.text())); } @@ -1213,7 +1199,7 @@ public class TermReadSvcImpl implements ITermReadSvc { /* * We treat the regex filter as a match on the regex * anywhere in the property string. The spec does not - * say whether or not this is the right behaviour, but + * say whether this is the right behaviour or not, but * there are examples that seem to suggest that it is. */ String value = theFilter.getValue(); @@ -1228,43 +1214,26 @@ public class TermReadSvcImpl implements ITermReadSvc { value = value.substring(1); } - if (isFullTextSetToUseElastic()) { - ElasticsearchNestedQueryBuilderUtil nestedQueryBuildUtil = new ElasticsearchNestedQueryBuilderUtil( - "myProperties", "myKey", theFilter.getProperty(), - "myValueString", value); - - JsonObject nestedQueryJO = nestedQueryBuildUtil.toGson(); - - ourLog.debug("Build nested Elasticsearch query: {}", nestedQueryJO); - theB.must(theF.extension(ElasticsearchExtension.get()).fromJson(nestedQueryJO)); - return; - + theB.must(theF.regexp() + .field(CONCEPT_PROPERTY_PREFIX_NAME + theFilter.getProperty()) + .matching(value) ); } - // native Lucene configured - Query termPropKeyQuery = new TermQuery(new Term(IDX_PROP_KEY, theFilter.getProperty())); - Query regexpValueQuery = new RegexpQuery(new Term(IDX_PROP_VALUE_STRING, value)); - theB.must(theF.nested().objectField("myProperties").nest( - theF.bool() - .must(theF.extension(LuceneExtension.get()).fromLuceneQuery(termPropKeyQuery)) - .must(theF.extension(LuceneExtension.get()).fromLuceneQuery(regexpValueQuery)) - )); - } + private void handleFilterLoincCopyright(SearchPredicateFactory theF, BooleanPredicateClausesStep theB, + ValueSet.ConceptSetFilterComponent theFilter) { - - private void handleFilterLoincCopyright(SearchPredicateFactory theF, BooleanPredicateClausesStep theB, ValueSet.ConceptSetFilterComponent theFilter) { if (theFilter.getOp() == ValueSet.FilterOperator.EQUAL) { String copyrightFilterValue = defaultString(theFilter.getValue()).toLowerCase(); switch (copyrightFilterValue) { case "3rdparty": logFilteringValueOnProperty(theFilter.getValue(), theFilter.getProperty()); - addFilterLoincCopyright3rdParty(theF, theB, theFilter); + addFilterLoincCopyright3rdParty(theF, theB); break; case "loinc": logFilteringValueOnProperty(theFilter.getValue(), theFilter.getProperty()); - addFilterLoincCopyrightLoinc(theF, theB, theFilter); + addFilterLoincCopyrightLoinc(theF, theB); break; default: throwInvalidRequestForValueOnProperty(theFilter.getValue(), theFilter.getProperty()); @@ -1275,21 +1244,13 @@ public class TermReadSvcImpl implements ITermReadSvc { } } - private void addFilterLoincCopyrightLoinc(SearchPredicateFactory theF, - BooleanPredicateClausesStep theB, ValueSet.ConceptSetFilterComponent theFilter) { - - theB.mustNot(theF.match().field(IDX_PROP_KEY).matching("EXTERNAL_COPYRIGHT_NOTICE")); + private void addFilterLoincCopyrightLoinc(SearchPredicateFactory theF, BooleanPredicateClausesStep theB) { + theB.mustNot(theF.exists().field(CONCEPT_PROPERTY_PREFIX_NAME + "EXTERNAL_COPYRIGHT_NOTICE")); } - private void addFilterLoincCopyright3rdParty(SearchPredicateFactory thePredicateFactory, - BooleanPredicateClausesStep theBooleanClause, ValueSet.ConceptSetFilterComponent theFilter) { - //TODO GGG HS These used to be Term term = new Term(TermConceptPropertyBinder.CONCEPT_FIELD_PROPERTY_PREFIX + "EXTERNAL_COPYRIGHT_NOTICE", ".*");, which was lucene-specific. - //TODO GGG HS ask diederik if this is equivalent. - //This old .* regex is the same as an existence check on a field, which I've implemented here. -// theBooleanClause.must(thePredicateFactory.exists().field("EXTERNAL_COPYRIGHT_NOTICE")); - - theBooleanClause.must(thePredicateFactory.match().field(IDX_PROP_KEY).matching("EXTERNAL_COPYRIGHT_NOTICE")); + private void addFilterLoincCopyright3rdParty(SearchPredicateFactory theF, BooleanPredicateClausesStep theB) { + theB.must(theF.exists().field(CONCEPT_PROPERTY_PREFIX_NAME + "EXTERNAL_COPYRIGHT_NOTICE")); } @SuppressWarnings("EnumSwitchStatementWhichMissesCases") @@ -1308,21 +1269,21 @@ public class TermReadSvcImpl implements ITermReadSvc { } private void addLoincFilterAncestorEqual(String theSystem, SearchPredicateFactory f, BooleanPredicateClausesStep b, ValueSet.ConceptSetFilterComponent theFilter) { - long parentPid = getAncestorCodePid(theSystem, theFilter.getProperty(), theFilter.getValue()); - b.must(f.match().field("myParentPids").matching(String.valueOf(parentPid))); + addLoincFilterAncestorEqual(theSystem, f, b, theFilter.getProperty(), theFilter.getValue()); } + private void addLoincFilterAncestorEqual(String theSystem, SearchPredicateFactory f, BooleanPredicateClausesStep b, String theProperty, String theValue) { + List terms = getAncestorTerms(theSystem, theProperty, theValue); + b.must(f.bool(innerB -> terms.forEach(term -> innerB.should(f.match().field(term.field()).matching(term.text()))))); + } private void addLoincFilterAncestorIn(String theSystem, SearchPredicateFactory f, BooleanPredicateClausesStep b, ValueSet.ConceptSetFilterComponent theFilter) { String[] values = theFilter.getValue().split(","); - List ancestorCodePidList = new ArrayList<>(); + List terms = new ArrayList<>(); for (String value : values) { - ancestorCodePidList.add(getAncestorCodePid(theSystem, theFilter.getProperty(), value)); + terms.addAll(getAncestorTerms(theSystem, theFilter.getProperty(), value)); } - - b.must(f.bool(innerB -> ancestorCodePidList.forEach( - ancestorPid -> innerB.should(f.match().field("myParentPids").matching(String.valueOf(ancestorPid))) - ))); + b.must(f.bool(innerB -> terms.forEach(term -> innerB.should(f.match().field(term.field()).matching(term.text()))))); } @@ -1342,32 +1303,20 @@ public class TermReadSvcImpl implements ITermReadSvc { private void addLoincFilterParentChildIn(SearchPredicateFactory f, BooleanPredicateClausesStep b, ValueSet.ConceptSetFilterComponent theFilter) { String[] values = theFilter.getValue().split(","); - b.minimumShouldMatchNumber(1); + List terms = new ArrayList<>(); for (String value : values) { logFilteringValueOnProperty(value, theFilter.getProperty()); - b.should(getPropertyNameValueNestedPredicate(f, theFilter.getProperty(), value)); + terms.add(getPropertyTerm(theFilter.getProperty(), value)); } + + b.must(f.bool(innerB -> terms.forEach(term -> innerB.should(f.match().field(term.field()).matching(term.text()))))); } private void addLoincFilterParentChildEqual(SearchPredicateFactory f, BooleanPredicateClausesStep b, String theProperty, String theValue) { logFilteringValueOnProperty(theValue, theProperty); - b.must(getPropertyNameValueNestedPredicate(f, theProperty, theValue)); + b.must(f.match().field(CONCEPT_PROPERTY_PREFIX_NAME + theProperty).matching(theValue)); } - /** - * A nested predicate is required for both predicates to be applied to same property, otherwise if properties - * propAA:valueAA and propBB:valueBB are defined, a search for propAA:valueBB would be a match - * @see "https://docs.jboss.org/hibernate/search/6.0/reference/en-US/html_single/#search-dsl-predicate-nested" - */ - private PredicateFinalStep getPropertyNameValueNestedPredicate(SearchPredicateFactory f, String theProperty, String theValue) { - return f.nested().objectField(IDX_PROPERTIES).nest( - f.bool() - .must(f.match().field(IDX_PROP_KEY).matching(theProperty)) - .must(f.match().field(IDX_PROP_VALUE_STRING).field(IDX_PROP_DISPLAY_STRING).matching(theValue)) - ); - } - - private void handleFilterConceptAndCode(String theSystem, SearchPredicateFactory f, BooleanPredicateClausesStep b, ValueSet.ConceptSetFilterComponent theFilter) { TermConcept code = findCodeForFilterCriteria(theSystem, theFilter); @@ -1427,14 +1376,24 @@ public class TermReadSvcImpl implements ITermReadSvc { ); } - private long getAncestorCodePid(String theSystem, String theProperty, String theValue) { + private Term getPropertyTerm(String theProperty, String theValue) { + return new Term(CONCEPT_PROPERTY_PREFIX_NAME + theProperty, theValue); + } + + + private List getAncestorTerms(String theSystem, String theProperty, String theValue) { + List retVal = new ArrayList<>(); + TermConcept code = findCode(theSystem, theValue) .orElseThrow(() -> new InvalidRequestException("Invalid filter criteria - code does not exist: {" + Constants.codeSystemWithDefaultDescription(theSystem) + "}" + theValue)); + retVal.add(new Term("myParentPids", "" + code.getId())); logFilteringValueOnProperty(theValue, theProperty); - return code.getId(); + + return retVal; } + @SuppressWarnings("EnumSwitchStatementWhichMissesCases") private void handleFilterLoincDescendant(String theSystem, SearchPredicateFactory f, BooleanPredicateClausesStep b, ValueSet.ConceptSetFilterComponent theFilter) { switch (theFilter.getOp()) { @@ -1640,7 +1599,7 @@ public class TermReadSvcImpl implements ITermReadSvc { IBaseResource valueSet = myDaoRegistry.getResourceDao("ValueSet").read(theValueSetId, theRequestDetails); ValueSet canonicalValueSet = myVersionCanonicalizer.valueSetToCanonical(valueSet); Optional optionalTermValueSet = fetchValueSetEntity(canonicalValueSet); - if (!optionalTermValueSet.isPresent()) { + if (optionalTermValueSet.isEmpty()) { return myContext.getLocalizer().getMessage(TermReadSvcImpl.class, "valueSetNotFoundInTerminologyDatabase", theValueSetId); } @@ -1666,7 +1625,7 @@ public class TermReadSvcImpl implements ITermReadSvc { public boolean isValueSetPreExpandedForCodeValidation(ValueSet theValueSet) { Optional optionalTermValueSet = fetchValueSetEntity(theValueSet); - if (!optionalTermValueSet.isPresent()) { + if (optionalTermValueSet.isEmpty()) { ourLog.warn("ValueSet is not present in terminology tables. Will perform in-memory code validation. {}", getValueSetInfo(theValueSet)); return false; } @@ -1684,13 +1643,11 @@ public class TermReadSvcImpl implements ITermReadSvc { private Optional fetchValueSetEntity(ValueSet theValueSet) { ResourcePersistentId valueSetResourcePid = getValueSetResourcePersistentId(theValueSet); - Optional optionalTermValueSet = myTermValueSetDao.findByResourcePid(valueSetResourcePid.getIdAsLong()); - return optionalTermValueSet; + return myTermValueSetDao.findByResourcePid(valueSetResourcePid.getIdAsLong()); } private ResourcePersistentId getValueSetResourcePersistentId(ValueSet theValueSet) { - ResourcePersistentId valueSetResourcePid = myIdHelperService.resolveResourcePersistentIds(RequestPartitionId.allPartitions(), theValueSet.getIdElement().getResourceType(), theValueSet.getIdElement().getIdPart()); - return valueSetResourcePid; + return myIdHelperService.resolveResourcePersistentIds(RequestPartitionId.allPartitions(), theValueSet.getIdElement().getResourceType(), theValueSet.getIdElement().getIdPart()); } protected IValidationSupport.CodeValidationResult validateCodeIsInPreExpandedValueSet( @@ -1726,7 +1683,7 @@ public class TermReadSvcImpl implements ITermReadSvc { return null; } - TermValueSet valueSetEntity = myTermValueSetDao.findByResourcePid(valueSetResourcePid.getIdAsLong()).orElseThrow(() -> new IllegalStateException()); + TermValueSet valueSetEntity = myTermValueSetDao.findByResourcePid(valueSetResourcePid.getIdAsLong()).orElseThrow(IllegalStateException::new); String timingDescription = toHumanReadableExpansionTimestamp(valueSetEntity); String msg = myContext.getLocalizer().getMessage(TermReadSvcImpl.class, "validationPerformedAgainstPreExpansion", timingDescription); @@ -1753,7 +1710,7 @@ public class TermReadSvcImpl implements ITermReadSvc { .setCodeSystemVersion(concepts.get(0).getSystemVersion()) .setMessage(msg); } - + // Ok, we failed List outcome = myValueSetConceptDao.findByTermValueSetIdSystemOnly(Pageable.ofSize(1), valueSetEntity.getId(), theSystem); String append; @@ -1813,18 +1770,6 @@ public class TermReadSvcImpl implements ITermReadSvc { } } - private CodeSystem.ConceptDefinitionComponent findCode(List theConcepts, String theCode) { - for (CodeSystem.ConceptDefinitionComponent next : theConcepts) { - if (theCode.equals(next.getCode())) { - return next; - } - CodeSystem.ConceptDefinitionComponent val = findCode(next.getConcept(), theCode); - if (val != null) { - return val; - } - } - return null; - } @Override public Optional findCode(String theCodeSystem, String theCode) { @@ -1909,7 +1854,7 @@ public class TermReadSvcImpl implements ITermReadSvc { StopWatch stopwatch = new StopWatch(); Optional concept = fetchLoadedCode(theCodeSystemResourcePid, theCode); - if (!concept.isPresent()) { + if (concept.isEmpty()) { return Collections.emptySet(); } @@ -1941,7 +1886,7 @@ public class TermReadSvcImpl implements ITermReadSvc { Stopwatch stopwatch = Stopwatch.createStarted(); Optional concept = fetchLoadedCode(theCodeSystemResourcePid, theCode); - if (!concept.isPresent()) { + if (concept.isEmpty()) { return Collections.emptySet(); } @@ -2003,7 +1948,7 @@ public class TermReadSvcImpl implements ITermReadSvc { StopWatch sw = new StopWatch(); TermValueSet valueSetToExpand = txTemplate.execute(t -> { Optional optionalTermValueSet = getNextTermValueSetNotExpanded(); - if (!optionalTermValueSet.isPresent()) { + if (optionalTermValueSet.isEmpty()) { return null; } @@ -2071,9 +2016,9 @@ public class TermReadSvcImpl implements ITermReadSvc { boolean haveCodeableConcept = codeableConcept != null && codeableConcept.getCoding().size() > 0; Coding canonicalCodingToValidate = myVersionCanonicalizer.codingToCanonical((IBaseCoding) theCodingToValidate); - boolean haveCoding = canonicalCodingToValidate != null && canonicalCodingToValidate.isEmpty() == false; + boolean haveCoding = canonicalCodingToValidate != null && !canonicalCodingToValidate.isEmpty(); - boolean haveCode = theCodeToValidate != null && theCodeToValidate.isEmpty() == false; + boolean haveCode = theCodeToValidate != null && !theCodeToValidate.isEmpty(); if (!haveCodeableConcept && !haveCoding && !haveCode) { throw new InvalidRequestException(Msg.code(899) + "No code, coding, or codeableConcept provided to validate"); @@ -2183,7 +2128,7 @@ public class TermReadSvcImpl implements ITermReadSvc { } else { optionalExistingTermValueSetByUrl = myTermValueSetDao.findTermValueSetByUrlAndNullVersion(url); } - if (!optionalExistingTermValueSetByUrl.isPresent()) { + if (optionalExistingTermValueSetByUrl.isEmpty()) { myTermValueSetDao.save(termValueSet); @@ -2557,9 +2502,9 @@ public class TermReadSvcImpl implements ITermReadSvc { boolean haveCodeableConcept = codeableConcept != null && codeableConcept.getCoding().size() > 0; Coding coding = myVersionCanonicalizer.codingToCanonical((IBaseCoding) theCoding); - boolean haveCoding = coding != null && coding.isEmpty() == false; + boolean haveCoding = coding != null && !coding.isEmpty(); - boolean haveCode = theCode != null && theCode.isEmpty() == false; + boolean haveCode = theCode != null && !theCode.isEmpty(); if (!haveCodeableConcept && !haveCoding && !haveCode) { throw new InvalidRequestException(Msg.code(906) + "No code, coding, or codeableConcept provided to validate."); @@ -2622,7 +2567,7 @@ public class TermReadSvcImpl implements ITermReadSvc { public Optional findCurrentTermValueSet(String theUrl) { if (TermReadSvcUtil.isLoincUnversionedValueSet(theUrl)) { Optional vsIdOpt = TermReadSvcUtil.getValueSetId(theUrl); - if (!vsIdOpt.isPresent()) { + if (vsIdOpt.isEmpty()) { return Optional.empty(); } @@ -2763,7 +2708,7 @@ public class TermReadSvcImpl implements ITermReadSvc { IConnectionPoolInfoProvider connectionPoolInfoProvider = new ConnectionPoolInfoProvider(myHibernatePropertiesProvider.getDataSource()); Optional maxConnectionsOpt = connectionPoolInfoProvider.getTotalConnectionSize(); - if ( ! maxConnectionsOpt.isPresent() ) { + if (maxConnectionsOpt.isEmpty()) { return DEFAULT_MASS_INDEXER_OBJECT_LOADING_THREADS; } @@ -2894,7 +2839,7 @@ public class TermReadSvcImpl implements ITermReadSvc { /** * Properties returned from method buildSearchScroll */ - private final class SearchProperties { + private static final class SearchProperties { private SearchScroll mySearchScroll; private Optional myExpansionStepOpt; private List myIncludeOrExcludeCodes; diff --git a/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtil.java b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtil.java new file mode 100644 index 00000000000..9ae87b24435 --- /dev/null +++ b/hapi-fhir-jpaserver-base/src/main/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtil.java @@ -0,0 +1,56 @@ +package ca.uhn.fhir.jpa.util; + +/*- + * #%L + * HAPI FHIR Storage api + * %% + * Copyright (C) 2014 - 2022 Smile CDR, Inc. + * %% + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * #L% + */ + +import com.google.gson.JsonArray; +import com.google.gson.JsonObject; +import com.google.gson.JsonPrimitive; + +/** + * The whole purpose of his class is to ease construction of a non-trivial gson.JsonObject, + * which can't be done the easy way in this case (using a JSON string), because there are + * valid regex strings which break gson, as this: ".*\\^Donor$" + */ +public class RegexpGsonBuilderUtil { + + + private RegexpGsonBuilderUtil() { } + + /** + * Builds a json object as this sample: + * {"regexp":{" + thePropName + ":{"value":" + theValue + "}}} + */ + public static JsonObject toGson(String thePropName, String theValue) { + JsonObject valueJO = new JsonObject(); + valueJO.add("value", new JsonPrimitive(theValue)); + + JsonObject systemValueJO = new JsonObject(); + systemValueJO.add(thePropName, valueJO); + + JsonObject regexpJO = new JsonObject(); + regexpJO.add("regexp", systemValueJO); + + JsonArray a = new JsonArray(); + return regexpJO; + } + + +} diff --git a/hapi-fhir-jpaserver-base/src/test/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtilTest.java b/hapi-fhir-jpaserver-base/src/test/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtilTest.java new file mode 100644 index 00000000000..14dd3496fca --- /dev/null +++ b/hapi-fhir-jpaserver-base/src/test/java/ca/uhn/fhir/jpa/util/RegexpGsonBuilderUtilTest.java @@ -0,0 +1,57 @@ +package ca.uhn.fhir.jpa.util; + +import com.google.gson.Gson; +import com.google.gson.JsonObject; +import com.google.gson.JsonSyntaxException; +import com.google.gson.stream.MalformedJsonException; +import org.junit.jupiter.api.Test; + +import static ca.uhn.fhir.jpa.entity.TermConceptPropertyBinder.CONCEPT_PROPERTY_PREFIX_NAME; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +class RegexpGsonBuilderUtilTest { + + private static final String PROP_NAME = "myValueString"; + + // a valid regex string which breaks gson + private static final String GSON_FAILING_REGEX = ".*\\^Donor$"; + + + @Test + void testBuildUsingTestedClass() { + String propertyValue = "propAAA"; + String expectedGson = "{\"regexp\":{\"P:" + PROP_NAME + "\":{\"value\":\"" + propertyValue + "\"}}}"; + + assertEquals(expectedGson, RegexpGsonBuilderUtil.toGson( + CONCEPT_PROPERTY_PREFIX_NAME + PROP_NAME, propertyValue).toString()); + } + + /** + * This test demonstrates that valid regex strings break gson library. + * If one day this test fails would mean the library added support for this kind of strings + * so the tested class could be removed and the code using it just build a gson.JsonObject from a JSON string + */ + @Test + void testShowGsonLibFailing() { + String workingRegex = ".*[abc]?jk"; + String workingRegexQuery = "{'regexp':{'P:SYSTEM':{'value':'" + workingRegex + "'}}}"; + + JsonObject jsonObj = new Gson().fromJson(workingRegexQuery, JsonObject.class); + assertFalse(jsonObj.isJsonNull()); + + // same json structure fails with some valid regex strings + String failingRegexQuery = "{'regexp':{'P:SYSTEM':{'value':'" + GSON_FAILING_REGEX + "'}}}"; + + JsonSyntaxException thrown = assertThrows( + JsonSyntaxException.class, + () -> new Gson().fromJson(failingRegexQuery, JsonObject.class)); + + assertTrue(thrown.getCause() instanceof MalformedJsonException); + } + + + +} diff --git a/hapi-fhir-jpaserver-elastic-test-utilities/src/test/java/ca/uhn/fhir/jpa/term/ValueSetExpansionR4ElasticsearchIT.java b/hapi-fhir-jpaserver-elastic-test-utilities/src/test/java/ca/uhn/fhir/jpa/term/ValueSetExpansionR4ElasticsearchIT.java index e2f9a4c4dcb..ff0d7bdd406 100644 --- a/hapi-fhir-jpaserver-elastic-test-utilities/src/test/java/ca/uhn/fhir/jpa/term/ValueSetExpansionR4ElasticsearchIT.java +++ b/hapi-fhir-jpaserver-elastic-test-utilities/src/test/java/ca/uhn/fhir/jpa/term/ValueSetExpansionR4ElasticsearchIT.java @@ -36,7 +36,6 @@ import org.hl7.fhir.r4.model.Enumerations; import org.hl7.fhir.r4.model.ValueSet; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeEach; -import org.junit.jupiter.api.Disabled; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; import org.mockito.Answers; @@ -287,7 +286,6 @@ public class ValueSetExpansionR4ElasticsearchIT extends BaseJpaTest { * Reproduced: https://github.com/hapifhir/hapi-fhir/issues/3992 */ @Test - @Disabled("until bug gets fixed") public void testExpandValueSetWithMoreThanElasticDefaultNestedObjectCount() { CodeSystem codeSystem = new CodeSystem(); codeSystem.setUrl(CS_URL); @@ -301,8 +299,13 @@ public class ValueSetExpansionR4ElasticsearchIT extends BaseJpaTest { codeSystemVersion.setResource(csResource); TermConcept tc = new TermConcept(codeSystemVersion, "test-code-1"); - // need to be more than elastic [index.mapping.nested_objects.limit] index level setting (default = 10_000) - addTerConceptProperties(tc, 10_100); + +// need to be more than elastic [index.mapping.nested_objects.limit] index level setting (default = 10_000) +// however, the new mapping (not nested, multivalued) groups properties by key, and there can't be more than 1000 +// properties (keys), so we test here with a number of properties > 10_000 but grouped in max 200 property keys +// (current loinc version (2.73) has 82 property keys maximum in a TermConcept) + addTermConceptProperties(tc, 10_100, 200); + codeSystemVersion.getConcepts().add(tc); ValueSet valueSet = new ValueSet(); @@ -320,10 +323,21 @@ public class ValueSetExpansionR4ElasticsearchIT extends BaseJpaTest { } - private void addTerConceptProperties(TermConcept theTermConcept, int theCount) { - for (int i = 0; i < theCount; i++) { - String suff = String.format("%05d", i); - theTermConcept.addPropertyString("prop-" + suff, "value-" + suff); + private void addTermConceptProperties(TermConcept theTermConcept, int thePropertiesCount, int thePropertyKeysCount) { + int valuesPerPropKey = thePropertiesCount / thePropertyKeysCount; + + int propsCreated = 0; + while(propsCreated < thePropertiesCount) { + + int propKeysCreated = 0; + while(propKeysCreated < thePropertyKeysCount && propsCreated < thePropertiesCount) { + String propKey = String.format("%05d", propKeysCreated); + String propSeq = String.format("%05d", propsCreated); + theTermConcept.addPropertyString("prop-key-" + propKey, "value-" + propSeq); + + propKeysCreated++; + propsCreated++; + } } } diff --git a/hapi-fhir-jpaserver-test-utilities/src/main/java/ca/uhn/fhir/jpa/term/AbstractValueSetHSearchExpansionR4Test.java b/hapi-fhir-jpaserver-test-utilities/src/main/java/ca/uhn/fhir/jpa/term/AbstractValueSetHSearchExpansionR4Test.java index fcd8e882449..17d7d0bf9a5 100644 --- a/hapi-fhir-jpaserver-test-utilities/src/main/java/ca/uhn/fhir/jpa/term/AbstractValueSetHSearchExpansionR4Test.java +++ b/hapi-fhir-jpaserver-test-utilities/src/main/java/ca/uhn/fhir/jpa/term/AbstractValueSetHSearchExpansionR4Test.java @@ -108,7 +108,7 @@ import static org.mockito.Mockito.when; */ //@ExtendWith(SpringExtension.class) //@ContextConfiguration(classes = {TestR4Config.class, TestHSearchAddInConfig.DefaultLuceneHeap.class}) -//@ContextConfiguration(classes = {TestR4Config.class, TestHSearchAddInConfig.DefaultLuceneHeap.class}) +//@ContextConfiguration(classes = {TestR4Config.class, TestHSearchAddInConfig.Elasticsearch.class}) public abstract class AbstractValueSetHSearchExpansionR4Test extends BaseJpaTest { private static final Logger ourLog = LoggerFactory.getLogger(AbstractValueSetHSearchExpansionR4Test.class); diff --git a/hapi-fhir-storage/src/main/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtil.java b/hapi-fhir-storage/src/main/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtil.java deleted file mode 100644 index 3106e49eee8..00000000000 --- a/hapi-fhir-storage/src/main/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtil.java +++ /dev/null @@ -1,94 +0,0 @@ -package ca.uhn.fhir.jpa.search; - -/*- - * #%L - * HAPI FHIR Storage api - * %% - * Copyright (C) 2014 - 2022 Smile CDR, Inc. - * %% - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - * #L% - */ - -import com.google.gson.JsonArray; -import com.google.gson.JsonObject; - -/** - * The whole purpose of his class is to ease construction of a non-trivial gson.JsonObject, - * which can't be done the easy way in this case (using a JSON string), because there are - * valid regex strings which break gson, as this: ".*\\^Donor$" - */ -public class ElasticsearchNestedQueryBuilderUtil { - - private final String myNestedObjName; - private final String myNestedKeyPropName; - private final String myNestedKeyPropValue; - private final String myNestedValuePropName; - private final String myNestedValuePropValue; - - private final String myNestedPropertyKeyPath; - private final String myNestedPropertyValuePath; - - private JsonObject builtJsonObj = new JsonObject(); - - public ElasticsearchNestedQueryBuilderUtil(String theNestedObjName, String theNestedKeyPropName, - String theNestedKeyPropValue, String theNestedValuePropName, String theNestedValuePropValue) { - - myNestedObjName = theNestedObjName ; - myNestedKeyPropName = theNestedKeyPropName; - myNestedKeyPropValue = theNestedKeyPropValue; - myNestedValuePropName = theNestedValuePropName; - myNestedValuePropValue = theNestedValuePropValue; - - myNestedPropertyKeyPath = myNestedObjName + "." + myNestedKeyPropName; - myNestedPropertyValuePath = myNestedObjName + "." + myNestedValuePropName; - - buildJsonObj(); - } - - - private void buildJsonObj() { - JsonObject matchPropJO = new JsonObject(); - matchPropJO.addProperty(myNestedObjName + "." + myNestedKeyPropName, myNestedKeyPropValue); - JsonObject matchJO = new JsonObject(); - matchJO.add("match", matchPropJO); - - JsonObject regexpPropJO = new JsonObject(); - regexpPropJO.addProperty(myNestedObjName + "." + myNestedValuePropName, myNestedValuePropValue); - JsonObject regexpJO = new JsonObject(); - regexpJO.add("regexp", regexpPropJO); - - JsonArray mustPropJA = new JsonArray(); - mustPropJA.add(matchJO); - mustPropJA.add(regexpJO); - - JsonObject mustPropJO = new JsonObject(); - mustPropJO.add("must", mustPropJA); - - JsonObject boolJO = new JsonObject(); - boolJO.add("bool", mustPropJO); - - JsonObject nestedJO = new JsonObject(); - nestedJO.addProperty("path", myNestedObjName); - nestedJO.add("query", boolJO); - - builtJsonObj.add("nested", nestedJO); - } - - public JsonObject toGson() { return builtJsonObj; } - - public String getNestedPropertyKeyPath() { return myNestedPropertyKeyPath; } - - public String getNestedPropertyValuePath() { return myNestedPropertyValuePath; } - -} diff --git a/hapi-fhir-storage/src/test/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtilTest.java b/hapi-fhir-storage/src/test/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtilTest.java deleted file mode 100644 index d95ef6f099b..00000000000 --- a/hapi-fhir-storage/src/test/java/ca/uhn/fhir/jpa/search/ElasticsearchNestedQueryBuilderUtilTest.java +++ /dev/null @@ -1,87 +0,0 @@ -package ca.uhn.fhir.jpa.search; - -import com.google.gson.Gson; -import com.google.gson.JsonArray; -import com.google.gson.JsonObject; -import com.google.gson.JsonSyntaxException; -import com.google.gson.stream.MalformedJsonException; -import org.junit.jupiter.api.Test; - -import static org.junit.jupiter.api.Assertions.assertEquals; -import static org.junit.jupiter.api.Assertions.assertFalse; -import static org.junit.jupiter.api.Assertions.assertThrows; -import static org.junit.jupiter.api.Assertions.assertTrue; - -class ElasticsearchNestedQueryBuilderUtilTest { - - private static final String NESTED_KEY_PROP_NAME = "myKey"; - private static final String NESTED_VALUE_PROP_NAME = "myValueString"; - private static final String NESTED_OBJECT_NAME = "myProperties"; - - // a valid regex string which breaks gson - private static final String GSON_FAILING_REGEX = ".*\\^Donor$"; - - - private ElasticsearchNestedQueryBuilderUtil testedUtil; - - - @Test - void testBuildUsingTestedClass() { - String propertyValue = "propAAA"; - - testedUtil = new ElasticsearchNestedQueryBuilderUtil( - NESTED_OBJECT_NAME, NESTED_KEY_PROP_NAME, propertyValue, NESTED_VALUE_PROP_NAME, GSON_FAILING_REGEX); - - assertFalse(testedUtil.toGson().isJsonNull()); - - JsonObject generatedJson = testedUtil.toGson(); - JsonObject nestedJO = generatedJson.getAsJsonObject("nested"); - assertEquals(NESTED_OBJECT_NAME, nestedJO.get("path").getAsString()); - - JsonObject queryJO = nestedJO.getAsJsonObject("query"); - JsonObject boolJO = queryJO.getAsJsonObject("bool"); - JsonArray mustJA = boolJO.getAsJsonArray("must"); - - JsonObject matchPropKeyJE = (JsonObject) mustJA.get(0); - JsonObject propKeyJO = matchPropKeyJE.getAsJsonObject("match"); - assertEquals(propertyValue, propKeyJO.get(testedUtil.getNestedPropertyKeyPath()).getAsString()); - - JsonObject regexpPropValueJE = (JsonObject) mustJA.get(1); - JsonObject propValueJO = regexpPropValueJE.getAsJsonObject("regexp"); - assertEquals(GSON_FAILING_REGEX, propValueJO.get(testedUtil.getNestedPropertyValuePath()).getAsString()); - } - - - /** - * This test demonstrates that valid regex strings break gson library. - * If one day this test fails would mean the library added support for this kind of strings - * so the tested class could be removed and the code using it just build a gson.JsonObject from a JSON string - */ - @Test - void testShowGsonLibFailing() { - String workingRegex = ".*[abc]?jk"; - String workingNestedQuery = - "{'nested': { 'path': 'myProperties', 'query': { 'bool': { 'must': [" + - "{'match': {'myProperties.myKey': 'propAAA' }}," + - "{'regexp': {'myProperties.myValueString': '" + workingRegex + "'}}" + - "]}}}}"; - - JsonObject jsonObj = new Gson().fromJson(workingNestedQuery, JsonObject.class); - assertFalse(jsonObj.isJsonNull()); - - // same json structure fails with some valid regex strings - String nestedQuery = - "{'nested': { 'path': 'myProperties', 'query': { 'bool': { 'must': [" + - "{'match': {'myProperties.myKey': 'propAAA' }}," + - "{'regexp': {'myProperties.myValueString': '" + GSON_FAILING_REGEX + "'}}" + - "]}}}}"; - -// MalformedJsonException thrown = assertThrows( - JsonSyntaxException thrown = assertThrows( - JsonSyntaxException.class, - () -> new Gson().fromJson(nestedQuery, JsonObject.class)); - - assertTrue(thrown.getCause() instanceof MalformedJsonException); - } - -}