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.RuntimeSearchParam; 023import ca.uhn.fhir.i18n.Msg; 024import ca.uhn.fhir.interceptor.model.RequestPartitionId; 025import ca.uhn.fhir.jpa.search.builder.sql.SearchQueryBuilder; 026import ca.uhn.fhir.jpa.searchparam.SearchParameterMap; 027import ca.uhn.fhir.jpa.util.CoordCalculator; 028import ca.uhn.fhir.model.api.IQueryParameterType; 029import ca.uhn.fhir.model.dstu2.resource.Location; 030import com.healthmarketscience.sqlbuilder.BinaryCondition; 031import com.healthmarketscience.sqlbuilder.ComboCondition; 032import com.healthmarketscience.sqlbuilder.Condition; 033import com.healthmarketscience.sqlbuilder.dbspec.basic.DbColumn; 034import org.hibernate.search.engine.spatial.GeoBoundingBox; 035 036public class CoordsPredicateBuilder extends BaseSearchParamPredicateBuilder { 037 038 private final DbColumn myColumnLatitude; 039 private final DbColumn myColumnLongitude; 040 041 /** 042 * Constructor 043 */ 044 public CoordsPredicateBuilder(SearchQueryBuilder theSearchSqlBuilder) { 045 super(theSearchSqlBuilder, theSearchSqlBuilder.addTable("HFJ_SPIDX_COORDS")); 046 047 myColumnLatitude = getTable().addColumn("SP_LATITUDE"); 048 myColumnLongitude = getTable().addColumn("SP_LONGITUDE"); 049 } 050 051 public DbColumn getColumnLatitude() { 052 return myColumnLatitude; 053 } 054 055 public DbColumn getColumnLongitude() { 056 return myColumnLongitude; 057 } 058 059 public Condition createPredicateCoords( 060 SearchParameterMap theParams, 061 IQueryParameterType theParam, 062 String theResourceName, 063 RuntimeSearchParam theSearchParam, 064 CoordsPredicateBuilder theFrom, 065 RequestPartitionId theRequestPartitionId) { 066 067 ParsedLocationParam params = ParsedLocationParam.from(theParams, theParam); 068 double distanceKm = params.getDistanceKm(); 069 double latitudeValue = params.getLatitudeValue(); 070 double longitudeValue = params.getLongitudeValue(); 071 072 Condition latitudePredicate; 073 Condition longitudePredicate; 074 if (distanceKm == 0.0) { 075 latitudePredicate = theFrom.createPredicateLatitudeExact(latitudeValue); 076 longitudePredicate = theFrom.createPredicateLongitudeExact(longitudeValue); 077 } else if (distanceKm < 0.0) { 078 throw new IllegalArgumentException(Msg.code(1233) + "Invalid " + Location.SP_NEAR_DISTANCE + " parameter '" 079 + distanceKm + "' must be >= 0.0"); 080 } else if (distanceKm > CoordCalculator.MAX_SUPPORTED_DISTANCE_KM) { 081 throw new IllegalArgumentException(Msg.code(1234) + "Invalid " + Location.SP_NEAR_DISTANCE + " parameter '" 082 + distanceKm + "' must be <= " + CoordCalculator.MAX_SUPPORTED_DISTANCE_KM); 083 } else { 084 GeoBoundingBox box = CoordCalculator.getBox(latitudeValue, longitudeValue, distanceKm); 085 latitudePredicate = theFrom.createLatitudePredicateFromBox(box); 086 longitudePredicate = theFrom.createLongitudePredicateFromBox(box); 087 } 088 ComboCondition singleCode = ComboCondition.and(latitudePredicate, longitudePredicate); 089 return combineWithHashIdentityPredicate(theResourceName, theSearchParam.getName(), singleCode); 090 } 091 092 public Condition createPredicateLatitudeExact(double theLatitudeValue) { 093 return BinaryCondition.equalTo(myColumnLatitude, generatePlaceholder(theLatitudeValue)); 094 } 095 096 public Condition createPredicateLongitudeExact(double theLongitudeValue) { 097 return BinaryCondition.equalTo(myColumnLongitude, generatePlaceholder(theLongitudeValue)); 098 } 099 100 public Condition createLatitudePredicateFromBox(GeoBoundingBox theBox) { 101 return ComboCondition.and( 102 BinaryCondition.greaterThanOrEq( 103 myColumnLatitude, 104 generatePlaceholder(theBox.bottomRight().latitude())), 105 BinaryCondition.lessThanOrEq( 106 myColumnLatitude, generatePlaceholder(theBox.topLeft().latitude()))); 107 } 108 109 public Condition createLongitudePredicateFromBox(GeoBoundingBox theBox) { 110 if (theBox.bottomRight().longitude() < theBox.topLeft().longitude()) { 111 return ComboCondition.or( 112 BinaryCondition.greaterThanOrEq( 113 myColumnLongitude, 114 generatePlaceholder(theBox.bottomRight().longitude())), 115 BinaryCondition.lessThanOrEq( 116 myColumnLongitude, 117 generatePlaceholder(theBox.topLeft().longitude()))); 118 } else { 119 return ComboCondition.and( 120 BinaryCondition.greaterThanOrEq( 121 myColumnLongitude, 122 generatePlaceholder(theBox.topLeft().longitude())), 123 BinaryCondition.lessThanOrEq( 124 myColumnLongitude, 125 generatePlaceholder(theBox.bottomRight().longitude()))); 126 } 127 } 128}