Refresh Token using Polly with Typed Client
Asked Answered
A

1

5

I have a Typed Client which i have configured in the services and i am using Polly to make retries for transient faults.

Aim: I want to make use of Polly to implement refresh token, whenever there is a 401 response from the target site, i want Polly to refresh the token and continue the initial request again.

The problem is the typed client has all the api methods and the refresh token method, when the request is initiated from the typed client how do i access the typed client again to call the refresh token and continue the initial request?

The 'Context' in onRetry provides some support to add any object to the dictionary, but i am unable to access the SetPolicyExecutionContext('someContext') method and i do not want to add this on all the methods before initiating the call as there's whole lot of API.

// In Service Configuration

// Refresh token policy

var refreshTokenPolicy = Polly.Policy.HandleResult<HttpResponseMessage>(r => r.StatusCode == HttpStatusCode.Unauthorized)
.RetryAsync(1, (response, retrycount, context)) =>
{
    if(response.Result.StatusCode == HttpStatusCode.Unauthorized)
    {
         // Perform refresh token
    }
}

// Typed Client 
services.AddHttpClient<TypedClient>();

public class TypedClient
{
    private static HttpClient _client;
    public TypedClient(HttpClient client)
    {
        _client = client;
    }

    public string ActualCall()
    {
        // some action
    }

    public string RefreshToken()
    {
        // Refresh the token and return
    }
}
Aggravation answered 10/7, 2019 at 17:52 Comment(2)
Do any of these three blog posts covering the topic help? 1) nodogmablog.bryanhogan.net/2017/05/… ; 2) diaryofadev.net/2017/05/oath-with-polly ; 3) jerriepelser.com/blog/refresh-google-access-token-with-polly . Number 3) sounds very similar.Grosbeak
@mountaintraveller None of these are what OP is doing specifically. OP has proper separation of concerns; all of these sources shove polly right into the actions of the HttpClient instead of wrapping Polly around the Typed HttpClientVaasa
S
9

You can useAddPolicyHandler which has an overload that passesIServiceProvider. So all you need to do is something like:

services.AddHttpClient<TypedClient>()
    .AddPolicyHandler((provider, request) =>
    {
        return Policy.HandleResult<HttpResponseMessage>(r => r.StatusCode == HttpStatusCode.Unauthorized)
            .RetryAsync(1, (response, retryCount, context) =>
            {
                var client = provider.GetRequiredService<TypedClient>();
                // refresh auth token.
            });
        });
    });
Subsistent answered 10/7, 2019 at 18:43 Comment(4)
This was very helpful. I ended up keeping an AccessToken private property with the most current token received. It worked out well.Structuralism
Isn't it asking for trouble to call the same get service from Polly? TypeClient.Get falls into Polly, Polly then calls TypedClient.Get, which then fails and falls into Polly again...Vaasa
How do I update Authorization header in HttpClient which will retry a request? provider.GetRequiredService<TypedClient>() will return a new instance of HttpClient and retry will still fail as far as I understandFinancier
This does not address the original question: how do you change the default header authentication token from here? This will get a new token just for the current request. So this answer will fail every time you try to contact the server, then find a token, then try again. Very, very inefficient.Leis

© 2022 - 2024 — McMap. All rights reserved.