001/*- 002 * #%L 003 * HAPI FHIR - Core Library 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.util; 021 022import ca.uhn.fhir.context.BaseRuntimeChildDefinition; 023import ca.uhn.fhir.context.BaseRuntimeElementDefinition; 024import ca.uhn.fhir.context.FhirContext; 025import ca.uhn.fhir.context.FhirVersionEnum; 026import ca.uhn.fhir.context.RuntimeResourceDefinition; 027import ca.uhn.fhir.model.primitive.IdDt; 028import jakarta.annotation.Nonnull; 029import jakarta.annotation.Nullable; 030import org.apache.commons.lang3.Validate; 031import org.hl7.fhir.instance.model.api.IBase; 032import org.hl7.fhir.instance.model.api.IBaseBackboneElement; 033import org.hl7.fhir.instance.model.api.IBaseBundle; 034import org.hl7.fhir.instance.model.api.IBaseParameters; 035import org.hl7.fhir.instance.model.api.IBaseResource; 036import org.hl7.fhir.instance.model.api.IIdType; 037import org.hl7.fhir.instance.model.api.IPrimitiveType; 038 039import java.util.Date; 040import java.util.Objects; 041 042/** 043 * This class can be used to build a Bundle resource to be used as a FHIR transaction. Convenience methods provide 044 * support for setting various bundle fields and working with bundle parts such as metadata and entry 045 * (method and search). 046 * 047 * <p> 048 * <p> 049 * This is not yet complete, and doesn't support all FHIR features. <b>USE WITH CAUTION</b> as the API 050 * may change. 051 * 052 * @since 5.1.0 053 */ 054public class BundleBuilder { 055 056 private final FhirContext myContext; 057 private final IBaseBundle myBundle; 058 private final RuntimeResourceDefinition myBundleDef; 059 private final BaseRuntimeChildDefinition myEntryChild; 060 private final BaseRuntimeChildDefinition myMetaChild; 061 private final BaseRuntimeChildDefinition mySearchChild; 062 private final BaseRuntimeElementDefinition<?> myEntryDef; 063 private final BaseRuntimeElementDefinition<?> myMetaDef; 064 private final BaseRuntimeElementDefinition mySearchDef; 065 private final BaseRuntimeChildDefinition myEntryResourceChild; 066 private final BaseRuntimeChildDefinition myEntryFullUrlChild; 067 private final BaseRuntimeChildDefinition myEntryRequestChild; 068 private final BaseRuntimeElementDefinition<?> myEntryRequestDef; 069 private final BaseRuntimeChildDefinition myEntryRequestUrlChild; 070 private final BaseRuntimeChildDefinition myEntryRequestMethodChild; 071 private final BaseRuntimeElementDefinition<?> myEntryRequestMethodDef; 072 private final BaseRuntimeChildDefinition myEntryRequestIfNoneExistChild; 073 074 /** 075 * Constructor 076 */ 077 public BundleBuilder(FhirContext theContext) { 078 myContext = theContext; 079 080 myBundleDef = myContext.getResourceDefinition("Bundle"); 081 myBundle = (IBaseBundle) myBundleDef.newInstance(); 082 083 myEntryChild = myBundleDef.getChildByName("entry"); 084 myEntryDef = myEntryChild.getChildByName("entry"); 085 086 mySearchChild = myEntryDef.getChildByName("search"); 087 mySearchDef = mySearchChild.getChildByName("search"); 088 089 if (myContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.DSTU3)) { 090 myMetaChild = myBundleDef.getChildByName("meta"); 091 myMetaDef = myMetaChild.getChildByName("meta"); 092 } else { 093 myMetaChild = null; 094 myMetaDef = null; 095 } 096 097 myEntryResourceChild = myEntryDef.getChildByName("resource"); 098 myEntryFullUrlChild = myEntryDef.getChildByName("fullUrl"); 099 100 myEntryRequestChild = myEntryDef.getChildByName("request"); 101 myEntryRequestDef = myEntryRequestChild.getChildByName("request"); 102 103 myEntryRequestUrlChild = myEntryRequestDef.getChildByName("url"); 104 105 myEntryRequestMethodChild = myEntryRequestDef.getChildByName("method"); 106 myEntryRequestMethodDef = myEntryRequestMethodChild.getChildByName("method"); 107 108 myEntryRequestIfNoneExistChild = myEntryRequestDef.getChildByName("ifNoneExist"); 109 } 110 111 /** 112 * Sets the specified primitive field on the bundle with the value provided. 113 * 114 * @param theFieldName Name of the primitive field. 115 * @param theFieldValue Value of the field to be set. 116 */ 117 public BundleBuilder setBundleField(String theFieldName, String theFieldValue) { 118 BaseRuntimeChildDefinition typeChild = myBundleDef.getChildByName(theFieldName); 119 Validate.notNull(typeChild, "Unable to find field %s", theFieldName); 120 121 IPrimitiveType<?> type = (IPrimitiveType<?>) 122 typeChild.getChildByName(theFieldName).newInstance(typeChild.getInstanceConstructorArguments()); 123 type.setValueAsString(theFieldValue); 124 typeChild.getMutator().setValue(myBundle, type); 125 return this; 126 } 127 128 /** 129 * Sets the specified primitive field on the search entry with the value provided. 130 * 131 * @param theSearch Search part of the entry 132 * @param theFieldName Name of the primitive field. 133 * @param theFieldValue Value of the field to be set. 134 */ 135 public BundleBuilder setSearchField(IBase theSearch, String theFieldName, String theFieldValue) { 136 BaseRuntimeChildDefinition typeChild = mySearchDef.getChildByName(theFieldName); 137 Validate.notNull(typeChild, "Unable to find field %s", theFieldName); 138 139 IPrimitiveType<?> type = (IPrimitiveType<?>) 140 typeChild.getChildByName(theFieldName).newInstance(typeChild.getInstanceConstructorArguments()); 141 type.setValueAsString(theFieldValue); 142 typeChild.getMutator().setValue(theSearch, type); 143 return this; 144 } 145 146 public BundleBuilder setSearchField(IBase theSearch, String theFieldName, IPrimitiveType<?> theFieldValue) { 147 BaseRuntimeChildDefinition typeChild = mySearchDef.getChildByName(theFieldName); 148 Validate.notNull(typeChild, "Unable to find field %s", theFieldName); 149 150 typeChild.getMutator().setValue(theSearch, theFieldValue); 151 return this; 152 } 153 154 /** 155 * Adds a FHIRPatch patch bundle to the transaction 156 * 157 * @param theTarget The target resource ID to patch 158 * @param thePatch The FHIRPath Parameters resource 159 * @since 6.3.0 160 */ 161 public PatchBuilder addTransactionFhirPatchEntry(IIdType theTarget, IBaseParameters thePatch) { 162 Validate.notNull(theTarget, "theTarget must not be null"); 163 Validate.notBlank(theTarget.getResourceType(), "theTarget must contain a resource type"); 164 Validate.notBlank(theTarget.getIdPart(), "theTarget must contain an ID"); 165 166 IPrimitiveType<?> url = addAndPopulateTransactionBundleEntryRequest( 167 thePatch, 168 theTarget.getValue(), 169 theTarget.toUnqualifiedVersionless().getValue(), 170 "PATCH"); 171 172 return new PatchBuilder(url); 173 } 174 175 /** 176 * Adds a FHIRPatch patch bundle to the transaction. This method is intended for conditional PATCH operations. If you 177 * know the ID of the resource you wish to patch, use {@link #addTransactionFhirPatchEntry(IIdType, IBaseParameters)} 178 * instead. 179 * 180 * @param thePatch The FHIRPath Parameters resource 181 * @see #addTransactionFhirPatchEntry(IIdType, IBaseParameters) 182 * @since 6.3.0 183 */ 184 public PatchBuilder addTransactionFhirPatchEntry(IBaseParameters thePatch) { 185 IPrimitiveType<?> url = addAndPopulateTransactionBundleEntryRequest(thePatch, null, null, "PATCH"); 186 187 return new PatchBuilder(url); 188 } 189 190 /** 191 * Adds an entry containing an update (PUT) request. 192 * Also sets the Bundle.type value to "transaction" if it is not already set. 193 * 194 * @param theResource The resource to update 195 */ 196 public UpdateBuilder addTransactionUpdateEntry(IBaseResource theResource) { 197 Validate.notNull(theResource, "theResource must not be null"); 198 199 IIdType id = getIdTypeForUpdate(theResource); 200 201 String requestUrl = id.toUnqualifiedVersionless().getValue(); 202 String fullUrl = id.getValue(); 203 String verb = "PUT"; 204 205 IPrimitiveType<?> url = addAndPopulateTransactionBundleEntryRequest(theResource, fullUrl, requestUrl, verb); 206 207 return new UpdateBuilder(url); 208 } 209 210 @Nonnull 211 private IPrimitiveType<?> addAndPopulateTransactionBundleEntryRequest( 212 IBaseResource theResource, String theFullUrl, String theRequestUrl, String theHttpVerb) { 213 setBundleField("type", "transaction"); 214 215 IBase request = addEntryAndReturnRequest(theResource, theFullUrl); 216 217 // Bundle.entry.request.url 218 IPrimitiveType<?> url = 219 (IPrimitiveType<?>) myContext.getElementDefinition("uri").newInstance(); 220 url.setValueAsString(theRequestUrl); 221 myEntryRequestUrlChild.getMutator().setValue(request, url); 222 223 // Bundle.entry.request.method 224 addRequestMethod(request, theHttpVerb); 225 return url; 226 } 227 228 /** 229 * Adds an entry containing an update (UPDATE) request without the body of the resource. 230 * Also sets the Bundle.type value to "transaction" if it is not already set. 231 * 232 * @param theResource The resource to update. 233 */ 234 public void addTransactionUpdateIdOnlyEntry(IBaseResource theResource) { 235 setBundleField("type", "transaction"); 236 237 Validate.notNull(theResource, "theResource must not be null"); 238 239 IIdType id = getIdTypeForUpdate(theResource); 240 String requestUrl = id.toUnqualifiedVersionless().getValue(); 241 String fullUrl = id.getValue(); 242 String httpMethod = "PUT"; 243 244 addIdOnlyEntry(requestUrl, httpMethod, fullUrl); 245 } 246 247 /** 248 * Adds an entry containing an create (POST) request. 249 * Also sets the Bundle.type value to "transaction" if it is not already set. 250 * 251 * @param theResource The resource to create 252 */ 253 public CreateBuilder addTransactionCreateEntry(IBaseResource theResource) { 254 return addTransactionCreateEntry(theResource, null); 255 } 256 257 /** 258 * Adds an entry containing an create (POST) request. 259 * Also sets the Bundle.type value to "transaction" if it is not already set. 260 * 261 * @param theResource The resource to create 262 * @param theFullUrl The fullUrl to attach to the entry. If null, will default to the resource ID. 263 */ 264 public CreateBuilder addTransactionCreateEntry(IBaseResource theResource, @Nullable String theFullUrl) { 265 setBundleField("type", "transaction"); 266 267 IBase request = addEntryAndReturnRequest( 268 theResource, 269 theFullUrl != null ? theFullUrl : theResource.getIdElement().getValue()); 270 271 String resourceType = myContext.getResourceType(theResource); 272 273 // Bundle.entry.request.url 274 addRequestUrl(request, resourceType); 275 276 // Bundle.entry.request.method 277 addRequestMethod(request, "POST"); 278 279 return new CreateBuilder(request); 280 } 281 282 /** 283 * Adds an entry containing a create (POST) request without the body of the resource. 284 * Also sets the Bundle.type value to "transaction" if it is not already set. 285 * 286 * @param theResource The resource to create 287 */ 288 public void addTransactionCreateEntryIdOnly(IBaseResource theResource) { 289 setBundleField("type", "transaction"); 290 291 String requestUrl = myContext.getResourceType(theResource); 292 String fullUrl = theResource.getIdElement().getValue(); 293 String httpMethod = "POST"; 294 295 addIdOnlyEntry(requestUrl, httpMethod, fullUrl); 296 } 297 298 private void addIdOnlyEntry(String theRequestUrl, String theHttpMethod, String theFullUrl) { 299 IBase entry = addEntry(); 300 301 // Bundle.entry.request 302 IBase request = myEntryRequestDef.newInstance(); 303 myEntryRequestChild.getMutator().setValue(entry, request); 304 305 // Bundle.entry.request.url 306 addRequestUrl(request, theRequestUrl); 307 308 // Bundle.entry.request.method 309 addRequestMethod(request, theHttpMethod); 310 311 // Bundle.entry.fullUrl 312 addFullUrl(entry, theFullUrl); 313 } 314 315 /** 316 * Adds an entry containing a delete (DELETE) request. 317 * Also sets the Bundle.type value to "transaction" if it is not already set. 318 * <p> 319 * Note that the resource is only used to extract its ID and type, and the body of the resource is not included in the entry, 320 * 321 * @param theCondition The conditional URL, e.g. "Patient?identifier=foo|bar" 322 * @since 6.8.0 323 */ 324 public DeleteBuilder addTransactionDeleteConditionalEntry(String theCondition) { 325 Validate.notBlank(theCondition, "theCondition must not be blank"); 326 327 setBundleField("type", "transaction"); 328 return addDeleteEntry(theCondition); 329 } 330 331 /** 332 * Adds an entry containing a delete (DELETE) request. 333 * Also sets the Bundle.type value to "transaction" if it is not already set. 334 * <p> 335 * Note that the resource is only used to extract its ID and type, and the body of the resource is not included in the entry, 336 * 337 * @param theResource The resource to delete. 338 */ 339 public DeleteBuilder addTransactionDeleteEntry(IBaseResource theResource) { 340 String resourceType = myContext.getResourceType(theResource); 341 String idPart = theResource.getIdElement().toUnqualifiedVersionless().getIdPart(); 342 return addTransactionDeleteEntry(resourceType, idPart); 343 } 344 345 /** 346 * Adds an entry containing a delete (DELETE) request. 347 * Also sets the Bundle.type value to "transaction" if it is not already set. 348 * <p> 349 * Note that the resource is only used to extract its ID and type, and the body of the resource is not included in the entry, 350 * 351 * @param theResourceId The resource ID to delete. 352 * @return 353 */ 354 public DeleteBuilder addTransactionDeleteEntry(IIdType theResourceId) { 355 String resourceType = theResourceId.getResourceType(); 356 String idPart = theResourceId.getIdPart(); 357 return addTransactionDeleteEntry(resourceType, idPart); 358 } 359 360 /** 361 * Adds an entry containing a delete (DELETE) request. 362 * Also sets the Bundle.type value to "transaction" if it is not already set. 363 * 364 * @param theResourceType The type resource to delete. 365 * @param theIdPart the ID of the resource to delete. 366 */ 367 public DeleteBuilder addTransactionDeleteEntry(String theResourceType, String theIdPart) { 368 setBundleField("type", "transaction"); 369 IdDt idDt = new IdDt(theIdPart); 370 371 String deleteUrl = idDt.toUnqualifiedVersionless() 372 .withResourceType(theResourceType) 373 .getValue(); 374 375 return addDeleteEntry(deleteUrl); 376 } 377 378 /** 379 * Adds an entry containing a delete (DELETE) request. 380 * Also sets the Bundle.type value to "transaction" if it is not already set. 381 * 382 * @param theMatchUrl The match URL, e.g. <code>Patient?identifier=http://foo|123</code> 383 * @since 6.3.0 384 */ 385 public BaseOperationBuilder addTransactionDeleteEntryConditional(String theMatchUrl) { 386 Validate.notBlank(theMatchUrl, "theMatchUrl must not be null or blank"); 387 return addDeleteEntry(theMatchUrl); 388 } 389 390 @Nonnull 391 private DeleteBuilder addDeleteEntry(String theDeleteUrl) { 392 IBase request = addEntryAndReturnRequest(); 393 394 // Bundle.entry.request.url 395 addRequestUrl(request, theDeleteUrl); 396 397 // Bundle.entry.request.method 398 addRequestMethod(request, "DELETE"); 399 400 return new DeleteBuilder(); 401 } 402 403 private IIdType getIdTypeForUpdate(IBaseResource theResource) { 404 IIdType id = theResource.getIdElement(); 405 if (id.hasIdPart() && !id.hasResourceType()) { 406 String resourceType = myContext.getResourceType(theResource); 407 id = id.withResourceType(resourceType); 408 } 409 return id; 410 } 411 412 private void addFullUrl(IBase theEntry, String theFullUrl) { 413 IPrimitiveType<?> fullUrl = 414 (IPrimitiveType<?>) myContext.getElementDefinition("uri").newInstance(); 415 fullUrl.setValueAsString(theFullUrl); 416 myEntryFullUrlChild.getMutator().setValue(theEntry, fullUrl); 417 } 418 419 private void addRequestUrl(IBase request, String theRequestUrl) { 420 IPrimitiveType<?> url = 421 (IPrimitiveType<?>) myContext.getElementDefinition("uri").newInstance(); 422 url.setValueAsString(theRequestUrl); 423 myEntryRequestUrlChild.getMutator().setValue(request, url); 424 } 425 426 private void addRequestMethod(IBase theRequest, String theMethod) { 427 IPrimitiveType<?> method = (IPrimitiveType<?>) 428 myEntryRequestMethodDef.newInstance(myEntryRequestMethodChild.getInstanceConstructorArguments()); 429 method.setValueAsString(theMethod); 430 myEntryRequestMethodChild.getMutator().setValue(theRequest, method); 431 } 432 433 /** 434 * Adds an entry for a Collection bundle type 435 */ 436 public void addCollectionEntry(IBaseResource theResource) { 437 setType("collection"); 438 addEntryAndReturnRequest(theResource); 439 } 440 441 /** 442 * Adds an entry for a Document bundle type 443 */ 444 public void addDocumentEntry(IBaseResource theResource) { 445 setType("document"); 446 addEntryAndReturnRequest(theResource); 447 } 448 449 /** 450 * Creates new entry and adds it to the bundle 451 * 452 * @return Returns the new entry. 453 */ 454 public IBase addEntry() { 455 IBase entry = myEntryDef.newInstance(); 456 myEntryChild.getMutator().addValue(myBundle, entry); 457 return entry; 458 } 459 460 /** 461 * Creates new search instance for the specified entry. 462 * Note that this method does not work for DSTU2 model classes, it will only work 463 * on DSTU3+. 464 * 465 * @param entry Entry to create search instance for 466 * @return Returns the search instance 467 */ 468 public IBaseBackboneElement addSearch(IBase entry) { 469 Validate.isTrue( 470 myContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.DSTU3), 471 "This method may only be called for FHIR version DSTU3 and above"); 472 473 IBase searchInstance = mySearchDef.newInstance(); 474 mySearchChild.getMutator().setValue(entry, searchInstance); 475 return (IBaseBackboneElement) searchInstance; 476 } 477 478 private IBase addEntryAndReturnRequest(IBaseResource theResource) { 479 IIdType id = theResource.getIdElement(); 480 if (id.hasVersionIdPart()) { 481 id = id.toVersionless(); 482 } 483 return addEntryAndReturnRequest(theResource, id.getValue()); 484 } 485 486 private IBase addEntryAndReturnRequest(IBaseResource theResource, String theFullUrl) { 487 Validate.notNull(theResource, "theResource must not be null"); 488 489 IBase entry = addEntry(); 490 491 // Bundle.entry.fullUrl 492 addFullUrl(entry, theFullUrl); 493 494 // Bundle.entry.resource 495 myEntryResourceChild.getMutator().setValue(entry, theResource); 496 497 // Bundle.entry.request 498 IBase request = myEntryRequestDef.newInstance(); 499 myEntryRequestChild.getMutator().setValue(entry, request); 500 return request; 501 } 502 503 public IBase addEntryAndReturnRequest() { 504 IBase entry = addEntry(); 505 506 // Bundle.entry.request 507 IBase request = myEntryRequestDef.newInstance(); 508 myEntryRequestChild.getMutator().setValue(entry, request); 509 return request; 510 } 511 512 public IBaseBundle getBundle() { 513 return myBundle; 514 } 515 516 /** 517 * Convenience method which auto-casts the results of {@link #getBundle()} 518 * 519 * @since 6.3.0 520 */ 521 public <T extends IBaseBundle> T getBundleTyped() { 522 return (T) myBundle; 523 } 524 525 /** 526 * Note that this method does not work for DSTU2 model classes, it will only work 527 * on DSTU3+. 528 */ 529 public BundleBuilder setMetaField(String theFieldName, IBase theFieldValue) { 530 Validate.isTrue( 531 myContext.getVersion().getVersion().isEqualOrNewerThan(FhirVersionEnum.DSTU3), 532 "This method may only be called for FHIR version DSTU3 and above"); 533 534 BaseRuntimeChildDefinition.IMutator mutator = 535 myMetaDef.getChildByName(theFieldName).getMutator(); 536 mutator.setValue(myBundle.getMeta(), theFieldValue); 537 return this; 538 } 539 540 /** 541 * Sets the specified entry field. 542 * 543 * @param theEntry The entry instance to set values on 544 * @param theEntryChildName The child field name of the entry instance to be set 545 * @param theValue The field value to set 546 */ 547 public void addToEntry(IBase theEntry, String theEntryChildName, IBase theValue) { 548 addToBase(theEntry, theEntryChildName, theValue, myEntryDef); 549 } 550 551 /** 552 * Sets the specified search field. 553 * 554 * @param theSearch The search instance to set values on 555 * @param theSearchFieldName The child field name of the search instance to be set 556 * @param theSearchFieldValue The field value to set 557 */ 558 public void addToSearch(IBase theSearch, String theSearchFieldName, IBase theSearchFieldValue) { 559 addToBase(theSearch, theSearchFieldName, theSearchFieldValue, mySearchDef); 560 } 561 562 private void addToBase( 563 IBase theBase, String theSearchChildName, IBase theValue, BaseRuntimeElementDefinition mySearchDef) { 564 BaseRuntimeChildDefinition defn = mySearchDef.getChildByName(theSearchChildName); 565 Validate.notNull(defn, "Unable to get child definition %s from %s", theSearchChildName, theBase); 566 defn.getMutator().addValue(theBase, theValue); 567 } 568 569 /** 570 * Creates a new primitive. 571 * 572 * @param theTypeName The element type for the primitive 573 * @param <T> Actual type of the parameterized primitive type interface 574 * @return Returns the new empty instance of the element definition. 575 */ 576 public <T> IPrimitiveType<T> newPrimitive(String theTypeName) { 577 BaseRuntimeElementDefinition primitiveDefinition = myContext.getElementDefinition(theTypeName); 578 Validate.notNull(primitiveDefinition, "Unable to find definition for %s", theTypeName); 579 return (IPrimitiveType<T>) primitiveDefinition.newInstance(); 580 } 581 582 /** 583 * Creates a new primitive instance of the specified element type. 584 * 585 * @param theTypeName Element type to create 586 * @param theInitialValue Initial value to be set on the new instance 587 * @param <T> Actual type of the parameterized primitive type interface 588 * @return Returns the newly created instance 589 */ 590 public <T> IPrimitiveType<T> newPrimitive(String theTypeName, T theInitialValue) { 591 IPrimitiveType<T> retVal = newPrimitive(theTypeName); 592 retVal.setValue(theInitialValue); 593 return retVal; 594 } 595 596 /** 597 * Sets a value for <code>Bundle.type</code>. That this is a coded field so {@literal theType} 598 * must be an actual valid value for this field or a {@link ca.uhn.fhir.parser.DataFormatException} 599 * will be thrown. 600 */ 601 public void setType(String theType) { 602 setBundleField("type", theType); 603 } 604 605 /** 606 * Adds an identifier to <code>Bundle.identifier</code> 607 * 608 * @param theSystem The system 609 * @param theValue The value 610 * @since 6.4.0 611 */ 612 public void setIdentifier(@Nullable String theSystem, @Nullable String theValue) { 613 FhirTerser terser = myContext.newTerser(); 614 IBase identifier = terser.addElement(myBundle, "identifier"); 615 terser.setElement(identifier, "system", theSystem); 616 terser.setElement(identifier, "value", theValue); 617 } 618 619 /** 620 * Sets the timestamp in <code>Bundle.timestamp</code> 621 * 622 * @since 6.4.0 623 */ 624 public void setTimestamp(@Nonnull IPrimitiveType<Date> theTimestamp) { 625 FhirTerser terser = myContext.newTerser(); 626 terser.setElement(myBundle, "Bundle.timestamp", theTimestamp.getValueAsString()); 627 } 628 629 /** 630 * Adds a profile URL to <code>Bundle.meta.profile</code> 631 * 632 * @since 7.4.0 633 */ 634 public void addProfile(String theProfile) { 635 FhirTerser terser = myContext.newTerser(); 636 terser.addElement(myBundle, "Bundle.meta.profile", theProfile); 637 } 638 639 public class DeleteBuilder extends BaseOperationBuilder { 640 641 // nothing yet 642 643 } 644 645 public class PatchBuilder extends BaseOperationBuilderWithConditionalUrl<PatchBuilder> { 646 647 PatchBuilder(IPrimitiveType<?> theUrl) { 648 super(theUrl); 649 } 650 } 651 652 public class UpdateBuilder extends BaseOperationBuilderWithConditionalUrl<UpdateBuilder> { 653 UpdateBuilder(IPrimitiveType<?> theUrl) { 654 super(theUrl); 655 } 656 } 657 658 public class CreateBuilder extends BaseOperationBuilder { 659 private final IBase myRequest; 660 661 CreateBuilder(IBase theRequest) { 662 myRequest = theRequest; 663 } 664 665 /** 666 * Make this create a Conditional Create 667 */ 668 public CreateBuilder conditional(String theConditionalUrl) { 669 BaseRuntimeElementDefinition<?> stringDefinition = 670 Objects.requireNonNull(myContext.getElementDefinition("string")); 671 IPrimitiveType<?> ifNoneExist = (IPrimitiveType<?>) stringDefinition.newInstance(); 672 ifNoneExist.setValueAsString(theConditionalUrl); 673 674 myEntryRequestIfNoneExistChild.getMutator().setValue(myRequest, ifNoneExist); 675 676 return this; 677 } 678 } 679 680 public abstract class BaseOperationBuilder { 681 682 /** 683 * Returns a reference to the BundleBuilder instance. 684 * <p> 685 * Calling this method has no effect at all, it is only 686 * provided for easy method chaning if you want to build 687 * your bundle as a single fluent call. 688 * 689 * @since 6.3.0 690 */ 691 public BundleBuilder andThen() { 692 return BundleBuilder.this; 693 } 694 } 695 696 public abstract class BaseOperationBuilderWithConditionalUrl<T extends BaseOperationBuilder> 697 extends BaseOperationBuilder { 698 699 private final IPrimitiveType<?> myUrl; 700 701 BaseOperationBuilderWithConditionalUrl(IPrimitiveType<?> theUrl) { 702 myUrl = theUrl; 703 } 704 705 /** 706 * Make this update a Conditional Update 707 */ 708 @SuppressWarnings("unchecked") 709 public T conditional(String theConditionalUrl) { 710 myUrl.setValueAsString(theConditionalUrl); 711 return (T) this; 712 } 713 } 714}