Skip to content
This repository was archived by the owner on Nov 19, 2024. It is now read-only.

Commit 3b939ee

Browse files
authored
Merge pull request #33 from espenekvang/feature/distributed-cache-exception
Log exception from distributed cache
2 parents 951f552 + 38ab40e commit 3b939ee

File tree

3 files changed

+116
-4
lines changed

3 files changed

+116
-4
lines changed

src/Duende.AccessTokenManagement/ClientCredentialsTokenManagementService.cs

+24-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
// Copyright (c) Brock Allen & Dominick Baier. All rights reserved.
22
// Licensed under the Apache License, Version 2.0. See LICENSE in the project root for license information.
33

4+
using System;
45
using System.Threading;
56
using System.Threading.Tasks;
67
using Microsoft.Extensions.Logging;
@@ -46,10 +47,19 @@ public async Task<ClientCredentialsToken> GetAccessTokenAsync(
4647

4748
if (parameters.ForceRenewal == false)
4849
{
49-
var item = await _tokenCache.GetAsync(clientName, parameters, cancellationToken).ConfigureAwait(false);
50-
if (item != null)
50+
try
5151
{
52-
return item;
52+
var item = await _tokenCache.GetAsync(clientName, parameters, cancellationToken).ConfigureAwait(false);
53+
if (item != null)
54+
{
55+
return item;
56+
}
57+
}
58+
catch (Exception e)
59+
{
60+
_logger.LogError(e,
61+
"Error trying to obtain token from cache for client {clientName}. Error = {error}. Will obtain new token.",
62+
clientName, e.Message);
5363
}
5464
}
5565

@@ -65,7 +75,17 @@ public async Task<ClientCredentialsToken> GetAccessTokenAsync(
6575
return token;
6676
}
6777

68-
await _tokenCache.SetAsync(clientName, token, parameters, cancellationToken).ConfigureAwait(false);
78+
try
79+
{
80+
await _tokenCache.SetAsync(clientName, token, parameters, cancellationToken).ConfigureAwait(false);
81+
}
82+
catch (Exception e)
83+
{
84+
_logger.LogError(e,
85+
"Error trying to set token in cache for client {clientName}. Error = {error}",
86+
clientName, e.Message);
87+
}
88+
6989
return token;
7090
}).ConfigureAwait(false);
7191
}

test/Tests/ClientTokenManagementTests.cs

+41
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
using System.Text.Json;
33
using IdentityModel;
44
using IdentityModel.Client;
5+
using Microsoft.Extensions.Caching.Distributed;
56
using Microsoft.Extensions.DependencyInjection;
67
using RichardSzalay.MockHttp;
78

@@ -433,6 +434,46 @@ public async Task Service_should_hit_network_only_once_and_then_use_cache()
433434
token.AccessTokenType.ShouldBe("token_type");
434435
mockHttp.GetMatchCount(mockedRequest).ShouldBe(1);
435436
}
437+
438+
[Fact]
439+
public async Task Service_should_hit_network_when_cache_throws_exception()
440+
{
441+
var services = new ServiceCollection();
442+
443+
services.AddTransient<IDistributedCache, TestDistributedCache>();
444+
services.AddClientCredentialsTokenManagement()
445+
.AddClient("test", client =>
446+
{
447+
client.TokenEndpoint = "https://as/connect/token";
448+
client.ClientId = "client_id";
449+
});
450+
451+
var response = new
452+
{
453+
access_token = "access_token",
454+
token_type = "token_type",
455+
expires_in = 3600,
456+
scope = "scope"
457+
};
458+
459+
var mockHttp = new MockHttpMessageHandler();
460+
mockHttp.Fallback.Throw(new InvalidOperationException("No matching mock handler"));
461+
462+
var mockedRequest = mockHttp.Expect("/connect/token")
463+
.Respond("application/json", JsonSerializer.Serialize(response));
464+
465+
services.AddHttpClient(ClientCredentialsTokenManagementDefaults.BackChannelHttpClientName)
466+
.ConfigurePrimaryHttpMessageHandler(() => mockHttp);
467+
468+
var provider = services.BuildServiceProvider();
469+
var sut = provider.GetRequiredService<IClientCredentialsTokenManagementService>();
470+
471+
var token = await sut.GetAccessTokenAsync("test");
472+
token.IsError.ShouldBeFalse();
473+
token.AccessToken.ShouldBe("access_token");
474+
token.AccessTokenType.ShouldBe("token_type");
475+
mockHttp.GetMatchCount(mockedRequest).ShouldBe(1);
476+
}
436477

437478
[Fact]
438479
public async Task Service_should_always_hit_network_with_force_renewal()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
using Microsoft.Extensions.Caching.Distributed;
2+
3+
namespace Duende.AccessTokenManagement.Tests;
4+
5+
internal class TestDistributedCache : IDistributedCache
6+
{
7+
private readonly Dictionary<string, byte[]> _cache = new();
8+
9+
public byte[] Get(string key)
10+
{
11+
return _cache[key];
12+
}
13+
14+
public Task<byte[]> GetAsync(string key, CancellationToken token = new CancellationToken())
15+
{
16+
return Task.FromResult(_cache[key]);
17+
}
18+
19+
public void Refresh(string key)
20+
{
21+
}
22+
23+
public Task RefreshAsync(string key, CancellationToken token = new CancellationToken())
24+
{
25+
Refresh(key);
26+
return Task.CompletedTask;
27+
}
28+
29+
public void Remove(string key)
30+
{
31+
_cache.Remove(key);
32+
}
33+
34+
public Task RemoveAsync(string key, CancellationToken token = new CancellationToken())
35+
{
36+
Remove(key);
37+
return Task.CompletedTask;
38+
}
39+
40+
public void Set(string key, byte[] value, DistributedCacheEntryOptions options)
41+
{
42+
_cache.Add(key, value);
43+
}
44+
45+
public Task SetAsync(string key, byte[] value, DistributedCacheEntryOptions options,
46+
CancellationToken token = new CancellationToken())
47+
{
48+
Set(key, value, options);
49+
return Task.CompletedTask;
50+
}
51+
}

0 commit comments

Comments
 (0)