001/*-
002 * #%L
003 * HAPI FHIR JPA Server
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 *      http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.jpa.search.builder.predicate;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.interceptor.model.RequestPartitionId;
025import ca.uhn.fhir.jpa.dao.predicate.SearchFilterParser;
026import ca.uhn.fhir.jpa.dao.predicate.SearchFuzzUtil;
027import ca.uhn.fhir.jpa.search.builder.sql.SearchQueryBuilder;
028import ca.uhn.fhir.model.api.IQueryParameterType;
029import ca.uhn.fhir.rest.param.ParamPrefixEnum;
030import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
031import com.healthmarketscience.sqlbuilder.BinaryCondition;
032import com.healthmarketscience.sqlbuilder.ComboCondition;
033import com.healthmarketscience.sqlbuilder.Condition;
034import com.healthmarketscience.sqlbuilder.dbspec.basic.DbColumn;
035import org.slf4j.Logger;
036import org.slf4j.LoggerFactory;
037import org.springframework.beans.factory.annotation.Autowired;
038
039import java.math.BigDecimal;
040import java.math.MathContext;
041
042import static org.apache.commons.lang3.ObjectUtils.defaultIfNull;
043
044public class NumberPredicateBuilder extends BaseSearchParamPredicateBuilder {
045
046        private static final Logger ourLog = LoggerFactory.getLogger(NumberPredicateBuilder.class);
047        private final DbColumn myColumnValue;
048
049        @Autowired
050        private FhirContext myFhirContext;
051
052        /**
053         * Constructor
054         */
055        public NumberPredicateBuilder(SearchQueryBuilder theSearchSqlBuilder) {
056                super(theSearchSqlBuilder, theSearchSqlBuilder.addTable("HFJ_SPIDX_NUMBER"));
057
058                myColumnValue = getTable().addColumn("SP_VALUE");
059        }
060
061        public Condition createPredicateNumeric(
062                        String theResourceName,
063                        String theParamName,
064                        SearchFilterParser.CompareOperation theOperation,
065                        BigDecimal theValue,
066                        RequestPartitionId theRequestPartitionId,
067                        IQueryParameterType theActualParam) {
068                Condition numericPredicate = createPredicateNumeric(
069                                this, theOperation, theValue, myColumnValue, "invalidNumberPrefix", myFhirContext, theActualParam);
070                return combineWithHashIdentityPredicate(theResourceName, theParamName, numericPredicate);
071        }
072
073        public DbColumn getColumnValue() {
074                return myColumnValue;
075        }
076
077        static Condition createPredicateNumeric(
078                        BaseSearchParamPredicateBuilder theIndexTable,
079                        SearchFilterParser.CompareOperation theOperation,
080                        BigDecimal theValue,
081                        DbColumn theColumn,
082                        String theInvalidValueKey,
083                        FhirContext theFhirContext,
084                        IQueryParameterType theActualParam) {
085                Condition num;
086
087                // Per discussions with Grahame Grieve and James Agnew on 11/13/19, modified logic for EQUAL and NOT_EQUAL
088                // operators below so as to
089                //   use exact value matching.  The "fuzz amount" matching is still used with the APPROXIMATE operator.
090                SearchFilterParser.CompareOperation operation =
091                                defaultIfNull(theOperation, SearchFilterParser.CompareOperation.eq);
092                switch (operation) {
093                        case gt:
094                                num = BinaryCondition.greaterThan(theColumn, theIndexTable.generatePlaceholder(theValue));
095                                break;
096                        case ge:
097                                num = BinaryCondition.greaterThanOrEq(theColumn, theIndexTable.generatePlaceholder(theValue));
098                                break;
099                        case lt:
100                                num = BinaryCondition.lessThan(theColumn, theIndexTable.generatePlaceholder(theValue));
101                                break;
102                        case le:
103                                num = BinaryCondition.lessThanOrEq(theColumn, theIndexTable.generatePlaceholder(theValue));
104                                break;
105                        case eq:
106                                num = BinaryCondition.equalTo(theColumn, theIndexTable.generatePlaceholder(theValue));
107                                break;
108                        case ne:
109                                num = BinaryCondition.notEqualTo(theColumn, theIndexTable.generatePlaceholder(theValue));
110                                break;
111                        case ap:
112                                BigDecimal mul = SearchFuzzUtil.calculateFuzzAmount(ParamPrefixEnum.APPROXIMATE, theValue);
113                                BigDecimal low = theValue.subtract(mul, MathContext.DECIMAL64);
114                                BigDecimal high = theValue.add(mul, MathContext.DECIMAL64);
115                                Condition lowPred = BinaryCondition.greaterThanOrEq(theColumn, theIndexTable.generatePlaceholder(low));
116                                Condition highPred = BinaryCondition.lessThanOrEq(theColumn, theIndexTable.generatePlaceholder(high));
117                                num = ComboCondition.and(lowPred, highPred);
118                                ourLog.trace("Searching for {} <= val <= {}", low, high);
119                                break;
120                        default:
121                                String paramValue = theActualParam.getValueAsQueryToken(theFhirContext);
122                                String msg = theIndexTable
123                                                .getFhirContext()
124                                                .getLocalizer()
125                                                .getMessage(NumberPredicateBuilder.class, theInvalidValueKey, operation, paramValue);
126                                throw new InvalidRequestException(Msg.code(1235) + msg);
127                }
128
129                return num;
130        }
131}