001/*- 002 * #%L 003 * Smile CDR - CDR 004 * %% 005 * Copyright (C) 2016 - 2025 Smile CDR, Inc. 006 * %% 007 * All rights reserved. 008 * #L% 009 */ 010package ca.cdr.api.pub.hl7v2.out; 011 012import ca.cdr.api.util.PublicApiConstants; 013import ca.uhn.fhir.context.FhirContext; 014import jakarta.annotation.Nonnull; 015import jakarta.annotation.Nullable; 016import org.apache.commons.lang3.Validate; 017import org.hl7.fhir.instance.model.api.IBaseReference; 018import org.hl7.fhir.instance.model.api.IBaseResource; 019import org.hl7.fhir.instance.model.api.IIdType; 020 021import java.util.HashMap; 022import java.util.List; 023import java.util.Map; 024 025public class OutboundMappingInstructions { 026 027 public static final OutboundMappingInstructions DEFAULT = new OutboundMappingInstructions.Builder().build(); 028 private final Map<IIdType, Long> myUseSourceDataResourceVersions; 029 030 /** 031 * Constructor 032 */ 033 private OutboundMappingInstructions(Map<IIdType, Long> theUseSourceDataResourceVersions) { 034 myUseSourceDataResourceVersions = theUseSourceDataResourceVersions; 035 } 036 037 @Nullable 038 public Long getSourceDataResourceVersion(@Nonnull IIdType theId) { 039 IIdType storageId = theId.toUnqualifiedVersionless(); 040 return myUseSourceDataResourceVersions.get(storageId); 041 } 042 043 /** 044 * Factory class for {@link OutboundMappingInstructions} 045 */ 046 public static class Builder { 047 048 private Map<IIdType, Long> myUseSourceDataResourceVersions; 049 050 /** 051 * When resolving FHIR resources to use as source data for generating HL7 v2.x messages, this method can be 052 * used to specify that a specific version of a given resource should be used. This can be used to achieve 053 * point-in-time architecture with your messages. 054 * 055 * @param theId The complete ID, must include the resource type, resource ID, and version. 056 * @return Returns a reference to the builder for easy method chaining 057 */ 058 public Builder addSourceDataResourceVersion(@Nonnull IIdType theId) { 059 if (!theId.hasResourceType()) { 060 throw new IllegalArgumentException("ID does not have a resource type specified"); 061 } 062 if (!theId.hasIdPart()) { 063 throw new IllegalArgumentException("ID does not have an ID part specified"); 064 } 065 if (!theId.isVersionIdPartValidLong()) { 066 throw new IllegalArgumentException("ID does not have a numeric version specified"); 067 } 068 IIdType storageId = theId.toUnqualifiedVersionless(); 069 if (myUseSourceDataResourceVersions == null) { 070 myUseSourceDataResourceVersions = new HashMap<>(); 071 } 072 myUseSourceDataResourceVersions.put(storageId, theId.getVersionIdPartAsLong()); 073 return this; 074 } 075 076 public OutboundMappingInstructions build() { 077 if (myUseSourceDataResourceVersions == null) { 078 myUseSourceDataResourceVersions = Map.of(); 079 } 080 return new OutboundMappingInstructions(myUseSourceDataResourceVersions); 081 } 082 083 /** 084 * Examine a MessageHeader resource to extract the list of resources found in 085 * <code>MessageHeader.focus</code>, and for any versioned references apply 086 * them to {@link #addSourceDataResourceVersion(IIdType)}. 087 * 088 * @param theMessageHeader The MessageHeader to extract from 089 */ 090 public OutboundMappingInstructions.Builder addSourceDataResourceVersionsFromMessageHeaderFocus( 091 @Nonnull IBaseResource theMessageHeader) { 092 093 FhirContext ctx = FhirContext.forCached(theMessageHeader.getStructureFhirVersionEnum()); 094 String resourceType = ctx.getResourceType(theMessageHeader); 095 Validate.isTrue( 096 resourceType.equals("MessageHeader"), 097 "theMessageHeader must be a MessageHeader resource, found: %s", 098 resourceType); 099 100 List<IBaseReference> focalReferences = ctx.newTerser() 101 .getValues( 102 theMessageHeader, 103 PublicApiConstants.FHIRPATH_MESSAGEHEADER_FOCUS, 104 IBaseReference.class, 105 false); 106 for (IBaseReference next : focalReferences) { 107 IIdType referenceId = next.getReferenceElement(); 108 if (referenceId.hasResourceType() && referenceId.hasIdPart() && referenceId.hasVersionIdPart()) { 109 addSourceDataResourceVersion(referenceId); 110 } 111 } 112 113 return this; 114 } 115 } 116}