From 0ab5243ae6dbe64792bd69f3007f3bac51d56008 Mon Sep 17 00:00:00 2001 From: James Thompson Date: Sat, 3 Feb 2024 13:45:14 +1100 Subject: [PATCH 1/4] #921 work on implementing S134 --- Steeltoe.Debug.ruleset | 1 + Steeltoe.Release.ruleset | 1 + .../PropertyPlaceHolderHelper.cs | 97 +++++++++---------- src/Common/src/Common/Net/InetUtils.cs | 65 +++++-------- .../Common/Reflection/ReflectionHelpers.cs | 17 ++-- .../ConfigServerConfigurationProvider.cs | 27 +++--- .../Encryption/EncryptionResolverProvider.cs | 28 ++++-- .../PlaceholderResolverProvider.cs | 28 ++++-- .../ConnectionStringPostProcessor.cs | 11 +-- .../MongoDb/MongoDbConnectionStringBuilder.cs | 13 +-- .../Redis/RedisConnectionStringBuilder.cs | 47 ++++----- .../src/Eureka/AppInfo/Applications.cs | 9 +- src/Discovery/src/Eureka/DiscoveryClient.cs | 11 +-- .../Info/Contributor/GitInfoContributor.cs | 41 ++++---- .../Metrics/MetricsEndpointMiddleware.cs | 15 ++- .../Metrics/Observer/EventCounterListener.cs | 11 +-- .../Web/Hypermedia/HypermediaService.cs | 19 ++-- .../Exporters/WavefrontMetricsExporter.cs | 69 +++++++------ 18 files changed, 236 insertions(+), 274 deletions(-) diff --git a/Steeltoe.Debug.ruleset b/Steeltoe.Debug.ruleset index 24c4c6487d..2b34016877 100644 --- a/Steeltoe.Debug.ruleset +++ b/Steeltoe.Debug.ruleset @@ -24,6 +24,7 @@ + diff --git a/Steeltoe.Release.ruleset b/Steeltoe.Release.ruleset index 821ac4d5cc..abbf5b73be 100644 --- a/Steeltoe.Release.ruleset +++ b/Steeltoe.Release.ruleset @@ -17,6 +17,7 @@ + diff --git a/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs b/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs index c0817522c0..a9ecb0b93f 100644 --- a/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs +++ b/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs @@ -95,76 +95,69 @@ private static string ParseStringValue(string property, IConfiguration configura } var result = new StringBuilder(property); - - while (startIndex != -1) + int endIndex; + while (startIndex != -1 && (endIndex = FindEndIndex(result, startIndex)) != -1) { - int endIndex = FindEndIndex(result, startIndex); - - if (endIndex != -1) - { - string placeholder = result.Substring(startIndex + Prefix.Length, endIndex); - string originalPlaceholder = placeholder; + string placeholder = result.Substring(startIndex + Prefix.Length, endIndex); - if (!visitedPlaceHolders.Add(originalPlaceholder)) - { - throw new InvalidOperationException($"Found circular placeholder reference '{originalPlaceholder}' in property definitions."); - } + string originalPlaceholder = placeholder; - // Recursive invocation, parsing placeholders contained in the placeholder key. - placeholder = ParseStringValue(placeholder, configuration, visitedPlaceHolders); + if (!visitedPlaceHolders.Add(originalPlaceholder)) + { + throw new InvalidOperationException($"Found circular placeholder reference '{originalPlaceholder}' in property definitions."); + } - // Handle array references foo:bar[1]:baz format -> foo:bar:1:baz - string lookup = placeholder.Replace('[', ':').Replace("]", string.Empty, StringComparison.Ordinal); + // Recursive invocation, parsing placeholders contained in the placeholder key. + placeholder = ParseStringValue(placeholder, configuration, visitedPlaceHolders); - // Now obtain the value for the fully resolved key... - string propVal = configuration[lookup]; + // Handle array references foo:bar[1]:baz format -> foo:bar:1:baz + string lookup = placeholder.Replace('[', ':').Replace("]", string.Empty, StringComparison.Ordinal); - if (propVal == null) - { - int separatorIndex = placeholder.IndexOf(Separator, StringComparison.Ordinal); - - if (separatorIndex != -1) - { - string actualPlaceholder = placeholder.Substring(0, separatorIndex); - string defaultValue = placeholder.Substring(separatorIndex + Separator.Length); - propVal = configuration[actualPlaceholder] ?? defaultValue; - } - else if (useEmptyStringIfNotFound) - { - propVal = string.Empty; - } - } + // Now obtain the value for the fully resolved key... + string propVal = configuration[lookup]; - // Attempt to resolve as a spring-compatible placeholder - if (propVal == null) - { - // Replace Spring delimiters ('.') with MS-friendly delimiters (':') so Spring placeholders can also be resolved - lookup = placeholder.Replace('.', ':'); - propVal = configuration[lookup]; - } + if (propVal == null) + { + int separatorIndex = placeholder.IndexOf(Separator, StringComparison.Ordinal); - if (propVal != null) + if (separatorIndex != -1) { - // Recursive invocation, parsing placeholders contained in these - // previously resolved placeholder value. - propVal = ParseStringValue(propVal, configuration, visitedPlaceHolders); - result.Replace(startIndex, endIndex + Suffix.Length, propVal); - logger?.LogDebug("Resolved placeholder '{placeholder}'", placeholder); - startIndex = result.IndexOf(Prefix, startIndex + propVal.Length); + string actualPlaceholder = placeholder.Substring(0, separatorIndex); + string defaultValue = placeholder.Substring(separatorIndex + Separator.Length); + propVal = configuration[actualPlaceholder] ?? defaultValue; } - else + else if (useEmptyStringIfNotFound) { - // Proceed with unprocessed value. - startIndex = result.IndexOf(Prefix, endIndex + Prefix.Length); + propVal = string.Empty; } + } + + // Attempt to resolve as a spring-compatible placeholder + if (propVal == null) + { + // Replace Spring delimiters ('.') with MS-friendly delimiters (':') so Spring placeholders can also be resolved + lookup = placeholder.Replace('.', ':'); + propVal = configuration[lookup]; + } - visitedPlaceHolders.Remove(originalPlaceholder); + if (propVal != null) + { + // Recursive invocation, parsing placeholders contained in these + // previously resolved placeholder value. + propVal = ParseStringValue(propVal, configuration, visitedPlaceHolders); + result.Replace(startIndex, endIndex + Suffix.Length, propVal); + logger?.LogDebug("Resolved placeholder '{placeholder}'", placeholder); + startIndex = result.IndexOf(Prefix, startIndex + propVal.Length); } else { - startIndex = -1; + // Proceed with unprocessed value. + startIndex = result.IndexOf(Prefix, endIndex + Prefix.Length); } + + visitedPlaceHolders.Remove(originalPlaceholder); + } return result.ToString(); diff --git a/src/Common/src/Common/Net/InetUtils.cs b/src/Common/src/Common/Net/InetUtils.cs index 817e30dc0b..b90c2f05a9 100644 --- a/src/Common/src/Common/Net/InetUtils.cs +++ b/src/Common/src/Common/Net/InetUtils.cs @@ -46,39 +46,30 @@ public IPAddress FindFirstNonLoopbackAddress() try { int lowest = int.MaxValue; - NetworkInterface[] interfaces = NetworkInterface.GetAllNetworkInterfaces(); + NetworkInterface[] interfaces = NetworkInterface.GetAllNetworkInterfaces() + .Where(x => x.OperationalStatus == OperationalStatus.Up && !x.IsReceiveOnly && !IgnoreInterface(x.Name)) + .ToArray(); foreach (NetworkInterface @interface in interfaces) { - if (@interface.OperationalStatus == OperationalStatus.Up && !@interface.IsReceiveOnly) + _logger?.LogTrace("Testing interface: {name}, {id}", @interface.Name, @interface.Id); + + IPInterfaceProperties props = @interface.GetIPProperties(); + IPv4InterfaceProperties ipProps = props.GetIPv4Properties(); + + if (ipProps.Index < lowest || result == null) + { + lowest = ipProps.Index; + } + else { - _logger?.LogTrace("Testing interface: {name}, {id}", @interface.Name, @interface.Id); - - IPInterfaceProperties props = @interface.GetIPProperties(); - IPv4InterfaceProperties ipProps = props.GetIPv4Properties(); - - if (ipProps.Index < lowest || result == null) - { - lowest = ipProps.Index; - } - else - { - continue; - } - - if (!IgnoreInterface(@interface.Name)) - { - foreach (UnicastIPAddressInformation addressInfo in props.UnicastAddresses) - { - IPAddress address = addressInfo.Address; - - if (IsInet4Address(address) && !IsLoopbackAddress(address) && IsPreferredAddress(address)) - { - _logger?.LogTrace("Found non-loopback interface: {name}", @interface.Name); - result = address; - } - } - } + continue; + } + foreach (UnicastIPAddressInformation addressInfo in props.UnicastAddresses + .Where(x => IsInet4Address(x.Address) && !IsLoopbackAddress(x.Address) && IsPreferredAddress(x.Address))) + { + _logger?.LogTrace("Found non-loopback interface: {name}", @interface.Name); + result = addressInfo.Address; } } } @@ -195,30 +186,18 @@ internal HostInfo ConvertAddress(IPAddress address) internal IPAddress ResolveHostAddress(string hostName) { - IPAddress result = null; - try { IPAddress[] results = Dns.GetHostAddresses(hostName); - if (results.Length > 0) - { - foreach (IPAddress address in results) - { - if (address.AddressFamily == AddressFamily.InterNetwork) - { - result = address; - break; - } - } - } + return Array.Find(results, x => x.AddressFamily == AddressFamily.InterNetwork); } catch (Exception e) { _logger?.LogWarning(e, "Unable to resolve host address"); } - return result; + return null; } internal string ResolveHostName() diff --git a/src/Common/src/Common/Reflection/ReflectionHelpers.cs b/src/Common/src/Common/Reflection/ReflectionHelpers.cs index e68528e800..8c09f619ba 100644 --- a/src/Common/src/Common/Reflection/ReflectionHelpers.cs +++ b/src/Common/src/Common/Reflection/ReflectionHelpers.cs @@ -213,16 +213,17 @@ public static Type FindType(string[] assemblyNames, string[] typeNames) { Assembly assembly = FindAssembly(assemblyName); - if (assembly != null) + if (assembly == null) { - foreach (string type in typeNames) - { - Type result = FindType(assembly, type); + continue; + } + foreach (string type in typeNames) + { + Type result = FindType(assembly, type); - if (result != null) - { - return result; - } + if (result != null) + { + return result; } } } diff --git a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs index 00c9146d94..37ee098157 100644 --- a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs +++ b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs @@ -535,21 +535,20 @@ private void AddConfigServerClientSettings(IDictionary data) Logger.LogInformation("Config Server returned status: {statusCode} invoking path: {requestUri}", response.StatusCode, WebUtility.UrlEncode(requestUri)); - if (response.StatusCode != HttpStatusCode.OK) - { - if (response.StatusCode == HttpStatusCode.NotFound) - { - return null; - } - - // Throw if status >= 400 - if (response.StatusCode >= HttpStatusCode.BadRequest) - { - // HttpClientErrorException - throw new HttpRequestException( - $"Config Server returned status: {response.StatusCode} invoking path: {WebUtility.UrlEncode(requestUri)}"); - } + if (response.StatusCode == HttpStatusCode.NotFound) + { + return null; + } + // Throw if status >= 400 + else if (response.StatusCode >= HttpStatusCode.BadRequest) + { + // HttpClientErrorException + throw new HttpRequestException( + $"Config Server returned status: {response.StatusCode} invoking path: {WebUtility.UrlEncode(requestUri)}"); + } + else if (response.StatusCode != HttpStatusCode.OK) + { return null; } diff --git a/src/Configuration/src/Encryption/EncryptionResolverProvider.cs b/src/Configuration/src/Encryption/EncryptionResolverProvider.cs index db31e4bfbc..94473cb521 100644 --- a/src/Configuration/src/Encryption/EncryptionResolverProvider.cs +++ b/src/Configuration/src/Encryption/EncryptionResolverProvider.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. +using System; using System.Text.RegularExpressions; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.Logging; @@ -193,22 +194,16 @@ public void Dispose() { HashSet disposables = []; - foreach (IConfigurationProvider provider in Providers) + foreach (IDisposable disposable in GetDisposables(Providers)) { - if (provider is IDisposable disposable) - { - disposables.Add(disposable); - } + disposables.Add(disposable); } if (Configuration != null) { - foreach (IConfigurationProvider provider in Configuration.Providers) + foreach (IDisposable disposable in GetDisposables(Configuration.Providers)) { - if (provider is IDisposable disposable) - { - disposables.Add(disposable); - } + disposables.Add(disposable); } } @@ -220,4 +215,17 @@ public void Dispose() _isDisposed = true; } } + + private HashSet GetDisposables(IEnumerable providers) + { + HashSet disposables = []; + foreach (IConfigurationProvider provider in providers) + { + if (provider is IDisposable disposable) + { + disposables.Add(disposable); + } + } + return disposables; + } } diff --git a/src/Configuration/src/Placeholder/PlaceholderResolverProvider.cs b/src/Configuration/src/Placeholder/PlaceholderResolverProvider.cs index 3d49dacd38..89abfc634a 100644 --- a/src/Configuration/src/Placeholder/PlaceholderResolverProvider.cs +++ b/src/Configuration/src/Placeholder/PlaceholderResolverProvider.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. +using System; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; @@ -173,22 +174,16 @@ public void Dispose() { HashSet disposables = []; - foreach (IConfigurationProvider provider in Providers) + foreach (IDisposable disposable in GetDisposables(Providers)) { - if (provider is IDisposable disposable) - { - disposables.Add(disposable); - } + disposables.Add(disposable); } if (Configuration != null) { - foreach (IConfigurationProvider provider in Configuration.Providers) + foreach (IDisposable disposable in GetDisposables(Configuration.Providers)) { - if (provider is IDisposable disposable) - { - disposables.Add(disposable); - } + disposables.Add(disposable); } } @@ -200,4 +195,17 @@ public void Dispose() _isDisposed = true; } } + + private HashSet GetDisposables(IEnumerable providers) + { + HashSet disposables = []; + foreach (IConfigurationProvider provider in providers) + { + if (provider is IDisposable disposable) + { + disposables.Add(disposable); + } + } + return disposables; + } } diff --git a/src/Connectors/src/Connectors/ConnectionStringPostProcessor.cs b/src/Connectors/src/Connectors/ConnectionStringPostProcessor.cs index f139c54458..4922ea877f 100644 --- a/src/Connectors/src/Connectors/ConnectionStringPostProcessor.cs +++ b/src/Connectors/src/Connectors/ConnectionStringPostProcessor.cs @@ -139,14 +139,11 @@ private void SetConnectionString(IDictionary configurationData, // Take the connection string from appsettings.json as baseline, then merge cloud-provided secrets into it. connectionStringBuilder.ConnectionString = secretValue; } - else + // Never merge separately-defined secrets from appsettings.json into the connection string. + // Earlier Steeltoe versions used to do that, which raised the question what takes precedence. + else if (!IsPartOfConnectionString(secretName)) { - // Never merge separately-defined secrets from appsettings.json into the connection string. - // Earlier Steeltoe versions used to do that, which raised the question what takes precedence. - if (!IsPartOfConnectionString(secretName)) - { - separateSecrets[secretName] = secretValue; - } + separateSecrets[secretName] = secretValue; } } } diff --git a/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs b/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs index 9cf767dc40..0db3867b44 100644 --- a/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs +++ b/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs @@ -170,16 +170,13 @@ private void FromConnectionString(string? connectionString, bool preserveUnknown NameValueCollection queryCollection = HttpUtility.ParseQueryString(uri.Query); - foreach (string? key in queryCollection.AllKeys) + foreach (string? key in queryCollection.AllKeys.Where(x => x != null)) { - if (key != null) - { - string? value = queryCollection.Get(key); + string? value = queryCollection.Get(key); - if (value != null) - { - _settings[key] = value; - } + if (key != null && value != null) + { + _settings[key] = value; } } } diff --git a/src/Connectors/src/Connectors/Redis/RedisConnectionStringBuilder.cs b/src/Connectors/src/Connectors/Redis/RedisConnectionStringBuilder.cs index 10c3d9b534..ee6a09d663 100644 --- a/src/Connectors/src/Connectors/Redis/RedisConnectionStringBuilder.cs +++ b/src/Connectors/src/Connectors/Redis/RedisConnectionStringBuilder.cs @@ -87,36 +87,37 @@ private void FromConnectionString(string? connectionString) { _settings.Clear(); - if (!string.IsNullOrEmpty(connectionString)) + if (string.IsNullOrEmpty(connectionString)) { - foreach (string option in connectionString.Split(',').Where(element => !string.IsNullOrWhiteSpace(element))) + return; + } + foreach (string option in connectionString.Split(',').Where(element => !string.IsNullOrWhiteSpace(element))) + { + int equalsIndex = option.IndexOf('='); + + if (equalsIndex != -1) { - int equalsIndex = option.IndexOf('='); + string name = option[..equalsIndex].Trim(); + string value = option[(equalsIndex + 1)..].Trim(); + _settings[name] = value; + } + else if (option.Contains(',')) + { + // Redis allows multiple servers in the connection string, but we haven't found any service bindings that actually use that. + throw new NotImplementedException("Support for multiple servers is not implemented. Please open a GitHub issue if you need this."); + } + else + { + string[] hostWithPort = option.Split(':', 2); + _settings[KnownKeywords.Host] = hostWithPort[0]; - if (equalsIndex != -1) + if (hostWithPort.Length > 1) { - string name = option[..equalsIndex].Trim(); - string value = option[(equalsIndex + 1)..].Trim(); - _settings[name] = value; - } - else - { - if (option.Contains(',')) - { - // Redis allows multiple servers in the connection string, but we haven't found any service bindings that actually use that. - throw new NotImplementedException("Support for multiple servers is not implemented. Please open a GitHub issue if you need this."); - } - - string[] hostWithPort = option.Split(':', 2); - _settings[KnownKeywords.Host] = hostWithPort[0]; - - if (hostWithPort.Length > 1) - { - _settings[KnownKeywords.Port] = hostWithPort[1]; - } + _settings[KnownKeywords.Port] = hostWithPort[1]; } } } + } private static void AssertIsKnownKeyword(string keyword) diff --git a/src/Discovery/src/Eureka/AppInfo/Applications.cs b/src/Discovery/src/Eureka/AppInfo/Applications.cs index 3bc2b5b9d9..46921e7793 100644 --- a/src/Discovery/src/Eureka/AppInfo/Applications.cs +++ b/src/Discovery/src/Eureka/AppInfo/Applications.cs @@ -259,14 +259,11 @@ private IList DoGetByVirtualHostName(string name, ConcurrentDictio { InstanceInfo inst = kvp.Value; - if (ReturnUpInstancesOnly) + if (ReturnUpInstancesOnly && inst.Status == InstanceStatus.Up) { - if (inst.Status == InstanceStatus.Up) - { - result.Add(inst); - } + result.Add(inst); } - else + else if(!ReturnUpInstancesOnly) { result.Add(inst); } diff --git a/src/Discovery/src/Eureka/DiscoveryClient.cs b/src/Discovery/src/Eureka/DiscoveryClient.cs index 0cd676a561..cee2e69626 100644 --- a/src/Discovery/src/Eureka/DiscoveryClient.cs +++ b/src/Discovery/src/Eureka/DiscoveryClient.cs @@ -259,15 +259,10 @@ private async void HandleInstanceStatusChanged(object sender, StatusChangedEvent logger.LogDebug("HandleInstanceStatusChanged {previousStatus}, {currentStatus}, {instanceId}, {dirty}", args.Previous, args.Current, args.InstanceId, info.IsDirty); - if (info.IsDirty) + if (info.IsDirty && await RegisterAsync(CancellationToken.None)) { - bool result = await RegisterAsync(CancellationToken.None); - - if (result) - { - info.IsDirty = false; - logger.LogInformation("HandleInstanceStatusChanged RegisterAsync succeeded"); - } + info.IsDirty = false; + logger.LogInformation("HandleInstanceStatusChanged RegisterAsync succeeded"); } } } diff --git a/src/Management/src/Endpoint/Info/Contributor/GitInfoContributor.cs b/src/Management/src/Endpoint/Info/Contributor/GitInfoContributor.cs index d657ff5191..126e31bf4c 100755 --- a/src/Management/src/Endpoint/Info/Contributor/GitInfoContributor.cs +++ b/src/Management/src/Endpoint/Info/Contributor/GitInfoContributor.cs @@ -54,34 +54,31 @@ public async Task ContributeAsync(IInfoBuilder builder, CancellationToken cancel { string[] lines = await File.ReadAllLinesAsync(propertiesPath, cancellationToken); - if (lines.Length > 0) + if (lines.Length == 0) { - var dictionary = new Dictionary(); - - foreach (string line in lines) - { - if (line.StartsWith('#') || !line.StartsWith("git.", StringComparison.OrdinalIgnoreCase)) - { - continue; - } - - string[] keyValuePair = line.Split('='); - - if (keyValuePair.Length != 2) - { - continue; - } + _logger.LogWarning("Unable to find valid GitInfo at {GitInfoLocation}", propertiesPath); + return null; + } + var dictionary = new Dictionary(); - string key = keyValuePair[0].Trim().Replace('.', ':'); - string value = keyValuePair[1].Replace("\\:", ":", StringComparison.Ordinal); + foreach (string line in lines.Where(x => x.StartsWith("git.", StringComparison.OrdinalIgnoreCase))) + { + string[] keyValuePair = line.Split('='); - dictionary[key] = value; + if (keyValuePair.Length != 2) + { + continue; } - var builder = new ConfigurationBuilder(); - builder.AddInMemoryCollection(dictionary); - return builder.Build(); + string key = keyValuePair[0].Trim().Replace('.', ':'); + string value = keyValuePair[1].Replace("\\:", ":", StringComparison.Ordinal); + + dictionary[key] = value; } + + var builder = new ConfigurationBuilder(); + builder.AddInMemoryCollection(dictionary); + return builder.Build(); } else { diff --git a/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs b/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs index e5e2e14575..3e17dea2c3 100644 --- a/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs +++ b/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs @@ -69,18 +69,15 @@ internal IList> ParseTags(IQueryCollection query) { var results = new List>(); - foreach (KeyValuePair parameter in query) + foreach (KeyValuePair parameter in query.Where(X => X.Key.Equals("tag", StringComparison.OrdinalIgnoreCase))) { - if (parameter.Key.Equals("tag", StringComparison.OrdinalIgnoreCase)) + foreach (string? value in parameter.Value) { - foreach (string? value in parameter.Value) - { - KeyValuePair? pair = ParseTag(value); + KeyValuePair? pair = ParseTag(value); - if (pair != null && !results.Contains(pair.Value)) - { - results.Add(pair.Value); - } + if (pair != null && !results.Contains(pair.Value)) + { + results.Add(pair.Value); } } } diff --git a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs index 93c737068f..867f597c58 100644 --- a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs +++ b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs @@ -63,21 +63,18 @@ protected override void OnEventWritten(EventWrittenEventArgs eventData) { ArgumentGuard.NotNull(eventData); - if (!_isInitialized) + if (!_isInitialized || (string.Equals(eventData.EventName, EventName, StringComparison.OrdinalIgnoreCase) && eventData.Payload != null)) { return; } try { - if (string.Equals(eventData.EventName, EventName, StringComparison.OrdinalIgnoreCase) && eventData.Payload != null) + foreach (IDictionary? payload in eventData.Payload) { - foreach (IDictionary? payload in eventData.Payload) + if (payload != null) { - if (payload != null) - { - ExtractAndRecordMetric(eventData.EventSource.Name, payload); - } + ExtractAndRecordMetric(eventData.EventSource.Name, payload); } } } diff --git a/src/Management/src/Endpoint/Web/Hypermedia/HypermediaService.cs b/src/Management/src/Endpoint/Web/Hypermedia/HypermediaService.cs index d55039e648..e961d64aff 100644 --- a/src/Management/src/Endpoint/Web/Hypermedia/HypermediaService.cs +++ b/src/Management/src/Endpoint/Web/Hypermedia/HypermediaService.cs @@ -71,19 +71,16 @@ public Links Invoke(string baseUrl) { selfLink = new Link(baseUrl); } - else + else if (!string.IsNullOrEmpty(endpointOptions.Id)) { - if (!string.IsNullOrEmpty(endpointOptions.Id)) + if (!links.Entries.ContainsKey(endpointOptions.Id)) { - if (!links.Entries.ContainsKey(endpointOptions.Id)) - { - string linkPath = $"{baseUrl.TrimEnd('/')}/{endpointOptions.Path}"; - links.Entries.Add(endpointOptions.Id, new Link(linkPath)); - } - else - { - _logger.LogWarning("Duplicate endpoint ID detected: {DuplicateEndpointId}", endpointOptions.Id); - } + string linkPath = $"{baseUrl.TrimEnd('/')}/{endpointOptions.Path}"; + links.Entries.Add(endpointOptions.Id, new Link(linkPath)); + } + else + { + _logger.LogWarning("Duplicate endpoint ID detected: {DuplicateEndpointId}", endpointOptions.Id); } } } diff --git a/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs b/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs index a0b9b09163..4f9bfb69f4 100644 --- a/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs +++ b/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. +using System.Runtime.CompilerServices; using Microsoft.Extensions.Logging; using OpenTelemetry; using OpenTelemetry.Metrics; @@ -58,50 +59,19 @@ public override ExportResult Export(in Batch batch) foreach (Metric metric in batch) { - bool isLong = ((int)metric.MetricType & 0b_0000_1111) == 0x0a; // I8 : signed 8 byte integer - bool isSum = metric.MetricType.IsSum(); try { - if (!metric.MetricType.IsHistogram()) + foreach (ref readonly MetricPoint metricPoint in metric.GetMetricPoints()) { - foreach (ref readonly MetricPoint metricPoint in metric.GetMetricPoints()) + long timestamp = metricPoint.EndTime.ToUnixTimeMilliseconds(); + IDictionary tags = GetTags(metricPoint.Tags); + foreach (var item in GetMetrics(metric, metricPoint)) { - long timestamp = metricPoint.EndTime.ToUnixTimeMilliseconds(); - double doubleValue; - - if (isLong) - { - doubleValue = isSum ? metricPoint.GetSumLong() : metricPoint.GetGaugeLastValueLong(); - } - else - { - doubleValue = isSum ? metricPoint.GetSumDouble() : metricPoint.GetGaugeLastValueDouble(); - } - - IDictionary tags = GetTags(metricPoint.Tags); - - _wavefrontSender.SendMetric(metric.Name.ToLowerInvariant(), doubleValue, timestamp, Options.Source, tags); - + _wavefrontSender.SendMetric(item.Key, item.Value, timestamp, Options.Source, tags); metricCount++; } } - else - { - foreach (ref readonly MetricPoint metricPoint in metric.GetMetricPoints()) - { - long timestamp = metricPoint.EndTime.ToUnixTimeMilliseconds(); - - IDictionary tags = GetTags(metricPoint.Tags); - - _wavefrontSender.SendMetric($"{metric.Name.ToLowerInvariant()}_count", metricPoint.GetHistogramCount(), timestamp, Options.Source, - tags); - - _wavefrontSender.SendMetric($"{metric.Name.ToLowerInvariant()}_sum", metricPoint.GetHistogramSum(), timestamp, Options.Source, tags); - - metricCount += 2; - } - } } catch (Exception exception) { @@ -113,6 +83,33 @@ public override ExportResult Export(in Batch batch) return ExportResult.Success; } + private List> GetMetrics(Metric metric, MetricPoint metricPoint) + { + bool isLong = ((int)metric.MetricType & 0b_0000_1111) == 0x0a; // I8 : signed 8 byte integer + bool isSum = metric.MetricType.IsSum(); + List> metrics = []; + if (!metric.MetricType.IsHistogram()) + { + double doubleValue; + + if (isLong) + { + doubleValue = isSum ? metricPoint.GetSumLong() : metricPoint.GetGaugeLastValueLong(); + } + else + { + doubleValue = isSum ? metricPoint.GetSumDouble() : metricPoint.GetGaugeLastValueDouble(); + } + metrics.Add(new KeyValuePair(metric.Name.ToLowerInvariant(), doubleValue)); + } + else + { + metrics.Add(new KeyValuePair($"{metric.Name.ToLowerInvariant()}_count", metricPoint.GetHistogramCount())); + metrics.Add(new KeyValuePair($"{metric.Name.ToLowerInvariant()}_sum", metricPoint.GetHistogramSum())); + } + return metrics; + } + private IDictionary GetTags(ReadOnlyTagCollection inputTags) { IDictionary tags = inputTags.AsDictionary(); From 8119e2005afb573fa861948c15c5b50d84962aea Mon Sep 17 00:00:00 2001 From: James Thompson Date: Sat, 3 Feb 2024 14:35:48 +1100 Subject: [PATCH 2/4] #935 enable S3995 --- Steeltoe.Release.ruleset | 1 + .../PropertyPlaceHolderHelper.cs | 62 +++++++++---------- .../ConfigServerConfigurationProvider.cs | 62 ++++++++++--------- src/Discovery/src/Eureka/DiscoveryClient.cs | 28 +++------ .../src/Eureka/EurekaPostConfigurer.cs | 2 + .../src/Eureka/Transport/EurekaHttpClient.cs | 8 +-- .../Metrics/Observer/EventCounterListener.cs | 30 ++++++--- .../ThreadDump/EventPipeThreadDumper.cs | 2 + 8 files changed, 98 insertions(+), 97 deletions(-) diff --git a/Steeltoe.Release.ruleset b/Steeltoe.Release.ruleset index abbf5b73be..e8b6bf3f74 100644 --- a/Steeltoe.Release.ruleset +++ b/Steeltoe.Release.ruleset @@ -43,6 +43,7 @@ + diff --git a/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs b/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs index a9ecb0b93f..66a356bb49 100644 --- a/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs +++ b/src/Common/src/Common/Configuration/PropertyPlaceHolderHelper.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. +using System.Linq; using System.Text; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.Logging; @@ -77,23 +78,13 @@ public static IEnumerable> GetResolvedConfiguration private static string ParseStringValue(string property, IConfiguration configuration, ISet visitedPlaceHolders, ILogger logger = null, bool useEmptyStringIfNotFound = false) { - if (configuration == null) - { - return property; - } - - if (string.IsNullOrEmpty(property)) + if (configuration == null || string.IsNullOrEmpty(property)) { return property; } int startIndex = property.IndexOf(Prefix, StringComparison.Ordinal); - if (startIndex == -1) - { - return property; - } - var result = new StringBuilder(property); int endIndex; while (startIndex != -1 && (endIndex = FindEndIndex(result, startIndex)) != -1) @@ -117,29 +108,7 @@ private static string ParseStringValue(string property, IConfiguration configura // Now obtain the value for the fully resolved key... string propVal = configuration[lookup]; - if (propVal == null) - { - int separatorIndex = placeholder.IndexOf(Separator, StringComparison.Ordinal); - - if (separatorIndex != -1) - { - string actualPlaceholder = placeholder.Substring(0, separatorIndex); - string defaultValue = placeholder.Substring(separatorIndex + Separator.Length); - propVal = configuration[actualPlaceholder] ?? defaultValue; - } - else if (useEmptyStringIfNotFound) - { - propVal = string.Empty; - } - } - - // Attempt to resolve as a spring-compatible placeholder - if (propVal == null) - { - // Replace Spring delimiters ('.') with MS-friendly delimiters (':') so Spring placeholders can also be resolved - lookup = placeholder.Replace('.', ':'); - propVal = configuration[lookup]; - } + propVal ??= ResolveValue(configuration, placeholder, propVal, useEmptyStringIfNotFound); if (propVal != null) { @@ -163,6 +132,31 @@ private static string ParseStringValue(string property, IConfiguration configura return result.ToString(); } + private static string ResolveValue(IConfiguration configuration, string placeholder, string propVal, bool useEmptyStringIfNotFound) + { + int separatorIndex = placeholder.IndexOf(Separator, StringComparison.Ordinal); + + if (separatorIndex != -1) + { + string actualPlaceholder = placeholder.Substring(0, separatorIndex); + string defaultValue = placeholder.Substring(separatorIndex + Separator.Length); + propVal = configuration[actualPlaceholder] ?? defaultValue; + } + else if (useEmptyStringIfNotFound) + { + propVal = string.Empty; + } + + // Attempt to resolve as a spring-compatible placeholder + if (propVal == null) + { + // Replace Spring delimiters ('.') with MS-friendly delimiters (':') so Spring placeholders can also be resolved + string lookup = placeholder.Replace('.', ':'); + propVal = configuration[lookup]; + } + return propVal; + } + private static int FindEndIndex(StringBuilder property, int startIndex) { int index = startIndex + Prefix.Length; diff --git a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs index 37ee098157..c7d890576d 100644 --- a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs +++ b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs @@ -277,34 +277,7 @@ public override void Load() if (updateDictionary) { - var data = new Dictionary(StringComparer.OrdinalIgnoreCase); - - if (!string.IsNullOrEmpty(env.State)) - { - data["spring:cloud:config:client:state"] = env.State; - } - - if (!string.IsNullOrEmpty(env.Version)) - { - data["spring:cloud:config:client:version"] = env.Version; - } - - IList sources = env.PropertySources; - int index = sources.Count - 1; - - for (; index >= 0; index--) - { - AddPropertySource(sources[index], data); - } - - // Adds client settings (e.g. spring:cloud:config:uri, etc.) back to the (new) Data dictionary - AddConfigServerClientSettings(data); - - if (!AreDictionariesEqual(Data, data)) - { - Data = data; - OnReload(); - } + UpdateDictionary(env); } return env; @@ -326,6 +299,39 @@ public override void Load() return null; } + private void UpdateDictionary(ConfigEnvironment env) + { + + var data = new Dictionary(StringComparer.OrdinalIgnoreCase); + + if (!string.IsNullOrEmpty(env.State)) + { + data["spring:cloud:config:client:state"] = env.State; + } + + if (!string.IsNullOrEmpty(env.Version)) + { + data["spring:cloud:config:client:version"] = env.Version; + } + + IList sources = env.PropertySources; + int index = sources.Count - 1; + + for (; index >= 0; index--) + { + AddPropertySource(sources[index], data); + } + + // Adds client settings (e.g. spring:cloud:config:uri, etc.) back to the (new) Data dictionary + AddConfigServerClientSettings(data); + + if (!AreDictionariesEqual(Data, data)) + { + Data = data; + OnReload(); + } + } + private static bool AreDictionariesEqual(IDictionary first, IDictionary second) { return first.Count == second.Count && first.Keys.All(firstKey => diff --git a/src/Discovery/src/Eureka/DiscoveryClient.cs b/src/Discovery/src/Eureka/DiscoveryClient.cs index cee2e69626..de0b64599f 100644 --- a/src/Discovery/src/Eureka/DiscoveryClient.cs +++ b/src/Discovery/src/Eureka/DiscoveryClient.cs @@ -168,17 +168,14 @@ public IList GetInstancesByVipAddressAndAppName(string vipAddress, return result; } - foreach (Application app in localRegionApps.GetRegisteredApplications()) + foreach (InstanceInfo instance in localRegionApps.GetRegisteredApplications().SelectMany(x => x.Instances)) { - foreach (InstanceInfo instance in app.Instances) - { - string instanceVipAddress = secure ? instance.SecureVipAddress : instance.VipAddress; + string instanceVipAddress = secure ? instance.SecureVipAddress : instance.VipAddress; - if (vipAddress.Equals(instanceVipAddress, StringComparison.OrdinalIgnoreCase) && - appName.Equals(instance.AppName, StringComparison.OrdinalIgnoreCase)) - { - result.Add(instance); - } + if (vipAddress.Equals(instanceVipAddress, StringComparison.OrdinalIgnoreCase) && + appName.Equals(instance.AppName, StringComparison.OrdinalIgnoreCase)) + { + result.Add(instance); } } @@ -429,16 +426,9 @@ protected internal async Task FetchFullRegistryAsync(CancellationT long startingCounter = registryFetchCounter; Applications fetched = null; - EurekaHttpResponse resp; - - if (string.IsNullOrEmpty(ClientConfiguration.RegistryRefreshSingleVipAddress)) - { - resp = await HttpClient.GetApplicationsAsync(cancellationToken); - } - else - { - resp = await HttpClient.GetVipAsync(ClientConfiguration.RegistryRefreshSingleVipAddress, cancellationToken); - } + EurekaHttpResponse resp = string.IsNullOrEmpty(ClientConfiguration.RegistryRefreshSingleVipAddress) + ? await HttpClient.GetApplicationsAsync(cancellationToken) + : await HttpClient.GetVipAsync(ClientConfiguration.RegistryRefreshSingleVipAddress, cancellationToken); logger.LogDebug("FetchFullRegistry returned: {StatusCode}, {Response}", resp.StatusCode, resp.Response != null ? resp.Response.ToString() : "null"); diff --git a/src/Discovery/src/Eureka/EurekaPostConfigurer.cs b/src/Discovery/src/Eureka/EurekaPostConfigurer.cs index 3ac74562ec..c9e7b6156d 100644 --- a/src/Discovery/src/Eureka/EurekaPostConfigurer.cs +++ b/src/Discovery/src/Eureka/EurekaPostConfigurer.cs @@ -71,7 +71,9 @@ public static void UpdateConfiguration(EurekaServiceInfo si, EurekaClientOptions /// /// Information about this application instance. /// +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high public static void UpdateConfiguration(IConfiguration configuration, EurekaInstanceOptions options, IApplicationInstanceInfo instanceInfo) +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high { string defaultIdEnding = $":{EurekaInstanceConfiguration.DefaultAppName}:{EurekaInstanceConfiguration.DefaultNonSecurePort}"; diff --git a/src/Discovery/src/Eureka/Transport/EurekaHttpClient.cs b/src/Discovery/src/Eureka/Transport/EurekaHttpClient.cs index 898da98fb3..a8dd00559b 100644 --- a/src/Discovery/src/Eureka/Transport/EurekaHttpClient.cs +++ b/src/Discovery/src/Eureka/Transport/EurekaHttpClient.cs @@ -25,9 +25,9 @@ public class EurekaHttpClient : IEurekaHttpClient private const int DefaultGetAccessTokenTimeout = 10000; // Milliseconds private static readonly char[] ColonDelimit = - { + [ ':' - }; + ]; private readonly IOptionsMonitor _configurationOptions; @@ -190,10 +190,6 @@ public virtual async Task> SendHeartBeatAsync(s { string responseBody = await response.Content.ReadAsStringAsync(cancellationToken); - if (response.IsSuccessStatusCode && string.IsNullOrEmpty(responseBody)) - { - // request was successful but body was empty. This is OK, we don't need a response body - } logger?.LogError(exception, "Failed to read heartbeat response. Response code: {responseCode}, Body: {responseBody}", response.StatusCode, responseBody); diff --git a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs index 867f597c58..af9fed8248 100644 --- a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs +++ b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs @@ -3,6 +3,7 @@ // See the LICENSE file in the project root for more information. using System.Collections.Concurrent; +using System.Collections.Generic; using System.Diagnostics.Metrics; using System.Diagnostics.Tracing; using System.Globalization; @@ -179,17 +180,10 @@ private void ExtractAndRecordMetric(string eventSourceName, IDictionary> labels = MakeLabels(payload.Value?.ToString()); + if (labels.Any()) { - string[] keyValuePairStrings = metadata.Split(','); - - foreach (string keyValuePairString in keyValuePairStrings) - { - string[] keyValuePair = keyValuePairString.Split(':'); - labelSet.Add(KeyValuePair.Create(keyValuePair[0], (object?)keyValuePair[1])); - } + labelSet.AddRange(labels); } break; @@ -214,6 +208,22 @@ private void ExtractAndRecordMetric(string eventSourceName, IDictionary> MakeLabels(string? metadata) + { + List> labelSet = new List>(); + if (!string.IsNullOrEmpty(metadata)) + { + string[] keyValuePairStrings = metadata.Split(','); + + foreach (string keyValuePairString in keyValuePairStrings) + { + string[] keyValuePair = keyValuePairString.Split(':'); + labelSet.Add(KeyValuePair.Create(keyValuePair[0], (object?)keyValuePair[1])); + } + } + return labelSet; + } + private Measurement ObserveDouble(string name, List> labelSet) { return new Measurement(_lastDoubleValue[name], labelSet); diff --git a/src/Management/src/Endpoint/ThreadDump/EventPipeThreadDumper.cs b/src/Management/src/Endpoint/ThreadDump/EventPipeThreadDumper.cs index 3a940e8b36..22e85cbdbe 100644 --- a/src/Management/src/Endpoint/ThreadDump/EventPipeThreadDumper.cs +++ b/src/Management/src/Endpoint/ThreadDump/EventPipeThreadDumper.cs @@ -338,7 +338,9 @@ private bool TryParseParameters(string input, ref string remaining, [NotNullWhen } // Much of this code is from PerfView/TraceLog.cs +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high private SourceLocation? GetSourceLine(TraceEventStackSource stackSource, StackSourceFrameIndex frameIndex, SymbolReader reader) +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high { TraceLog log = stackSource.TraceLog; uint codeAddress = (uint)frameIndex - (uint)StackSourceFrameIndex.Start; From 75be8dd2b4acfdf86048bfa8d6444019f7d181eb Mon Sep 17 00:00:00 2001 From: James Thompson Date: Sat, 3 Feb 2024 15:13:23 +1100 Subject: [PATCH 3/4] More work on S134 & S3776 --- .../DynamicTypeAccess/PackageResolver.cs | 2 + .../Common/Reflection/ReflectionHelpers.cs | 16 +++-- .../ConfigServerConfigurationProvider.cs | 53 +++++++------- .../MongoDb/MongoDbConnectionStringBuilder.cs | 71 ++++++++++--------- .../PostgreSql/PostgreSqlConnectorTests.cs | 4 ++ .../Loggers/LoggersEndpointMiddleware.cs | 32 +++++---- .../Metrics/MetricsEndpointMiddleware.cs | 2 +- .../Metrics/Observer/EventCounterListener.cs | 10 +-- .../TracingBaseServiceCollectionExtensions.cs | 2 + .../Exporters/WavefrontMetricsExporter.cs | 34 ++++++--- src/Management/src/Wavefront/MetricReading.cs | 25 +++++++ .../ClientCertificateAuthenticationTests.cs | 4 ++ 12 files changed, 159 insertions(+), 96 deletions(-) create mode 100644 src/Management/src/Wavefront/MetricReading.cs diff --git a/src/Common/src/Abstractions/DynamicTypeAccess/PackageResolver.cs b/src/Common/src/Abstractions/DynamicTypeAccess/PackageResolver.cs index 571f16d5bd..769b412f95 100644 --- a/src/Common/src/Abstractions/DynamicTypeAccess/PackageResolver.cs +++ b/src/Common/src/Abstractions/DynamicTypeAccess/PackageResolver.cs @@ -88,6 +88,7 @@ protected TypeAccessor ResolveType(params string[] typeNames) foreach (string typeName in typeNames) { +#pragma warning disable S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply try { Type type = assembly.GetType(typeName, true)!; @@ -97,6 +98,7 @@ protected TypeAccessor ResolveType(params string[] typeNames) { exceptions.Add(exception); } +#pragma warning restore S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply } } catch (Exception exception) when (exception is ArgumentException or IOException or BadImageFormatException) diff --git a/src/Common/src/Common/Reflection/ReflectionHelpers.cs b/src/Common/src/Common/Reflection/ReflectionHelpers.cs index 8c09f619ba..34c6abeb44 100644 --- a/src/Common/src/Common/Reflection/ReflectionHelpers.cs +++ b/src/Common/src/Common/Reflection/ReflectionHelpers.cs @@ -463,12 +463,7 @@ private static void TryLoadAssembliesWithAttribute() try { Assembly assemblyRef = loadContext.LoadFromAssemblyPath(assembly); - - // haven't been able to get actual type comparison to work (assembly of the attribute not found?), falling back on matching the type name - if (CustomAttributeData.GetCustomAttributes(assemblyRef).Any(attr => attr.AttributeType.FullName == typeof(T).FullName)) - { - FindAssembly(filename); - } + FindAssemblyByFullName(assemblyRef, filename); } catch { @@ -478,6 +473,15 @@ private static void TryLoadAssembliesWithAttribute() } } + private static void FindAssemblyByFullName(Assembly assemblyRef, string filename) + { + // haven't been able to get actual type comparison to work (assembly of the attribute not found?), falling back on matching the type name + if (CustomAttributeData.GetCustomAttributes(assemblyRef).Any(attr => attr.AttributeType.FullName == typeof(T).FullName)) + { + FindAssembly(filename); + } + } + /// /// Build a list of file paths that are relevant to this task. /// diff --git a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs index c7d890576d..fe9ecdf0d7 100644 --- a/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs +++ b/src/Configuration/src/ConfigServer/ConfigServerConfigurationProvider.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. +using System; using System.Globalization; using System.Net; using System.Net.Http.Json; @@ -226,21 +227,19 @@ public override void Load() { return await DoLoadAsync(updateDictionary, cancellationToken); } - catch (ConfigServerException e) + catch (ConfigServerException e) when (attempts < Settings.RetryAttempts) { Logger.LogInformation(e, "Failed fetching configuration from server at: {uri}.", Settings.Uri); attempts++; - if (attempts < Settings.RetryAttempts) - { - Thread.CurrentThread.Join(backOff); - int nextBackOff = (int)(backOff * Settings.RetryMultiplier); - backOff = Math.Min(nextBackOff, Settings.RetryMaxInterval); - } - else - { - throw; - } + Thread.CurrentThread.Join(backOff); + int nextBackOff = (int)(backOff * Settings.RetryMultiplier); + backOff = Math.Min(nextBackOff, Settings.RetryMaxInterval); + } + catch (ConfigServerException e) when (attempts >= Settings.RetryAttempts) + { + Logger.LogInformation(e, "Failed fetching configuration for the final time from server at: {uri} .", Settings.Uri); + throw; } } while (true); @@ -275,10 +274,7 @@ public override void Load() Logger.LogInformation("Located environment name: {name}, profiles: {profiles}, labels: {label}, version: {version}, state: {state}", env.Name, env.Profiles, env.Label, env.Version, env.State); - if (updateDictionary) - { - UpdateDictionary(env); - } + UpdateDictionaryData(updateDictionary, env); return env; } @@ -299,9 +295,12 @@ public override void Load() return null; } - private void UpdateDictionary(ConfigEnvironment env) + private void UpdateDictionaryData(bool updateDictionary, ConfigEnvironment env) { - + if (updateDictionary) + { + return; + } var data = new Dictionary(StringComparer.OrdinalIgnoreCase); if (!string.IsNullOrEmpty(env.State)) @@ -383,16 +382,7 @@ internal void UpdateSettingsFromDiscovery(IEnumerable instance settings.Username = username; settings.Password = password; } - - if (metaData.TryGetValue("configPath", out string? path)) - { - if (uri.EndsWith('/') && path.StartsWith('/')) - { - uri = uri.Substring(0, uri.Length - 1); - } - - uri += path; - } + uri += ExtractPath(metaData, uri); } endpoints.Append(uri); @@ -406,6 +396,15 @@ internal void UpdateSettingsFromDiscovery(IEnumerable instance } } + private string ExtractPath(IDictionary metaData, string uri) + { + if (metaData.TryGetValue("configPath", out string? path)) + { + return uri.EndsWith('/') && path.StartsWith('/')? path.Substring(1, path.Length - 1) : path; + } + return string.Empty; + } + internal async Task ProvideRuntimeReplacementsAsync(IDiscoveryClient? discoveryClientFromDi, CancellationToken cancellationToken) { if (_configServerDiscoveryService is not null) diff --git a/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs b/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs index 0db3867b44..034ee88344 100644 --- a/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs +++ b/src/Connectors/src/Connectors/MongoDb/MongoDbConnectionStringBuilder.cs @@ -117,7 +117,9 @@ private string ToConnectionString() return builder.Uri.AbsoluteUri; } +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high private void FromConnectionString(string? connectionString, bool preserveUnknownSettings) +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high { if (preserveUnknownSettings) { @@ -131,53 +133,54 @@ private void FromConnectionString(string? connectionString, bool preserveUnknown _settings.Clear(); } - if (!string.IsNullOrEmpty(connectionString)) + if (string.IsNullOrEmpty(connectionString)) { - if (connectionString.Contains(',')) - { - // MongoDB allows multiple servers in the connection string, but we haven't found any service bindings that actually use that. - throw new NotImplementedException("Support for multiple servers is not implemented. Please open a GitHub issue if you need this."); - } + return; + } + if (connectionString.Contains(',')) + { + // MongoDB allows multiple servers in the connection string, but we haven't found any service bindings that actually use that. + throw new NotImplementedException("Support for multiple servers is not implemented. Please open a GitHub issue if you need this."); + } - // MongoDB allows semicolon as separator for query string parameters, to provide backwards compatibility. - connectionString = connectionString.Replace(';', '&'); + // MongoDB allows semicolon as separator for query string parameters, to provide backwards compatibility. + connectionString = connectionString.Replace(';', '&'); - var uri = new Uri(connectionString); + var uri = new Uri(connectionString); - if (!string.IsNullOrEmpty(uri.UserInfo)) - { - string[] parts = uri.UserInfo.Split(':', 2); + if (!string.IsNullOrEmpty(uri.UserInfo)) + { + string[] parts = uri.UserInfo.Split(':', 2); - _settings[KnownKeywords.Username] = Uri.UnescapeDataString(parts[0]); + _settings[KnownKeywords.Username] = Uri.UnescapeDataString(parts[0]); - if (parts.Length == 2) - { - _settings[KnownKeywords.Password] = Uri.UnescapeDataString(parts[1]); - } + if (parts.Length == 2) + { + _settings[KnownKeywords.Password] = Uri.UnescapeDataString(parts[1]); } + } - _settings[KnownKeywords.Server] = uri.Host; + _settings[KnownKeywords.Server] = uri.Host; - if (uri.Port != -1) - { - _settings[KnownKeywords.Port] = uri.Port.ToString(CultureInfo.InvariantCulture); - } + if (uri.Port != -1) + { + _settings[KnownKeywords.Port] = uri.Port.ToString(CultureInfo.InvariantCulture); + } - if (uri.AbsolutePath.StartsWith('/') && uri.AbsolutePath.Length > 1) - { - _settings[KnownKeywords.AuthenticationDatabase] = Uri.UnescapeDataString(uri.AbsolutePath[1..]); - } + if (uri.AbsolutePath.StartsWith('/') && uri.AbsolutePath.Length > 1) + { + _settings[KnownKeywords.AuthenticationDatabase] = Uri.UnescapeDataString(uri.AbsolutePath[1..]); + } - NameValueCollection queryCollection = HttpUtility.ParseQueryString(uri.Query); + NameValueCollection queryCollection = HttpUtility.ParseQueryString(uri.Query); - foreach (string? key in queryCollection.AllKeys.Where(x => x != null)) - { - string? value = queryCollection.Get(key); + foreach (string? key in queryCollection.AllKeys.Where(x => x != null)) + { + string? value = queryCollection.Get(key); - if (key != null && value != null) - { - _settings[key] = value; - } + if (key != null && value != null) + { + _settings[key] = value; } } } diff --git a/src/Connectors/test/Connectors.Test/PostgreSql/PostgreSqlConnectorTests.cs b/src/Connectors/test/Connectors.Test/PostgreSql/PostgreSqlConnectorTests.cs index 3516f465ad..82e7cf92c5 100644 --- a/src/Connectors/test/Connectors.Test/PostgreSql/PostgreSqlConnectorTests.cs +++ b/src/Connectors/test/Connectors.Test/PostgreSql/PostgreSqlConnectorTests.cs @@ -760,10 +760,12 @@ private static IEnumerable ExtractConnectionStringParameters(string? con string name = nameValuePair[0]; string value = nameValuePair[1]; +#pragma warning disable S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply if (TempFileKeys.Contains(name)) { value = File.ReadAllText(value); } +#pragma warning restore S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply value = value.Replace("\n", Environment.NewLine, StringComparison.Ordinal); @@ -787,10 +789,12 @@ private static void CleanupTempFiles(params string?[] connectionStrings) string key = pair[0]; string value = pair[1]; +#pragma warning disable S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply if (TempFileKeys.Contains(key) && File.Exists(value)) { File.Delete(value); } +#pragma warning restore S134 // Control flow statements "if", "switch", "for", "foreach", "while", "do" and "try" should not be nested too deeply } } } diff --git a/src/Management/src/Endpoint/Loggers/LoggersEndpointMiddleware.cs b/src/Management/src/Endpoint/Loggers/LoggersEndpointMiddleware.cs index 61f7d41c5e..81c60fd252 100644 --- a/src/Management/src/Endpoint/Loggers/LoggersEndpointMiddleware.cs +++ b/src/Management/src/Endpoint/Loggers/LoggersEndpointMiddleware.cs @@ -57,25 +57,31 @@ protected override async Task InvokeEndpointHandlerAsync(HttpCo string loggerName = remaining.Value!.TrimStart('/'); Dictionary change = await DeserializeRequestAsync(request.Body); + return CheckLogger(loggerName, change); + } + } - change.TryGetValue("configuredLevel", out string? level); + return new LoggersRequest(); + } - _logger.LogDebug("Change Request: {name}, {level}", loggerName, level ?? "RESET"); + private LoggersRequest? CheckLogger(string loggerName, Dictionary change) + { - if (!string.IsNullOrEmpty(loggerName)) - { - if (!string.IsNullOrEmpty(level) && LoggerLevels.StringToLogLevel(level) == null) - { - _logger.LogDebug("Invalid LogLevel specified: {level}", level); - return null; - } + change.TryGetValue("configuredLevel", out string? level); - return new LoggersRequest(loggerName, level); - } - } + _logger.LogDebug("Change Request: {name}, {level}", loggerName, level ?? "RESET"); + + if (string.IsNullOrEmpty(loggerName)) + { + return new LoggersRequest(); + } + if (!string.IsNullOrEmpty(level) && LoggerLevels.StringToLogLevel(level) == null) + { + _logger.LogDebug("Invalid LogLevel specified: {level}", level); + return null; } - return new LoggersRequest(); + return new LoggersRequest(loggerName, level); } private async Task> DeserializeRequestAsync(Stream stream) diff --git a/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs b/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs index 3e17dea2c3..1d09252d48 100644 --- a/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs +++ b/src/Management/src/Endpoint/Metrics/MetricsEndpointMiddleware.cs @@ -69,7 +69,7 @@ internal IList> ParseTags(IQueryCollection query) { var results = new List>(); - foreach (KeyValuePair parameter in query.Where(X => X.Key.Equals("tag", StringComparison.OrdinalIgnoreCase))) + foreach (KeyValuePair parameter in query.Where(x => x.Key.Equals("tag", StringComparison.OrdinalIgnoreCase))) { foreach (string? value in parameter.Value) { diff --git a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs index af9fed8248..d9c55ae173 100644 --- a/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs +++ b/src/Management/src/Endpoint/Metrics/Observer/EventCounterListener.cs @@ -64,13 +64,13 @@ protected override void OnEventWritten(EventWrittenEventArgs eventData) { ArgumentGuard.NotNull(eventData); - if (!_isInitialized || (string.Equals(eventData.EventName, EventName, StringComparison.OrdinalIgnoreCase) && eventData.Payload != null)) - { - return; - } - try { + + if (!_isInitialized || string.Equals(eventData.EventName, EventName, StringComparison.OrdinalIgnoreCase) || eventData.Payload == null) + { + return; + } foreach (IDictionary? payload in eventData.Payload) { if (payload != null) diff --git a/src/Management/src/Tracing/TracingBaseServiceCollectionExtensions.cs b/src/Management/src/Tracing/TracingBaseServiceCollectionExtensions.cs index c00c05e9c7..fd6b7173f4 100644 --- a/src/Management/src/Tracing/TracingBaseServiceCollectionExtensions.cs +++ b/src/Management/src/Tracing/TracingBaseServiceCollectionExtensions.cs @@ -50,7 +50,9 @@ public static IServiceCollection AddDistributedTracing(this IServiceCollection s /// /// configured for distributed tracing. /// +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high public static IServiceCollection AddDistributedTracing(this IServiceCollection services, Action? action) +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high { ArgumentGuard.NotNull(services); diff --git a/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs b/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs index 4f9bfb69f4..b02b412052 100644 --- a/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs +++ b/src/Management/src/Wavefront/Exporters/WavefrontMetricsExporter.cs @@ -62,15 +62,17 @@ public override ExportResult Export(in Batch batch) try { + List metricReadings = new List(); foreach (ref readonly MetricPoint metricPoint in metric.GetMetricPoints()) { long timestamp = metricPoint.EndTime.ToUnixTimeMilliseconds(); IDictionary tags = GetTags(metricPoint.Tags); - foreach (var item in GetMetrics(metric, metricPoint)) - { - _wavefrontSender.SendMetric(item.Key, item.Value, timestamp, Options.Source, tags); - metricCount++; - } + metricReadings.AddRange(GetMetricData(metric, metricPoint, timestamp, tags)); + } + foreach (MetricReading metricReading in metricReadings) + { + _wavefrontSender.SendMetric(metricReading.Key, metricReading.Value, metricReading.Timestamp, Options.Source, metricReading.Tags); + metricCount++; } } catch (Exception exception) @@ -83,11 +85,11 @@ public override ExportResult Export(in Batch batch) return ExportResult.Success; } - private List> GetMetrics(Metric metric, MetricPoint metricPoint) + private List GetMetricData(Metric metric, MetricPoint metricPoint, long timestamp, IDictionary tags) { bool isLong = ((int)metric.MetricType & 0b_0000_1111) == 0x0a; // I8 : signed 8 byte integer bool isSum = metric.MetricType.IsSum(); - List> metrics = []; + List metrics = []; if (!metric.MetricType.IsHistogram()) { double doubleValue; @@ -100,12 +102,24 @@ private List> GetMetrics(Metric metric, MetricPoint { doubleValue = isSum ? metricPoint.GetSumDouble() : metricPoint.GetGaugeLastValueDouble(); } - metrics.Add(new KeyValuePair(metric.Name.ToLowerInvariant(), doubleValue)); + metrics.Add(new MetricReading(metric.Name.ToLowerInvariant(), doubleValue) + { + Tags = tags, + Timestamp = timestamp + }); } else { - metrics.Add(new KeyValuePair($"{metric.Name.ToLowerInvariant()}_count", metricPoint.GetHistogramCount())); - metrics.Add(new KeyValuePair($"{metric.Name.ToLowerInvariant()}_sum", metricPoint.GetHistogramSum())); + metrics.Add(new MetricReading($"{metric.Name.ToLowerInvariant()}_count", metricPoint.GetHistogramCount()) + { + Tags = tags, + Timestamp = timestamp + }); + metrics.Add(new MetricReading($"{metric.Name.ToLowerInvariant()}_sum", metricPoint.GetHistogramSum()) + { + Tags = tags, + Timestamp = timestamp + }); } return metrics; } diff --git a/src/Management/src/Wavefront/MetricReading.cs b/src/Management/src/Wavefront/MetricReading.cs new file mode 100644 index 0000000000..d5e49ad0af --- /dev/null +++ b/src/Management/src/Wavefront/MetricReading.cs @@ -0,0 +1,25 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the Apache 2.0 License. +// See the LICENSE file in the project root for more information. + +namespace Steeltoe.Management.Wavefront; +internal class MetricReading +{ + public MetricReading(string key, double value, long? timestamp, IDictionary tags) + { + Key = key; + Value = value; + Timestamp = timestamp; + Tags = tags; + } + public MetricReading(string key, double value) + { + Key = key; + Value = value; + } + + public string Key { get; internal set; } + public double Value { get; internal set; } + public long? Timestamp { get; internal set; } + public IDictionary Tags { get; internal set; } = new Dictionary(); +} diff --git a/src/Security/test/Authentication.Mtls.Test/ClientCertificateAuthenticationTests.cs b/src/Security/test/Authentication.Mtls.Test/ClientCertificateAuthenticationTests.cs index 7abe07486c..8adcc2b6b7 100644 --- a/src/Security/test/Authentication.Mtls.Test/ClientCertificateAuthenticationTests.cs +++ b/src/Security/test/Authentication.Mtls.Test/ClientCertificateAuthenticationTests.cs @@ -359,7 +359,9 @@ public async Task VerifyACustomHeaderFailsIfTheHeaderIsNotPresent() } [Fact] +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high public async Task VerifyNoEventWireUpWithAValidCertificateCreatesADefaultUser() +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high { TestServer server = CreateServer(new MutualTlsAuthenticationOptions { @@ -512,7 +514,9 @@ public async Task VerifyValidationEventPrincipalIsPropagated() Assert.Single(responseAsXml.Elements("claim")); } +#pragma warning disable S3776 // Cognitive Complexity of methods should not be too high private static TestServer CreateServer(MutualTlsAuthenticationOptions configureOptions, X509Certificate2 clientCertificate = null, Uri baseAddress = null, +#pragma warning restore S3776 // Cognitive Complexity of methods should not be too high bool wireUpHeaderMiddleware = false, string headerName = "") { IWebHostBuilder builder = new WebHostBuilder().Configure(app => From 6a7db302f7099a8fe4c42105d043173b16de3ce6 Mon Sep 17 00:00:00 2001 From: James Thompson Date: Sun, 4 Feb 2024 20:49:01 +1100 Subject: [PATCH 4/4] Code style fix --- src/Management/src/Wavefront/MetricReading.cs | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/src/Management/src/Wavefront/MetricReading.cs b/src/Management/src/Wavefront/MetricReading.cs index d5e49ad0af..a739d7715d 100644 --- a/src/Management/src/Wavefront/MetricReading.cs +++ b/src/Management/src/Wavefront/MetricReading.cs @@ -3,8 +3,14 @@ // See the LICENSE file in the project root for more information. namespace Steeltoe.Management.Wavefront; + internal class MetricReading { + public string Key { get; internal set; } + public double Value { get; internal set; } + public long? Timestamp { get; internal set; } + public IDictionary Tags { get; internal set; } = new Dictionary(); + public MetricReading(string key, double value, long? timestamp, IDictionary tags) { Key = key; @@ -12,14 +18,10 @@ public MetricReading(string key, double value, long? timestamp, IDictionary Tags { get; internal set; } = new Dictionary(); }