-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathConfigurationVisualizer.cs
308 lines (268 loc) · 13.4 KB
/
ConfigurationVisualizer.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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
/*
* Configuration Visualizer
* Creates a visual of your program configuration (IConfiguration)
* (C) 2024 Charles Burns
* MIT license
* This code is meant as a small public service for use by the fine C# community.
*/
using System.Text;
using Microsoft.Extensions.Configuration;
namespace YourNamespace.ExtensionMethodLogic;
/// <summary>Builds a text visualization of an IConfiguration object</summary>
internal static class ConfigurationVisualizer
{
/// <summary>Configure the ignore list most useful to your environment. This is just a sample.</summary>
private readonly static HashSet<string> DefaultVendorPrefixes = new(StringComparer.OrdinalIgnoreCase) {
"ZES_ENABLE_SYSMAN",
"STTASKFOLDERPATH",
"SNOW_AGENT",
"PSModulePath",
"AllowedHosts",
"PROCESSOR_",
"PATHEXT",
"NEXTHINK",
"IJ_RESTARTER_LOG",
"DriverData",
"ComSpec",
"LOGONSERVER",
"Chocolatey",
"PSLockDownPolicy",
"CommonProgram",
"ProgramFiles",
"SESSIONNAME",
"USERDOMAIN"
};
/// <summary>The visualization uses several symbols. This method prints the legend of those symbols.</summary>
/// <param name="stringBuilder">The visualization is build into this</param>
/// <param name="symbols">The symbols for which to generate a legend</param>
private static void AppendLegend(StringBuilder stringBuilder, IVisualizationSymbols symbols)
{
stringBuilder.AppendLine("\nConfiguration Details:");
stringBuilder.AppendLine("====================");
stringBuilder.AppendLine($"{symbols.Section} = Section");
stringBuilder.AppendLine($"{symbols.Provider} = Configuration Source");
stringBuilder.AppendLine($"{symbols.Active} = Active Value");
stringBuilder.AppendLine($"{symbols.Overridden} = Overridden Value");
stringBuilder.AppendLine("Indentation indicates nesting level");
}
/// <summary>Adds a configuration providers list to the visualization. Providers could be data files with configuration information, such as appsettings.json, or they might be middleware that converts other data sources into configuration information, such as EnvironmentVariablesConfigurationProvider.</summary>
/// <param name="stringBuilder"></param>
/// <param name="providers"></param>
/// <param name="symbols"></param>
private static void AppendProvidersList(
StringBuilder stringBuilder,
Dictionary<string, string> providers,
IVisualizationSymbols symbols)
{
stringBuilder.AppendLine("\nConfiguration Providers (in order of precedence):");
stringBuilder.AppendLine("============================================");
if(providers.Any()) {
foreach(var provider in providers)
stringBuilder.AppendLine($"{symbols.Provider} {provider.Value}");
}
else
stringBuilder.AppendLine("No provider information available");
stringBuilder.AppendLine();
}
/// <summary>Appends the section header to the visualization output.</summary>
/// <param name="stringBuilder">StringBuilder collecting the visualization output</param>
/// <param name="indentation">Current indentation string</param>
/// <param name="section">The configuration section being processed</param>
/// <param name="symbols">Symbols to use for visualization</param>
private static void AppendSectionHeader(
StringBuilder stringBuilder,
string indentation,
IConfigurationSection section,
IVisualizationSymbols symbols)
{
stringBuilder.Append($"{indentation}{symbols.Section} {section.Key}");
}
/// <summary>Appends a section's value and its provider history to the visualization output.</summary>
/// <param name="stringBuilder">StringBuilder collecting the visualization output</param>
/// <param name="indentation">Current indentation string</param>
/// <param name="currentPath">Full configuration path of the current section</param>
/// <param name="value">The section's value</param>
/// <param name="valueProviders">Dictionary mapping configuration paths to their provider history</param>
/// <param name="providers">Dictionary mapping provider keys to their display names</param>
/// <param name="symbols">Symbols to use for visualization</param>
private static void AppendSectionValue(
StringBuilder stringBuilder,
string indentation,
string currentPath,
string value,
Dictionary<string, List<(string Provider, string Value)>> valueProviders,
Dictionary<string, string> providers,
IVisualizationSymbols symbols)
{
stringBuilder.Append($" = {value}");
if(!valueProviders.TryGetValue(currentPath, out var sources))
return;
if(sources.Count > 1)
AppendValueHistory(stringBuilder, indentation, sources, providers, value, symbols);
else if(sources.Count == 1) {
var (providerKey, _) = sources[0];
stringBuilder.Append($" {symbols.Provider} [{providers[providerKey]}]");
}
stringBuilder.AppendLine();
}
/// <summary>Appends the history of a specific value to the visualization. This is useful because .NET configuration values can be set and then overridden. For example, a default connection string might be set in appsettings.json, and then overridden in appsettings.Development.json. This may be confusing to a developer who sets the value in appsettings.json and expects it to be honored. With this history, the evolution of each value is clear.</summary>
private static void AppendValueHistory(
StringBuilder stringBuilder,
string indentation,
List<(string Provider, string Value)> sources,
Dictionary<string, string> providers,
string currentValue,
IVisualizationSymbols symbols)
{
stringBuilder.AppendLine();
stringBuilder.Append($"{indentation} {symbols.HistoryArrow} Value history (most recent first):");
var isFirst = true;
foreach(var (providerKey, historicalValue) in sources) {
stringBuilder.AppendLine();
stringBuilder.Append($"{indentation} {symbols.Provider} ");
if(isFirst)
stringBuilder.Append($"({symbols.Active} active) [{providers[providerKey]}] = {currentValue}");
else
stringBuilder.Append($"({symbols.Overridden} overridden) [{providers[providerKey]}] = {historicalValue}");
isFirst = false;
}
}
/// <summary>Builds a full configuration path by combining a parent path and key.</summary>
/// <param name="parentPath">Full configuration path of the parent section, or null if root</param>
/// <param name="key">The key of the current section</param>
/// <returns>The full configuration path for the current section</returns>
private static string BuildPath(string? parentPath, string key) =>
string.IsNullOrEmpty(parentPath) ? key : $"{parentPath}:{key}";
/// <summary>Collects information about each provider which contributed to the configuration information</summary>
/// <param name="configRoot">This is the root of the IConfiguration hierarchy</param>
private static (Dictionary<string, string> Providers, Dictionary<string, List<(string Provider, string Value)>> ValueProviders) CollectProviderInformation(IConfigurationRoot configRoot)
{
var providers = new Dictionary<string, string>();
var valueProviders = new Dictionary<string, List<(string Provider, string Value)>>();
foreach(var provider in configRoot.Providers.Reverse()) {
var providerKey = provider.ToString();
var providerName = provider switch {
FileConfigurationProvider fileProvider => fileProvider.Source.Path,
_ => provider.GetType().Name
};
if(providerKey is null || providerName is null)
continue;
providers[providerKey] = providerName;
foreach(var child in configRoot.GetChildren())
CollectValues(child, "", provider, providerKey, valueProviders);
}
return (providers, valueProviders);
}
private static void CollectValues(
IConfigurationSection section,
string parentPath,
IConfigurationProvider provider,
string providerKey,
Dictionary<string, List<(string Provider, string Value)>> valueProviders)
{
var currentPath = string.IsNullOrEmpty(parentPath) ? section.Key : $"{parentPath}:{section.Key}";
if(provider.TryGet(currentPath, out var value)) {
if(!valueProviders.ContainsKey(currentPath))
valueProviders[currentPath] = new List<(string, string)>();
if(value is not null)
valueProviders[currentPath].Add((providerKey, value));
}
foreach(var child in section.GetChildren())
CollectValues(child, currentPath, provider, providerKey, valueProviders);
}
internal static string GenerateVisualization(IConfiguration configuration, bool excludeVendorSections, IVisualizationSymbols symbols, IEnumerable<string>? excludePrefixes = null)
{
if(symbols is VisualizationSymbols.Unicode)
Console.OutputEncoding = Encoding.UTF8;
var vendorPrefixes = excludePrefixes?.ToHashSet(StringComparer.OrdinalIgnoreCase) ?? DefaultVendorPrefixes;
var stringBuilder = new StringBuilder();
stringBuilder.AppendLine("Configuration Structure:\n=======================");
var (providers, valueProviders) = configuration is IConfigurationRoot configRoot
? CollectProviderInformation(configRoot)
: (new Dictionary<string, string>(), new Dictionary<string, List<(string Provider, string Value)>>());
AppendProvidersList(stringBuilder, providers, symbols);
foreach(var child in configuration.GetChildren())
TraverseSection(
child, 0, null, stringBuilder,
valueProviders, providers, vendorPrefixes, excludeVendorSections, symbols);
AppendLegend(stringBuilder, symbols);
return stringBuilder.ToString();
}
/// <summary>Determines if a configuration section is a leaf node with a value.</summary>
/// <param name="section">The configuration section to check</param>
/// <returns>True if the section has a value and no children, false otherwise</returns>
private static bool IsLeafWithValue(IConfigurationSection section) => !string.IsNullOrEmpty(section.Value) && !section.GetChildren().Any();
/// <summary>Determines if a configuration section should be skipped based on vendor prefix rules.</summary>
/// <param name="section">The configuration section to check</param>
/// <param name="parentPath">Full configuration path of the parent section, or null if root</param>
/// <param name="vendorPrefixes">Set of configuration path prefixes to potentially exclude</param>
/// <param name="excludeVendorSections">Whether to exclude sections matching vendor prefixes</param>
/// <returns>True if the section should be skipped, false otherwise</returns>
private static bool ShouldSkipSection(IConfigurationSection section, string? parentPath, HashSet<string> vendorPrefixes, bool excludeVendorSections)
{
if(!excludeVendorSections)
return false;
var currentPath = BuildPath(parentPath, section.Key);
return vendorPrefixes.Any(prefix => currentPath.StartsWith(prefix, StringComparison.OrdinalIgnoreCase));
}
/// <summary>Recursively traverses and visualizes a configuration section, its value, and any children.</summary>
/// <param name="section">The configuration section to process</param>
/// <param name="depth">Current indentation depth</param>
/// <param name="parentPath">Full configuration path of the parent section, or null if root</param>
/// <param name="stringBuilder">StringBuilder collecting the visualization output</param>
/// <param name="valueProviders">Dictionary mapping configuration paths to their provider history</param>
/// <param name="providers">Dictionary mapping provider keys to their display names</param>
/// <param name="vendorPrefixes">Set of configuration path prefixes to potentially exclude</param>
/// <param name="excludeVendorSections">Whether to exclude sections matching vendor prefixes</param>
/// <param name="symbols">Symbols to use for visualization</param>
private static void TraverseSection(
IConfigurationSection section,
int depth,
string? parentPath,
StringBuilder stringBuilder,
Dictionary<string, List<(string Provider, string Value)>> valueProviders,
Dictionary<string, string> providers,
HashSet<string> vendorPrefixes,
bool excludeVendorSections,
IVisualizationSymbols symbols)
{
if(ShouldSkipSection(section, parentPath, vendorPrefixes, excludeVendorSections))
return;
var currentPath = BuildPath(parentPath, section.Key);
var indentation = new string(' ', depth * 2);
AppendSectionHeader(stringBuilder, indentation, section, symbols);
if(IsLeafWithValue(section))
AppendSectionValue(stringBuilder, indentation, currentPath, section.Value!, valueProviders, providers, symbols);
foreach(var child in section.GetChildren())
TraverseSection(child, depth + 1, currentPath, stringBuilder, valueProviders, providers, vendorPrefixes, excludeVendorSections, symbols);
}
}
internal interface IVisualizationSymbols
{
string Section { get; }
string Provider { get; }
string Active { get; }
string Overridden { get; }
string HistoryArrow { get; }
}
internal static class VisualizationSymbols
{
/// <summary>These symbols will be used in the output to attempt to graphically represent it as much as possible for text. Terminals which do not support emoticons will need to use the Ascii symbols.</summary>
internal class Unicode : IVisualizationSymbols
{
public string Section => "📂";
public string Provider => "📄";
public string Active => "✓";
public string Overridden => "↪";
public string HistoryArrow => "↳";
}
/// <summary>These symbols will be used in the output to attempt to graphically represent it as much as possible for ASCII text. Terminals that support unicode characters should opt for the Unicode symbols as they look much nicer.</summary>
internal class Ascii : IVisualizationSymbols
{
public string Section => "+";
public string Provider => "*";
public string Active => ">";
public string Overridden => "-";
public string HistoryArrow => "\\";
}
}