Skip to content
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

Retry on configurable exception #6991

Draft
wants to merge 15 commits into
base: main
Choose a base branch
from

Conversation

YuriyHolinko
Copy link

@YuriyHolinko YuriyHolinko commented Jan 6, 2025

Number of retryable exceptions is very limited in the current logic so we have data loss in case of any other(not mentioned in the current java code) IO exception happen.
As we might have different networks we might experience different exceptions. In my environment I caught a few exceptions that very likely need to be retried and they are not listed as retryable in current code.
since each environment is different I suggest to have an ability to configure retryable exceptions

the change is fully backward compatible and does not change default behaviour of the library.

@YuriyHolinko YuriyHolinko requested a review from a team as a code owner January 6, 2025 21:07
Copy link

linux-foundation-easycla bot commented Jan 6, 2025

CLA Signed

The committers listed above are authorized under a signed CLA.

@YuriyHolinko
Copy link
Author

Resolves #6962

Copy link

codecov bot commented Jan 6, 2025

Codecov Report

All modified and coverable lines are covered by tests ✅

Project coverage is 89.92%. Comparing base (81d165d) to head (8913b52).

Additional details and impacted files
@@             Coverage Diff              @@
##               main    #6991      +/-   ##
============================================
+ Coverage     89.88%   89.92%   +0.03%     
- Complexity     6655     6665      +10     
============================================
  Files           748      748              
  Lines         20077    20085       +8     
  Branches       1969     1970       +1     
============================================
+ Hits          18047    18062      +15     
+ Misses         1435     1426       -9     
- Partials        595      597       +2     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

RetryInterceptor::isRetryableException,
e ->
retryPolicy.getRetryExceptionPredicate().test(e)
|| RetryInterceptor.isRetryableException(e),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The OR here is interesting. It means a user can choose to expand the definition of what is retryable but not reduce it. I wonder if there are any cases when you would not want to retry when the default would retry. 🤔

Copy link
Author

@YuriyHolinko YuriyHolinko Jan 7, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

a user can choose to expand the definition of what is retryable but not reduce it

it's exactly the idea

I wonder if there are any cases when you would not want to retry when the default would retry

I would say no 🤔

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would say no 🤔

I think we might..

Suppose we want expose options to give full control to the user over what's retryable (as I alluded to in the end of this comment), we'd probably want to do something like:

  • Expose a single configurable predicate option of the form setRetryPredicate(Predicate<Throwable>)
  • Funnel all failed requests through this predicate, whether they resolved a response with a status code or ended with an exception
  • This means we'd need to translate requests with a non-200 status code to an equivalent exception to pass to the predicate
  • If the user doesn't define their own predicate, default to one that retriable when status is retryable (429, 502, 503, 504) or when the exception is retryable (like one of the SocketTimeoutException we've discussed).

In this case, its possible that a user doesn't want to retry on a particular response status code like 502, even when the default behavior is to retry on it.

Copy link
Author

@YuriyHolinko YuriyHolinko Jan 11, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I did not fully understand the idea

It means a user can choose to expand the definition of what is retryable but not reduce it. I wonder if there are any cases when you would not want to retry when the default would retry.

the current implementation allows only to extend retry conditions, but we can change it to override or extend so user will have this option and it's probably what you meant by this comment below

Expose a single configurable predicate option of the form setRetryPredicate(Predicate)

so we could achieve it

Funnel all failed requests through this predicate, whether they resolved a response with a status code or ended with an exception

we probably should not mix status codes and exception in the same predicate but use two predicates separately. what do you think ? 🤔

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we probably should not mix status codes and exception in the same predicate but use two predicates separately. what do you think ? 🤔

I've seen HTTP clients generate exceptions when the status code is 4xx or 5xx, and this is essentially going in that direction. All non-200 results flow flow through a Predicate<Throwable>, and the caller can setup different rules for different Throwables. Supposing we translate non-200 responses to an exception class like our HttpExportException, example usage might look like:

// Override default retry policy to retry on all all 5xx errors, and on any IOException
builder.setRetryPredicate(throwable -> {
  if (thowable instanceOf HttpExportException) {
    int httpStatusCode = ((HttpExportException) throwable).getResponse().statusCode();
    return httpStatusCode >= 500;
  }
  return throwable instanceOf IOException;
});

I suppose its not super important if exceptions and error responses are handled in a single predicate vs. two separate predicates, but I think my point here is still valid, which suggests we should give the user the ability to not retry on some IOException we normally retry by default:

If the user doesn't define their own predicate, default to one that retriable when status is retryable (429, 502, 503, 504) or when the exception is retryable (like one of the SocketTimeoutException we've discussed).

Copy link
Author

@YuriyHolinko YuriyHolinko Jan 15, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am a bit confused.
Do you want to say that currently we do not retry on default retryable status codes?

so in case of 429, 502, 503, 504 we throw exception in line 91 and line 96 is never executed

try {
response = chain.proceed(chain.request());
} catch (IOException e) {
exception = e;
}
if (response != null) {
boolean retryable = Boolean.TRUE.equals(isRetryable.apply(response));
if (logger.isLoggable(Level.FINER)) {
logger.log(
Level.FINER,
"Attempt "
+ attempt
+ " returned "
+ (retryable ? "retryable" : "non-retryable")
+ " response: "
+ responseStringRepresentation(response));
}
if (!retryable) {
return response;
}
}

@jack-berg
Copy link
Member

Thanks for the PR!

In my environment I caught a few exceptions that very likely need to be retried and they are not listed as retryable in current code. since each environment is different I suggest to have an ability to configure retryable exceptions

Wondering if you could elaborate on these, since its possible that the errors aren't actually environment-specific and everyone could benefit from them. My initial inclination was that we should just update the static definition of what constitutes a retryable exception, but I'm open to being wrong.

@YuriyHolinko
Copy link
Author

YuriyHolinko commented Jan 7, 2025

hey @jack-berg

Wondering if you could elaborate on these, since its possible that the errors aren't actually environment-specific and everyone could benefit from them. My #6962 (comment) was that we should just update the static definition of what constitutes a retryable exception, but I'm open to being wrong.

3 exceptions from me:

  1. DNS issues. my services are running on popular cloud providers and using their DNS services but sporadically I encounter issues like this.
java.net.UnknownHostException: xxxxxx.com
	at java.base/java.net.InetAddress$CachedAddresses.get(InetAddress.java:801)
	at java.base/java.net.InetAddress.getAllByName0(InetAddress.java:1533)
	at java.base/java.net.InetAddress.getAllByName(InetAddress.java:1385)
	at java.base/java.net.InetAddress.getAllByName(InetAddress.java:1306)
	at okhttp3.Dns$Companion$DnsSystem.lookup(Dns.kt:49)
	at okhttp3.internal.connection.RouteSelector.resetNextInetSocketAddress(RouteSelector.kt:164)
java.io.InterruptedIOException: timeout 	
    at okhttp3.internal.connection.RealCall.timeoutExit(RealCall.kt:398)
java.net.SocketTimeoutException: timeout 	
    at okio.SocketAsyncTimeout.newTimeoutException(JvmOkio.kt:143) 	
    at okio.AsyncTimeout.access$newTimeoutException(AsyncTimeout.kt:162) 	
    at okio.AsyncTimeout$source$1.read(AsyncTimeout.kt:340) 	
    at okio.RealBufferedSource.indexOf(RealBufferedSource.kt:449) 	
    at okio.RealBufferedSource.readUtf8LineStrict(RealBufferedSource.kt:333) 	
    at okhttp3.internal.http1.HeadersReader.readLine(HeadersReader.kt:29)

Also recently we had network issues(retryable) when using SSL, but it was luckily solved by java upgrade so I can neglect it but it might be useful for some users with some java versions

I don't mind to put all of that into "static" definition but the reason I want to have it configurable is the ability to apply a quick fix when a new retryable exception is discovered. Also I don't know all the exceptions that other people encounter in their networks so the list of exceptions is not complete

So I can combine it all in static definition in addition to the current retryable exceptions, but I want to preserve the dynamic config as well

Tell me your thoughts about it

@YuriyHolinko YuriyHolinko marked this pull request as draft January 7, 2025 16:24
@YuriyHolinko YuriyHolinko marked this pull request as ready for review January 7, 2025 16:24
@YuriyHolinko YuriyHolinko requested a review from jack-berg January 7, 2025 16:24
@YuriyHolinko YuriyHolinko changed the title Retry on configurable exception Retry on configurable exception Jan 7, 2025
@YuriyHolinko
Copy link
Author

there is flaky test in of the checks, not related to my change because it's in metrics product 🙈
could anyone tell if I can rerun it somehow without any new commits ?

@jack-berg
Copy link
Member

java.net.SocketTimeoutException: timeout
java.io.InterruptedIOException: timeout

Hmm.. let's think about these. They are clearly the result of some sort of timeout occurring. We take the arguments for setTimeout and setConnectTimeout and apply them to the OkHttpClient here:

        new OkHttpClient.Builder()
            .dispatcher(OkHttpUtil.newDispatcher())
            .connectTimeout(Duration.ofNanos(connectionTimeoutNanos))
            .callTimeout(Duration.ofNanos(timeoutNanos));

The callTimeout represents the total allotted time for everything to resolve with the call, including resolving DNS, connecting, writing request body, reading response body, and any additional retry requests. So if this is exceeded, it won't do any good to retry because there's no allotted time left for the additional attempts to resolve.

The connectTimeout is a little different. It represents the max amount of time connecting a TCP socket to the target host. If this is less than callTimeout, then there is still time remaining in the allotted callTimeout for a retry to exceed. And so I think its correct to retry if this occurs, and if we look at RetryInterceptor, we see there's the attempt to retry on this type of situation.

So I think its appropriate to extend the condition to include the exception you're seeing:

    if (e instanceof SocketTimeoutException) {
      String message = e.getMessage();
      // Connect timeouts can produce SocketTimeoutExceptions with no message, or with "connect
      // timed out", or "timeout"
      return message == null || message.toLowerCase(Locale.ROOT).contains("connect timed out") || message.toLowerCase(Locale.ROOT).contains("timeout");
    }

There's two additional OkHttpClient settings that we don't configure: readTimeout and writeTimeout. Both of these default to 10s, and presumably produce SocketTimeoutException similar to the ones we already (attempt to) retry on.

This still doesn't address the java.io.InterruptedIOException: timeout exception you've included. I wonder if you have any more of that stack trace to include? But either way, I'm inclined to again expand the general RetryInterceptor#isRetryableException to include this as well since it seems like another variation of the types of timeout exceptions we're trying to retry on.

java.net.UnknownHostException: xxxxxx.com

This is the last one that's unaddressed. This exception is thrown when DNS lookup fails for the given host. I know that the java runtime caches DNS results, but wasn't sure what it does with DNS lookup failures. I did some searching and found that negative DNS cache TTL is controlled by a property called networkaddress.cache.negative.ttl which defaults to 10 seconds.

This indicates that we should in fact retry when UnknownHostException occurs because there's a chance that the error is transient and succeeds with the next attempt.

I don't mind to put all of that into "static" definition but the reason I want to have it configurable is the ability to apply a quick fix when a new retryable exception is discovered.

This is a good point. One downside I can think of to adding the proposed RetryPolicyBuilder#setRetryExceptionPredicate method is that it may lead users to think that they have control over other aspects of retry, like which HTTP / gRPCstatus codes are retryable. There's a spec issue open about this, so its possible that we make this configurable in the future. I wonder if we would want a holistic approach to configuring retry conditions, giving the user the ability to choose based on exception and status code, or whether we'd stick with distinct configuration options.

@YuriyHolinko
Copy link
Author

YuriyHolinko commented Jan 11, 2025

thanks for sharing your ideas

here we have two stack traces

  1. java.net.SocketTimeoutException: timeout is wrapped by another exception
java.io.InterruptedIOException: timeout
	at okhttp3.internal.connection.RealCall.timeoutExit(RealCall.kt:398)
	at okhttp3.internal.connection.RealCall.callDone(RealCall.kt:360)
	at okhttp3.internal.connection.RealCall.noMoreExchanges$okhttp(RealCall.kt:325)
	at okhttp3.internal.connection.RealCall.getResponseWithInterceptorChain$okhttp(RealCall.kt:209)
	at okhttp3.internal.connection.RealCall$AsyncCall.run(RealCall.kt:517)
	at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
	at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
	at java.base/java.lang.Thread.run(Thread.java:829)
Caused by: java.net.SocketTimeoutException: timeout
	at okio.SocketAsyncTimeout.newTimeoutException(JvmOkio.kt:143)
	at okio.AsyncTimeout.access$newTimeoutException(AsyncTimeout.kt:162)
	at okio.AsyncTimeout$source$1.read(AsyncTimeout.kt:338)
	at okio.RealBufferedSource.indexOf(RealBufferedSource.kt:449)
	at okio.RealBufferedSource.readUtf8LineStrict(RealBufferedSource.kt:333)
	at okhttp3.internal.http1.HeadersReader.readLine(HeadersReader.kt:29)
	at okhttp3.internal.http1.Http1ExchangeCodec.readResponseHeaders(Http1ExchangeCodec.kt:178)
	at okhttp3.internal.connection.Exchange.readResponseHeaders(Exchange.kt:106)
	at okhttp3.internal.http.CallServerInterceptor.intercept(CallServerInterceptor.kt:79)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.connection.ConnectInterceptor.intercept(ConnectInterceptor.kt:34)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.cache.CacheInterceptor.intercept(CacheInterceptor.kt:95)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.http.BridgeInterceptor.intercept(BridgeInterceptor.kt:83)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.http.RetryAndFollowUpInterceptor.intercept(RetryAndFollowUpInterceptor.kt:76)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at io.opentelemetry.exporter.sender.okhttp.internal.RetryInterceptor.intercept(RetryInterceptor.java:83)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.connection.RealCall.getResponseWithInterceptorChain$okhttp(RealCall.kt:201)
	... 4 more
java.io.InterruptedIOException: timeout
	at okhttp3.internal.connection.RealCall.timeoutExit(RealCall.kt:398)
	at okhttp3.internal.connection.RealCall.callDone(RealCall.kt:360)
	at okhttp3.internal.connection.RealCall.noMoreExchanges$okhttp(RealCall.kt:325)
	at okhttp3.internal.connection.RealCall.getResponseWithInterceptorChain$okhttp(RealCall.kt:209)
	at okhttp3.internal.connection.RealCall$AsyncCall.run(RealCall.kt:517)
	at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1136)
	at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:635)
	at java.base/java.lang.Thread.run(Thread.java:840)
Caused by: java.net.SocketException: Socket closed
	at java.base/sun.nio.ch.NioSocketImpl.endConnect(NioSocketImpl.java:536)
	at java.base/sun.nio.ch.NioSocketImpl.connect(NioSocketImpl.java:620)
	at java.base/java.net.SocksSocketImpl.connect(SocksSocketImpl.java:327)
	at java.base/java.net.Socket.connect(Socket.java:633)
	at okhttp3.internal.platform.Platform.connectSocket(Platform.kt:120)
	at okhttp3.internal.connection.RealConnection.connectSocket(RealConnection.kt:295)
	at okhttp3.internal.connection.RealConnection.connect(RealConnection.kt:207)
	at okhttp3.internal.connection.ExchangeFinder.findConnection(ExchangeFinder.kt:226)
	at okhttp3.internal.connection.ExchangeFinder.findHealthyConnection(ExchangeFinder.kt:106)
	at okhttp3.internal.connection.ExchangeFinder.find(ExchangeFinder.kt:74)
	at okhttp3.internal.connection.RealCall.initExchange$okhttp(RealCall.kt:255)
	at okhttp3.internal.connection.ConnectInterceptor.intercept(ConnectInterceptor.kt:32)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.cache.CacheInterceptor.intercept(CacheInterceptor.kt:95)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.http.BridgeInterceptor.intercept(BridgeInterceptor.kt:83)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.http.RetryAndFollowUpInterceptor.intercept(RetryAndFollowUpInterceptor.kt:76)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at io.opentelemetry.exporter.sender.okhttp.internal.RetryInterceptor.intercept(RetryInterceptor.java:83)
	at okhttp3.internal.http.RealInterceptorChain.proceed(RealInterceptorChain.kt:109)
	at okhttp3.internal.connection.RealCall.getResponseWithInterceptorChain$okhttp(RealCall.kt:201)
	... 4 more
	Suppressed: java.net.SocketTimeoutException: Connect timed out
		at java.base/sun.nio.ch.NioSocketImpl.timedFinishConnect(NioSocketImpl.java:551)
		at java.base/sun.nio.ch.NioSocketImpl.connect(NioSocketImpl.java:602)
		... 24 more

in case of the first stack trace I believe we will retry on java.net.SocketTimeoutException: timeout (when we add it)
but in the second case we need to add java.net.SocketException: Socket closed to retryable exceptions because the expected exception java.net.SocketTimeoutException: Connect timed out was suppressed so we don't consider it in the predicate and do not retry on it eventually

there might be more edge cases and we have to be ready for them, so I will keep watching

One downside I can think of to adding the proposed RetryPolicyBuilder#setRetryExceptionPredicate method is that it may lead users to think that they have control over other aspects of retry, like which HTTP / gRPCstatus codes are retryable

Could you tell why it's downside? we could start with exceptions and later in future add response codes for http and gRPC

There's a open-telemetry/opentelemetry-specification#3876 open about this, so its possible that we make this configurable in the future.

thanks for sharing I will have a look at the docs. Just wondering if the case with exceptions should be in spec, as for me spec is not language specific, but I am not sure I fully understand the open-telemetry project. Tell me please if there was something similar in the past where we had a difference in spec using different languages or we probably have spec per language 🤔

@jack-berg
Copy link
Member

I think the fasted way to get get results for you is to split up this work. Have one PR that expands the existing retry predicate to ensure that we retry on the types of exception's you've seen in the real world. Have a second PR where we propose extending RetryPolicy with one or more configurable predicates so the user can exert more control.

Given our strong API backwards compatibility guarantees, we're generally fairly slow / methodical / careful with our API design. But I don't want this process to slow down getting concrete results for you, and I believe that it makes sense to expand the set of retryable exceptions to reflect the ones you've seen.

@YuriyHolinko YuriyHolinko force-pushed the configurable-retry-predicate branch from c37e78d to a367900 Compare January 16, 2025 11:38
@YuriyHolinko
Copy link
Author

YuriyHolinko commented Jan 16, 2025

I have made some changes in the PR where added predicate to RetryPolicy so we can extend or override a list of retryable exceptions.
The change is fully backward compatible. Other default retryable exceptions will be added in another PR

@@ -66,6 +72,9 @@ public static RetryPolicyBuilder builder() {
/** Returns the backoff multiplier. */
public abstract double getBackoffMultiplier();

/** Returns the predicate if exception is retryable. */
public abstract Predicate<IOException> getRetryExceptionPredicate();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can't have a default retry predicate because the default predicate varies by sender implementation. Instead, let's make this method @Nullable, and use null to indicate that the "protocol and sender specific default retry predicate should be used".

I sketched out a commit of how this works: 317397a

Notice how JdkHttpSender and OkHttpSender each have their own default predicate. Because the RetryPolicy is more generic than JdkHttpSender, OkHttpSender, and even more generic than OTLP, it can't contain protocol and sender specific logic.

Suggested change
public abstract Predicate<IOException> getRetryExceptionPredicate();
@Nullable
public abstract Predicate<IOException> getRetryExceptionPredicate();

@YuriyHolinko YuriyHolinko marked this pull request as draft January 18, 2025 00:15
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Data loss if issues on TCP protocol layer or failures on network link. Retry policy is ignored
2 participants