Does OkHttpClient have a max retry count

20,316

Solution 1

There are more docs here https://square.github.io/okhttp/3.x/okhttp/okhttp3/OkHttpClient.Builder.html#retryOnConnectionFailure-boolean-

Configure this client to retry or not when a connectivity problem is encountered. By default, this client silently recovers from the following problems:

  • Unreachable IP addresses. If the URL's host has multiple IP addresses, failure to reach any individual IP address doesn't fail the overall request. This can increase availability of multi-homed services.
  • Stale pooled connections. The ConnectionPool reuses sockets to decrease request latency, but these connections will occasionally time out.

  • Unreachable proxy servers. A ProxySelector can be used to attempt multiple proxy servers in sequence, eventually falling back to a direct connection.

Set this to false to avoid retrying requests when doing so is destructive. In this case the calling application should do its own recovery of connectivity failures.

But generally, I believe it is intended to retry when there is an existing stale connection, or alternate paths that can be retried. Not to retry exactly the same thing indefinitely.

Also see ConnectionSpecSelector.connectionFailed

Solution 2

I made an workaround below:

@Override
public Response intercept(Chain chain) throws IOException {
  Request request = chain.request();
  // try the request
  Response response = doRequest(chain,request);
  int tryCount = 0;
  while (response == null && tryCount <= RetryCount) {
    String url = request.url().toString();
    url = switchServer(url);
    Request newRequest = request.newBuilder().url(url).build();
    tryCount++;
    // retry the request
    response = doRequest(chain,newRequest);
  }
  if(response == null){//important ,should throw an exception here
      throw new IOException();
  }
  return response;
}

private Response doRequest(Chain chain,Request request){
  Response response = null;
  try{
      response = chain.proceed(request);
  }catch (Exception e){
  }
  return response;
}

Solution 3

There's no in built method to set the maximum limit, but you can add an interceptor like below.

client.interceptors().add(new Interceptor() {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();

        // try the request
        Response response = chain.proceed(request);

        int tryCount = 0;
        int maxLimit = 3; //Set your max limit here

        while (!response.isSuccessful() && tryCount < maxLimit) {

            Log.d("intercept", "Request failed - " + tryCount);

            tryCount++;

            // retry the request
            response = chain.proceed(request);
        }

        // otherwise just pass the original response on
        return response;
    }
});

More detail on interceptos can be found here.

Solution 4

According to source code of RetryAndFollowUpInterceptor it is 20

  /**
   * How many redirects and auth challenges should we attempt? Chrome follows 21 redirects; Firefox,
   * curl, and wget follow 20; Safari follows 16; and HTTP/1.0 recommends 5.
   */
  private static final int MAX_FOLLOW_UPS = 20;
Share:
20,316
RajV
Author by

RajV

Updated on July 11, 2022

Comments

  • RajV
    RajV almost 2 years

    I am setting the retry on connection failure option for OkHttpClient.

    client = new OkHttpClient();
    client.setRetryOnConnectionFailure(true);
    

    I will like to know how many times it will keep trying. Looking at the source code I did not see any maximum limit. How do I configure the client to stop trying after a few attempts?

  • RajV
    RajV almost 8 years
    response.isSuccessful() checks for HTTP status code. Will that work for connection failure?
  • Prerak Sola
    Prerak Sola almost 8 years
    Yes. if the response code is 2xx or 3xx, isSuccessful() will return true. For rest, it will return false. And connection failures are mostly 4xx, so it should work.
  • RajV
    RajV almost 8 years
    TCP connection failure means no HTTP request could be sent or response obtained. I am not sure how the status code will be 4xx.
  • RajV
    RajV almost 8 years
    This is my understanding too. In addition, it will also retry a failed SSL handshake. But it does not seem to retry an outright TCP connection failure.
  • Liangliang Sha
    Liangliang Sha almost 8 years
    RajV's right,the chain.proceed(request) will throw IOE exception instead of return false when the connection is failure, so Prerak Sola's answer can only retry based on the connection is success
  • Ahmad Shahwaiz
    Ahmad Shahwaiz almost 3 years
    i think this code might throw error to first close the response then make new request.