001/*- 002 * #%L 003 * HAPI FHIR - Server 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.server.messaging; 021 022import ca.uhn.fhir.i18n.Msg; 023import ca.uhn.fhir.model.api.IModelJson; 024import ca.uhn.fhir.rest.api.RestOperationTypeEnum; 025import com.fasterxml.jackson.annotation.JsonProperty; 026import com.google.common.annotations.VisibleForTesting; 027import jakarta.annotation.Nullable; 028import org.apache.commons.lang3.ObjectUtils; 029import org.apache.commons.lang3.Validate; 030 031import java.util.Collections; 032import java.util.HashMap; 033import java.util.Map; 034import java.util.Objects; 035import java.util.Optional; 036 037import static org.apache.commons.lang3.StringUtils.defaultString; 038 039@SuppressWarnings("WeakerAccess") 040public abstract class BaseResourceMessage implements IResourceMessage, IModelJson { 041 042 @JsonProperty("operationType") 043 protected BaseResourceModifiedMessage.OperationTypeEnum myOperationType; 044 045 @JsonProperty("attributes") 046 private Map<String, String> myAttributes; 047 048 @JsonProperty("transactionId") 049 private String myTransactionId; 050 051 @JsonProperty("mediaType") 052 private String myMediaType; 053 054 /** 055 * This is used by any message going to kafka for topic partition selection purposes. 056 */ 057 @JsonProperty("messageKey") 058 private String myMessageKey; 059 060 /** 061 * Returns an attribute stored in this message. 062 * <p> 063 * Attributes are just a spot for user data of any kind to be 064 * added to the message for pasing along the subscription processing 065 * pipeline (typically by interceptors). Values will be carried from the beginning to the end. 066 * </p> 067 * <p> 068 * Note that messages are designed to be passed into queueing systems 069 * and serialized as JSON. As a result, only strings are currently allowed 070 * as values. 071 * </p> 072 */ 073 public Optional<String> getAttribute(String theKey) { 074 Validate.notBlank(theKey); 075 if (myAttributes == null) { 076 return Optional.empty(); 077 } 078 return Optional.ofNullable(myAttributes.get(theKey)); 079 } 080 081 /** 082 * Sets an attribute stored in this message. 083 * <p> 084 * Attributes are just a spot for user data of any kind to be 085 * added to the message for passing along the subscription processing 086 * pipeline (typically by interceptors). Values will be carried from the beginning to the end. 087 * </p> 088 * <p> 089 * Note that messages are designed to be passed into queueing systems 090 * and serialized as JSON. As a result, only strings are currently allowed 091 * as values. 092 * </p> 093 * 094 * @param theKey The key (must not be null or blank) 095 * @param theValue The value (must not be null) 096 */ 097 public void setAttribute(String theKey, String theValue) { 098 Validate.notBlank(theKey); 099 Validate.notNull(theValue); 100 if (myAttributes == null) { 101 myAttributes = new HashMap<>(); 102 } 103 myAttributes.put(theKey, theValue); 104 } 105 106 /** 107 * Copies any attributes from the given message into this messsage. 108 * 109 * @see #setAttribute(String, String) 110 * @see #getAttribute(String) 111 */ 112 public void copyAdditionalPropertiesFrom(BaseResourceMessage theMsg) { 113 if (theMsg.myAttributes != null) { 114 if (myAttributes == null) { 115 myAttributes = new HashMap<>(); 116 } 117 myAttributes.putAll(theMsg.myAttributes); 118 } 119 } 120 121 /** 122 * Returns the {@link OperationTypeEnum} that is occurring to the Resource of the message 123 * 124 * @return the operation type. 125 */ 126 public BaseResourceModifiedMessage.OperationTypeEnum getOperationType() { 127 return myOperationType; 128 } 129 130 /** 131 * Sets the {@link OperationTypeEnum} occuring to the resource of the message. 132 * 133 * @param theOperationType The operation type to set. 134 */ 135 public void setOperationType(BaseResourceModifiedMessage.OperationTypeEnum theOperationType) { 136 myOperationType = theOperationType; 137 } 138 139 /** 140 * Retrieve the transaction ID related to this message. 141 * 142 * @return the transaction ID, or null. 143 */ 144 @Nullable 145 public String getTransactionId() { 146 return myTransactionId; 147 } 148 149 /** 150 * Adds a transaction ID to this message. This ID can be used for many purposes. For example, performing tracing 151 * across asynchronous hooks, tying data together, or downstream logging purposes. 152 * <p> 153 * One current internal implementation uses this field to tie back MDM processing results (which are asynchronous) 154 * to the original transaction log that caused the MDM processing to occur. 155 * 156 * @param theTransactionId An ID representing a transaction of relevance to this message. 157 */ 158 public void setTransactionId(String theTransactionId) { 159 myTransactionId = theTransactionId; 160 } 161 162 public String getMediaType() { 163 return myMediaType; 164 } 165 166 public void setMediaType(String theMediaType) { 167 myMediaType = theMediaType; 168 } 169 170 @Deprecated 171 @Nullable 172 public String getMessageKeyOrNull() { 173 return getMessageKey(); 174 } 175 176 @Nullable 177 public String getMessageKey() { 178 return myMessageKey; 179 } 180 181 public void setMessageKey(String theMessageKey) { 182 myMessageKey = theMessageKey; 183 } 184 185 /** 186 * Returns {@link #getMessageKey()} or {@link #getMessageKeyDefaultValue()} when {@link #getMessageKey()} returns <code>null</code>. 187 * 188 * @return the message key value or default 189 */ 190 @Nullable 191 public String getMessageKeyOrDefault() { 192 return defaultString(getMessageKey(), getMessageKeyDefaultValue()); 193 } 194 195 /** 196 * Provides a fallback value when method {@link #getMessageKey()} returns <code>null</code>. 197 * 198 * @return null by default 199 */ 200 @Nullable 201 protected String getMessageKeyDefaultValue() { 202 return null; 203 } 204 205 public enum OperationTypeEnum { 206 CREATE(RestOperationTypeEnum.CREATE), 207 UPDATE(RestOperationTypeEnum.UPDATE), 208 DELETE(RestOperationTypeEnum.DELETE), 209 MANUALLY_TRIGGERED(RestOperationTypeEnum.UPDATE), 210 TRANSACTION(RestOperationTypeEnum.UPDATE); 211 212 private final RestOperationTypeEnum myRestOperationTypeEnum; 213 214 OperationTypeEnum(RestOperationTypeEnum theRestOperationTypeEnum) { 215 myRestOperationTypeEnum = theRestOperationTypeEnum; 216 } 217 218 public static OperationTypeEnum from(RestOperationTypeEnum theRestOperationType) { 219 switch (theRestOperationType) { 220 case CREATE: 221 return CREATE; 222 case UPDATE: 223 return UPDATE; 224 case DELETE: 225 return DELETE; 226 default: 227 throw new IllegalArgumentException( 228 Msg.code(2348) + "Unsupported operation type: " + theRestOperationType); 229 } 230 } 231 232 public RestOperationTypeEnum asRestOperationType() { 233 return myRestOperationTypeEnum; 234 } 235 } 236 237 @VisibleForTesting 238 public Map<String, String> getAttributes() { 239 return ObjectUtils.defaultIfNull(myAttributes, Collections.emptyMap()); 240 } 241 242 @Override 243 public boolean equals(Object theO) { 244 if (this == theO) return true; 245 if (theO == null || getClass() != theO.getClass()) return false; 246 BaseResourceMessage that = (BaseResourceMessage) theO; 247 return getOperationType() == that.getOperationType() 248 && Objects.equals(getAttributes(), that.getAttributes()) 249 && Objects.equals(getTransactionId(), that.getTransactionId()) 250 && Objects.equals(getMediaType(), that.getMediaType()) 251 && Objects.equals(getMessageKey(), that.getMessageKey()); 252 } 253 254 @Override 255 public int hashCode() { 256 return Objects.hash(getOperationType(), getAttributes(), getTransactionId(), getMediaType()); 257 } 258}