001/*- 002 * #%L 003 * HAPI FHIR Storage api 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.graphql; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.context.RuntimeResourceDefinition; 024import ca.uhn.fhir.context.RuntimeSearchParam; 025import ca.uhn.fhir.i18n.Msg; 026import ca.uhn.fhir.interceptor.model.RequestPartitionId; 027import ca.uhn.fhir.jpa.api.dao.DaoRegistry; 028import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao; 029import ca.uhn.fhir.jpa.api.svc.ISearchCoordinatorSvc; 030import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc; 031import ca.uhn.fhir.jpa.searchparam.SearchParameterMap; 032import ca.uhn.fhir.model.api.IQueryParameterOr; 033import ca.uhn.fhir.model.valueset.BundleTypeEnum; 034import ca.uhn.fhir.rest.api.BundleLinks; 035import ca.uhn.fhir.rest.api.CacheControlDirective; 036import ca.uhn.fhir.rest.api.Constants; 037import ca.uhn.fhir.rest.api.IVersionSpecificBundleFactory; 038import ca.uhn.fhir.rest.api.server.IBundleProvider; 039import ca.uhn.fhir.rest.api.server.RequestDetails; 040import ca.uhn.fhir.rest.param.DateOrListParam; 041import ca.uhn.fhir.rest.param.DateParam; 042import ca.uhn.fhir.rest.param.NumberOrListParam; 043import ca.uhn.fhir.rest.param.NumberParam; 044import ca.uhn.fhir.rest.param.QuantityOrListParam; 045import ca.uhn.fhir.rest.param.QuantityParam; 046import ca.uhn.fhir.rest.param.ReferenceOrListParam; 047import ca.uhn.fhir.rest.param.ReferenceParam; 048import ca.uhn.fhir.rest.param.SpecialOrListParam; 049import ca.uhn.fhir.rest.param.SpecialParam; 050import ca.uhn.fhir.rest.param.StringOrListParam; 051import ca.uhn.fhir.rest.param.StringParam; 052import ca.uhn.fhir.rest.param.TokenOrListParam; 053import ca.uhn.fhir.rest.param.TokenParam; 054import ca.uhn.fhir.rest.server.IPagingProvider; 055import ca.uhn.fhir.rest.server.RestfulServerUtils; 056import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 057import ca.uhn.fhir.rest.server.util.ISearchParamRegistry; 058import ca.uhn.fhir.rest.server.util.ResourceSearchParams; 059import org.apache.commons.lang3.Validate; 060import org.hl7.fhir.exceptions.FHIRException; 061import org.hl7.fhir.instance.model.api.IBaseBundle; 062import org.hl7.fhir.instance.model.api.IBaseReference; 063import org.hl7.fhir.instance.model.api.IBaseResource; 064import org.hl7.fhir.instance.model.api.IIdType; 065import org.hl7.fhir.utilities.graphql.Argument; 066import org.hl7.fhir.utilities.graphql.IGraphQLStorageServices; 067import org.hl7.fhir.utilities.graphql.Value; 068import org.springframework.beans.factory.annotation.Autowired; 069import org.springframework.transaction.annotation.Propagation; 070import org.springframework.transaction.annotation.Transactional; 071 072import java.util.List; 073import java.util.Optional; 074import java.util.Set; 075import java.util.TreeSet; 076import java.util.stream.Collectors; 077 078import static ca.uhn.fhir.rest.api.Constants.PARAM_COUNT; 079import static ca.uhn.fhir.rest.api.Constants.PARAM_FILTER; 080 081public class DaoRegistryGraphQLStorageServices implements IGraphQLStorageServices { 082 083 // the constant hasn't already been defined in org.hl7.fhir.core so we define it here 084 static final String SEARCH_ID_PARAM = "search-id"; 085 static final String SEARCH_OFFSET_PARAM = "search-offset"; 086 087 private static final int MAX_SEARCH_SIZE = 500; 088 089 @Autowired 090 private FhirContext myContext; 091 092 @Autowired 093 private DaoRegistry myDaoRegistry; 094 095 @Autowired 096 private ISearchParamRegistry mySearchParamRegistry; 097 098 @Autowired 099 protected ISearchCoordinatorSvc mySearchCoordinatorSvc; 100 101 @Autowired 102 private IRequestPartitionHelperSvc myPartitionHelperSvc; 103 104 @Autowired 105 private IPagingProvider myPagingProvider; 106 107 private IFhirResourceDao<? extends IBaseResource> getDao(String theResourceType) { 108 RuntimeResourceDefinition typeDef = myContext.getResourceDefinition(theResourceType); 109 return myDaoRegistry.getResourceDaoOrNull(typeDef.getImplementingClass()); 110 } 111 112 private String graphqlArgumentToSearchParam(String name) { 113 if (name.startsWith("_")) { 114 return name; 115 } else { 116 return name.replaceAll("_", "-"); 117 } 118 } 119 120 private String searchParamToGraphqlArgument(String name) { 121 return name.replaceAll("-", "_"); 122 } 123 124 private SearchParameterMap buildSearchParams(String theType, List<Argument> theSearchParams) { 125 List<Argument> resourceSearchParam = theSearchParams.stream() 126 .filter(it -> !PARAM_COUNT.equals(it.getName())) 127 .collect(Collectors.toList()); 128 129 FhirContext fhirContext = myContext; 130 RuntimeResourceDefinition typeDef = fhirContext.getResourceDefinition(theType); 131 132 SearchParameterMap params = new SearchParameterMap(); 133 ResourceSearchParams searchParams = mySearchParamRegistry.getRuntimeSearchParams( 134 typeDef.getName(), ISearchParamRegistry.SearchParamLookupContextEnum.SEARCH); 135 136 for (Argument nextArgument : resourceSearchParam) { 137 138 if (nextArgument.getName().equals(PARAM_FILTER)) { 139 String value = nextArgument.getValues().get(0).getValue(); 140 params.add(PARAM_FILTER, new StringParam(value)); 141 continue; 142 } 143 144 String searchParamName = graphqlArgumentToSearchParam(nextArgument.getName()); 145 RuntimeSearchParam searchParam = searchParams.get(searchParamName); 146 if (searchParam == null) { 147 Set<String> graphqlArguments = searchParams.getSearchParamNames().stream() 148 .map(this::searchParamToGraphqlArgument) 149 .collect(Collectors.toSet()); 150 String msg = myContext 151 .getLocalizer() 152 .getMessageSanitized( 153 DaoRegistryGraphQLStorageServices.class, 154 "invalidGraphqlArgument", 155 nextArgument.getName(), 156 new TreeSet<>(graphqlArguments)); 157 throw new InvalidRequestException(Msg.code(1275) + msg); 158 } 159 160 IQueryParameterOr<?> queryParam; 161 162 switch (searchParam.getParamType()) { 163 case NUMBER: 164 NumberOrListParam numberOrListParam = new NumberOrListParam(); 165 for (Value value : nextArgument.getValues()) { 166 numberOrListParam.addOr(new NumberParam(value.getValue())); 167 } 168 queryParam = numberOrListParam; 169 break; 170 case DATE: 171 DateOrListParam dateOrListParam = new DateOrListParam(); 172 for (Value value : nextArgument.getValues()) { 173 dateOrListParam.addOr(new DateParam(value.getValue())); 174 } 175 queryParam = dateOrListParam; 176 break; 177 case STRING: 178 StringOrListParam stringOrListParam = new StringOrListParam(); 179 for (Value value : nextArgument.getValues()) { 180 stringOrListParam.addOr(new StringParam(value.getValue())); 181 } 182 queryParam = stringOrListParam; 183 break; 184 case TOKEN: 185 TokenOrListParam tokenOrListParam = new TokenOrListParam(); 186 for (Value value : nextArgument.getValues()) { 187 TokenParam tokenParam = new TokenParam(); 188 tokenParam.setValueAsQueryToken(fhirContext, searchParamName, null, value.getValue()); 189 tokenOrListParam.addOr(tokenParam); 190 } 191 queryParam = tokenOrListParam; 192 break; 193 case REFERENCE: 194 ReferenceOrListParam referenceOrListParam = new ReferenceOrListParam(); 195 for (Value value : nextArgument.getValues()) { 196 referenceOrListParam.addOr(new ReferenceParam(value.getValue())); 197 } 198 queryParam = referenceOrListParam; 199 break; 200 case QUANTITY: 201 QuantityOrListParam quantityOrListParam = new QuantityOrListParam(); 202 for (Value value : nextArgument.getValues()) { 203 quantityOrListParam.addOr(new QuantityParam(value.getValue())); 204 } 205 queryParam = quantityOrListParam; 206 break; 207 case SPECIAL: 208 SpecialOrListParam specialOrListParam = new SpecialOrListParam(); 209 for (Value value : nextArgument.getValues()) { 210 specialOrListParam.addOr(new SpecialParam().setValue(value.getValue())); 211 } 212 queryParam = specialOrListParam; 213 break; 214 case COMPOSITE: 215 case URI: 216 case HAS: 217 default: 218 throw new InvalidRequestException(Msg.code(1276) 219 + String.format( 220 "%s parameters are not yet supported in GraphQL", searchParam.getParamType())); 221 } 222 223 params.add(searchParamName, queryParam); 224 } 225 226 return params; 227 } 228 229 @Transactional(propagation = Propagation.NEVER) 230 @Override 231 public void listResources( 232 Object theAppInfo, String theType, List<Argument> theSearchParams, List<IBaseResource> theMatches) 233 throws FHIRException { 234 SearchParameterMap params = buildSearchParams(theType, theSearchParams); 235 params.setLoadSynchronousUpTo(MAX_SEARCH_SIZE); 236 237 RequestDetails requestDetails = (RequestDetails) theAppInfo; 238 IBundleProvider response = getDao(theType).search(params, requestDetails); 239 Integer size = response.size(); 240 // We set size to null in SearchCoordinatorSvcImpl.executeQuery() if matching results exceeds count 241 // so don't throw here 242 if ((response.preferredPageSize() != null && size != null && response.preferredPageSize() < size) 243 || size == null) { 244 size = response.preferredPageSize(); 245 } 246 247 Validate.notNull(size, "size is null"); 248 theMatches.addAll(response.getResources(0, size)); 249 } 250 251 @Transactional(propagation = Propagation.REQUIRED) 252 @Override 253 public IBaseResource lookup(Object theAppInfo, String theType, String theId) throws FHIRException { 254 IIdType refId = myContext.getVersion().newIdType(); 255 refId.setValue(theType + "/" + theId); 256 return lookup(theAppInfo, refId); 257 } 258 259 private IBaseResource lookup(Object theAppInfo, IIdType theRefId) { 260 IFhirResourceDao<? extends IBaseResource> dao = getDao(theRefId.getResourceType()); 261 RequestDetails requestDetails = (RequestDetails) theAppInfo; 262 return dao.read(theRefId, requestDetails, false); 263 } 264 265 @Transactional(propagation = Propagation.REQUIRED) 266 @Override 267 public ReferenceResolution lookup(Object theAppInfo, IBaseResource theContext, IBaseReference theReference) 268 throws FHIRException { 269 IBaseResource outcome = lookup(theAppInfo, theReference.getReferenceElement()); 270 if (outcome == null) { 271 return null; 272 } 273 return new ReferenceResolution(theContext, outcome); 274 } 275 276 private Optional<String> getArgument(List<Argument> params, String name) { 277 return params.stream() 278 .filter(it -> name.equals(it.getName())) 279 .map(it -> it.getValues().get(0).getValue()) 280 .findAny(); 281 } 282 283 @Transactional(propagation = Propagation.NEVER) 284 @Override 285 public IBaseBundle search(Object theAppInfo, String theType, List<Argument> theSearchParams) throws FHIRException { 286 RequestDetails requestDetails = (RequestDetails) theAppInfo; 287 288 Optional<String> searchIdArgument = getArgument(theSearchParams, SEARCH_ID_PARAM); 289 Optional<String> searchOffsetArgument = getArgument(theSearchParams, SEARCH_OFFSET_PARAM); 290 291 String searchId; 292 int searchOffset; 293 int pageSize; 294 IBundleProvider response; 295 296 if (searchIdArgument.isPresent() && searchOffsetArgument.isPresent()) { 297 searchId = searchIdArgument.get(); 298 searchOffset = Integer.parseInt(searchOffsetArgument.get()); 299 300 response = Optional.ofNullable(myPagingProvider.retrieveResultList(requestDetails, searchId)) 301 .orElseThrow(() -> { 302 String msg = myContext 303 .getLocalizer() 304 .getMessageSanitized( 305 DaoRegistryGraphQLStorageServices.class, 306 "invalidGraphqlCursorArgument", 307 searchId); 308 return new InvalidRequestException(Msg.code(2076) + msg); 309 }); 310 311 pageSize = 312 Optional.ofNullable(response.preferredPageSize()).orElseGet(myPagingProvider::getDefaultPageSize); 313 } else { 314 pageSize = getArgument(theSearchParams, "_count") 315 .map(Integer::parseInt) 316 .orElseGet(myPagingProvider::getDefaultPageSize); 317 318 SearchParameterMap params = buildSearchParams(theType, theSearchParams); 319 params.setCount(pageSize); 320 321 CacheControlDirective cacheControlDirective = new CacheControlDirective(); 322 cacheControlDirective.parse(requestDetails.getHeaders(Constants.HEADER_CACHE_CONTROL)); 323 324 RequestPartitionId requestPartitionId = 325 myPartitionHelperSvc.determineReadPartitionForRequestForSearchType(requestDetails, theType, params); 326 response = mySearchCoordinatorSvc.registerSearch( 327 getDao(theType), params, theType, cacheControlDirective, requestDetails, requestPartitionId); 328 329 searchOffset = 0; 330 searchId = myPagingProvider.storeResultList(requestDetails, response); 331 } 332 333 // response.size() may return {@literal null}, in that case use pageSize 334 String serverBase = requestDetails.getFhirServerBase(); 335 Optional<Integer> numTotalResults = Optional.ofNullable(response.size()); 336 int numToReturn = numTotalResults 337 .map(integer -> Math.min(pageSize, integer - searchOffset)) 338 .orElse(pageSize); 339 340 BundleLinks links = new BundleLinks( 341 requestDetails.getServerBaseForRequest(), 342 null, 343 RestfulServerUtils.prettyPrintResponse(requestDetails.getServer(), requestDetails), 344 BundleTypeEnum.SEARCHSET); 345 346 // RestfulServerUtils.createLinkSelf not suitable here 347 String linkFormat = "%s/%s?_format=application/json&search-id=%s&search-offset=%d&_count=%d"; 348 349 String linkSelf = String.format(linkFormat, serverBase, theType, searchId, searchOffset, pageSize); 350 links.setSelf(linkSelf); 351 352 boolean hasNext = numTotalResults 353 .map(total -> (searchOffset + numToReturn) < total) 354 .orElse(true); 355 356 if (hasNext) { 357 String linkNext = 358 String.format(linkFormat, serverBase, theType, searchId, searchOffset + numToReturn, pageSize); 359 links.setNext(linkNext); 360 } 361 362 if (searchOffset > 0) { 363 String linkPrev = String.format( 364 linkFormat, serverBase, theType, searchId, Math.max(0, searchOffset - pageSize), pageSize); 365 links.setPrev(linkPrev); 366 } 367 368 List<IBaseResource> resourceList = response.getResources(searchOffset, numToReturn + searchOffset); 369 370 IVersionSpecificBundleFactory bundleFactory = myContext.newBundleFactory(); 371 bundleFactory.addRootPropertiesToBundle(response.getUuid(), links, response.size(), response.getPublished()); 372 bundleFactory.addResourcesToBundle(resourceList, BundleTypeEnum.SEARCHSET, serverBase, null, null); 373 374 IBaseResource result = bundleFactory.getResourceBundle(); 375 return (IBaseBundle) result; 376 } 377}