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.uhn.fhir.context.FhirContext; 013import ca.uhn.fhir.jpa.api.dao.DaoRegistry; 014import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao; 015import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscription; 016import ca.uhn.fhir.rest.api.server.SystemRequestDetails; 017import ca.uhn.hl7v2.HL7Exception; 018import org.apache.commons.lang3.Validate; 019import org.hl7.fhir.instance.model.api.IBaseReference; 020import org.hl7.fhir.instance.model.api.IBaseResource; 021import org.hl7.fhir.instance.model.api.IIdType; 022 023import java.util.List; 024 025/** 026 * This interface is intended to be implemented by custom HL7v2 outbound mapper 027 * classes. Implementations receive a focal resource (which was the target of 028 * a Subscription) and produce a {@link MappingTarget} containing an HL7v2 029 * message to send. 030 */ 031public interface IHl7V2OutboundCustomMapper { 032 033 /** 034 * Convert a resource into a collection of HL7v2 message(s) to send. 035 * <p> 036 * Note that transmission of the generated messages to the receiving system will happen synchronously, 037 * in the order that the messages are returned in the list. If multiple messages are returned and a message 038 * fails to be sent (i.e. because the receiving system rejects it or is unreachable), the entire collection 039 * may be resent, including messages which have already previously been sent. 040 * </p> 041 * 042 * @param theConversionContext An object containing various information about the requested conversion 043 * @return A collection of {@link MappingTarget} instances containing HL7v2 message(s) to send. Implementations return an empty list or {@literal null} if they wish to not send any message. 044 */ 045 List<MappingTarget<?>> convert(ConversionContext theConversionContext) throws HL7Exception; 046 047 /** 048 * This class contains parameters for the {@link ConversionContext} method. 049 */ 050 class ConversionContext { 051 052 private final CanonicalSubscription mySubscription; 053 private final IBaseResource myFocalResource; 054 private final IHl7V2OutboundMapperSvc myHl7V2OutboundMapperSvc; 055 private final DaoRegistry myDaoRegistry; 056 057 /** 058 * Constructor 059 */ 060 public ConversionContext( 061 CanonicalSubscription theSubscription, 062 IBaseResource theFocalResource, 063 IHl7V2OutboundMapperSvc theHl7V2OutboundMapperSvc, 064 DaoRegistry theDaoRegistry) { 065 mySubscription = theSubscription; 066 myFocalResource = theFocalResource; 067 myHl7V2OutboundMapperSvc = theHl7V2OutboundMapperSvc; 068 myDaoRegistry = theDaoRegistry; 069 } 070 071 /** 072 * @return The outbound mapper service, which is used to create {@link MappingTarget} instances, and can be used to apply Smile CDR default segment or message mappings. 073 */ 074 public IHl7V2OutboundMapperSvc getHl7V2OutboundMapperSvc() { 075 return myHl7V2OutboundMapperSvc; 076 } 077 078 /** 079 * @return The FHIR subscription which triggered this action 080 */ 081 public CanonicalSubscription getSubscription() { 082 return mySubscription; 083 } 084 085 /** 086 * @return The resource which triggered the Subscription and caused this conversion 087 */ 088 public IBaseResource getFocalResource() { 089 return myFocalResource; 090 } 091 092 /** 093 * This method may be used to resolve a resource from storage 094 */ 095 @SuppressWarnings({"unchecked", "rawtypes"}) 096 public <T extends IBaseResource> T resolveResource(IIdType theResourceId) { 097 Validate.notNull(theResourceId, "theResourceId must not be null"); 098 Validate.isTrue(theResourceId.hasResourceType(), "theResourceId must contain a resource type"); 099 Validate.isTrue(theResourceId.hasIdPart(), "theResourceId must contain a resource ID"); 100 101 IFhirResourceDao dao = myDaoRegistry.getResourceDao(theResourceId.getResourceType()); 102 return (T) dao.read(theResourceId, new SystemRequestDetails()); 103 } 104 105 /** 106 * Given a MessageHeader resource in {@literal theMessageHeader}, searches all 107 * repetitions of {@literal MessageHeader.focus} for references to resource of 108 * type {@literal theResourceType}, then resolves and returns the first. 109 * 110 * @param theResourceType The target resource type to search for, e.g. {@literal Patient.class} 111 * @param theMessageHeader The MessageHeader resource to search 112 */ 113 public <T extends IBaseResource> T resolveFirstResourceFromMessageHeaderFocus( 114 Class<T> theResourceType, IBaseResource theMessageHeader) { 115 FhirContext ctx = FhirContext.forCached(theMessageHeader.getStructureFhirVersionEnum()); 116 String type = ctx.getResourceType(theResourceType); 117 List<IBaseReference> focusReferences = 118 ctx.newTerser().getValues(theMessageHeader, "focus", IBaseReference.class, false); 119 IIdType firstId = focusReferences.stream() 120 .map(t -> t.getReferenceElement()) 121 .filter(t -> type.equals(t.getResourceType())) 122 .findFirst() 123 .orElseThrow(); 124 return resolveResource(firstId); 125 } 126 } 127}