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.subscription.model; 021 022import ca.uhn.fhir.context.FhirContext; 023import ca.uhn.fhir.interceptor.model.RequestPartitionId; 024import ca.uhn.fhir.parser.IParser; 025import ca.uhn.fhir.rest.api.EncodingEnum; 026import ca.uhn.fhir.rest.server.messaging.BaseResourceMessage; 027import ca.uhn.fhir.rest.server.messaging.IResourceMessage; 028import com.fasterxml.jackson.annotation.JsonIgnore; 029import com.fasterxml.jackson.annotation.JsonProperty; 030import jakarta.annotation.Nullable; 031import org.apache.commons.lang3.StringUtils; 032import org.apache.commons.lang3.builder.ToStringBuilder; 033import org.hl7.fhir.instance.model.api.IBaseResource; 034import org.hl7.fhir.instance.model.api.IIdType; 035 036import static org.apache.commons.lang3.StringUtils.isNotBlank; 037 038@SuppressWarnings("WeakerAccess") 039public class ResourceDeliveryMessage extends BaseResourceMessage implements IResourceMessage { 040 041 @JsonProperty("canonicalSubscription") 042 private CanonicalSubscription mySubscription; 043 044 @JsonProperty("partitionId") 045 private RequestPartitionId myPartitionId; 046 047 @JsonProperty("payload") 048 private String myPayloadString; 049 050 @JsonProperty("payloadId") 051 private String myPayloadId; 052 053 @JsonIgnore 054 private transient IBaseResource myPayloadDecoded; 055 056 /** 057 * Constructor 058 */ 059 public ResourceDeliveryMessage() { 060 super(); 061 myPartitionId = RequestPartitionId.defaultPartition(); 062 } 063 064 public IBaseResource getPayload(FhirContext theCtx) { 065 IBaseResource retVal = myPayloadDecoded; 066 if (retVal == null && isNotBlank(myPayloadString)) { 067 IParser parser = EncodingEnum.detectEncoding(myPayloadString).newParser(theCtx); 068 retVal = parser.parseResource(myPayloadString); 069 myPayloadDecoded = retVal; 070 } 071 return retVal; 072 } 073 074 public String getPayloadString() { 075 if (this.myPayloadString != null) { 076 return this.myPayloadString; 077 } 078 079 return ""; 080 } 081 082 public IIdType getPayloadId(FhirContext theCtx) { 083 IIdType retVal = null; 084 if (myPayloadId != null) { 085 retVal = theCtx.getVersion().newIdType().setValue(myPayloadId); 086 } 087 return retVal; 088 } 089 090 public CanonicalSubscription getSubscription() { 091 return mySubscription; 092 } 093 094 public void setSubscription(CanonicalSubscription theSubscription) { 095 mySubscription = theSubscription; 096 } 097 098 public void setPayload(FhirContext theCtx, IBaseResource thePayload, EncodingEnum theEncoding) { 099 /* 100 * Note that we populate the raw string but don't keep the parsed resource around when we set this. This 101 * has two reasons: 102 * - If we build up a big queue of these on an in-memory queue, we aren't taking up double the memory 103 * - If use a serializing queue, we aren't behaving differently (and therefore possibly missing things 104 * in tests) 105 */ 106 myPayloadString = theEncoding.newParser(theCtx).encodeResourceToString(thePayload); 107 myPayloadId = thePayload.getIdElement().toUnqualifiedVersionless().getValue(); 108 } 109 110 public void setPayloadToNull() { 111 myPayloadString = null; 112 } 113 114 @Override 115 public String getPayloadId() { 116 return myPayloadId; 117 } 118 119 public void setPayloadId(IIdType thePayloadId) { 120 myPayloadId = null; 121 if (thePayloadId != null) { 122 myPayloadId = thePayloadId.getValue(); 123 } 124 } 125 126 public RequestPartitionId getRequestPartitionId() { 127 return myPartitionId; 128 } 129 130 public void setPartitionId(RequestPartitionId thePartitionId) { 131 myPartitionId = thePartitionId; 132 } 133 134 @Override 135 public String toString() { 136 return new ToStringBuilder(this) 137 .append("mySubscription", mySubscription == null ? "null" : mySubscription.getIdElementString()) 138 // it isn't safe to log payloads 139 .append("myPayloadString", "[Not Logged]") 140 .append("myPayload", myPayloadDecoded) 141 .append("myPayloadId", myPayloadId) 142 .append("myPartitionId", myPartitionId) 143 .append("myOperationType", getOperationType()) 144 .toString(); 145 } 146 147 /** 148 * Helper method to fetch the subscription ID 149 */ 150 public String getSubscriptionId(FhirContext theFhirContext) { 151 String retVal = null; 152 if (getSubscription() != null) { 153 IIdType idElement = getSubscription().getIdElement(theFhirContext); 154 if (idElement != null) { 155 retVal = idElement.getValue(); 156 } 157 } 158 return retVal; 159 } 160 161 @Nullable 162 @Override 163 public String getMessageKeyOrDefault() { 164 return StringUtils.defaultString(super.getMessageKeyOrNull(), myPayloadId); 165 } 166}