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.interceptor; 021 022import ca.uhn.fhir.rest.client.api.IClientInterceptor; 023import ca.uhn.fhir.rest.client.api.IHttpRequest; 024import ca.uhn.fhir.rest.client.api.IHttpResponse; 025 026/** 027 * This is a client interceptor that captures the current request and response 028 * in a ThreadLocal variable, meaning that it can work in multithreaded 029 * environments without mixing up requests. 030 * <p> 031 * Use this with caution, since <b>this interceptor does not automatically clean up</b> 032 * the ThreadLocal after setting it. You must make sure to call 033 * {@link #clearThreadLocals()} after a given request has been completed, 034 * or you will end up leaving stale request/response objects associated 035 * with threads that no longer need them. 036 * </p> 037 * 038 * @see CapturingInterceptor for an equivalent interceptor that does not use a ThreadLocal 039 * @since 3.5.0 040 */ 041public class ThreadLocalCapturingInterceptor implements IClientInterceptor { 042 043 private final ThreadLocal<IHttpRequest> myRequestThreadLocal = new ThreadLocal<>(); 044 private final ThreadLocal<IHttpResponse> myResponseThreadLocal = new ThreadLocal<>(); 045 private boolean myBufferResponse; 046 047 /** 048 * This method should be called at the end of any request process, in 049 * order to clear the last request and response from the current thread. 050 */ 051 public void clearThreadLocals() { 052 myRequestThreadLocal.remove(); 053 myResponseThreadLocal.remove(); 054 } 055 056 public IHttpRequest getRequestForCurrentThread() { 057 return myRequestThreadLocal.get(); 058 } 059 060 public IHttpResponse getResponseForCurrentThread() { 061 return myResponseThreadLocal.get(); 062 } 063 064 @Override 065 public void interceptRequest(IHttpRequest theRequest) { 066 myRequestThreadLocal.set(theRequest); 067 } 068 069 @Override 070 public void interceptResponse(IHttpResponse theResponse) { 071 if (isBufferResponse()) { 072 CapturingInterceptor.bufferResponse(theResponse); 073 } 074 myResponseThreadLocal.set(theResponse); 075 } 076 077 /** 078 * Should we buffer (capture) the response body? This defaults to 079 * <code>false</code>. Set to <code>true</code> if you are planning on 080 * examining response bodies after the response processing is complete. 081 */ 082 public boolean isBufferResponse() { 083 return myBufferResponse; 084 } 085 086 /** 087 * Should we buffer (capture) the response body? This defaults to 088 * <code>false</code>. Set to <code>true</code> if you are planning on 089 * examining response bodies after the response processing is complete. 090 */ 091 public ThreadLocalCapturingInterceptor setBufferResponse(boolean theBufferResponse) { 092 myBufferResponse = theBufferResponse; 093 return this; 094 } 095}