-
Notifications
You must be signed in to change notification settings - Fork 38.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow multiple executions of ClientHttpRequestInterceptors #34169
Comments
I found a workaround. It's not an ideal solution, but it does change the behavior to be chain like. Define and add this interceptor as the first interceptor in the chain: private static final ClientHttpRequestInterceptor FORCED_CHAINING = (request, body, terminalExecution) -> {
ClientHttpRequestExecution first = terminalExecution;
try {
Class<?> targetClass = Class.forName("org.springframework.http.client.InterceptingClientHttpRequest$InterceptingRequestExecution");
if (targetClass.isInstance(terminalExecution)) {
Field iteratorField = targetClass.getDeclaredField("iterator");
iteratorField.setAccessible(true);
Iterator<ClientHttpRequestInterceptor> iterator = (Iterator<ClientHttpRequestInterceptor>) iteratorField.get(terminalExecution);
// A chained version of ClientHttpRequestExecution
class InterceptedExecution implements ClientHttpRequestExecution {
ClientHttpRequestInterceptor interceptor;
ClientHttpRequestExecution next;
@Override public ClientHttpResponse execute(HttpRequest request, byte[] body) throws IOException {
// when executing, run the interceptor and point it to the next intercepting execution or the terminal if no next.
return interceptor.intercept(request, body, next == null ? terminalExecution : next);
}
}
// Build the intercepting execution chain in iteration order
InterceptedExecution previous = null;
while (iterator.hasNext()) {
InterceptedExecution current = new InterceptedExecution();
current.interceptor = iterator.next();
if (previous != null) previous.next = current; else first = current;
previous = current;
}
}
} catch (Exception ignored) {}
return first.execute(request, body);
}; Application code must add this interceptor as the first in the list, and it will then manage the execution of the remaining list items in chained fashion (followed by the terminal request execution.) This interceptor will not affect request processing if the concrete spring class is not the specific target class of the workaround. However, this uses reflection to grab the private iterator using within the target spring class and this exhausts that iterator. That kills two birds with one stone; it causes the spring class to think there are no interceptors left for it to run and it allows this code to discover and build a proper chain structure for remaining interceptors. Dropping this interceptor into the issue demo code results in the expected output. |
Hello @Russell-Allen and thanks for the detailed report. Now we are working on the next 7.0 major generation and this change seems acceptable there. |
org.springframework.http.client.ClientHttpRequestInterceptor
javadoc describes the interceptor as a "chain", and I assumed that theClientHttpRequestExecution execution
argument was a reference to the next link/interceptor in the chain (or the terminal request execution.)The above assumption fails if the interceptor makes more than one call to
execution.execute(request,body)
. That is, all calls after the first will skip any following interceptors.Issue Demo Code
Root Cause?
I traced the issue down to the
org.springframework.http.client.InterceptingClientHttpRequest.InterceptingRequestExecution
class which is the type being passed into the interceptors as theexecution
argument. Here's an annotated snippet of that class showing the issue:The use of an iterator and
iterator.hasNext()
results in a visitor behavioral pattern instead of an execution chain.My expected behavior for a chain pattern is that any node in the chain has a reference to the next node in the chain and that there is no state outside of the chain arguments. It seems quite odd that calls to execute the next interceptor in the chain has the side-effect of changing the next reference.
Example Use Case
Using a request interceptor, detect a request that results in a
401 Unauthorized
response and replay the request with a refreshed token in the authorization header. For example:Workarounds
RestClient
into the repeating interceptor - If the interceptor had a reference to the otherwise fully configured rest client, it could use it to make follow on requests. Note that those requests would repeat the interceptor chain from the start instead of here forward. It also tightly couples the injected rest client to the interceptor; in fact, that injection would likely need to happen per request which would require the calling code to create an instance of the repeating interceptor (with a reference to the in flight rest client) and add that instance to the rest client (circular references!). This smells.InterceptingClientHttpRequestFactory
which is easy enough to replicate and have it use a fixed version ofInterceptingClientHttpRequest
, but ... I don't see a way to set my version intoDefaultRestClient.interceptingRequestFactory
; a lazily set field in a package-private final class. While I could use reflection to overcome the accessibility issues, its a brittle solution that will earn me the stink eye from peers.The text was updated successfully, but these errors were encountered: