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.fhirgw.json; 011 012import ca.cdr.api.fhirgw.model.RetryStrategyEnum; 013import ca.cdr.api.model.json.IModelJson; 014import ca.uhn.fhir.model.api.annotation.ExampleSupplier; 015import ca.uhn.fhir.rest.api.EncodingEnum; 016import com.fasterxml.jackson.annotation.JsonProperty; 017import io.swagger.v3.oas.annotations.media.Schema; 018import org.springframework.lang.NonNull; 019 020import java.util.Arrays; 021import java.util.Collections; 022import java.util.List; 023import java.util.function.Supplier; 024 025/** 026 * Contains the definition for an individual Smile CDR FHIR Gateway target server 027 */ 028@Schema( 029 name = "GatewayTarget", 030 description = "Contains the definition for an individual Smile CDR FHIR Gateway target server") 031@ExampleSupplier(GatewayTargetJson.ExampleSupplier.class) 032public class GatewayTargetJson implements IModelJson, Cloneable { 033 034 public static final int DEFAULT_TIMEOUT = 60 * 1000; 035 public static final Boolean DEFAULT_USE_HTTP_POST_FOR_ALL_SEARCHES = Boolean.FALSE; 036 public static final Boolean DEFAULT_SERVER_CAPABILITY_STATEMENT_VALIDATION_ENABLED = Boolean.TRUE; 037 public static final Boolean DEFAULT_ALLOWED_TO_FAIL = Boolean.FALSE; 038 039 @JsonProperty("id") 040 @Schema(description = "An internal ID for the target server") 041 private String myId; 042 043 @JsonProperty("baseUrl") 044 @Schema(description = "The base URL for the target server") 045 private String myBaseUrl; 046 047 @JsonProperty("fixedEndpointUrl") 048 @Schema( 049 description = 050 "If specified, will create a mapping between the baseUrl of the target, and this endpoint. When a bundle link is received that refers to this fixed endpoint, we will associate it to this target.") 051 private String myFixedEndpointUrl; 052 053 @JsonProperty("httpBasicCredentials") 054 @Schema( 055 description = 056 "If specified, these credentials (in the form `username:password`) will be passed in all client requests to the target server") 057 private String myHttpBasicCredentials; 058 059 @JsonProperty("connectTimeoutMillis") 060 @Schema( 061 description = 062 "Specifies a connection timeout (in millis) to use for communication with the target server. Default is " 063 + DEFAULT_TIMEOUT) 064 private Integer myConnectTimeoutMillis; 065 066 @JsonProperty("socketTimeoutMillis") 067 @Schema( 068 description = 069 "Specifies a socket timeout (in millis) to use for communication with the target server. Default is " 070 + DEFAULT_TIMEOUT) 071 private Integer mySocketTimeoutMillis; 072 073 @JsonProperty("clientInterceptorClasses") 074 @Schema( 075 description = 076 "If specified, signals to FHIR Gateway to load a comma-delimited list of CLIENT_RESPONSE interceptor classes that will allow the user to overwrite the HTTP response") 077 private String myClientInterceptorClasses; 078 079 public String getClientInterceptorClasses() { 080 return myClientInterceptorClasses; 081 } 082 083 public GatewayTargetJson setClientInterceptorClasses(String myClientInterceptorClasses) { 084 this.myClientInterceptorClasses = myClientInterceptorClasses; 085 return this; 086 } 087 088 @JsonProperty("resourceIdPrefix") 089 @Schema( 090 description = 091 "If specified, provides a prefix that will be added to all resource IDs and local references for the target server before returning to the client") 092 private String myResourceIdPrefix; 093 094 @JsonProperty("useHttpPostForAllSearches") 095 @Schema( 096 description = 097 "If set to true, FHIR search and paging operations against the target server will be performed using an HTTP POST instead of a GET. Default is `false`") 098 private Boolean myUseHttpPostForAllSearches; 099 100 @JsonProperty("serverCapabilityStatementValidationEnabled") 101 @Schema( 102 description = 103 "If set to false, FHIR Gateway will not validate the target server's CapabilityStatement with a request to /metadata. Default is `true`") 104 private Boolean myServerCapabilityStatementValidationEnabled; 105 106 @JsonProperty("alternateValidationPath") 107 @Schema( 108 description = 109 "An alternate validation path which can be used to validate the target server using a HTTP GET request.") 110 private String alternateValidationPath; 111 112 @JsonProperty("headersToForward") 113 @Schema( 114 description = 115 "Any headers specified by name will be copied from the incoming client request and added to requests to the target server") 116 private List<String> myHeadersToForward; 117 118 @JsonProperty("allowedToFail") 119 @Schema( 120 description = 121 "If set to true, FHIR search operations against the target server that fail will not return an error to the client, unless all targets for a given request have failed. This flag does not apply to read routes. Default is `false`") 122 private Boolean myAllowedToFail; 123 124 @JsonProperty("forcedEncoding") 125 @Schema( 126 description = 127 "If set, any requests containing a payload will have their payload re-encoded to the defined content-type before being forwarded to the target server") 128 private EncodingEnum myForcedEncoding; 129 130 @JsonProperty("retryStrategy") 131 @Schema( 132 description = 133 "If set, failed requests to this target will be retried using the specified retry configurations.") 134 private GatewayTargetRetryStrategyJson myRetryStrategy; 135 136 public Boolean getAllowedToFail() { 137 if (myAllowedToFail == null) { 138 myAllowedToFail = DEFAULT_ALLOWED_TO_FAIL; 139 } 140 return myAllowedToFail; 141 } 142 143 public void setAllowedToFail(Boolean theAllowedToFail) { 144 myAllowedToFail = theAllowedToFail; 145 } 146 147 public List<String> getHeadersToForward() { 148 if (myHeadersToForward == null) { 149 myHeadersToForward = Collections.emptyList(); 150 } 151 return myHeadersToForward; 152 } 153 154 public void setHeadersToForward(@NonNull List<String> theHeaders) { 155 myHeadersToForward = Collections.unmodifiableList(theHeaders); 156 } 157 158 public Boolean getUseHttpPostForAllSearches() { 159 if (myUseHttpPostForAllSearches == null) { 160 myUseHttpPostForAllSearches = DEFAULT_USE_HTTP_POST_FOR_ALL_SEARCHES; 161 } 162 return myUseHttpPostForAllSearches; 163 } 164 165 public void setUseHttpPostForAllSearches(Boolean theUseHttpPostForAllSearches) { 166 myUseHttpPostForAllSearches = theUseHttpPostForAllSearches; 167 } 168 169 public Boolean getServerCapabilityStatementValidationEnabled() { 170 if (myServerCapabilityStatementValidationEnabled == null) { 171 myServerCapabilityStatementValidationEnabled = DEFAULT_SERVER_CAPABILITY_STATEMENT_VALIDATION_ENABLED; 172 } 173 return myServerCapabilityStatementValidationEnabled; 174 } 175 176 public GatewayTargetJson setServerCapabilityStatementValidationEnabled( 177 Boolean theServerCapabilityStatementValidationEnabled) { 178 myServerCapabilityStatementValidationEnabled = theServerCapabilityStatementValidationEnabled; 179 return this; 180 } 181 182 public String getAlternateValidationPath() { 183 return alternateValidationPath; 184 } 185 186 public GatewayTargetJson setAlternateValidationPath(String theAlternateValidationPath) { 187 alternateValidationPath = theAlternateValidationPath; 188 return this; 189 } 190 191 public String getResourceIdPrefix() { 192 return myResourceIdPrefix; 193 } 194 195 public GatewayTargetJson setResourceIdPrefix(String theResourceIdPrefix) { 196 myResourceIdPrefix = theResourceIdPrefix; 197 return this; 198 } 199 200 public Integer getConnectTimeoutMillis() { 201 return myConnectTimeoutMillis; 202 } 203 204 public GatewayTargetJson setConnectTimeoutMillis(Integer theConnectTimeoutMillis) { 205 myConnectTimeoutMillis = theConnectTimeoutMillis; 206 return this; 207 } 208 209 public Integer getSocketTimeoutMillis() { 210 return mySocketTimeoutMillis; 211 } 212 213 public GatewayTargetJson setSocketTimeoutMillis(Integer theSocketTimeoutMillis) { 214 mySocketTimeoutMillis = theSocketTimeoutMillis; 215 return this; 216 } 217 218 public String getHttpBasicCredentials() { 219 return myHttpBasicCredentials; 220 } 221 222 public GatewayTargetJson setHttpBasicCredentials(String theHttpBasicCredentials) { 223 myHttpBasicCredentials = theHttpBasicCredentials; 224 return this; 225 } 226 227 public String getId() { 228 return myId; 229 } 230 231 public GatewayTargetJson setId(String theId) { 232 myId = theId; 233 return this; 234 } 235 236 public String getBaseUrl() { 237 return myBaseUrl; 238 } 239 240 public GatewayTargetJson setBaseUrl(String theBaseUrl) { 241 myBaseUrl = theBaseUrl; 242 return this; 243 } 244 245 public int getConnectTimeoutOrDefault() { 246 if (getConnectTimeoutMillis() != null) { 247 return getConnectTimeoutMillis(); 248 } 249 return DEFAULT_TIMEOUT; 250 } 251 252 public int getSocketTimeoutOrDefault() { 253 if (getSocketTimeoutMillis() != null) { 254 return getSocketTimeoutMillis(); 255 } 256 return DEFAULT_TIMEOUT; 257 } 258 259 public EncodingEnum getForcedEncoding() { 260 return myForcedEncoding; 261 } 262 263 public void setForcedEncoding(EncodingEnum theForcedEncoding) { 264 myForcedEncoding = theForcedEncoding; 265 } 266 267 public GatewayTargetRetryStrategyJson getRetryStrategy() { 268 return myRetryStrategy; 269 } 270 271 public void setRetryStrategy(GatewayTargetRetryStrategyJson theRetryStrategy) { 272 myRetryStrategy = theRetryStrategy; 273 } 274 275 @Override 276 public GatewayTargetJson clone() { 277 try { 278 return (GatewayTargetJson) super.clone(); 279 } catch (CloneNotSupportedException theE) { 280 throw new UnsupportedOperationException(); 281 } 282 } 283 284 public void setFixedEndpointUrl(String theFixedTargetUrl) { 285 myFixedEndpointUrl = theFixedTargetUrl; 286 } 287 288 public String getFixedEndpointUrl() { 289 return myFixedEndpointUrl; 290 } 291 292 public static class ExampleSupplier implements Supplier<GatewayTargetJson> { 293 294 @Override 295 public GatewayTargetJson get() { 296 GatewayTargetRetryStrategyJson gatewayTargetRetryStrategyJson = new GatewayTargetRetryStrategyJson(); 297 gatewayTargetRetryStrategyJson.setBackoffStrategy(RetryStrategyEnum.EXPONENTIAL); 298 gatewayTargetRetryStrategyJson.setBackoffInterval(100); 299 gatewayTargetRetryStrategyJson.setMaxRetries(2); 300 gatewayTargetRetryStrategyJson.setErrorRetryClasses( 301 Collections.singletonList("ca.uhn.fhir.rest.server.exceptions.MethodNotAllowedException")); 302 303 GatewayTargetJson targetJson = new GatewayTargetJson(); 304 targetJson.setRetryStrategy(gatewayTargetRetryStrategyJson); 305 targetJson.setBaseUrl("http://localhost:8000").setId("target1").setAllowedToFail(true); 306 targetJson.setHeadersToForward(Arrays.asList("Sample-Header-1", "Sample-Header-2")); 307 308 return targetJson; 309 } 310 } 311}