001/* 002 * #%L 003 * HAPI FHIR - Client Framework 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.rest.client.method; 021 022import ca.uhn.fhir.context.ConfigurationException; 023import ca.uhn.fhir.context.FhirContext; 024import ca.uhn.fhir.context.FhirVersionEnum; 025import ca.uhn.fhir.context.RuntimeResourceDefinition; 026import ca.uhn.fhir.i18n.Msg; 027import ca.uhn.fhir.model.api.IResource; 028import ca.uhn.fhir.model.api.Include; 029import ca.uhn.fhir.model.api.ResourceMetadataKeyEnum; 030import ca.uhn.fhir.model.api.TagList; 031import ca.uhn.fhir.model.primitive.IdDt; 032import ca.uhn.fhir.model.primitive.InstantDt; 033import ca.uhn.fhir.parser.IParser; 034import ca.uhn.fhir.rest.annotation.At; 035import ca.uhn.fhir.rest.annotation.ConditionalUrlParam; 036import ca.uhn.fhir.rest.annotation.Count; 037import ca.uhn.fhir.rest.annotation.Elements; 038import ca.uhn.fhir.rest.annotation.IdParam; 039import ca.uhn.fhir.rest.annotation.IncludeParam; 040import ca.uhn.fhir.rest.annotation.Offset; 041import ca.uhn.fhir.rest.annotation.Operation; 042import ca.uhn.fhir.rest.annotation.OperationParam; 043import ca.uhn.fhir.rest.annotation.OptionalParam; 044import ca.uhn.fhir.rest.annotation.RawParam; 045import ca.uhn.fhir.rest.annotation.RequiredParam; 046import ca.uhn.fhir.rest.annotation.ResourceParam; 047import ca.uhn.fhir.rest.annotation.Since; 048import ca.uhn.fhir.rest.annotation.Sort; 049import ca.uhn.fhir.rest.annotation.TransactionParam; 050import ca.uhn.fhir.rest.annotation.Validate; 051import ca.uhn.fhir.rest.api.Constants; 052import ca.uhn.fhir.rest.api.EncodingEnum; 053import ca.uhn.fhir.rest.api.MethodOutcome; 054import ca.uhn.fhir.rest.api.PatchTypeEnum; 055import ca.uhn.fhir.rest.api.RestOperationTypeEnum; 056import ca.uhn.fhir.rest.api.SummaryEnum; 057import ca.uhn.fhir.rest.api.ValidationModeEnum; 058import ca.uhn.fhir.rest.client.api.IHttpRequest; 059import ca.uhn.fhir.rest.client.method.OperationParameter.IOperationParamConverter; 060import ca.uhn.fhir.rest.param.ParameterUtil; 061import ca.uhn.fhir.rest.param.binder.CollectionBinder; 062import ca.uhn.fhir.util.DateUtils; 063import ca.uhn.fhir.util.ParametersUtil; 064import ca.uhn.fhir.util.ReflectionUtil; 065import ca.uhn.fhir.util.UrlUtil; 066import org.apache.commons.lang3.StringUtils; 067import org.hl7.fhir.instance.model.api.IAnyResource; 068import org.hl7.fhir.instance.model.api.IBaseOperationOutcome; 069import org.hl7.fhir.instance.model.api.IBaseResource; 070import org.hl7.fhir.instance.model.api.IIdType; 071 072import java.io.IOException; 073import java.io.InputStream; 074import java.io.PushbackInputStream; 075import java.lang.annotation.Annotation; 076import java.lang.reflect.Method; 077import java.util.ArrayList; 078import java.util.Collection; 079import java.util.Date; 080import java.util.List; 081import java.util.Map; 082import java.util.Map.Entry; 083 084import static org.apache.commons.lang3.StringUtils.isNotBlank; 085 086public class MethodUtil { 087 088 private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(MethodUtil.class); 089 090 /** Non instantiable */ 091 private MethodUtil() { 092 // nothing 093 } 094 095 public static void addAcceptHeaderToRequest( 096 EncodingEnum theEncoding, IHttpRequest theHttpRequest, FhirContext theContext) { 097 if (theEncoding == null) { 098 if (theContext.getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2_1) == false) { 099 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.HEADER_ACCEPT_VALUE_XML_OR_JSON_LEGACY); 100 } else { 101 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.HEADER_ACCEPT_VALUE_XML_OR_JSON_NON_LEGACY); 102 } 103 } else if (theEncoding == EncodingEnum.JSON) { 104 if (theContext.getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2_1) == false) { 105 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.CT_FHIR_JSON); 106 } else { 107 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.HEADER_ACCEPT_VALUE_JSON_NON_LEGACY); 108 } 109 } else if (theEncoding == EncodingEnum.XML) { 110 if (theContext.getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2_1) == false) { 111 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.CT_FHIR_XML); 112 } else { 113 theHttpRequest.addHeader(Constants.HEADER_ACCEPT, Constants.HEADER_ACCEPT_VALUE_XML_NON_LEGACY); 114 } 115 } 116 } 117 118 public static HttpGetClientInvocation createConformanceInvocation(FhirContext theContext) { 119 return new HttpGetClientInvocation(theContext, "metadata"); 120 } 121 122 public static HttpPostClientInvocation createCreateInvocation(IBaseResource theResource, FhirContext theContext) { 123 return createCreateInvocation(theResource, null, theContext); 124 } 125 126 public static HttpPostClientInvocation createCreateInvocation( 127 IBaseResource theResource, String theResourceBody, FhirContext theContext) { 128 RuntimeResourceDefinition def = theContext.getResourceDefinition(theResource); 129 String resourceName = def.getName(); 130 131 StringBuilder urlExtension = new StringBuilder(); 132 urlExtension.append(resourceName); 133 134 HttpPostClientInvocation retVal; 135 if (StringUtils.isBlank(theResourceBody)) { 136 retVal = new HttpPostClientInvocation(theContext, theResource, urlExtension.toString()); 137 } else { 138 retVal = new HttpPostClientInvocation(theContext, theResourceBody, false, urlExtension.toString()); 139 } 140 141 retVal.setOmitResourceId(true); 142 143 return retVal; 144 } 145 146 public static HttpPostClientInvocation createCreateInvocation( 147 IBaseResource theResource, 148 String theResourceBody, 149 FhirContext theContext, 150 Map<String, List<String>> theIfNoneExistParams) { 151 HttpPostClientInvocation retVal = createCreateInvocation(theResource, theResourceBody, theContext); 152 retVal.setIfNoneExistParams(theIfNoneExistParams); 153 return retVal; 154 } 155 156 public static HttpPostClientInvocation createCreateInvocation( 157 IBaseResource theResource, String theResourceBody, FhirContext theContext, String theIfNoneExistUrl) { 158 HttpPostClientInvocation retVal = createCreateInvocation(theResource, theResourceBody, theContext); 159 retVal.setIfNoneExistString(theIfNoneExistUrl); 160 return retVal; 161 } 162 163 public static HttpPatchClientInvocation createPatchInvocation( 164 FhirContext theContext, IIdType theId, PatchTypeEnum thePatchType, String theBody) { 165 return PatchMethodBinding.createPatchInvocation(theContext, theId, thePatchType, theBody); 166 } 167 168 public static HttpPatchClientInvocation createPatchInvocation( 169 FhirContext theContext, 170 PatchTypeEnum thePatchType, 171 String theBody, 172 String theResourceType, 173 Map<String, List<String>> theMatchParams) { 174 return PatchMethodBinding.createPatchInvocation( 175 theContext, thePatchType, theBody, theResourceType, theMatchParams); 176 } 177 178 public static HttpPatchClientInvocation createPatchInvocation( 179 FhirContext theContext, String theUrl, PatchTypeEnum thePatchType, String theBody) { 180 return PatchMethodBinding.createPatchInvocation(theContext, theUrl, thePatchType, theBody); 181 } 182 183 public static HttpPutClientInvocation createUpdateInvocation( 184 FhirContext theContext, 185 IBaseResource theResource, 186 String theResourceBody, 187 Map<String, List<String>> theMatchParams) { 188 String resourceType = theContext.getResourceType(theResource); 189 190 StringBuilder b = createUrl(resourceType, theMatchParams); 191 192 HttpPutClientInvocation retVal; 193 if (StringUtils.isBlank(theResourceBody)) { 194 retVal = new HttpPutClientInvocation(theContext, theResource, b.toString()); 195 } else { 196 retVal = new HttpPutClientInvocation(theContext, theResourceBody, false, b.toString()); 197 } 198 199 return retVal; 200 } 201 202 public static HttpPutClientInvocation createUpdateInvocation( 203 FhirContext theContext, IBaseResource theResource, String theResourceBody, String theMatchUrl) { 204 HttpPutClientInvocation retVal; 205 if (StringUtils.isBlank(theResourceBody)) { 206 retVal = new HttpPutClientInvocation(theContext, theResource, theMatchUrl); 207 } else { 208 retVal = new HttpPutClientInvocation(theContext, theResourceBody, false, theMatchUrl); 209 } 210 211 return retVal; 212 } 213 214 public static HttpPutClientInvocation createUpdateInvocation( 215 IBaseResource theResource, String theResourceBody, IIdType theId, FhirContext theContext) { 216 String resourceName = theContext.getResourceType(theResource); 217 StringBuilder urlBuilder = new StringBuilder(); 218 urlBuilder.append(resourceName); 219 urlBuilder.append('/'); 220 urlBuilder.append(theId.getIdPart()); 221 String urlExtension = urlBuilder.toString(); 222 223 HttpPutClientInvocation retVal; 224 if (StringUtils.isBlank(theResourceBody)) { 225 retVal = new HttpPutClientInvocation(theContext, theResource, urlExtension); 226 } else { 227 retVal = new HttpPutClientInvocation(theContext, theResourceBody, false, urlExtension); 228 } 229 230 retVal.setForceResourceId(theId); 231 232 if (theId.hasVersionIdPart()) { 233 retVal.addHeader(Constants.HEADER_IF_MATCH, '"' + theId.getVersionIdPart() + '"'); 234 } 235 236 return retVal; 237 } 238 239 public static HttpPutClientInvocation createUpdateHistoryRewriteInvocation( 240 IBaseResource theResource, String theResourceBody, IIdType theId, FhirContext theContext) { 241 String resourceName = theContext.getResourceType(theResource); 242 StringBuilder urlBuilder = new StringBuilder(); 243 urlBuilder.append(resourceName); 244 urlBuilder.append('/'); 245 urlBuilder.append(theId.getIdPart()); 246 if (theId.hasVersionIdPart()) { 247 urlBuilder.append('/'); 248 urlBuilder.append(Constants.PARAM_HISTORY); 249 urlBuilder.append('/'); 250 urlBuilder.append(theId.getVersionIdPart()); 251 } 252 253 String urlExtension = urlBuilder.toString(); 254 255 HttpPutClientInvocation retVal; 256 if (StringUtils.isBlank(theResourceBody)) { 257 retVal = new HttpPutClientInvocation(theContext, theResource, urlExtension); 258 } else { 259 retVal = new HttpPutClientInvocation(theContext, theResourceBody, false, urlExtension); 260 } 261 262 return retVal; 263 } 264 265 public static StringBuilder createUrl(String theResourceType, Map<String, List<String>> theMatchParams) { 266 StringBuilder b = new StringBuilder(); 267 268 b.append(theResourceType); 269 270 boolean haveQuestionMark = false; 271 for (Entry<String, List<String>> nextEntry : theMatchParams.entrySet()) { 272 for (String nextValue : nextEntry.getValue()) { 273 b.append(haveQuestionMark ? '&' : '?'); 274 haveQuestionMark = true; 275 b.append(UrlUtil.escapeUrlParam(nextEntry.getKey())); 276 b.append('='); 277 b.append(UrlUtil.escapeUrlParam(nextValue)); 278 } 279 } 280 return b; 281 } 282 283 @SuppressWarnings("unchecked") 284 public static List<IParameter> getResourceParameters( 285 final FhirContext theContext, 286 Method theMethod, 287 Object theProvider, 288 RestOperationTypeEnum theRestfulOperationTypeEnum) { 289 List<IParameter> parameters = new ArrayList<>(); 290 291 Class<?>[] parameterTypes = theMethod.getParameterTypes(); 292 int paramIndex = 0; 293 for (Annotation[] annotations : theMethod.getParameterAnnotations()) { 294 295 IParameter param = null; 296 Class<?> parameterType = parameterTypes[paramIndex]; 297 Class<? extends java.util.Collection<?>> outerCollectionType = null; 298 Class<? extends java.util.Collection<?>> innerCollectionType = null; 299 if (TagList.class.isAssignableFrom(parameterType)) { 300 // TagList is handled directly within the method bindings 301 param = new NullParameter(); 302 } else { 303 if (Collection.class.isAssignableFrom(parameterType)) { 304 innerCollectionType = (Class<? extends java.util.Collection<?>>) parameterType; 305 parameterType = ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex); 306 } 307 if (Collection.class.isAssignableFrom(parameterType)) { 308 outerCollectionType = innerCollectionType; 309 innerCollectionType = (Class<? extends java.util.Collection<?>>) parameterType; 310 parameterType = ReflectionUtil.getGenericCollectionTypeOfMethodParameter(theMethod, paramIndex); 311 } 312 if (Collection.class.isAssignableFrom(parameterType)) { 313 throw new ConfigurationException( 314 Msg.code(1433) + "Argument #" + paramIndex + " of Method '" + theMethod.getName() 315 + "' in type '" 316 + theMethod.getDeclaringClass().getCanonicalName() 317 + "' is of an invalid generic type (can not be a collection of a collection of a collection)"); 318 } 319 } 320 321 if (parameterType.equals(SummaryEnum.class)) { 322 param = new SummaryEnumParameter(); 323 } else if (parameterType.equals(PatchTypeEnum.class)) { 324 param = new PatchTypeParameter(); 325 } else { 326 for (int i = 0; i < annotations.length && param == null; i++) { 327 Annotation nextAnnotation = annotations[i]; 328 329 if (nextAnnotation instanceof RequiredParam) { 330 SearchParameter parameter = new SearchParameter(); 331 parameter.setName(((RequiredParam) nextAnnotation).name()); 332 parameter.setRequired(true); 333 parameter.setDeclaredTypes(((RequiredParam) nextAnnotation).targetTypes()); 334 parameter.setCompositeTypes(((RequiredParam) nextAnnotation).compositeTypes()); 335 parameter.setChainlists(((RequiredParam) nextAnnotation).chainWhitelist()); 336 parameter.setType(theContext, parameterType, innerCollectionType, outerCollectionType); 337 param = parameter; 338 } else if (nextAnnotation instanceof OptionalParam) { 339 SearchParameter parameter = new SearchParameter(); 340 parameter.setName(((OptionalParam) nextAnnotation).name()); 341 parameter.setRequired(false); 342 parameter.setDeclaredTypes(((OptionalParam) nextAnnotation).targetTypes()); 343 parameter.setCompositeTypes(((OptionalParam) nextAnnotation).compositeTypes()); 344 parameter.setChainlists(((OptionalParam) nextAnnotation).chainWhitelist()); 345 parameter.setType(theContext, parameterType, innerCollectionType, outerCollectionType); 346 param = parameter; 347 } else if (nextAnnotation instanceof RawParam) { 348 param = new RawParamsParmeter(); 349 } else if (nextAnnotation instanceof IncludeParam) { 350 Class<? extends Collection<Include>> instantiableCollectionType; 351 Class<?> specType; 352 353 if (parameterType == String.class) { 354 instantiableCollectionType = null; 355 specType = String.class; 356 } else if ((parameterType != Include.class) 357 || innerCollectionType == null 358 || outerCollectionType != null) { 359 throw new ConfigurationException(Msg.code(1434) + "Method '" + theMethod.getName() 360 + "' is annotated with @" 361 + IncludeParam.class.getSimpleName() + " but has a type other than Collection<" 362 + Include.class.getSimpleName() + ">"); 363 } else { 364 instantiableCollectionType = (Class<? extends Collection<Include>>) 365 CollectionBinder.getInstantiableCollectionType( 366 innerCollectionType, "Method '" + theMethod.getName() + "'"); 367 specType = parameterType; 368 } 369 370 param = new IncludeParameter( 371 (IncludeParam) nextAnnotation, instantiableCollectionType, specType); 372 } else if (nextAnnotation instanceof ResourceParam) { 373 if (IBaseResource.class.isAssignableFrom(parameterType)) { 374 // good 375 } else if (String.class.equals(parameterType)) { 376 // good 377 } else { 378 StringBuilder b = new StringBuilder(); 379 b.append("Method '"); 380 b.append(theMethod.getName()); 381 b.append("' is annotated with @"); 382 b.append(ResourceParam.class.getSimpleName()); 383 b.append(" but has a type that is not an implementation of "); 384 b.append(IBaseResource.class.getCanonicalName()); 385 throw new ConfigurationException(Msg.code(1435) + b.toString()); 386 } 387 param = new ResourceParameter(parameterType); 388 } else if (nextAnnotation instanceof IdParam) { 389 param = new NullParameter(); 390 } else if (nextAnnotation instanceof Elements) { 391 param = new ElementsParameter(); 392 } else if (nextAnnotation instanceof Since) { 393 param = new SinceParameter(); 394 ((SinceParameter) param) 395 .setType(theContext, parameterType, innerCollectionType, outerCollectionType); 396 } else if (nextAnnotation instanceof At) { 397 param = new AtParameter(); 398 ((AtParameter) param) 399 .setType(theContext, parameterType, innerCollectionType, outerCollectionType); 400 } else if (nextAnnotation instanceof Count) { 401 param = new CountParameter(); 402 } else if (nextAnnotation instanceof Offset) { 403 param = new OffsetParameter(); 404 } else if (nextAnnotation instanceof Sort) { 405 param = new SortParameter(theContext); 406 } else if (nextAnnotation instanceof TransactionParam) { 407 param = new TransactionParameter(theContext); 408 } else if (nextAnnotation instanceof ConditionalUrlParam) { 409 param = new ConditionalParamBinder( 410 theRestfulOperationTypeEnum, ((ConditionalUrlParam) nextAnnotation).supportsMultiple()); 411 } else if (nextAnnotation instanceof OperationParam) { 412 Operation op = theMethod.getAnnotation(Operation.class); 413 param = new OperationParameter(theContext, op.name(), ((OperationParam) nextAnnotation)); 414 } else if (nextAnnotation instanceof Validate.Mode) { 415 if (parameterType.equals(ValidationModeEnum.class) == false) { 416 throw new ConfigurationException(Msg.code(1436) + "Parameter annotated with @" 417 + Validate.class.getSimpleName() + "." + Validate.Mode.class.getSimpleName() 418 + " must be of type " + ValidationModeEnum.class.getName()); 419 } 420 param = new OperationParameter( 421 theContext, Constants.EXTOP_VALIDATE, Constants.EXTOP_VALIDATE_MODE, 0, 1) 422 .setConverter(new IOperationParamConverter() { 423 @Override 424 public Object outgoingClient(Object theObject) { 425 return ParametersUtil.createString( 426 theContext, ((ValidationModeEnum) theObject).getCode()); 427 } 428 }); 429 } else if (nextAnnotation instanceof Validate.Profile) { 430 if (parameterType.equals(String.class) == false) { 431 throw new ConfigurationException(Msg.code(1437) + "Parameter annotated with @" 432 + Validate.class.getSimpleName() + "." + Validate.Profile.class.getSimpleName() 433 + " must be of type " + String.class.getName()); 434 } 435 param = new OperationParameter( 436 theContext, Constants.EXTOP_VALIDATE, Constants.EXTOP_VALIDATE_PROFILE, 0, 1) 437 .setConverter(new IOperationParamConverter() { 438 439 @Override 440 public Object outgoingClient(Object theObject) { 441 return ParametersUtil.createString(theContext, theObject.toString()); 442 } 443 }); 444 } else { 445 continue; 446 } 447 } 448 } 449 450 if (param == null) { 451 throw new ConfigurationException( 452 Msg.code(1438) + "Parameter #" + ((paramIndex + 1)) + "/" + (parameterTypes.length) 453 + " of method '" + theMethod.getName() + "' on type '" 454 + theMethod.getDeclaringClass().getCanonicalName() 455 + "' has no recognized FHIR interface parameter annotations. Don't know how to handle this parameter"); 456 } 457 458 param.initializeTypes(theMethod, outerCollectionType, innerCollectionType, parameterType); 459 parameters.add(param); 460 461 paramIndex++; 462 } 463 return parameters; 464 } 465 466 public static void parseClientRequestResourceHeaders( 467 IIdType theRequestedId, Map<String, List<String>> theHeaders, IBaseResource resource) { 468 List<String> lmHeaders = theHeaders.get(Constants.HEADER_LAST_MODIFIED_LOWERCASE); 469 if (lmHeaders != null && lmHeaders.size() > 0 && StringUtils.isNotBlank(lmHeaders.get(0))) { 470 String headerValue = lmHeaders.get(0); 471 Date headerDateValue; 472 try { 473 headerDateValue = DateUtils.parseDate(headerValue); 474 if (resource instanceof IResource) { 475 IResource iResource = (IResource) resource; 476 InstantDt existing = ResourceMetadataKeyEnum.UPDATED.get(iResource); 477 if (existing == null || existing.isEmpty()) { 478 InstantDt lmValue = new InstantDt(headerDateValue); 479 iResource.getResourceMetadata().put(ResourceMetadataKeyEnum.UPDATED, lmValue); 480 } 481 } else if (resource instanceof IAnyResource) { 482 IAnyResource anyResource = (IAnyResource) resource; 483 if (anyResource.getMeta().getLastUpdated() == null) { 484 anyResource.getMeta().setLastUpdated(headerDateValue); 485 } 486 } 487 } catch (Exception e) { 488 ourLog.warn("Unable to parse date string '{}'. Error is: {}", headerValue, e.toString()); 489 } 490 } 491 492 List<String> clHeaders = theHeaders.get(Constants.HEADER_CONTENT_LOCATION_LC); 493 if (clHeaders != null && clHeaders.size() > 0 && StringUtils.isNotBlank(clHeaders.get(0))) { 494 String headerValue = clHeaders.get(0); 495 if (isNotBlank(headerValue)) { 496 new IdDt(headerValue).applyTo(resource); 497 } 498 } 499 500 List<String> locationHeaders = theHeaders.get(Constants.HEADER_LOCATION_LC); 501 if (locationHeaders != null && locationHeaders.size() > 0 && StringUtils.isNotBlank(locationHeaders.get(0))) { 502 String headerValue = locationHeaders.get(0); 503 if (isNotBlank(headerValue)) { 504 new IdDt(headerValue).applyTo(resource); 505 } 506 } 507 508 IdDt existing = IdDt.of(resource); 509 510 List<String> eTagHeaders = theHeaders.get(Constants.HEADER_ETAG_LC); 511 String eTagVersion = null; 512 if (eTagHeaders != null && eTagHeaders.size() > 0) { 513 eTagVersion = ParameterUtil.parseETagValue(eTagHeaders.get(0)); 514 } 515 if (isNotBlank(eTagVersion)) { 516 if (existing == null || existing.isEmpty()) { 517 if (theRequestedId != null) { 518 theRequestedId.withVersion(eTagVersion).applyTo(resource); 519 } 520 } else if (existing.hasVersionIdPart() == false) { 521 existing.withVersion(eTagVersion).applyTo(resource); 522 } 523 } else if (existing == null || existing.isEmpty()) { 524 if (theRequestedId != null) { 525 theRequestedId.applyTo(resource); 526 } 527 } 528 } 529 530 public static MethodOutcome process2xxResponse( 531 FhirContext theContext, 532 int theResponseStatusCode, 533 String theResponseMimeType, 534 InputStream theResponseReader, 535 Map<String, List<String>> theHeaders) { 536 List<String> locationHeaders = new ArrayList<>(); 537 List<String> lh = theHeaders.get(Constants.HEADER_LOCATION_LC); 538 if (lh != null) { 539 locationHeaders.addAll(lh); 540 } 541 List<String> clh = theHeaders.get(Constants.HEADER_CONTENT_LOCATION_LC); 542 if (clh != null) { 543 locationHeaders.addAll(clh); 544 } 545 546 MethodOutcome retVal = new MethodOutcome(); 547 retVal.setResponseStatusCode(theResponseStatusCode); 548 if (locationHeaders.size() > 0) { 549 String locationHeader = locationHeaders.get(0); 550 BaseOutcomeReturningMethodBinding.parseContentLocation(theContext, retVal, locationHeader); 551 } 552 if (theResponseStatusCode != Constants.STATUS_HTTP_204_NO_CONTENT) { 553 EncodingEnum ct = EncodingEnum.forContentType(theResponseMimeType); 554 if (ct != null) { 555 PushbackInputStream reader = new PushbackInputStream(theResponseReader); 556 557 try { 558 int firstByte = reader.read(); 559 if (firstByte == -1) { 560 BaseOutcomeReturningMethodBinding.ourLog.debug("No content in response, not going to read"); 561 reader = null; 562 } else { 563 reader.unread(firstByte); 564 } 565 } catch (IOException e) { 566 BaseOutcomeReturningMethodBinding.ourLog.debug("No content in response, not going to read", e); 567 reader = null; 568 } 569 570 if (reader != null) { 571 IParser parser = ct.newParser(theContext); 572 IBaseResource outcome = parser.parseResource(reader); 573 if (outcome instanceof IBaseOperationOutcome) { 574 retVal.setOperationOutcome((IBaseOperationOutcome) outcome); 575 } else { 576 retVal.setResource(outcome); 577 } 578 } 579 580 } else { 581 BaseOutcomeReturningMethodBinding.ourLog.debug( 582 "Ignoring response content of type: {}", theResponseMimeType); 583 } 584 } 585 return retVal; 586 } 587}