001/*- 002 * #%L 003 * HAPI FHIR - Master Data Management 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.mdm.svc; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.interceptor.model.RequestPartitionId; 024import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc; 025import ca.uhn.fhir.jpa.searchparam.SearchParameterMap; 026import ca.uhn.fhir.mdm.api.IMdmLinkExpandSvc; 027import ca.uhn.fhir.mdm.log.Logs; 028import ca.uhn.fhir.model.api.IQueryParameterType; 029import ca.uhn.fhir.rest.api.server.RequestDetails; 030import ca.uhn.fhir.rest.param.BaseParam; 031import ca.uhn.fhir.rest.param.ReferenceParam; 032import ca.uhn.fhir.rest.param.TokenParam; 033import jakarta.annotation.Nonnull; 034import jakarta.annotation.Nullable; 035import org.hl7.fhir.instance.model.api.IAnyResource; 036import org.hl7.fhir.instance.model.api.IIdType; 037import org.slf4j.Logger; 038import org.springframework.beans.factory.annotation.Autowired; 039 040import java.util.ArrayList; 041import java.util.List; 042import java.util.Map; 043import java.util.Objects; 044import java.util.Set; 045 046import static org.apache.commons.lang3.StringUtils.isBlank; 047 048public class MdmSearchExpansionSvc { 049 private static final String EXPANSION_RESULTS = MdmSearchExpansionSvc.class.getName() + "_EXPANSION_RESULTS"; 050 private static final String RESOURCE_NAME = MdmSearchExpansionSvc.class.getName() + "_RESOURCE_NAME"; 051 private static final String QUERY_STRING = MdmSearchExpansionSvc.class.getName() + "_QUERY_STRING"; 052 private static final Logger ourLog = Logs.getMdmTroubleshootingLog(); 053 054 @Autowired 055 private FhirContext myFhirContext; 056 057 @Autowired 058 private IRequestPartitionHelperSvc myRequestPartitionHelperSvc; 059 060 @Autowired 061 private IMdmLinkExpandSvc myMdmLinkExpandSvc; 062 063 /** 064 * This method looks through all the reference parameters within a {@link SearchParameterMap} 065 * and performs MDM expansion. This means looking for any subject/patient parameters, and 066 * expanding them to include any linked and golden resource patients. So, for example, a 067 * search for <code>Encounter?subject=Patient/1</code> might be modified to be a search 068 * for <code>Encounter?subject=Patient/1,Patient/999</code> if 999 is linked to 1 by MDM. 069 * <p> 070 * This is an internal MDM service and its API is subject to change. Use with caution! 071 * </p> 072 * 073 * @param theRequestDetails The incoming request details 074 * @param theSearchParameterMap The parameter map to modify 075 * @param theParamTester Determines which parameters should be expanded 076 * @return Returns the results of the expansion, which are also stored in the {@link RequestDetails} userdata map, so this service will only be invoked a maximum of once per request. 077 * @since 8.0.0 078 */ 079 public MdmSearchExpansionResults expandSearchAndStoreInRequestDetails( 080 String theResourceName, 081 @Nullable RequestDetails theRequestDetails, 082 @Nonnull SearchParameterMap theSearchParameterMap, 083 IParamTester theParamTester) { 084 085 if (theRequestDetails == null) { 086 return null; 087 } 088 089 // Try to detect if the RequestDetails is being reused across multiple different queries, which 090 // can happen during CQL measure evaluation 091 { 092 String resourceName = theRequestDetails.getResourceName(); 093 String queryString = theSearchParameterMap.toNormalizedQueryString(myFhirContext); 094 if (!Objects.equals(resourceName, theRequestDetails.getUserData().get(RESOURCE_NAME)) 095 || !Objects.equals( 096 queryString, theRequestDetails.getUserData().get(QUERY_STRING))) { 097 theRequestDetails.getUserData().remove(EXPANSION_RESULTS); 098 } 099 } 100 101 MdmSearchExpansionResults expansionResults = getCachedExpansionResults(theRequestDetails); 102 if (expansionResults != null) { 103 return expansionResults; 104 } 105 106 expansionResults = new MdmSearchExpansionResults(); 107 108 final RequestPartitionId requestPartitionId = 109 myRequestPartitionHelperSvc.determineReadPartitionForRequestForSearchType( 110 theRequestDetails, theRequestDetails.getResourceName(), theSearchParameterMap); 111 112 for (Map.Entry<String, List<List<IQueryParameterType>>> set : theSearchParameterMap.entrySet()) { 113 String paramName = set.getKey(); 114 List<List<IQueryParameterType>> andList = set.getValue(); 115 for (List<IQueryParameterType> orList : andList) { 116 // here we will know if it's an _id param or not 117 // from theSearchParameterMap.keySet() 118 expandAnyReferenceParameters( 119 requestPartitionId, theResourceName, paramName, orList, theParamTester, expansionResults); 120 } 121 } 122 123 theRequestDetails.getUserData().put(EXPANSION_RESULTS, expansionResults); 124 125 /* 126 * Note: Do this at the end so that the query string reflects the post-translated 127 * query string 128 */ 129 String queryString = theSearchParameterMap.toNormalizedQueryString(myFhirContext); 130 theRequestDetails.getUserData().put(RESOURCE_NAME, theResourceName); 131 theRequestDetails.getUserData().put(QUERY_STRING, queryString); 132 133 return expansionResults; 134 } 135 136 private void expandAnyReferenceParameters( 137 RequestPartitionId theRequestPartitionId, 138 String theResourceName, 139 String theParamName, 140 List<IQueryParameterType> orList, 141 IParamTester theParamTester, 142 MdmSearchExpansionResults theResultsToPopulate) { 143 144 List<IQueryParameterType> toRemove = new ArrayList<>(); 145 List<IQueryParameterType> toAdd = new ArrayList<>(); 146 for (IQueryParameterType iQueryParameterType : orList) { 147 if (iQueryParameterType instanceof ReferenceParam) { 148 ReferenceParam refParam = (ReferenceParam) iQueryParameterType; 149 if (theParamTester.shouldExpand(theParamName, refParam)) { 150 ourLog.debug("Found a reference parameter to expand: {}", refParam); 151 // First, attempt to expand as a source resource. 152 IIdType sourceId = newId(refParam.getValue()); 153 Set<String> expandedResourceIds = 154 myMdmLinkExpandSvc.expandMdmBySourceResourceId(theRequestPartitionId, sourceId); 155 156 // If we failed, attempt to expand as a golden resource 157 if (expandedResourceIds.isEmpty()) { 158 expandedResourceIds = 159 myMdmLinkExpandSvc.expandMdmByGoldenResourceId(theRequestPartitionId, sourceId); 160 } 161 162 // Rebuild the search param list. 163 if (!expandedResourceIds.isEmpty()) { 164 ourLog.debug("Parameter has been expanded to: {}", String.join(", ", expandedResourceIds)); 165 toRemove.add(refParam); 166 for (String resourceId : expandedResourceIds) { 167 IIdType nextReference = 168 newId(addResourceTypeIfNecessary(refParam.getResourceType(), resourceId)); 169 toAdd.add(new ReferenceParam(nextReference)); 170 theResultsToPopulate.addExpandedId(sourceId, nextReference); 171 } 172 } 173 } 174 } else if (theParamName.equalsIgnoreCase(IAnyResource.SP_RES_ID)) { 175 expandIdParameter( 176 theRequestPartitionId, 177 iQueryParameterType, 178 toAdd, 179 toRemove, 180 theParamTester, 181 theResourceName, 182 theResultsToPopulate); 183 } 184 } 185 186 orList.removeAll(toRemove); 187 orList.addAll(toAdd); 188 } 189 190 private IIdType newId(String value) { 191 return myFhirContext.getVersion().newIdType(value); 192 } 193 194 private String addResourceTypeIfNecessary(String theResourceType, String theResourceId) { 195 if (theResourceId.contains("/") || isBlank(theResourceType)) { 196 return theResourceId; 197 } else { 198 return theResourceType + "/" + theResourceId; 199 } 200 } 201 202 /** 203 * Expands out the provided _id parameter into all the various 204 * ids of linked resources. 205 */ 206 private void expandIdParameter( 207 RequestPartitionId theRequestPartitionId, 208 IQueryParameterType theIdParameter, 209 List<IQueryParameterType> theAddList, 210 List<IQueryParameterType> theRemoveList, 211 IParamTester theParamTester, 212 String theResourceName, 213 MdmSearchExpansionResults theResultsToPopulate) { 214 // id parameters can either be StringParam (for $everything operation) 215 // or TokenParam (for searches) 216 // either case, we want to expand it out and grab all related resources 217 IIdType id; 218 Creator<? extends IQueryParameterType> creator; 219 boolean mdmExpand = false; 220 if (theIdParameter instanceof TokenParam) { 221 TokenParam param = (TokenParam) theIdParameter; 222 mdmExpand = theParamTester.shouldExpand(IAnyResource.SP_RES_ID, param); 223 String value = param.getValue(); 224 value = addResourceTypeIfNecessary(theResourceName, value); 225 id = newId(value); 226 creator = TokenParam::new; 227 } else { 228 creator = null; 229 id = null; 230 } 231 232 if (id == null) { 233 // in case the _id parameter type is different from the above 234 ourLog.warn( 235 "_id parameter of incorrect type. Expected StringParam or TokenParam, but got {}. No expansion will be done!", 236 theIdParameter.getClass().getSimpleName()); 237 } else if (mdmExpand) { 238 ourLog.debug("_id parameter must be expanded out from: {}", id.getValue()); 239 240 Set<String> expandedResourceIds = myMdmLinkExpandSvc.expandMdmBySourceResourceId(theRequestPartitionId, id); 241 242 if (expandedResourceIds.isEmpty()) { 243 expandedResourceIds = myMdmLinkExpandSvc.expandMdmByGoldenResourceId(theRequestPartitionId, id); 244 } 245 246 // Rebuild 247 if (!expandedResourceIds.isEmpty()) { 248 ourLog.debug("_id parameter has been expanded to: {}", expandedResourceIds); 249 250 // remove the original 251 theRemoveList.add(theIdParameter); 252 253 // add in all the linked values 254 expandedResourceIds.stream().map(creator::create).forEach(theAddList::add); 255 256 for (String expandedId : expandedResourceIds) { 257 theResultsToPopulate.addExpandedId( 258 id, newId(addResourceTypeIfNecessary(theResourceName, expandedId))); 259 } 260 } 261 } 262 // else - no expansion required 263 } 264 265 // A simple interface to turn ids into some form of IQueryParameterTypes 266 private interface Creator<T extends IQueryParameterType> { 267 T create(String id); 268 } 269 270 @FunctionalInterface 271 public interface IParamTester { 272 273 boolean shouldExpand(String theParamName, BaseParam theParam); 274 } 275 276 @Nullable 277 public static MdmSearchExpansionResults getCachedExpansionResults(@Nonnull RequestDetails theRequestDetails) { 278 MdmSearchExpansionResults expansionResults = 279 (MdmSearchExpansionResults) theRequestDetails.getUserData().get(EXPANSION_RESULTS); 280 return expansionResults; 281 } 282}