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

Eliminate token cache timeout setting in most cases #126

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 11 additions & 4 deletions src/Arc4u.Standard.OAuth2.AspNetCore.Adal/Token/Cache.cs
Original file line number Diff line number Diff line change
@@ -1,9 +1,7 @@
using Arc4u.Dependency;
using Arc4u.Dependency;
using Arc4u.Diagnostics;
using Microsoft.Extensions.Logging;
using Microsoft.IdentityModel.Clients.ActiveDirectory;
using System;
using System.Collections.Generic;

namespace Arc4u.OAuth2.Token.Adal
{
Expand Down Expand Up @@ -75,7 +73,16 @@ private void AfterAccessNotification(TokenCacheNotificationArgs args)
if (HasStateChanged)
{
Logger.Technical().From<Cache>().System($"Adding token information to the cache for the identifier: {_identifier}.").Log();
TokenCache.Put(_identifier, SerializeAdalV3());
var now = DateTimeOffset.UtcNow;
var maxExpires = now;
foreach (var tokenCacheItem in ReadItems())
{
if (tokenCacheItem.ExpiresOn > maxExpires)
{
maxExpires = tokenCacheItem.ExpiresOn;
}
}
TokenCache.Put(_identifier, maxExpires - now, SerializeAdalV3());
vvdb-architecture marked this conversation as resolved.
Show resolved Hide resolved
Logger.Technical().From<Cache>().System($"Added token information to the cache for the identifier: {_identifier}.").Log();

HasStateChanged = false;
Expand Down
101 changes: 48 additions & 53 deletions src/Arc4u.Standard.OAuth2.AspNetCore/AppPrincipalTransform.cs
Original file line number Diff line number Diff line change
@@ -1,11 +1,7 @@
using System.Collections.Generic;
using System;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Linq;
using System.Security.Claims;
using System.Threading.Tasks;
using Arc4u.Configuration;
using Arc4u.Dependency.Attribute;
using Arc4u.Diagnostics;
Expand All @@ -14,9 +10,9 @@
using Arc4u.OAuth2.Token;
using Arc4u.Security.Principal;
using Microsoft.AspNetCore.Authentication;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
using Microsoft.Extensions.DependencyInjection;

namespace Arc4u.OAuth2;

Expand Down Expand Up @@ -103,7 +99,7 @@ public async Task<ClaimsPrincipal> TransformAsync(ClaimsPrincipal principal)

#region Handling extra claims
private const string tokenExpirationClaimType = "exp";
private static readonly string[] ClaimsToExclude = { "aud", "iss", "iat", "nbf", "acr", "aio", "appidacr", "ipaddr", "scp", "sub", "tid", "uti", "unique_name", "apptype", "appid", "ver", "http://schemas.microsoft.com/ws/2008/06/identity/claims/authenticationinstant", "http://schemas.microsoft.com/identity/claims/scope" };
private static readonly HashSet<string> ClaimsToExclude = new HashSet<string>(StringComparer.OrdinalIgnoreCase) { "aud", "iss", "iat", "nbf", "acr", "aio", "appidacr", "ipaddr", "scp", "sub", "tid", "uti", "unique_name", "apptype", "appid", "ver", "http://schemas.microsoft.com/ws/2008/06/identity/claims/authenticationinstant", "http://schemas.microsoft.com/identity/claims/scope" };

/// <summary>
/// This code is similar to the code in AppPrincipalFactory where the claims are stored in a secureCache.
Expand All @@ -128,21 +124,13 @@ private async Task LoadExtraClaimsAsync(ClaimsIdentity? identity)
// Something already in the cache? Avoid a expensive backend call.
var cachedClaims = GetClaimsFromCache(cacheKey);

// check expirity.
var cachedExpiredClaim = cachedClaims.FirstOrDefault(c => c.ClaimType.Equals(tokenExpirationClaimType, StringComparison.OrdinalIgnoreCase));

if (cachedExpiredClaim is not null && long.TryParse(cachedExpiredClaim.Value, out var cachedExpiredTicks))
// if cachedClaims is not null, it means that the information was in the cache and didn't expire yet.
if (cachedClaims is not null)
{
var expDate = DateTimeOffset.FromUnixTimeSeconds(cachedExpiredTicks).UtcDateTime;
if (expDate > DateTime.UtcNow)
{
identity.AddClaims(cachedClaims.Where(c => c.ClaimType != tokenExpirationClaimType)
.Where(c => !ClaimsToExclude.Any(arg => arg.Equals(c.ClaimType)))
.Where(c => !identity.Claims.Any(c1 => c1.Type == c.ClaimType))
.Select(c => new Claim(c.ClaimType, c.Value)));

return;
}
identity.AddClaims(cachedClaims.Where(c => !ClaimsToExclude.Contains(c.ClaimType))
.Where(c => !identity.Claims.Any(c1 => StringComparer.OrdinalIgnoreCase.Equals(c1.Type, c.ClaimType)))
.Select(c => new Claim(c.ClaimType, c.Value)));
return;
}

// Add Telemetry.
Expand All @@ -154,41 +142,57 @@ private async Task LoadExtraClaimsAsync(ClaimsIdentity? identity)
// Should receive specific extra claims. This is the responsibility of the caller to provide the right claims.
// We expect the exp claim to be present.
// if not the persistence time will be the default one.
var claims = (await _claimsFiller.GetAsync(identity, settings, null).ConfigureAwait(false)).Where(c => !ClaimsToExclude.Any(arg => arg.Equals(c.ClaimType))).ToList();
var claims = (await _claimsFiller.GetAsync(identity, settings, null).ConfigureAwait(false)).Where(c => !ClaimsToExclude.Contains(c.ClaimType)).ToList();

// Load the claims into the identity but exclude the exp claim and the one already present.
identity.AddClaims(claims.Where(c => c.ClaimType != tokenExpirationClaimType)
.Where(c => !ClaimsToExclude.Any(arg => arg.Equals(c.ClaimType)))
.Where(c => !identity.Claims.Any(c1 => c1.Type == c.ClaimType))
identity.AddClaims(claims.Where(c => !StringComparer.OrdinalIgnoreCase.Equals(c.ClaimType, tokenExpirationClaimType))
.Where(c => !ClaimsToExclude.Contains(c.ClaimType))
.Where(c => !identity.Claims.Any(c1 => StringComparer.OrdinalIgnoreCase.Equals(c1.Type, c.ClaimType)))
vvdb-architecture marked this conversation as resolved.
Show resolved Hide resolved
.Select(c => new Claim(c.ClaimType, c.Value)));

SaveClaimsToCache(claims, cacheKey);
}

}
// Check expiry claim explicitly returned in the call to _claimsFiller.GetAsync.
var cachedExpiredClaim = claims.FirstOrDefault(c => StringComparer.OrdinalIgnoreCase.Equals(c.ClaimType, tokenExpirationClaimType))?.Value;

private List<ClaimDto> GetClaimsFromCache(string? cacheKey)
{
var claims = new List<ClaimDto>();
DateTimeOffset expDate;
if (cachedExpiredClaim is not null && long.TryParse(cachedExpiredClaim, NumberStyles.Integer, CultureInfo.InvariantCulture, out var cachedExpiredTicks))
{
expDate = DateTimeOffset.FromUnixTimeSeconds(cachedExpiredTicks);
}
else
{
// No expiry claim found in the call to _claimsFiller.GetAsync, try to locate one in the existing identity claims (which are normally obtained from the bearer token)
cachedExpiredClaim = identity.Claims.FirstOrDefault(c => StringComparer.OrdinalIgnoreCase.Equals(c.Type, tokenExpirationClaimType))?.Value;
if (cachedExpiredClaim is not null && long.TryParse(cachedExpiredClaim, NumberStyles.Integer, CultureInfo.InvariantCulture, out cachedExpiredTicks))
{
expDate = DateTimeOffset.FromUnixTimeSeconds(cachedExpiredTicks);
}
else
{
expDate = DateTimeOffset.UtcNow.Add(_options.MaxTime);
}
}
vvdb-architecture marked this conversation as resolved.
Show resolved Hide resolved

if (string.IsNullOrWhiteSpace(cacheKey))
{
return claims;
SaveClaimsToCache(claims, cacheKey, expDate - DateTimeOffset.UtcNow);
vvdb-architecture marked this conversation as resolved.
Show resolved Hide resolved
}
}

try
{
claims = _cacheHelper.GetCache().Get<List<ClaimDto>>(cacheKey) ?? new List<ClaimDto>();
}
catch (Exception ex)
private List<ClaimDto>? GetClaimsFromCache(string? cacheKey)
{
if (!string.IsNullOrWhiteSpace(cacheKey))
{
_logger.Technical().Exception(ex).Log();
try
{
return _cacheHelper.GetCache().Get<List<ClaimDto>>(cacheKey);
}
catch (Exception ex)
{
_logger.Technical().Exception(ex).Log();
}
Comment on lines +172 to +177
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Avoid catching general exceptions

Catching the general Exception type may mask specific issues and make debugging more difficult. Consider catching more specific exceptions that are expected from the cache retrieval operation.

}

return claims;
return null;
}

private void SaveClaimsToCache([DisallowNull] IEnumerable<ClaimDto> claims, string? cacheKey)
private void SaveClaimsToCache([DisallowNull] IEnumerable<ClaimDto> claims, string? cacheKey, TimeSpan timeout)
{
if (string.IsNullOrWhiteSpace(cacheKey))
{
Expand All @@ -199,16 +203,7 @@ private void SaveClaimsToCache([DisallowNull] IEnumerable<ClaimDto> claims, stri

try
{
var cachedExpiredClaim = claims.FirstOrDefault(c => c.ClaimType.Equals(tokenExpirationClaimType, StringComparison.OrdinalIgnoreCase));

// if no expiration claim exist we assume the lifetime of the extra claims is defined by the cache context for the principal.
if (cachedExpiredClaim is null)
{
cachedExpiredClaim = new ClaimDto(tokenExpirationClaimType, DateTimeOffset.UtcNow.Add(_cacheOptions.MaxTime).ToUnixTimeSeconds().ToString(CultureInfo.InvariantCulture));
claims = new List<ClaimDto>(claims) { cachedExpiredClaim };
}

_cacheHelper.GetCache().Put(cacheKey, _cacheOptions.MaxTime, claims);
_cacheHelper.GetCache().Put(cacheKey, timeout, claims);
}
catch (Exception ex)
{
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
using System;
using System.Linq;
using Arc4u.Configuration;
using Arc4u.OAuth2.Options;
using Microsoft.Extensions.Configuration;
Expand All @@ -13,9 +11,21 @@ public static void AddClaimsFiller(this IServiceCollection services, Action<Clai
var validate = new ClaimsFillerOptions();
options(validate);

if (validate.LoadClaimsFromClaimsFillerProvider && (null == validate.SettingsKeys || !validate.SettingsKeys.Any()))
if (validate.LoadClaimsFromClaimsFillerProvider)
{
throw new ConfigurationException("Settings key must be provided.");
string? configErrors = null;
if (null == validate.SettingsKeys || !validate.SettingsKeys.Any())
{
configErrors += "Settings key must be provided." + System.Environment.NewLine;
}
if (validate.LoadClaimsFromClaimsFillerProvider && validate.MaxTime == TimeSpan.Zero)
{
configErrors += $"If {nameof(validate.LoadClaimsFromClaimsFillerProvider)} is true, then {nameof(validate.MaxTime)} must be provided." + System.Environment.NewLine;
}
if (configErrors is not null)
{
throw new ConfigurationException(configErrors);
}
}

services.Configure<ClaimsFillerOptions>(options);
Expand All @@ -39,10 +49,10 @@ public static void AddClaimsFiller(this IServiceCollection services, IConfigurat
}
}

AddClaimsFiller(services, o =>
{
o.LoadClaimsFromClaimsFillerProvider = options.LoadClaimsFromClaimsFillerProvider;
o.SettingsKeys = options.SettingsKeys;
});
AddClaimsFiller(services, o =>
{
o.LoadClaimsFromClaimsFillerProvider = options.LoadClaimsFromClaimsFillerProvider;
o.SettingsKeys = options.SettingsKeys;
});
}
}
Original file line number Diff line number Diff line change
@@ -1,11 +1,8 @@
using System;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Globalization;
using System.Linq;
using System.Net.Http.Headers;
using System.Text;
using System.Threading.Tasks;
using Arc4u.Configuration;
using Arc4u.Dependency;
using Arc4u.Diagnostics;
Expand Down Expand Up @@ -89,7 +86,7 @@ public async Task Invoke(HttpContext context)
{
tokenInfo = await GetTokenFromCredentialAsync(credential, context.RequestServices).ConfigureAwait(false);

tokenCache?.Put(cacheKey, tokenInfo);
tokenCache?.Put(cacheKey, tokenInfo.ExpiresOnUtc - DateTime.UtcNow, tokenInfo);
}

if (tokenInfo is not null)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
using System.Collections.Generic;

namespace Arc4u.OAuth2.Options;

/// <summary>
Expand All @@ -10,6 +8,11 @@ public class ClaimsFillerOptions
// By default, a specific claims filler is needed to manage the right.
public bool LoadClaimsFromClaimsFillerProvider { get; set; } = true;

/// <summary>
/// If <see cref="LoadClaimsFromClaimsFillerProvider"/> is true, the extra claims are cached for <see cref="MaxTime"/>."/>
/// </summary>
public TimeSpan MaxTime { get; set; } = TimeSpan.FromMinutes(20);

/// <summary>
/// The settings key to load the claims from. => registered as a named <see cref="SimpleKeyValueSettings>"/>
/// By default no on behalf of scenario is defined => AOauth2 must be added to perform the on behalf of scenario.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,3 @@
using System;
using System.Threading.Tasks;
using Arc4u.Configuration;
using Arc4u.Dependency;
using Arc4u.Dependency.Attribute;
using Arc4u.Diagnostics;
Expand Down Expand Up @@ -84,7 +81,7 @@ public async Task<TokenInfo> GetTokenAsync(IKeyValueSettings settings, Credentia
try
{
_logger.Technical().System($"Save the token in the cache for {cacheKey}, will expire at {tokenInfo.ExpiresOnUtc} Utc.").Log();
TokenCache.Put(cacheKey, tokenInfo);
TokenCache.Put(cacheKey, tokenInfo.ExpiresOnUtc - DateTime.UtcNow, tokenInfo);
}
catch (Exception ex)
{
Expand Down
7 changes: 0 additions & 7 deletions src/Arc4u.Standard.OAuth2/Extensions/TokenCacheExtension.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
using System;
using Arc4u.Configuration;
using Arc4u.OAuth2.Options;
using Microsoft.Extensions.Configuration;
Expand Down Expand Up @@ -54,15 +53,9 @@ private static void AddTokenCache(IServiceCollection services, TokenCacheOptions
throw new ConfigurationException("TokenCacheOptions.CacheName is not defined in the configuration file.");
}

if (TimeSpan.Zero == tokenCacheOptions.MaxTime)
{
throw new ConfigurationException("TokenCacheOptions.MaxTime is not defined in the configuration file.");
}

services.Configure<TokenCacheOptions>(options =>
{
options.CacheName = tokenCacheOptions.CacheName;
options.MaxTime = tokenCacheOptions.MaxTime;
});
}
}
4 changes: 0 additions & 4 deletions src/Arc4u.Standard.OAuth2/Options/TokenCacheOptions.cs
Original file line number Diff line number Diff line change
@@ -1,9 +1,5 @@
using System;

namespace Arc4u.OAuth2.Options;
public class TokenCacheOptions
{
public TimeSpan MaxTime { get; set; } = TimeSpan.FromMinutes(20);

public string CacheName { get; set; }
}
6 changes: 2 additions & 4 deletions src/Arc4u.Standard.OAuth2/Token/ApplicationCache.cs
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,6 @@
using Arc4u.OAuth2.Options;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
using System;
using System.Collections.Generic;

namespace Arc4u.OAuth2.Token
{
Expand Down Expand Up @@ -40,7 +38,7 @@ public void DeleteItem(string id)
_logger.Technical().From<ApplicationCache>().System($"Deleted information from the token cache for the id: {id}.").Log();
}

public void Put<T>(string key, T data)
public void Put<T>(string key, TimeSpan timeout, T data)
{
if (null == data)
{
Expand All @@ -49,7 +47,7 @@ public void Put<T>(string key, T data)
}

_logger.Technical().From<ApplicationCache>().System($"Adding token data information to the cache: {key}.").Log();
_cache.Put(GetKey(key), _tokenCacheOptions.MaxTime, data);
_cache.Put(GetKey(key), timeout, data);
_logger.Technical().From<ApplicationCache>().System($"Added token data information to the cache: {key}.").Log();
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@
using Arc4u.Dependency.Attribute;
using Arc4u.Diagnostics;
using Microsoft.Extensions.Logging;
using System;

namespace Arc4u.OAuth2.Token
{
Expand Down Expand Up @@ -37,7 +36,7 @@ public void DeleteItem(string key)
Clear(key);
}

public void Put<T>(string key, T data)
public void Put<T>(string key, TimeSpan timeout, T data)
{
try
{
Expand Down
12 changes: 10 additions & 2 deletions src/Arc4u.Standard.OAuth2/Token/ITokenCache.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
namespace Arc4u.OAuth2.Token
namespace Arc4u.OAuth2.Token
{
public interface ITokenCache
{
Expand All @@ -9,7 +9,15 @@ public interface ITokenCache
void DeleteItem(string key);


void Put<T>(string key, T data);
/// <summary>
/// Set or overwrite the item with key <paramref name="key"/> with <paramref name="data"/>.
/// The item expires after <paramref name="timeout"/>
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="key"></param>
/// <param name="timeout"></param>
/// <param name="data"></param>
void Put<T>(string key, TimeSpan timeout, T data);

T Get<T>(string key);

Expand Down
Loading