001/* 002 * #%L 003 * HAPI FHIR - Client 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.client.impl; 021 022import ca.uhn.fhir.context.ConfigurationException; 023import ca.uhn.fhir.context.FhirContext; 024import ca.uhn.fhir.context.FhirVersionEnum; 025import ca.uhn.fhir.i18n.Msg; 026import ca.uhn.fhir.parser.DataFormatException; 027import ca.uhn.fhir.rest.api.Constants; 028import ca.uhn.fhir.rest.client.api.IGenericClient; 029import ca.uhn.fhir.rest.client.api.IHttpClient; 030import ca.uhn.fhir.rest.client.api.IRestfulClient; 031import ca.uhn.fhir.rest.client.api.IRestfulClientFactory; 032import ca.uhn.fhir.rest.client.api.ServerValidationModeEnum; 033import ca.uhn.fhir.rest.client.exceptions.FhirClientConnectionException; 034import ca.uhn.fhir.rest.client.exceptions.FhirClientInappropriateForServerException; 035import ca.uhn.fhir.rest.client.method.BaseMethodBinding; 036import ca.uhn.fhir.util.FhirTerser; 037import org.apache.commons.lang3.StringUtils; 038import org.apache.commons.lang3.Validate; 039import org.hl7.fhir.instance.model.api.IBaseResource; 040import org.hl7.fhir.instance.model.api.IPrimitiveType; 041 042import java.lang.reflect.InvocationHandler; 043import java.lang.reflect.Method; 044import java.lang.reflect.Proxy; 045import java.util.Collections; 046import java.util.HashMap; 047import java.util.HashSet; 048import java.util.Map; 049import java.util.Set; 050 051/** 052 * Base class for a REST client factory implementation 053 */ 054public abstract class RestfulClientFactory implements IRestfulClientFactory { 055 private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(RestfulClientFactory.class); 056 057 private final Set<String> myValidatedServerBaseUrls = Collections.synchronizedSet(new HashSet<>()); 058 private int myConnectionRequestTimeout = DEFAULT_CONNECTION_REQUEST_TIMEOUT; 059 private int myConnectTimeout = DEFAULT_CONNECT_TIMEOUT; 060 private int myConnectionTimeToLive = DEFAULT_CONNECTION_TTL; 061 private FhirContext myContext; 062 private final Map<Class<? extends IRestfulClient>, ClientInvocationHandlerFactory> myInvocationHandlers = 063 new HashMap<>(); 064 private ServerValidationModeEnum myServerValidationMode = DEFAULT_SERVER_VALIDATION_MODE; 065 private int mySocketTimeout = DEFAULT_SOCKET_TIMEOUT; 066 private String myProxyUsername; 067 private String myProxyPassword; 068 private int myPoolMaxTotal = DEFAULT_POOL_MAX; 069 private int myPoolMaxPerRoute = DEFAULT_POOL_MAX_PER_ROUTE; 070 071 /** 072 * Constructor 073 */ 074 public RestfulClientFactory() {} 075 076 /** 077 * Constructor 078 * 079 * @param theFhirContext The context 080 */ 081 public RestfulClientFactory(FhirContext theFhirContext) { 082 myContext = theFhirContext; 083 } 084 085 @Override 086 public synchronized int getConnectionRequestTimeout() { 087 return myConnectionRequestTimeout; 088 } 089 090 @Override 091 public synchronized int getConnectTimeout() { 092 return myConnectTimeout; 093 } 094 095 @Override 096 public synchronized int getConnectionTimeToLive() { 097 return myConnectionTimeToLive; 098 } 099 100 /** 101 * Return the proxy username to authenticate with the HTTP proxy 102 */ 103 protected synchronized String getProxyUsername() { 104 return myProxyUsername; 105 } 106 107 /** 108 * Return the proxy password to authenticate with the HTTP proxy 109 */ 110 protected synchronized String getProxyPassword() { 111 return myProxyPassword; 112 } 113 114 @Override 115 public synchronized void setProxyCredentials(String theUsername, String thePassword) { 116 myProxyUsername = theUsername; 117 myProxyPassword = thePassword; 118 } 119 120 @Override 121 public synchronized ServerValidationModeEnum getServerValidationMode() { 122 return myServerValidationMode; 123 } 124 125 @Override 126 public synchronized int getSocketTimeout() { 127 return mySocketTimeout; 128 } 129 130 @Override 131 public synchronized int getPoolMaxTotal() { 132 return myPoolMaxTotal; 133 } 134 135 @Override 136 public synchronized int getPoolMaxPerRoute() { 137 return myPoolMaxPerRoute; 138 } 139 140 @SuppressWarnings("unchecked") 141 private <T extends IRestfulClient> T instantiateProxy( 142 Class<T> theClientType, InvocationHandler theInvocationHandler) { 143 return (T) Proxy.newProxyInstance( 144 theClientType.getClassLoader(), new Class[] {theClientType}, theInvocationHandler); 145 } 146 147 /** 148 * Instantiates a new client instance 149 * 150 * @param theClientType The client type, which is an interface type to be instantiated 151 * @param theServerBase The URL of the base for the restful FHIR server to connect to 152 * @return A newly created client 153 * @throws ConfigurationException If the interface type is not an interface 154 */ 155 @Override 156 public synchronized <T extends IRestfulClient> T newClient(Class<T> theClientType, String theServerBase) { 157 validateConfigured(); 158 159 if (!theClientType.isInterface()) { 160 throw new ConfigurationException( 161 Msg.code(1354) + theClientType.getCanonicalName() + " is not an interface"); 162 } 163 164 ClientInvocationHandlerFactory invocationHandler = myInvocationHandlers.get(theClientType); 165 if (invocationHandler == null) { 166 IHttpClient httpClient = getHttpClient(theServerBase); 167 invocationHandler = new ClientInvocationHandlerFactory(httpClient, myContext, theServerBase, theClientType); 168 for (Method nextMethod : theClientType.getMethods()) { 169 BaseMethodBinding<?> binding = BaseMethodBinding.bindMethod(nextMethod, myContext, null); 170 invocationHandler.addBinding(nextMethod, binding); 171 } 172 myInvocationHandlers.put(theClientType, invocationHandler); 173 } 174 175 return instantiateProxy(theClientType, invocationHandler.newInvocationHandler(this)); 176 } 177 178 /** 179 * Called automatically before the first use of this factory to ensure that 180 * the configuration is sane. Subclasses may override, but should also call 181 * <code>super.validateConfigured()</code> 182 */ 183 protected void validateConfigured() { 184 if (getFhirContext() == null) { 185 throw new IllegalStateException(Msg.code(1355) + getClass().getSimpleName() 186 + " does not have FhirContext defined. This must be set via " 187 + getClass().getSimpleName() + "#setFhirContext(FhirContext)"); 188 } 189 } 190 191 @Override 192 public synchronized IGenericClient newGenericClient(String theServerBase) { 193 validateConfigured(); 194 IHttpClient httpClient = getHttpClient(theServerBase); 195 196 return new GenericClient(myContext, httpClient, theServerBase, this); 197 } 198 199 private String normalizeBaseUrlForMap(String theServerBase) { 200 String serverBase = theServerBase; 201 if (!serverBase.endsWith("/")) { 202 serverBase = serverBase + "/"; 203 } 204 return serverBase; 205 } 206 207 @Override 208 public synchronized void setConnectionRequestTimeout(int theConnectionRequestTimeout) { 209 myConnectionRequestTimeout = theConnectionRequestTimeout; 210 resetHttpClient(); 211 } 212 213 @Override 214 public synchronized void setConnectTimeout(int theConnectTimeout) { 215 myConnectTimeout = theConnectTimeout; 216 resetHttpClient(); 217 } 218 219 @Override 220 public synchronized void setConnectionTimeToLive(int theConnectionTimeToLive) { 221 myConnectionTimeToLive = theConnectionTimeToLive; 222 resetHttpClient(); 223 } 224 225 /** 226 * Sets the context associated with this client factory. Must not be called more than once. 227 */ 228 public void setFhirContext(FhirContext theContext) { 229 if (myContext != null && myContext != theContext) { 230 throw new IllegalStateException( 231 Msg.code(1356) 232 + "RestfulClientFactory instance is already associated with one FhirContext. RestfulClientFactory instances can not be shared."); 233 } 234 myContext = theContext; 235 } 236 237 /** 238 * Return the fhir context 239 * 240 * @return the fhir context 241 */ 242 public FhirContext getFhirContext() { 243 return myContext; 244 } 245 246 @Override 247 public synchronized void setServerValidationMode(ServerValidationModeEnum theServerValidationMode) { 248 Validate.notNull(theServerValidationMode, "theServerValidationMode may not be null"); 249 myServerValidationMode = theServerValidationMode; 250 } 251 252 @Override 253 public synchronized void setSocketTimeout(int theSocketTimeout) { 254 mySocketTimeout = theSocketTimeout; 255 resetHttpClient(); 256 } 257 258 @Override 259 public synchronized void setPoolMaxTotal(int thePoolMaxTotal) { 260 myPoolMaxTotal = thePoolMaxTotal; 261 resetHttpClient(); 262 } 263 264 @Override 265 public synchronized void setPoolMaxPerRoute(int thePoolMaxPerRoute) { 266 myPoolMaxPerRoute = thePoolMaxPerRoute; 267 resetHttpClient(); 268 } 269 270 @Deprecated // override deprecated method 271 @Override 272 public synchronized ServerValidationModeEnum getServerValidationModeEnum() { 273 return getServerValidationMode(); 274 } 275 276 @Deprecated // override deprecated method 277 @Override 278 public synchronized void setServerValidationModeEnum(ServerValidationModeEnum theServerValidationMode) { 279 setServerValidationMode(theServerValidationMode); 280 } 281 282 @Override 283 public void validateServerBaseIfConfiguredToDoSo( 284 String theServerBase, IHttpClient theHttpClient, IRestfulClient theClient) { 285 String serverBase = normalizeBaseUrlForMap(theServerBase); 286 287 switch (getServerValidationMode()) { 288 case NEVER: 289 break; 290 291 case ONCE: 292 synchronized (myValidatedServerBaseUrls) { 293 if (myValidatedServerBaseUrls.add(serverBase)) { 294 validateServerBase(serverBase, theHttpClient, theClient); 295 } 296 } 297 break; 298 } 299 } 300 301 @SuppressWarnings("unchecked") 302 @Override 303 public void validateServerBase(String theServerBase, IHttpClient theHttpClient, IRestfulClient theClient) { 304 GenericClient client = new GenericClient(myContext, theHttpClient, theServerBase, this); 305 306 client.setInterceptorService(theClient.getInterceptorService()); 307 client.setEncoding(theClient.getEncoding()); 308 client.setDontValidateConformance(true); 309 310 IBaseResource conformance; 311 try { 312 String capabilityStatementResourceName = "CapabilityStatement"; 313 if (myContext.getVersion().getVersion().isOlderThan(FhirVersionEnum.DSTU3)) { 314 capabilityStatementResourceName = "Conformance"; 315 } 316 317 @SuppressWarnings("rawtypes") 318 Class implementingClass; 319 try { 320 implementingClass = myContext 321 .getResourceDefinition(capabilityStatementResourceName) 322 .getImplementingClass(); 323 } catch (DataFormatException e) { 324 if (!myContext.getVersion().getVersion().isOlderThan(FhirVersionEnum.DSTU3)) { 325 capabilityStatementResourceName = "Conformance"; 326 implementingClass = myContext 327 .getResourceDefinition(capabilityStatementResourceName) 328 .getImplementingClass(); 329 } else { 330 throw e; 331 } 332 } 333 try { 334 conformance = (IBaseResource) 335 client.fetchConformance().ofType(implementingClass).execute(); 336 } catch (FhirClientConnectionException e) { 337 if (!myContext.getVersion().getVersion().isOlderThan(FhirVersionEnum.DSTU3) 338 && e.getCause() instanceof DataFormatException) { 339 capabilityStatementResourceName = "CapabilityStatement"; 340 implementingClass = myContext 341 .getResourceDefinition(capabilityStatementResourceName) 342 .getImplementingClass(); 343 conformance = (IBaseResource) 344 client.fetchConformance().ofType(implementingClass).execute(); 345 } else { 346 throw e; 347 } 348 } 349 } catch (FhirClientConnectionException e) { 350 String msg = myContext 351 .getLocalizer() 352 .getMessage( 353 RestfulClientFactory.class, 354 "failedToRetrieveConformance", 355 theServerBase + Constants.URL_TOKEN_METADATA); 356 throw new FhirClientConnectionException(Msg.code(1357) + msg, e); 357 } 358 359 FhirTerser t = myContext.newTerser(); 360 String serverFhirVersionString = null; 361 Object value = t.getSingleValueOrNull(conformance, "fhirVersion"); 362 if (value instanceof IPrimitiveType) { 363 serverFhirVersionString = ((IPrimitiveType<?>) value).getValueAsString(); 364 } 365 FhirVersionEnum serverFhirVersionEnum = null; 366 if (StringUtils.isBlank(serverFhirVersionString)) { 367 // we'll be lenient and accept this 368 ourLog.debug("Server conformance statement does not indicate the FHIR version"); 369 } else { 370 if (serverFhirVersionString.equals(FhirVersionEnum.DSTU2.getFhirVersionString())) { 371 serverFhirVersionEnum = FhirVersionEnum.DSTU2; 372 } else if (serverFhirVersionString.equals(FhirVersionEnum.DSTU2_1.getFhirVersionString())) { 373 serverFhirVersionEnum = FhirVersionEnum.DSTU2_1; 374 } else if (serverFhirVersionString.equals(FhirVersionEnum.DSTU3.getFhirVersionString())) { 375 serverFhirVersionEnum = FhirVersionEnum.DSTU3; 376 } else if (serverFhirVersionString.equals(FhirVersionEnum.R4.getFhirVersionString())) { 377 serverFhirVersionEnum = FhirVersionEnum.R4; 378 } else { 379 // we'll be lenient and accept this 380 ourLog.debug( 381 "Server conformance statement indicates unknown FHIR version: {}", serverFhirVersionString); 382 } 383 } 384 385 if (serverFhirVersionEnum != null) { 386 FhirVersionEnum contextFhirVersion = myContext.getVersion().getVersion(); 387 if (!contextFhirVersion.isEquivalentTo(serverFhirVersionEnum)) { 388 throw new FhirClientInappropriateForServerException(Msg.code(1358) 389 + myContext 390 .getLocalizer() 391 .getMessage( 392 RestfulClientFactory.class, 393 "wrongVersionInConformance", 394 theServerBase + Constants.URL_TOKEN_METADATA, 395 serverFhirVersionString, 396 serverFhirVersionEnum, 397 contextFhirVersion)); 398 } 399 } 400 401 String serverBase = normalizeBaseUrlForMap(theServerBase); 402 myValidatedServerBaseUrls.add(serverBase); 403 } 404 405 /** 406 * Get the http client for the given server base 407 * 408 * @param theServerBase the server base 409 * @return the http client 410 */ 411 protected abstract IHttpClient getHttpClient(String theServerBase); 412 413 /** 414 * Reset the http client. This method is used when parameters have been set and a 415 * new http client needs to be created 416 */ 417 protected abstract void resetHttpClient(); 418}