-
Notifications
You must be signed in to change notification settings - Fork 415
/
Copy pathBaseConfigurationManager.cs
188 lines (163 loc) · 8.53 KB
/
BaseConfigurationManager.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
using System;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.IdentityModel.Logging;
using Microsoft.IdentityModel.Tokens.Configuration;
namespace Microsoft.IdentityModel.Tokens
{
/// <summary>
/// Represents a generic configuration manager.
/// </summary>
public abstract class BaseConfigurationManager
{
private TimeSpan _automaticRefreshInterval = DefaultAutomaticRefreshInterval;
private TimeSpan _refreshInterval = DefaultRefreshInterval;
private TimeSpan _lastKnownGoodLifetime = DefaultLastKnownGoodConfigurationLifetime;
private BaseConfiguration _lastKnownGoodConfiguration;
private DateTime? _lastKnownGoodConfigFirstUse;
internal EventBasedLRUCache<BaseConfiguration, DateTime> _lastKnownGoodConfigurationCache;
/// <summary>
/// Gets or sets the <see cref="TimeSpan"/> that controls how often an automatic metadata refresh should occur.
/// </summary>
public TimeSpan AutomaticRefreshInterval
{
get { return _automaticRefreshInterval; }
set
{
if (value < MinimumAutomaticRefreshInterval)
throw LogHelper.LogExceptionMessage(new ArgumentOutOfRangeException(nameof(value), LogHelper.FormatInvariant(LogMessages.IDX10108, LogHelper.MarkAsNonPII(MinimumAutomaticRefreshInterval), LogHelper.MarkAsNonPII(value))));
_automaticRefreshInterval = value;
}
}
/// <summary>
/// Default time interval (12 hours) after which a new configuration is obtained automatically.
/// </summary>
public static readonly TimeSpan DefaultAutomaticRefreshInterval = new TimeSpan(0, 12, 0, 0);
/// <summary>
/// Default time interval (1 hour) for which the last known good configuration remains valid.
/// </summary>
public static readonly TimeSpan DefaultLastKnownGoodConfigurationLifetime = new TimeSpan(0, 1, 0, 0);
/// <summary>
/// Default time interval (5 minutes) that must pass for <see cref="RequestRefresh"/> to obtain a new configuration.
/// </summary>
public static readonly TimeSpan DefaultRefreshInterval = new TimeSpan(0, 0, 5, 0);
/// <summary>
/// Initializes a new instance of the <see cref="BaseConfigurationManager"/> class.
/// </summary>
public BaseConfigurationManager()
: this(new LKGConfigurationCacheOptions())
{
}
/// <summary>
/// Initializes a new instance of the <see cref="BaseConfigurationManager"/> class.
/// </summary>
/// <param name="options">The event queue task creation option.</param>
public BaseConfigurationManager(LKGConfigurationCacheOptions options)
{
if (options == null)
throw LogHelper.LogArgumentNullException(nameof(options));
_lastKnownGoodConfigurationCache = new EventBasedLRUCache<BaseConfiguration, DateTime>(
options.LastKnownGoodConfigurationSizeLimit,
options.TaskCreationOptions,
options.BaseConfigurationComparer,
options.RemoveExpiredValues);
}
/// <summary>
/// Obtains an updated version of <see cref="BaseConfiguration"/> if the appropriate refresh interval has passed.
/// This method may return a cached version of the configuration.
/// </summary>
/// <param name="cancel">A cancellation token that can be used to cancel the asynchronous operation.</param>
/// <returns>Configuration of type Configuration.</returns>
/// <remarks>This method on the base class throws a <see cref="NotImplementedException"/>
/// as it is meant to be overridden by the class that extends it.</remarks>
public virtual Task<BaseConfiguration> GetBaseConfigurationAsync(CancellationToken cancel)
{
throw LogHelper.LogExceptionMessage(
new NotImplementedException(
LogHelper.FormatInvariant(
LogMessages.IDX10267,
LogHelper.MarkAsNonPII("public virtual Task<BaseConfiguration> GetBaseConfigurationAsync(CancellationToken cancel)"),
LogHelper.MarkAsNonPII(GetType().FullName))));
}
/// <summary>
/// Gets all valid last known good configurations.
/// </summary>
/// <returns>A collection of all valid last known good configurations.</returns>
internal BaseConfiguration[] GetValidLkgConfigurations()
{
return _lastKnownGoodConfigurationCache.ToArray().Where(x => x.Value.Value > DateTime.UtcNow).Select(x => x.Key).ToArray();
}
/// <summary>
/// The last known good configuration or LKG (a configuration retrieved in the past that we were able to successfully validate a token against).
/// </summary>
public BaseConfiguration LastKnownGoodConfiguration
{
get
{
return _lastKnownGoodConfiguration;
}
set
{
_lastKnownGoodConfiguration = value ?? throw LogHelper.LogArgumentNullException(nameof(value));
_lastKnownGoodConfigFirstUse = DateTime.UtcNow;
// LRU cache will remove the expired configuration
_lastKnownGoodConfigurationCache.SetValue(_lastKnownGoodConfiguration, DateTime.UtcNow + LastKnownGoodLifetime, DateTime.UtcNow + LastKnownGoodLifetime);
}
}
/// <summary>
/// The length of time that a last known good configuration is valid for.
/// </summary>
public TimeSpan LastKnownGoodLifetime
{
get { return _lastKnownGoodLifetime; }
set
{
if (value < TimeSpan.Zero)
throw LogHelper.LogExceptionMessage(new ArgumentOutOfRangeException(nameof(value), LogHelper.FormatInvariant(LogMessages.IDX10110, value)));
_lastKnownGoodLifetime = value;
}
}
/// <summary>
/// The metadata address to retrieve the configuration from.
/// </summary>
public string MetadataAddress { get; set; }
/// <summary>
/// Minimum time interval (5 minutes) for automatic refresh. <see cref="AutomaticRefreshInterval"/> cannot be set to less than this value.
/// </summary>
public static readonly TimeSpan MinimumAutomaticRefreshInterval = new TimeSpan(0, 0, 5, 0);
/// <summary>
/// Minimum time interval (1 second) that must pass before calling <see cref="RequestRefresh"/> to obtain new configuration.
/// </summary>
public static readonly TimeSpan MinimumRefreshInterval = new TimeSpan(0, 0, 0, 1);
/// <summary>
/// The minimum time between retrievals, in the event that a retrieval failed, or that a refresh was explicitly requested.
/// </summary>
public TimeSpan RefreshInterval
{
get { return _refreshInterval; }
set
{
if (value < MinimumRefreshInterval)
throw LogHelper.LogExceptionMessage(new ArgumentOutOfRangeException(nameof(value), LogHelper.FormatInvariant(LogMessages.IDX10107, LogHelper.MarkAsNonPII(MinimumRefreshInterval), LogHelper.MarkAsNonPII(value))));
_refreshInterval = value;
}
}
/// <summary>
/// Gets or sets a value indicating whether to use the last known good configuration. Default is true.
/// </summary>
public bool UseLastKnownGoodConfiguration { get; set; } = true;
/// <summary>
/// Gets a value indicating whether the last known good configuration is still valid, depends on when the LKG was first used and it's lifetime.
/// </summary>
// The _lastKnownGoodConfiguration private variable is accessed rather than the property (LastKnownGoodConfiguration) as we do not want this access
// to trigger a change in _lastKnownGoodConfigFirstUse.
public bool IsLastKnownGoodValid => _lastKnownGoodConfiguration != null && (_lastKnownGoodConfigFirstUse == null || DateTime.UtcNow < _lastKnownGoodConfigFirstUse + LastKnownGoodLifetime);
/// <summary>
/// Indicate that the configuration may be stale (as indicated by failing to process incoming tokens).
/// </summary>
public abstract void RequestRefresh();
}
}