From d8e6a31ebd0abe82788d985e66b1e4fd02fb6702 Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Fri, 29 Nov 2024 17:32:30 +0100 Subject: [PATCH 01/31] wip --- .../Arcus.WebApi.Hosting.AzureFunctions.csproj | 1 - ...unctionsWorkerApplicationBuilderExtensions.cs | 6 ++++-- .../Extensions/IServiceCollectionExtensions.cs | 11 ++++++++--- .../Arcus.WebApi.Hosting.csproj | 4 ---- .../Extensions/MvcOptionsExtensions.cs | 16 ++++++++++++---- .../Arcus.WebApi.Logging.csproj | 1 - .../ExceptionHandlingMiddleware.cs | 8 ++------ .../RequestTrackingAttribute.cs | 13 ++++++++----- 8 files changed, 34 insertions(+), 26 deletions(-) diff --git a/src/Arcus.WebApi.Hosting.AzureFunctions/Arcus.WebApi.Hosting.AzureFunctions.csproj b/src/Arcus.WebApi.Hosting.AzureFunctions/Arcus.WebApi.Hosting.AzureFunctions.csproj index b6e4bc4e..fac533cd 100644 --- a/src/Arcus.WebApi.Hosting.AzureFunctions/Arcus.WebApi.Hosting.AzureFunctions.csproj +++ b/src/Arcus.WebApi.Hosting.AzureFunctions/Arcus.WebApi.Hosting.AzureFunctions.csproj @@ -26,7 +26,6 @@ - diff --git a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs index d45a7387..02915ee9 100644 --- a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs +++ b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs @@ -1,6 +1,5 @@ using System; using Arcus.WebApi.Hosting.AzureFunctions.Formatting; -using GuardNet; using Microsoft.Azure.Functions.Worker; // ReSharper disable once CheckNamespace @@ -20,7 +19,10 @@ public static class IFunctionsWorkerApplicationBuilderExtensions public static IFunctionsWorkerApplicationBuilder UseOnlyJsonFormatting( this IFunctionsWorkerApplicationBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the JSON formatting middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the JSON formatting middleware"); + } builder.UseMiddleware(); return builder; diff --git a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs index e982a74d..66dd8edb 100644 --- a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs @@ -1,7 +1,6 @@ using System; using System.Text.Json; using Azure.Core.Serialization; -using GuardNet; using Microsoft.Azure.Functions.Worker; using Microsoft.Extensions.DependencyInjection; @@ -24,8 +23,14 @@ public static IServiceCollection ConfigureJsonFormatting( this IFunctionsWorkerApplicationBuilder builder, Action configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires an Azure Functions application builder instance to add the JSON serializer to the application services"); - Guard.NotNull(configureOptions, nameof(configureOptions), "Requires a function to configure the JSON serialization options to add the JSON serializer to the application services"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires an Azure Functions application builder instance to add the JSON serializer to the application services"); + } + if (configureOptions is null) + { + throw new ArgumentNullException(nameof(configureOptions), "Requires a function to configure the JSON serialization options to add the JSON serializer to the application services"); + } var options = new JsonSerializerOptions(); configureOptions(options); diff --git a/src/Arcus.WebApi.Hosting/Arcus.WebApi.Hosting.csproj b/src/Arcus.WebApi.Hosting/Arcus.WebApi.Hosting.csproj index 7c0576a7..bd0eebf2 100644 --- a/src/Arcus.WebApi.Hosting/Arcus.WebApi.Hosting.csproj +++ b/src/Arcus.WebApi.Hosting/Arcus.WebApi.Hosting.csproj @@ -28,8 +28,4 @@ - - - - diff --git a/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs index bf41937e..464672c2 100644 --- a/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs @@ -1,7 +1,6 @@ using System; using System.Linq; using System.Text.Json; -using GuardNet; using Microsoft.AspNetCore.Mvc.Formatters; using Microsoft.Extensions.DependencyInjection; @@ -20,7 +19,10 @@ public static class MvcOptionsExtensions /// Thrown when the is null. public static MvcOptions OnlyAllowJsonFormatting(this MvcOptions options) { - Guard.NotNull(options, nameof(options), "Requires MVC options to restrict the formatting to only JSON formatting"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires MVC options to restrict the formatting to only JSON formatting"); + } IInputFormatter[] allButJsonInputFormatters = options.InputFormatters.Where(formatter => !(formatter is SystemTextJsonInputFormatter)) @@ -46,8 +48,14 @@ public static MvcOptions OnlyAllowJsonFormatting(this MvcOptions options) [Obsolete("Use the " + nameof(MvcCoreMvcBuilderExtensions) + "." + nameof(MvcCoreMvcBuilderExtensions.AddJsonOptions) + " instead to configure the JSON formatters")] public static MvcOptions ConfigureJsonFormatting(this MvcOptions options, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires MVC options to configure the JSON formatters"); - Guard.NotNull(configureOptions, nameof(configureOptions), "Requires a function to configure the JSON formatters in the MVC options"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires MVC options to configure the JSON formatters"); + } + if (configureOptions is null) + { + throw new ArgumentNullException(nameof(configureOptions), "Requires a function to configure the JSON formatters in the MVC options"); + } SystemTextJsonInputFormatter[] onlyJsonInputFormatters = options.InputFormatters.OfType() diff --git a/src/Arcus.WebApi.Logging/Arcus.WebApi.Logging.csproj b/src/Arcus.WebApi.Logging/Arcus.WebApi.Logging.csproj index 1fc6c463..06cce636 100644 --- a/src/Arcus.WebApi.Logging/Arcus.WebApi.Logging.csproj +++ b/src/Arcus.WebApi.Logging/Arcus.WebApi.Logging.csproj @@ -37,7 +37,6 @@ - diff --git a/src/Arcus.WebApi.Logging/ExceptionHandlingMiddleware.cs b/src/Arcus.WebApi.Logging/ExceptionHandlingMiddleware.cs index bba462ba..5ef28f96 100644 --- a/src/Arcus.WebApi.Logging/ExceptionHandlingMiddleware.cs +++ b/src/Arcus.WebApi.Logging/ExceptionHandlingMiddleware.cs @@ -3,7 +3,6 @@ using System.Threading.Tasks; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; -using GuardNet; using Microsoft.Extensions.Logging.Abstractions; // ReSharper disable once CheckNamespace @@ -45,11 +44,8 @@ public ExceptionHandlingMiddleware(RequestDelegate next, string categoryName) /// When the is null. public ExceptionHandlingMiddleware(RequestDelegate next, Func getLoggingCategory) { - Guard.NotNull(next, nameof(next), "The next request delegate in the application request pipeline cannot be null"); - Guard.NotNull(getLoggingCategory, nameof(getLoggingCategory), "The retrieval of the logging category function cannot be null"); - - _next = next; - _getLoggingCategory = getLoggingCategory; + _next = next ?? throw new ArgumentNullException(nameof(next), "The next request delegate in the application request pipeline cannot be null"); + _getLoggingCategory = getLoggingCategory ?? throw new ArgumentNullException(nameof(getLoggingCategory), "The retrieval of the logging category function cannot be null"); } /// diff --git a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs index dd16b4a9..1019613c 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs @@ -21,9 +21,10 @@ public class RequestTrackingAttribute : Attribute /// Thrown when the is outside the range of the enumeration. public RequestTrackingAttribute(Exclude filter) { - Guard.For( - () => !Enum.IsDefined(typeof(Exclude), filter) || filter is Exclude.None, - $"Requires the exclusion filter to be within these bounds of the enumeration '{ExcludeFilterNames}'; 'None' is not allowed"); + if (!Enum.IsDefined(typeof(Exclude), filter) || filter is Exclude.None) + { + throw new ArgumentOutOfRangeException(nameof(filter), $"Requires the exclusion filter to be within these bounds of the enumeration '{ExcludeFilterNames}'; 'None' is not allowed",); + } Filter = filter; } @@ -44,8 +45,10 @@ public RequestTrackingAttribute(HttpStatusCode trackedStatusCode) : this((int) t /// Thrown when the is outside the expected range (100-599). public RequestTrackingAttribute(int trackedStatusCode) { - Guard.NotLessThan(trackedStatusCode, 100, nameof(trackedStatusCode), "Requires the allowed tracked HTTP status code to not be less than 100"); - Guard.NotGreaterThan(trackedStatusCode, 599, nameof(trackedStatusCode), "Requires the allowed tracked HTTP status code to not be greater than 599"); + if (trackedStatusCode < 100 || trackedStatusCode > 599) + { + throw new ArgumentOutOfRangeException(nameof(trackedStatusCode), "Requires the allowed tracked HTTP status code to be within the range of 100-599"); + } StatusCodeRange = new StatusCodeRange(trackedStatusCode); } From 011f946dd6f44a0d299ef1d4a814d67b15c8f6dd Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 3 Dec 2024 02:46:28 +0100 Subject: [PATCH 02/31] wip logging core --- .../Arcus.WebApi.Logging.Core.csproj | 1 - .../CorrelationInfoUpstreamServiceOptions.cs | 13 ++++--- .../Correlation/HttpCorrelation.cs | 37 ++++++++++++------- .../HttpCorrelationClientOptions.cs | 21 ++++++++--- .../HttpCorrelationInfoAccessor.cs | 12 ++++-- .../HttpCorrelationInfoOperationOptions.cs | 13 +++++-- .../HttpCorrelationInfoTransactionOptions.cs | 13 +++++-- .../HttpCorrelationMessageHandler.cs | 2 - 8 files changed, 76 insertions(+), 36 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj b/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj index c3cf166c..fe8921eb 100644 --- a/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj +++ b/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj @@ -38,7 +38,6 @@ - diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs index 90f4ba3b..550b3e6f 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs @@ -1,7 +1,4 @@ using System; -using Arcus.Observability.Correlation; -using GuardNet; -using Microsoft.Net.Http.Headers; namespace Arcus.WebApi.Logging.Core.Correlation { @@ -41,7 +38,10 @@ public string HeaderName get => _headerName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Requires a non-blank value for the operation parent ID request header name"); + if (string.IsNullOrWhiteSpace(value)) + { + throw new ArgumentException(message: "Requires a non-blank value for the operation parent ID request header name", paramName: nameof(value)); + } _headerName = value; } } @@ -58,7 +58,10 @@ public Func GenerateId get => _generateId; set { - Guard.NotNull(value, nameof (value), "Requires a function to generate the operation parent ID"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), "Requires a function to generate the operation parent ID"); + } _generateId = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs index 28276ff1..a66d37b2 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs @@ -2,7 +2,6 @@ using System.Threading.Tasks; using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; @@ -39,14 +38,16 @@ public HttpCorrelation( ILogger logger) : base(options?.Value, correlationInfoAccessor, logger) { - Guard.NotNull(httpContextAccessor, nameof(httpContextAccessor), "Requires a HTTP context accessor to get the current HTTP context"); - Guard.NotNull(correlationInfoAccessor, nameof(correlationInfoAccessor), "Requires a correlation info instance to set and retrieve the correlation information"); - Guard.NotNull(options, nameof(options), "Requires a value in the set of options to configure the correlation process"); - Guard.NotNull(options.Value, nameof(options), "Requires a value in the set of options to configure the correlation process"); - - _httpContextAccessor = httpContextAccessor; - _options = options.Value; - _correlationInfoAccessor = correlationInfoAccessor; + if (options is null) + { + throw new ArgumentNullException(paramName: nameof(options), message: "Requires a value in the set of options to configure the correlation process"); + } + + _httpContextAccessor = httpContextAccessor ?? throw new ArgumentNullException(paramName: nameof(httpContextAccessor), message: "Requires a HTTP context accessor to get the current HTTP context"); + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a correlation info instance to set and retrieve the correlation information"); + _options = options.Value ?? throw new ArgumentNullException(paramName: nameof(options.Value), message: "Requires a value in the set of options to configure the correlation process"); + + _logger = logger ?? NullLogger.Instance; } @@ -65,7 +66,10 @@ public CorrelationInfo GetCorrelationInfo() /// Thrown when the is null. public void SetCorrelationInfo(CorrelationInfo correlationInfo) { - Guard.NotNull(correlationInfo, nameof(correlationInfo)); + if (_correlationInfoAccessor is null) + { + throw new ArgumentNullException(paramName: nameof(correlationInfo), message: "Requires a correlation info instance to set and retrieve the correlation information"); + } _correlationInfoAccessor.SetCorrelationInfo(correlationInfo); } @@ -77,11 +81,16 @@ public void SetCorrelationInfo(CorrelationInfo correlationInfo) /// Thrown when the given doesn't have any response headers to set the correlation headers. public HttpCorrelationResult CorrelateHttpRequest() { - HttpContext httpContext = _httpContextAccessor.HttpContext; + HttpContext httpContext = _httpContextAccessor.HttpContext ?? throw new ArgumentNullException(paramName: nameof(HttpContext), "Requires a HTTP context from the HTTP context accessor to start correlating the HTTP request"); - Guard.NotNull(httpContext, nameof(httpContext), "Requires a HTTP context from the HTTP context accessor to start correlating the HTTP request"); - Guard.For(() => httpContext.Response is null, "Requires a 'Response'"); - Guard.For(() => httpContext.Response.Headers is null, "Requires a 'Response' object with headers"); + if (httpContext.Response is null) + { + throw new ArgumentException(message: "Requires a 'Response'", paramName: nameof(httpContext.Response)); + } + if (httpContext.Response.Headers is null) + { + throw new ArgumentException(message: "Requires a 'Response' object with headers", paramName: nameof(httpContext.Response.Headers)); + } HttpCorrelationResult result = TrySettingCorrelationFromRequest(httpContext.Request, httpContext.TraceIdentifier); if (result.IsSuccess) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index e2bf5ab3..739a4516 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -1,7 +1,6 @@ using System; using System.Collections.Generic; using System.Net.Http; -using GuardNet; using Microsoft.Extensions.Logging; namespace Arcus.WebApi.Logging.Core.Correlation @@ -25,7 +24,10 @@ public Func GenerateDependencyId get => _generateDependencyId; set { - Guard.NotNull(value, nameof(value), "Requires a function to generate the dependency ID used when tracking HTTP dependencies"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), message: "Requires a function to generate the dependency ID used when tracking HTTP dependencies"); + } _generateDependencyId = value; } } @@ -39,7 +41,10 @@ public string UpstreamServiceHeaderName get => _upstreamServiceHeaderName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), message: "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); + } _upstreamServiceHeaderName = value; } } @@ -53,7 +58,10 @@ public string TransactionIdHeaderName get => _transactionIdHeaderName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), message: "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); + } _transactionIdHeaderName = value; } } @@ -70,7 +78,10 @@ public string TransactionIdHeaderName /// Thrown when the is null. public void AddTelemetryContext(Dictionary telemetryContext) { - Guard.NotNull(telemetryContext, nameof(telemetryContext), "Requires a telemetry context dictionary to add to the HTTP dependency tracking"); + if (telemetryContext == null) + { + throw new ArgumentNullException(paramName: nameof(telemetryContext), message: "Requires a telemetry context dictionary to add to the HTTP dependency tracking"); + } foreach (KeyValuePair item in telemetryContext) { TelemetryContext[item.Key] = item.Value; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs index c0567d67..d4fbdd9b 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs @@ -1,7 +1,6 @@ using System; using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; // ReSharper disable once CheckNamespace @@ -21,7 +20,10 @@ public class HttpCorrelationInfoAccessor : IHttpCorrelationInfoAccessor, ICorrel /// Thrown when the is null. public HttpCorrelationInfoAccessor(IHttpContextAccessor contextAccessor) { - Guard.NotNull(contextAccessor, nameof(contextAccessor)); + if (contextAccessor is null) + { + throw new ArgumentNullException(paramName: nameof(contextAccessor)); + } _httpContextAccessor = contextAccessor; } @@ -42,7 +44,11 @@ public CorrelationInfo GetCorrelationInfo() /// Thrown when the is null. public void SetCorrelationInfo(CorrelationInfo correlationInfo) { - Guard.NotNull(correlationInfo, nameof(correlationInfo)); + if (correlationInfo is null) + { + throw new ArgumentNullException(paramName: nameof(correlationInfo)); + } + _httpContextAccessor.HttpContext?.Features?.Set(correlationInfo); } } diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs index 4edcf393..5b4af2f9 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; namespace Arcus.WebApi.Logging.Core.Correlation { @@ -27,7 +26,11 @@ public string HeaderName get => _headerName; set { - Guard.NotNullOrWhitespace(value, nameof (value), "Correlation operation header cannot be blank"); + if (string.IsNullOrWhiteSpace(value)) + { + throw new ArgumentException(message: "Correlation operation header cannot be blank", paramName: nameof(value)); + } + _headerName = value; } } @@ -43,7 +46,11 @@ public Func GenerateId get => _generateId; set { - Guard.NotNull(value, nameof (value), "Correlation function to generate an operation ID cannot be 'null'"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), message: "Correlation function to generate an operation ID cannot be 'null'"); + } + _generateId = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs index f81f669f..2113e972 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; namespace Arcus.WebApi.Logging.Core.Correlation { @@ -43,7 +42,11 @@ public string HeaderName get => _headerName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Correlation transaction header cannot be blank"); + if (string.IsNullOrWhiteSpace(value)) + { + throw new ArgumentException(message: "Header name cannot be blank", paramName: nameof(value)); + } + _headerName = value; } } @@ -62,7 +65,11 @@ public Func GenerateId get => this._generateId; set { - Guard.NotNull(value, nameof(value), "Correlation function to generate an transaction ID cannot be 'null'"); + if (value is null) + { + throw new ArgumentNullException(paramName: nameof(value), message: "Correlation function to generate an transaction ID cannot be 'null'"); + } + _generateId = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs index 1f1912b0..d3959354 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs @@ -5,8 +5,6 @@ using System.Threading.Tasks; using Arcus.Observability.Correlation; using Arcus.Observability.Telemetry.Core; -using GuardNet; -using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; namespace Arcus.WebApi.Logging.Core.Correlation From 0d6462a1b699f2ee84d6e0ec355c90a3e8639055 Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Wed, 4 Dec 2024 03:53:31 +0100 Subject: [PATCH 03/31] Finish Arcus.WebApi.Logging.Core --- .../HttpCorrelationMessageHandler.cs | 10 +-- .../Correlation/HttpCorrelationResult.cs | 37 +++++++-- .../Correlation/HttpCorrelationTemplate.cs | 23 +++--- .../Extensions/HttpClientExtensions.cs | 81 +++++++++++++++---- .../HttpCorrelationEnricherExtensions.cs | 11 ++- .../Extensions/IDictionaryExtensions.cs | 11 ++- .../Extensions/IHeaderDictionaryExtensions.cs | 11 ++- .../IHttpClientBuilderExtensions.cs | 11 ++- .../RequestTracking/RequestTrackingOptions.cs | 11 ++- .../RequestTrackingTemplate.cs | 11 ++- .../RequestTracking/StatusCodeRange.cs | 26 ++++-- 11 files changed, 178 insertions(+), 65 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs index d3959354..3bb72cc7 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs @@ -32,13 +32,9 @@ public HttpCorrelationMessageHandler( HttpCorrelationClientOptions options, ILogger logger) { - Guard.NotNull(correlationInfoAccessor, nameof(correlationInfoAccessor), "Requires a HTTP context accessor to retrieve the current HTTP correlation"); - Guard.NotNull(options, nameof(options), "Requires a set of additional user-configurable options to influence the HTTP dependency tracking"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to write the HTTP dependency telemetry"); - - _correlationInfoAccessor = correlationInfoAccessor; - _options = options; - _logger = logger; + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a HTTP context accessor to retrieve the current HTTP correlation"); + _options = options ?? throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of additional user-configurable options to influence the HTTP dependency tracking"); + _logger = logger ?? throw new ArgumentNullException(paramName: nameof(logger), message: "Requires a logger instance to write the HTTP dependency telemetry"); } /// diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs index be4659cf..74f0992b 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs @@ -1,7 +1,6 @@ using System; using System.Diagnostics; using Arcus.Observability.Correlation; -using GuardNet; using Microsoft.ApplicationInsights; using Microsoft.ApplicationInsights.DataContracts; using Microsoft.ApplicationInsights.Extensibility; @@ -19,8 +18,14 @@ public class HttpCorrelationResult : IDisposable private HttpCorrelationResult(bool isSuccess, string requestId, string errorMessage) { - Guard.For(() => isSuccess && errorMessage != null, new ArgumentException("Cannot create a successful HTTP correlation result with an error user message", nameof(errorMessage))); - Guard.For(() => !isSuccess && requestId != null, new ArgumentException("Cannot create a failed HTTP correlation result with a request ID", nameof(requestId))); + if (isSuccess && errorMessage != null) + { + throw new ArgumentException("Cannot create a successful HTTP correlation result with an error user message", nameof(errorMessage)); + } + if (!isSuccess && requestId != null) + { + throw new ArgumentException("Cannot create a failed HTTP correlation result with a request ID", nameof(requestId)); + } RequestId = requestId; ErrorMessage = errorMessage; @@ -82,7 +87,11 @@ public static HttpCorrelationResult Success(string requestId) /// Thrown when the is blank. public static HttpCorrelationResult Failure(string errorMessage) { - Guard.NotNullOrWhitespace(errorMessage, nameof(errorMessage), "Requires an error user message that describes why the HTTP correlation process failed on the current HTTP request"); + if (string.IsNullOrWhiteSpace(errorMessage)) + { + throw new ArgumentException(paramName: nameof(errorMessage), message: "Requires an error user message that describes why the HTTP correlation process failed on the current HTTP request"); + } + return new HttpCorrelationResult(isSuccess: false, requestId: null, errorMessage); } @@ -95,8 +104,14 @@ public static HttpCorrelationResult Failure(string errorMessage) /// Thrown when the is blank. public static HttpCorrelationResult Success(TelemetryClient client, string transactionId) { - Guard.NotNull(client, nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); - Guard.NotNullOrWhitespace(transactionId, nameof(transactionId), "Requires a non-blank transaction ID for the pending HTTP correlation"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); + } + if (string.IsNullOrWhiteSpace(transactionId)) + { + throw new ArgumentException(message: "Requires a non-blank transaction ID for the pending HTTP correlation", paramName: nameof(transactionId)); + } return Success(client, transactionId, operationParentId: null, traceParent: null); } @@ -114,8 +129,14 @@ public static HttpCorrelationResult Success(TelemetryClient client, string trans /// public static HttpCorrelationResult Success(TelemetryClient client, string transactionId, string operationParentId, string traceParent) { - Guard.NotNull(client, nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); - Guard.NotNullOrWhitespace(transactionId, nameof(transactionId), "Requires a non-blank transaction ID for the pending HTTP correlation"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); + } + if (string.IsNullOrWhiteSpace(transactionId)) + { + throw new ArgumentException(message: "Requires a non-blank transaction ID for the pending HTTP correlation", paramName: nameof(transactionId)); + } var telemetry = new RequestTelemetry(); telemetry.Context.Operation.Id = transactionId; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs index bfd91112..19c8faf9 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs @@ -5,7 +5,6 @@ using Microsoft.Extensions.Primitives; using System.Text.RegularExpressions; using Arcus.Observability.Correlation; -using GuardNet; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; using System.Collections.Generic; @@ -38,11 +37,8 @@ protected HttpCorrelationTemplate( IHttpCorrelationInfoAccessor correlationInfoAccessor, ILogger logger) { - Guard.NotNull(options, nameof(options), "Requires a set of options to configure the correlation process"); - Guard.NotNull(correlationInfoAccessor, nameof(correlationInfoAccessor), "Requires a correlation info instance to set and retrieve the correlation information"); - - _options = options; - _correlationInfoAccessor = correlationInfoAccessor; + _options = options ?? throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of options to configure the correlation process"); ; + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a correlation info instance to set and retrieve the correlation information"); Logger = logger ?? NullLogger.Instance; } @@ -68,7 +64,10 @@ protected HttpCorrelationTemplate( /// Thrown when the is null. public HttpCorrelationResult TrySettingCorrelationFromRequest(THttpRequest request, string traceIdentifier) { - Guard.NotNull(request, nameof(request), "Requires a HTTP request to determine the HTTP correlation of the application"); + if (request is null) + { + throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to determine the HTTP correlation of the application"); + } IHeaderDictionary requestHeaders = GetRequestHeaders(request); if (requestHeaders is null) @@ -405,8 +404,14 @@ private bool MatchesRequestIdFormat(string requestId, string headerName) /// Thrown when the or is null. public void SetCorrelationHeadersInResponse(THttpResponse response, HttpCorrelationResult result) { - Guard.NotNull(response, nameof(response), "Requires a HTTP response to set the HTTP correlation headers"); - Guard.NotNull(result, nameof(result), "Requires a HTTP correlation result to determine to set the HTTP correlation headers in the HTTP request"); + if (response is null) + { + throw new ArgumentNullException(paramName: nameof(response), message: "Requires a HTTP response to set the HTTP correlation headers"); + } + if (result is null) + { + throw new ArgumentNullException(paramName: nameof(result), message: "Requires a HTTP correlation result to determine to set the HTTP correlation headers in the HTTP request"); + } string requestId = result.RequestId; CorrelationInfo correlationInfo = _correlationInfoAccessor.GetCorrelationInfo(); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs index 5ec264e0..27bf3289 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs @@ -2,7 +2,6 @@ using Arcus.Observability.Correlation; using Arcus.Observability.Telemetry.Core; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; @@ -37,10 +36,22 @@ public static async Task SendAsync( IHttpCorrelationInfoAccessor correlationAccessor, ILogger logger) { - Guard.NotNull(client, nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - Guard.NotNull(request, nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - Guard.NotNull(correlationAccessor, nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + } + if (request is null) + { + throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + } + if (correlationAccessor is null) + { + throw new ArgumentNullException(paramName: nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); + } + if (logger is null) + { + throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + } return await SendAsync(client, request, correlationAccessor, logger, configureOptions: null); } @@ -70,10 +81,22 @@ public static async Task SendAsync( ILogger logger, Action configureOptions) { - Guard.NotNull(client, nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - Guard.NotNull(request, nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - Guard.NotNull(correlationAccessor, nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + } + if (request is null) + { + throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + } + if (correlationAccessor is null) + { + throw new ArgumentNullException(paramName: nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); + } + if (logger is null) + { + throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + } CorrelationInfo correlation = correlationAccessor.GetCorrelationInfo(); return await SendAsync(client, request, correlation, logger, configureOptions); @@ -102,10 +125,22 @@ public static async Task SendAsync( CorrelationInfo correlationInfo, ILogger logger) { - Guard.NotNull(client, nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - Guard.NotNull(request, nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - Guard.NotNull(correlationInfo, nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + } + if (request is null) + { + throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + } + if (correlationInfo is null) + { + throw new ArgumentNullException(paramName: nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); + } + if (logger is null) + { + throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + } return await SendAsync(client, request, correlationInfo, logger, configureOptions: null); } @@ -135,10 +170,22 @@ public static async Task SendAsync( ILogger logger, Action configureOptions) { - Guard.NotNull(client, nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - Guard.NotNull(request, nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - Guard.NotNull(correlationInfo, nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + if (client is null) + { + throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + } + if (request is null) + { + throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + } + if (correlationInfo is null) + { + throw new ArgumentNullException(paramName: nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); + } + if (logger is null) + { + throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + } var options = new HttpCorrelationClientOptions(); configureOptions?.Invoke(options); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs index 72c4f6e3..3cf1fd02 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs @@ -3,7 +3,6 @@ using Arcus.Observability.Telemetry.Serilog.Enrichers; using Arcus.WebApi.Logging.Core.Correlation; using Arcus.WebApi.Logging.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.DependencyInjection; @@ -27,8 +26,14 @@ public static class HttpCorrelationEnricherExtensions /// Thrown when the or is null. public static LoggerConfiguration WithHttpCorrelationInfo(this LoggerEnrichmentConfiguration enrichmentConfiguration, IServiceProvider serviceProvider) { - Guard.NotNull(enrichmentConfiguration, nameof(enrichmentConfiguration), "Requires a Serilog logger enrichment configuration to register the HTTP correlation as enrichment"); - Guard.NotNull(serviceProvider, nameof(serviceProvider), "Requires a service provider to retrieve the HTTP correlation from the registered services when enriching the Serilog with the HTTP correlation"); + if (enrichmentConfiguration is null) + { + throw new ArgumentNullException(paramName: nameof(enrichmentConfiguration), message: "Requires a Serilog logger enrichment configuration to register the HTTP correlation as enrichment"); + } + if (serviceProvider is null) + { + throw new ArgumentNullException(paramName: nameof(serviceProvider), message: "Requires a service provider to retrieve the HTTP correlation from the registered services when enriching the Serilog with the HTTP correlation"); + } var correlationInfoAccessor = serviceProvider.GetService(); if (correlationInfoAccessor is null) diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs index bf5c67ae..c997c674 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs @@ -1,5 +1,4 @@ using System.Linq; -using GuardNet; // ReSharper disable once CheckNamespace namespace System.Collections.Generic @@ -22,8 +21,14 @@ public static class IDictionaryExtensions /// public static IDictionary Where(this IDictionary dictionary, Func, bool> predicate) { - Guard.NotNull(dictionary, nameof(dictionary)); - Guard.NotNull(predicate, nameof(predicate)); + if (dictionary is null) + { + throw new ArgumentNullException(paramName: nameof(dictionary)); + } + if (predicate is null) + { + throw new ArgumentNullException(paramName: nameof(predicate)); + } return Enumerable.Where(dictionary, predicate) .ToDictionary(item => item.Key, item => item.Value); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/IHeaderDictionaryExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/IHeaderDictionaryExtensions.cs index 5879c488..706d7ecd 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/IHeaderDictionaryExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/IHeaderDictionaryExtensions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; using Microsoft.Extensions.Primitives; // ReSharper disable once CheckNamespace @@ -18,7 +17,10 @@ public static class IHeaderDictionaryExtensions /// Thrown when the is null. public static StringValues GetTraceParent(this IHeaderDictionary headers) { - Guard.NotNull(headers, nameof(headers), "Requires a HTTP request headers dictionary instance to retrieve the 'traceparent' header value"); + if (headers is null) + { + throw new ArgumentNullException(paramName: nameof(headers), message: "Requires a HTTP request headers dictionary instance to retrieve the 'traceparent' header value"); + } #if NET6_0 StringValues traceParent = headers.TraceParent; #else @@ -40,7 +42,10 @@ public static StringValues GetTraceParent(this IHeaderDictionary headers) /// Thrown when the is null. internal static StringValues GetTraceState(this IHeaderDictionary headers) { - Guard.NotNull(headers, nameof(headers), "Requires a HTTP request headers dictionary instance to retrieve the 'tracestate' header value"); + if (headers is null) + { + throw new ArgumentNullException(paramName: nameof(headers), message: "Requires a HTTP request headers dictionary instance to retrieve the 'tracestate' header value"); + } #if NET6_0 return headers.TraceState; #else diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/IHttpClientBuilderExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/IHttpClientBuilderExtensions.cs index 5e2f27af..e059c164 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/IHttpClientBuilderExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/IHttpClientBuilderExtensions.cs @@ -1,6 +1,5 @@ using System; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; @@ -24,7 +23,10 @@ public static class IHttpClientBuilderExtensions /// Thrown when the no was found in the dependency injection container. public static IHttpClientBuilder WithHttpCorrelationTracking(this IHttpClientBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a HTTP client builder instance to add the HTTP correlation message handler"); + if (builder is null) + { + throw new ArgumentNullException(paramName: nameof(builder), message: "Requires a HTTP client builder instance to add the HTTP correlation message handler"); + } return WithHttpCorrelationTracking(builder, configureOptions: null); } @@ -40,7 +42,10 @@ public static IHttpClientBuilder WithHttpCorrelationTracking(this IHttpClientBui /// Thrown when the no was found in the dependency injection container. public static IHttpClientBuilder WithHttpCorrelationTracking(this IHttpClientBuilder builder, Action configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires a HTTP client builder instance to add the HTTP correlation message handler"); + if (builder is null) + { + throw new ArgumentNullException(paramName: nameof(builder), message: "Requires a HTTP client builder instance to add the HTTP correlation message handler"); + } return builder.AddHttpMessageHandler(serviceProvider => { diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs index 50bf967c..65f8f0a4 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Collections.ObjectModel; using System.Net; -using GuardNet; namespace Arcus.WebApi.Logging { @@ -33,7 +32,10 @@ public int? RequestBodyBufferSize get => _requestBodyBufferSize; set { - Guard.For(() => value < 0, "Requires a request body buffer size greater than zero"); + if (value < 0) + { + throw new ArgumentOutOfRangeException(paramName: nameof(value), message: "Requires a request body buffer size greater than zero"); + } _requestBodyBufferSize = value; } } @@ -52,7 +54,10 @@ public int? ResponseBodyBufferSize get => _responseBodyBufferSize; set { - Guard.For(() => value < 0, "Requires a response body buffer size greater than zero"); + if (value < 0) + { + throw new ArgumentOutOfRangeException(paramName: nameof(value), message: "Requires a response body buffer size greater than zero"); + } _responseBodyBufferSize = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs index a4783dfe..28490da3 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs @@ -5,7 +5,6 @@ using System.Linq; using System.Net; using System.Threading.Tasks; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; @@ -25,7 +24,10 @@ public class RequestTrackingTemplate /// Thrown when the is null. protected RequestTrackingTemplate(RequestTrackingOptions options) { - Guard.NotNull(options, nameof(options), "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); + if (options is null) + { + throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); + } Options = options; } @@ -182,7 +184,10 @@ protected virtual IDictionary SanitizeRequestHeaders(IDict /// Thrown when the is null. protected async Task GetBodyAsync(Stream body, int? maxLength, string targetName, ILogger logger) { - Guard.NotNull(body, nameof(body), $"Requires a streamed body to read the string representation of the {targetName}"); + if (body is null) + { + throw new ArgumentNullException(paramName: nameof(body), message: $"Requires a streamed body to read the string representation of the {targetName}"); + } logger = logger ?? NullLogger.Instance; logger.LogTrace("Prepare for {Target} body to be tracked...", targetName); diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs index 0c28fcfb..c0f7fc4a 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; namespace Arcus.WebApi.Logging { @@ -30,9 +29,18 @@ public StatusCodeRange(int minimum) : this(minimum, minimum) /// public StatusCodeRange(int minimum, int maximum) { - Guard.NotLessThan(minimum, 100, nameof(minimum), "Requires the minimum HTTP status code threshold not be less than 100"); - Guard.NotGreaterThan(maximum, 599, nameof(maximum), "Requires the maximum HTTP status code threshold not be greater than 599"); - Guard.NotGreaterThan(minimum, maximum, nameof(minimum), "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); + if (minimum < 100) + { + throw new ArgumentOutOfRangeException(paramName: nameof(minimum), message: "Requires the minimum HTTP status code threshold to not be less than 100"); + } + if (maximum > 599) + { + throw new ArgumentOutOfRangeException(paramName: nameof(maximum), message: "Requires the maximum HTTP status code threshold to not be greater than 599"); + } + if (minimum > maximum) + { + throw new ArgumentOutOfRangeException(paramName: nameof(minimum), message: "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); + } Minimum = minimum; Maximum = maximum; @@ -58,8 +66,14 @@ public StatusCodeRange(int minimum, int maximum) /// Thrown when the is less than 100 or greater than 599. public bool IsWithinRange(int statusCode) { - Guard.NotLessThan(statusCode, 100, nameof(statusCode), "Requires the response HTTP status code not be less than 100"); - Guard.NotGreaterThan(statusCode, 599, nameof(statusCode), "Requires the response HTTP status code not be greater than 599"); + if (statusCode < 100) + { + throw new ArgumentOutOfRangeException(paramName: nameof(statusCode), message: "Requires the response HTTP status code to not be less than 100"); + } + if (statusCode > 599) + { + throw new ArgumentOutOfRangeException(paramName: nameof(statusCode), message: "Requires the response HTTP status code to not be greater than 599"); + } return Minimum <= statusCode && statusCode <= Maximum; } From 38ba612c144b8ac14a6e605e47be1bd1f63f940a Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Wed, 4 Dec 2024 04:30:06 +0100 Subject: [PATCH 04/31] Finish Arcus.WebApi.OpenApi.Extensions --- .../Arcus.WebApi.OpenApi.Extensions.csproj | 4 ---- ...ertificateAuthenticationOperationFilter.cs | 14 +++++--------- .../OAuthAuthorizeOperationFilter.cs | 14 +++++++++----- ...dAccessKeyAuthenticationOperationFilter.cs | 19 +++++++++++++------ 4 files changed, 27 insertions(+), 24 deletions(-) diff --git a/src/Arcus.WebApi.OpenApi.Extensions/Arcus.WebApi.OpenApi.Extensions.csproj b/src/Arcus.WebApi.OpenApi.Extensions/Arcus.WebApi.OpenApi.Extensions.csproj index 10bc60a1..b866e09a 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/Arcus.WebApi.OpenApi.Extensions.csproj +++ b/src/Arcus.WebApi.OpenApi.Extensions/Arcus.WebApi.OpenApi.Extensions.csproj @@ -31,10 +31,6 @@ - - - - diff --git a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs index 49a1da3c..010d8e69 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Linq; using Arcus.WebApi.Security.Authentication.Certificates; -using GuardNet; #if !NETSTANDARD2_1 using Microsoft.OpenApi.Models; #endif @@ -44,16 +43,13 @@ public CertificateAuthenticationOperationFilter( #endif ) { - Guard.NotNullOrWhitespace(securitySchemeName, - nameof(securitySchemeName), - "Requires a name for the Certificate security scheme"); - - _securitySchemeName = securitySchemeName; + _securitySchemeName = securitySchemeName ?? throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Certificate security scheme"); #if !NETSTANDARD2_1 - Guard.For( - () => !Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType), - "Requires a security scheme type for the Certificate authentication that is within the bounds of the enumeration"); + if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType)) + { + throw new ArgumentException(paramName: nameof(securitySchemeType), message: "Requires a security scheme type for the Certificate authentication that is within the bounds of the enumeration"); + } _securitySchemeType = securitySchemeType; #endif diff --git a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs index 5c7bf446..ab057270 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs @@ -3,7 +3,6 @@ using Swashbuckle.AspNetCore.SwaggerGen; using System.Collections.Generic; using System.Linq; -using GuardNet; using Swashbuckle.AspNetCore.Swagger; #if !NETSTANDARD2_1 using Microsoft.OpenApi.Models; @@ -31,12 +30,17 @@ public class OAuthAuthorizeOperationFilter : IOperationFilter /// public OAuthAuthorizeOperationFilter(IEnumerable scopes, string securitySchemaName = "oauth2") { - Guard.NotNull(scopes, nameof(scopes), "Requires a list of API scopes"); - Guard.For(() => scopes.Any(String.IsNullOrWhiteSpace), "Requires a list of non-blank API scopes"); - Guard.NotNullOrWhitespace(securitySchemaName, nameof(securitySchemaName), "Requires a name for the OAuth2 security scheme"); + if (scopes.Any(String.IsNullOrWhiteSpace)) + { + throw new ArgumentException(message: "Requires a list of non-blank API scopes", paramName: nameof(scopes)); + } + if (string.IsNullOrWhiteSpace(securitySchemaName)) + { + throw new ArgumentNullException(paramName: nameof(securitySchemaName), message: "Requires a name for the OAuth2 security scheme"); + } _securitySchemaName = securitySchemaName; - _scopes = scopes; + _scopes = scopes ?? throw new ArgumentNullException(paramName: nameof(scopes), message: "Requires a list of API scopes"); } /// diff --git a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs index b612b1ba..1c9174f2 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs @@ -1,11 +1,11 @@ using System.Collections.Generic; using System.Linq; using Arcus.WebApi.Security.Authentication.SharedAccessKey; -using GuardNet; #if !NETSTANDARD2_1 using System; using Microsoft.OpenApi.Models; #else +using System; using Swashbuckle.AspNetCore.Swagger; #endif using Swashbuckle.AspNetCore.SwaggerGen; @@ -34,10 +34,14 @@ public SharedAccessKeyAuthenticationOperationFilter( string securitySchemeName = DefaultSecuritySchemeName, SecuritySchemeType securitySchemeType = SecuritySchemeType.ApiKey) { - Guard.NotNullOrWhitespace(securitySchemeName, nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); - Guard.For( - () => !Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType), - "Requires a security scheme type for the Shared Access Key authentication that is within the bounds of the enumeration"); + if (string.IsNullOrWhiteSpace(securitySchemeName)) + { + throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Shared Access Key security scheme"); + } + if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType)) + { + throw new ArgumentException(message: "Requires a security scheme type for the Shared Access Key authentication that is within the bounds of the enumeration", paramName: nameof(securitySchemeType)); + } _securitySchemeName = securitySchemeName; _securitySchemeType = securitySchemeType; @@ -49,7 +53,10 @@ public SharedAccessKeyAuthenticationOperationFilter( /// The name of the security scheme. Default value is "sharedaccesskey". public SharedAccessKeyAuthenticationOperationFilter(string securitySchemeName) { - Guard.NotNullOrWhitespace(securitySchemeName, nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); + if (string.IsNullOrWhiteSpace(securitySchemeName)) + { + throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Shared Access Key security scheme"); + } _securitySchemeName = securitySchemeName; } From 333166d713e5826c402960336819a9c440c15026 Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 31 Dec 2024 10:45:11 +0100 Subject: [PATCH 05/31] remove guard.net --- .../RequestTrackingAttribute.cs | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs index 1019613c..ab2a6aa3 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs @@ -65,9 +65,18 @@ public RequestTrackingAttribute(int trackedStatusCode) /// public RequestTrackingAttribute(int minimumStatusCode, int maximumStatusCode) { - Guard.NotLessThan(minimumStatusCode, 100, nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold not be less than 100"); - Guard.NotGreaterThan(maximumStatusCode, 599, nameof(maximumStatusCode), "Requires the maximum HTTP status code threshold not be greater than 599"); - Guard.NotGreaterThan(minimumStatusCode, maximumStatusCode, nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); + if (minimumStatusCode < 100) + { + throw new ArgumentOutOfRangeException(nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold to not be less than 100"); + } + if (maximumStatusCode > 599) + { + throw new ArgumentOutOfRangeException(nameof(maximumStatusCode), "Requires the maximum HTTP status code threshold to not be greater than 599"); + } + if (minimumStatusCode >= maximumStatusCode) + { + throw new ArgumentOutOfRangeException(nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); + } StatusCodeRange = new StatusCodeRange(minimumStatusCode, maximumStatusCode); } From d6f6299dd9ddc7281c83340c7784cf5c9e9c77c7 Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 31 Dec 2024 10:45:25 +0100 Subject: [PATCH 06/31] remove guard.net package from logging.core --- src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj b/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj index c3cf166c..fe8921eb 100644 --- a/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj +++ b/src/Arcus.WebApi.Logging.Core/Arcus.WebApi.Logging.Core.csproj @@ -38,7 +38,6 @@ - From 6781a190939133308ebaf8370872a2855758398f Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 31 Dec 2024 11:31:31 +0100 Subject: [PATCH 07/31] remove guard.net from webapi.logging --- .../AzureFunctionsCorrelationMiddleware.cs | 11 +++-- .../AzureFunctionsHttpCorrelation.cs | 21 +++++++-- ...ureFunctionsHttpCorrelationInfoAccessor.cs | 4 +- .../AzureFunctionsInProcessHttpCorrelation.cs | 26 ++++++++--- ...tionsWorkerApplicationBuilderExtensions.cs | 46 +++++++++++++++---- .../IServiceCollectionExtensions.cs | 11 +++-- .../FunctionContextMiddleware.cs | 16 ++++--- .../Correlation/CorrelationMiddleware.cs | 28 +++++++---- .../IApplicationBuilderExtensions.cs | 31 ++++++++++--- .../IServiceCollectionExtensions.cs | 11 +++-- .../RequestTrackingAttribute.cs | 1 - .../RequestTrackingMiddleware.cs | 31 ++++++++++--- .../VersionTrackingMiddleware.cs | 26 ++++++++--- .../VersionTrackingOptions.cs | 7 ++- 14 files changed, 198 insertions(+), 72 deletions(-) diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs index 207d272c..76a2ff21 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs @@ -2,7 +2,6 @@ using System.Net; using System.Threading.Tasks; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Http; using Microsoft.Azure.Functions.Worker.Middleware; @@ -25,8 +24,14 @@ public class AzureFunctionsCorrelationMiddleware : IFunctionsWorkerMiddleware /// Thrown when the or is nul. public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next) { - Guard.NotNull(context, nameof(context), "Requires a function context instance of the current Azure Function invocation to HTTP correlate the HTTP request"); - Guard.NotNull(next, nameof(next), "Requires a 'next' function to chain this HTTP correlation middleware to the next action in the HTTP request pipeline"); + if (context is null) + { + throw new ArgumentNullException(nameof(context), "Requires a function context instance of the current Azure Function invocation to HTTP correlate the HTTP request"); + } + if (next is null) + { + throw new ArgumentNullException(nameof(next), "Requires a 'next' function to chain this HTTP correlation middleware to the next action in the HTTP request pipeline"); + } var service = context.InstanceServices.GetRequiredService(); HttpRequestData request = await DetermineHttpRequestAsync(context); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs index 7d02cfc4..2030d8d0 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs @@ -4,7 +4,6 @@ using System.Linq; using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.ApplicationInsights; using Microsoft.AspNetCore.Http; using Microsoft.Azure.Functions.Worker.Http; @@ -63,7 +62,10 @@ public AzureFunctionsHttpCorrelation( /// Thrown when the is null. protected override IHeaderDictionary GetRequestHeaders(HttpRequestData request) { - Guard.NotNull(request, nameof(request), "Requires a HTTP request instance to retrieve the HTTP request headers"); + if (request is null) + { + throw new ArgumentNullException(nameof(request), "Requires a HTTP request instance to retrieve the HTTP request headers"); + } Dictionary dictionary = request.Headers.ToDictionary( @@ -128,9 +130,18 @@ protected override HttpCorrelationResult CorrelateW3CForExistingParent(IHeaderDi /// Thrown when the or is blank. protected override void SetHttpResponseHeader(HttpResponseData response, string headerName, string headerValue) { - Guard.NotNull(response, nameof(response), "Requires a HTTP response to set the HTTP correlation headers"); - Guard.NotNullOrWhitespace(headerName, nameof(headerName), "Requires a non-blank HTTP correlation header name to set the HTTP correlation header in the HTTP request"); - Guard.NotNullOrWhitespace(headerValue, nameof(headerValue), "Requires a non-blank HTTP correlation header value to set the HTTP correlation header in the HTTP request"); + if (response is null) + { + throw new ArgumentNullException(nameof(response), "Requires a HTTP response to set the HTTP correlation headers"); + } + if (string.IsNullOrWhiteSpace(headerName)) + { + throw new ArgumentException("Requires a non-blank HTTP correlation header name to set the HTTP correlation header in the HTTP request", nameof(headerName)); + } + if (string.IsNullOrWhiteSpace(headerValue)) + { + throw new ArgumentException("Requires a non-blank HTTP correlation header value to set the HTTP correlation header in the HTTP request", nameof(headerValue)); + } response.Headers.Add(headerName, headerValue); } diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelationInfoAccessor.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelationInfoAccessor.cs index e8958943..3c15e266 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelationInfoAccessor.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelationInfoAccessor.cs @@ -1,7 +1,6 @@ using System; using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.Azure.Functions.Worker; namespace Arcus.WebApi.Logging.AzureFunctions.Correlation @@ -20,8 +19,7 @@ public class AzureFunctionsHttpCorrelationInfoAccessor : IHttpCorrelationInfoAcc /// Thrown when the is null. public AzureFunctionsHttpCorrelationInfoAccessor(IFunctionContextAccessor contextAccessor) { - Guard.NotNull(contextAccessor, nameof(contextAccessor), "Requires a function context accessor instance to get/set the correlation information in the function context"); - _contextAccessor = contextAccessor; + _contextAccessor = contextAccessor ?? throw new ArgumentNullException(nameof(contextAccessor), "Requires a function context accessor instance to get/set the correlation information in the function context"); } /// diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs index 7959ae8c..f57e32a9 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs @@ -2,7 +2,6 @@ using System.Threading.Tasks; using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Primitives; @@ -33,9 +32,18 @@ public AzureFunctionsInProcessHttpCorrelation( IHttpCorrelationInfoAccessor correlationInfoAccessor, ILogger logger) { - Guard.NotNull(options, nameof(options), "Requires a set of HTTP correlation options to determine where the correlation information should be added to the HTTP response headers"); - Guard.NotNull(correlationInfoAccessor, nameof(correlationInfoAccessor), "Requires a HTTP correlation accessor to retrieve the current correlation information"); - Guard.NotNull(logger, nameof(logger), "Requires a logging instance to write diagnostic trace messages while adding the correlation information to the HTTP response headers"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of HTTP correlation options to determine where the correlation information should be added to the HTTP response headers"); + } + if (correlationInfoAccessor is null) + { + throw new ArgumentNullException(nameof(correlationInfoAccessor), "Requires a HTTP correlation accessor to retrieve the current correlation information"); + } + if (logger is null) + { + throw new ArgumentNullException(nameof(logger), "Requires a logging instance to write diagnostic trace messages while adding the correlation information to the HTTP response headers"); + } _options = options; _correlationInfoAccessor = correlationInfoAccessor; @@ -57,8 +65,14 @@ public CorrelationInfo GetCorrelationInfo() /// Thrown when the is null or does not have a response present. public void AddCorrelationResponseHeaders(HttpContext httpContext) { - Guard.NotNull(httpContext, nameof(httpContext), "Requires a HTTP context to add the correlation information to the response headers"); - Guard.NotNull(httpContext.Response, nameof(httpContext), "Requires a HTTP response in the HTTP context to add the correlation information to the response headers"); + if (httpContext is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context to add the correlation information to the response headers"); + } + if (httpContext.Response is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP response in the HTTP context to add the correlation information to the response headers"); + } if (_options.Operation.IncludeInResponse) { diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs index ad7ae63e..5e6cc5a9 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs @@ -4,7 +4,6 @@ using Arcus.WebApi.Logging.AzureFunctions; using Arcus.WebApi.Logging.AzureFunctions.Correlation; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.ApplicationInsights; using Microsoft.Azure.Functions.Worker; using Microsoft.Extensions.DependencyInjection; @@ -27,7 +26,10 @@ public static class IFunctionsWorkerApplicationBuilderExtensions public static IFunctionsWorkerApplicationBuilder UseFunctionContext( this IFunctionsWorkerApplicationBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the function context middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the function context middleware"); + } builder.Services.AddSingleton(); builder.UseMiddleware(); @@ -43,7 +45,10 @@ public static IFunctionsWorkerApplicationBuilder UseFunctionContext( public static IFunctionsWorkerApplicationBuilder UseHttpCorrelation( this IFunctionsWorkerApplicationBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP correlation middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP correlation middleware"); + } return UseHttpCorrelation(builder, options => { }); } @@ -58,7 +63,10 @@ public static IFunctionsWorkerApplicationBuilder UseHttpCorrelation( this IFunctionsWorkerApplicationBuilder builder, Action configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP correlation middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP correlation middleware"); + } builder.Services.AddApplicationInsightsTelemetryWorkerService(); builder.Services.ConfigureFunctionsApplicationInsights(); @@ -98,7 +106,10 @@ public static IFunctionsWorkerApplicationBuilder UseHttpCorrelation( /// Thrown when the is null. public static IFunctionsWorkerApplicationBuilder UseExceptionHandling(this IFunctionsWorkerApplicationBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP exception handling middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP exception handling middleware"); + } return builder.UseMiddleware(); } @@ -111,7 +122,10 @@ public static IFunctionsWorkerApplicationBuilder UseExceptionHandling(this IFunc public static IFunctionsWorkerApplicationBuilder UseExceptionHandling(this IFunctionsWorkerApplicationBuilder builder) where TMiddleware : AzureFunctionsExceptionHandlingMiddleware { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP exception handling middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP exception handling middleware"); + } return builder.UseMiddleware(); } @@ -123,7 +137,10 @@ public static IFunctionsWorkerApplicationBuilder UseExceptionHandling configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + } return UseRequestTracking(builder, configureOptions); } @@ -152,7 +172,10 @@ public static IFunctionsWorkerApplicationBuilder UseRequestTracking this IFunctionsWorkerApplicationBuilder builder) where TMiddleware : AzureFunctionsRequestTrackingMiddleware { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + } return UseRequestTracking(builder, configureOptions: null); } @@ -169,7 +192,10 @@ public static IFunctionsWorkerApplicationBuilder UseRequestTracking Action configureOptions) where TMiddleware : AzureFunctionsRequestTrackingMiddleware { - Guard.NotNull(builder, nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); + } var options = new RequestTrackingOptions(); configureOptions?.Invoke(options); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs index d3a19aef..8a11d6bc 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs @@ -3,7 +3,6 @@ using Arcus.WebApi.Logging.AzureFunctions.Correlation; using Arcus.WebApi.Logging.Core.Correlation; using Arcus.WebApi.Logging.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Azure.Functions.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; @@ -25,7 +24,10 @@ public static class IServiceCollectionExtensions /// Thrown when the is null. public static IServiceCollection AddHttpCorrelation(this IFunctionsHostBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a function host builder instance to add the HTTP correlation services"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function host builder instance to add the HTTP correlation services"); + } return AddHttpCorrelation(builder, options => { }); } @@ -38,7 +40,10 @@ public static IServiceCollection AddHttpCorrelation(this IFunctionsHostBuilder b /// Thrown when the is null. public static IServiceCollection AddHttpCorrelation(this IFunctionsHostBuilder builder, Action configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires a function host builder instance to add the HTTP correlation services"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a function host builder instance to add the HTTP correlation services"); + } IServiceCollection services = builder.Services; diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs index 5e6f2d95..5f3ab2df 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs @@ -1,6 +1,5 @@ using System; using System.Threading.Tasks; -using GuardNet; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Middleware; @@ -19,9 +18,8 @@ public class FunctionContextMiddleware : IFunctionsWorkerMiddleware /// The instance to manage the in this request. /// Thrown when the is null. public FunctionContextMiddleware(IFunctionContextAccessor contextAccessor) - { - Guard.NotNull(contextAccessor, nameof(contextAccessor), "Requires a function context accessor to assign the current function context instance"); - _contextAccessor = contextAccessor; + { + _contextAccessor = contextAccessor ?? throw new ArgumentNullException(nameof(contextAccessor), "Requires a function context accessor to assign the current function context instance"); } /// @@ -32,8 +30,14 @@ public FunctionContextMiddleware(IFunctionContextAccessor contextAccessor) /// A that represents the asynchronous invocation. public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next) { - Guard.NotNull(context, nameof(context), "Requires a function context instance to assign the context to the function context accessor"); - Guard.NotNull(next, nameof(next), "Requires a 'next' function to chain this middleware to the next action in the HTTP request pipeline"); + if (context is null) + { + throw new ArgumentNullException(nameof(context), "Requires a function context instance to assign the context to the function context accessor"); + } + if (next is null) + { + throw new ArgumentNullException(nameof(next), "Requires a 'next' function to chain this middleware to the next action in the HTTP request pipeline"); + } _contextAccessor.FunctionContext = context; await next(context); diff --git a/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs b/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs index 124c182d..2a4caea3 100644 --- a/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs +++ b/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs @@ -1,7 +1,6 @@ using System; using System.Threading.Tasks; using Arcus.WebApi.Logging.Core.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; @@ -25,11 +24,8 @@ public CorrelationMiddleware( RequestDelegate next, ILogger logger) { - Guard.NotNull(next, nameof(next), "Requires a continuation delegate"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance"); - - _next = next; - _logger = logger; + _next = next ?? throw new ArgumentNullException(nameof(next), "Requires a continuation delegate"); + _logger = logger ?? throw new ArgumentNullException(nameof(logger), "Requires a logger instance"); } /// @@ -44,10 +40,22 @@ public CorrelationMiddleware( /// Thrown when the response headers are null. public async Task Invoke(HttpContext httpContext, HttpCorrelation service) { - Guard.NotNull(httpContext, nameof(httpContext)); - Guard.NotNull(service, nameof(service), "Requires the HTTP correlation service"); - Guard.For(() => httpContext.Response is null, "Requires a 'Response'"); - Guard.For(() => httpContext.Response.Headers is null, "Requires a 'Response' object with headers"); + if (httpContext is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context"); + } + if (service is null) + { + throw new ArgumentNullException(nameof(service), "Requires the HTTP correlation service"); + } + if (httpContext.Response is null) + { + throw new ArgumentException("Requires a 'Response'", nameof(httpContext)); + } + if (httpContext.Response.Headers is null) + { + throw new ArgumentException("Requires a 'Response' object with headers", nameof(httpContext)); + } using (HttpCorrelationResult result = service.CorrelateHttpRequest()) { diff --git a/src/Arcus.WebApi.Logging/Extensions/IApplicationBuilderExtensions.cs b/src/Arcus.WebApi.Logging/Extensions/IApplicationBuilderExtensions.cs index 5d81ba0a..92896acc 100644 --- a/src/Arcus.WebApi.Logging/Extensions/IApplicationBuilderExtensions.cs +++ b/src/Arcus.WebApi.Logging/Extensions/IApplicationBuilderExtensions.cs @@ -1,7 +1,6 @@ using System; using Arcus.WebApi.Logging; using Arcus.WebApi.Logging.Correlation; -using GuardNet; // ReSharper disable once CheckNamespace namespace Microsoft.AspNetCore.Builder @@ -19,7 +18,10 @@ public static class IApplicationBuilderExtensions /// Thrown when the is null. public static IApplicationBuilder UseExceptionHandling(this IApplicationBuilder app) { - Guard.NotNull(app, nameof(app), "Requires an application builder instance to add the exception middleware component"); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the exception middleware component"); + } return app.UseMiddleware(); } @@ -33,7 +35,10 @@ public static IApplicationBuilder UseExceptionHandling(this IApplicationBuilder public static IApplicationBuilder UseExceptionHandling(this IApplicationBuilder app) where TMiddleware : ExceptionHandlingMiddleware { - Guard.NotNull(app, nameof(app), "Requires an application builder instance to add the exception middleware component"); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the exception middleware component"); + } return app.UseMiddleware(); } @@ -47,7 +52,10 @@ public static IApplicationBuilder UseRequestTracking( this IApplicationBuilder app, Action configureOptions = null) { - Guard.NotNull(app, nameof(app)); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the request tracking middleware component"); + } return UseRequestTracking(app, configureOptions); } @@ -62,7 +70,10 @@ public static IApplicationBuilder UseRequestTracking( Action configureOptions = null) where TMiddleware : RequestTrackingMiddleware { - Guard.NotNull(app, nameof(app)); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the request tracking middleware component"); + } var options = new RequestTrackingOptions(); configureOptions?.Invoke(options); @@ -76,7 +87,10 @@ public static IApplicationBuilder UseRequestTracking( /// The builder to configure the application's request pipeline. public static IApplicationBuilder UseHttpCorrelation(this IApplicationBuilder app) { - Guard.NotNull(app, nameof(app)); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the request tracking middleware component"); + } return app.UseMiddleware(); } @@ -94,7 +108,10 @@ public static IApplicationBuilder UseHttpCorrelation(this IApplicationBuilder ap /// public static IApplicationBuilder UseVersionTracking(this IApplicationBuilder app, Action configureOptions = null) { - Guard.NotNull(app, nameof(app), "Requires an application builder to add the version tracking middleware"); + if (app is null) + { + throw new ArgumentNullException(nameof(app), "Requires an application builder instance to add the request tracking middleware component"); + } var options = new VersionTrackingOptions(); configureOptions?.Invoke(options); diff --git a/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs index 5666437c..2918c3de 100644 --- a/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs @@ -2,7 +2,6 @@ using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; using Arcus.WebApi.Logging.Correlation; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; @@ -21,7 +20,10 @@ public static class IServiceCollectionExtensions /// The services collection containing the dependency injection services. public static IServiceCollection AddHttpCorrelation(this IServiceCollection services) { - Guard.NotNull(services, nameof(services), "Requires a services collection to add the HTTP correlation services"); + if (services == null) + { + throw new ArgumentNullException(nameof(services), "Requires a services collection to add the HTTP correlation services"); + } return AddHttpCorrelation(services, configureOptions: (HttpCorrelationInfoOptions options) => { }); } @@ -36,7 +38,10 @@ public static IServiceCollection AddHttpCorrelation( this IServiceCollection services, Action configureOptions) { - Guard.NotNull(services, nameof(services), "Requires a services collection to add the HTTP correlation services"); + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Requires a services collection to add the HTTP correlation services"); + } services.AddHttpContextAccessor(); services.AddSingleton(serviceProvider => diff --git a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs index ab2a6aa3..3c10c42b 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs @@ -1,7 +1,6 @@ using System; using System.Linq; using System.Net; -using GuardNet; namespace Arcus.WebApi.Logging { diff --git a/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs b/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs index ae37f83d..f7e94052 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs @@ -5,7 +5,6 @@ using System.Threading.Tasks; using Arcus.Observability.Telemetry.Core; using Arcus.WebApi.Logging.Core.RequestTracking; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Http.Features; using Microsoft.AspNetCore.Routing; @@ -34,9 +33,18 @@ public RequestTrackingMiddleware( ILogger logger) : base(options) { - Guard.NotNull(options, nameof(options), "Requires a set of options to control the behavior of the HTTP tracking middleware"); - Guard.NotNull(next, nameof(next), "Requires a function pipeline to delegate the remainder of the request processing"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to write telemetry tracking during the request processing"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of options to control the behavior of the HTTP tracking middleware"); + } + if (next is null) + { + throw new ArgumentNullException(nameof(next), "Requires a function pipeline to delegate the remainder of the request processing"); + } + if (logger is null) + { + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to write telemetry tracking during the request processing"); + } _next = next; _logger = logger; @@ -50,9 +58,18 @@ public RequestTrackingMiddleware( /// Thrown when the is null. public async Task Invoke(HttpContext httpContext) { - Guard.NotNull(httpContext, nameof(httpContext), "Requires a HTTP context instance to track the incoming request and outgoing response"); - Guard.NotNull(httpContext.Request, nameof(httpContext), "Requires a HTTP request in the context to track the request"); - Guard.NotNull(httpContext.Response, nameof(httpContext), "Requires a HTTP response in the context to track the request"); + if (httpContext is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context instance to track the incoming request and outgoing response"); + } + if (httpContext.Request is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP request in the context to track the request"); + } + if (httpContext.Response is null) + { + throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP response in the context to track the request"); + } if (IsRequestPathOmitted(httpContext.Request.Path, _logger)) { diff --git a/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs b/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs index 086d7142..da8d62f2 100644 --- a/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs @@ -1,7 +1,6 @@ using System; using System.Threading.Tasks; using Arcus.Observability.Telemetry.Core; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; @@ -35,9 +34,18 @@ public VersionTrackingMiddleware( RequestDelegate next, ILogger logger) { - Guard.NotNull(appVersion, nameof(appVersion), "Requires an instance to retrieve the current application version to add the version to the response"); - Guard.NotNull(next, nameof(next), "Requires a continuation delegate to move towards the next functionality in the request pipeline"); - Guard.NotNull(options, nameof(options), "Requires version tracking options to specify how the application version should be tracked in the response"); + if (appVersion is null) + { + throw new ArgumentNullException(nameof(appVersion), "Requires an instance to retrieve the current application version to add the version to the response"); + } + if (next is null) + { + throw new ArgumentNullException(nameof(next), "Requires a continuation delegate to move towards the next functionality in the request pipeline"); + } + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires version tracking options to specify how the application version should be tracked in the response"); + } _appVersion = appVersion; _options = options; @@ -53,8 +61,14 @@ public VersionTrackingMiddleware( /// Thrown when the doesn't contain a response. public async Task Invoke(HttpContext context) { - Guard.NotNull(context, nameof(context), "Requires a HTTP context to add the application version to the response"); - Guard.For(() => context.Response is null, new ArgumentException("Requires a HTTP context with a response to add the application version", nameof(context))); + if (context is null) + { + throw new ArgumentNullException(nameof(context), "Requires a HTTP context to add the application version to the response"); + } + if (context.Response is null) + { + throw new ArgumentException("Requires a HTTP context with a response to add the application version", nameof(context)); + } context.Response.OnStarting(() => { diff --git a/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs b/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs index 0ecb5652..eb79e44d 100644 --- a/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs +++ b/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs @@ -1,4 +1,4 @@ -using GuardNet; +using System; namespace Arcus.WebApi.Logging { @@ -17,7 +17,10 @@ public string HeaderName get => _headerName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Requires a non-blank header name to add the current application version to the response"); + if (string.IsNullOrWhiteSpace(value)) + { + throw new ArgumentException("Requires a non-blank header name to add the current application version to the response", nameof(value)); + } _headerName = value; } } From d1c03500f96c6735dd8cee38e8b16d77e90c54db Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 31 Dec 2024 11:31:48 +0100 Subject: [PATCH 08/31] remove guard.net package from webapi.security --- src/Arcus.WebApi.Security/Arcus.WebApi.Security.csproj | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Arcus.WebApi.Security/Arcus.WebApi.Security.csproj b/src/Arcus.WebApi.Security/Arcus.WebApi.Security.csproj index 3dc15716..baf2c9c5 100644 --- a/src/Arcus.WebApi.Security/Arcus.WebApi.Security.csproj +++ b/src/Arcus.WebApi.Security/Arcus.WebApi.Security.csproj @@ -36,7 +36,6 @@ - From 0a8c36da22013adba5fb89e410c0bc8df11c87dc Mon Sep 17 00:00:00 2001 From: Joachim Goris Date: Tue, 31 Dec 2024 13:07:18 +0100 Subject: [PATCH 09/31] wip webapi.security --- .../CertificateAuthenticationConfig.cs | 23 +++++--- .../CertificateAuthenticationConfigBuilder.cs | 56 +++++++++++++++---- .../CertificateAuthenticationFilter.cs | 31 ++++++---- .../CertificateAuthenticationValidator.cs | 16 ++++-- .../ConfigurationValidationLocation.cs | 11 +++- .../SecretProviderValidationLocation.cs | 12 +++- .../JwtTokenAuthorizationOptions.cs | 30 +++++++--- 7 files changed, 127 insertions(+), 52 deletions(-) diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs index 60ba5456..67697747 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs @@ -4,7 +4,6 @@ using System.Security.Cryptography.X509Certificates; using System.Threading.Tasks; using Arcus.WebApi.Security.Authentication.Certificates.Interfaces; -using GuardNet; using Microsoft.Extensions.Logging; namespace Arcus.WebApi.Security.Authentication.Certificates @@ -27,10 +26,14 @@ public class CertificateAuthenticationConfig internal CertificateAuthenticationConfig( IDictionary locationAndKeyByRequirement) { - Guard.NotNull(locationAndKeyByRequirement, nameof(locationAndKeyByRequirement), "Location and key by certificate requirement dictionary cannot be 'null'"); - Guard.For( - () => locationAndKeyByRequirement.Any(keyValue => keyValue.Value.location is null || keyValue.Value.configuredKey is null), - "All locations and configured keys by certificate requirement cannot be 'null'"); + if (locationAndKeyByRequirement is null) + { + throw new ArgumentNullException(nameof(locationAndKeyByRequirement), "Location and key by certificate requirement dictionary cannot be 'null'"); + } + if (locationAndKeyByRequirement.Any(keyValue => keyValue.Value.location is null || keyValue.Value.configuredKey is null)) + { + throw new ArgumentException("All locations and configured keys by certificate requirement cannot be 'null'"); + } _locationAndKeyByRequirement = locationAndKeyByRequirement; } @@ -45,8 +48,14 @@ internal CertificateAuthenticationConfig( /// Thrown when the is null. internal async Task> GetAllExpectedCertificateValuesAsync(IServiceProvider services, ILogger logger) { - Guard.NotNull(services, nameof(services), "Request services cannot be 'null'"); - Guard.NotNull(logger, nameof(logger), "Logger cannot be 'null'"); + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Request services cannot be 'null'"); + } + if (logger is null) + { + throw new ArgumentNullException(nameof(logger), "Logger cannot be 'null'"); + } var expectedValuesByRequirement = await Task.WhenAll( diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs index 03b24d43..778c56bc 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Security.Cryptography.X509Certificates; using Arcus.WebApi.Security.Authentication.Certificates.Interfaces; -using GuardNet; namespace Arcus.WebApi.Security.Authentication.Certificates { @@ -29,7 +28,10 @@ public CertificateAuthenticationConfigBuilder() /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithSubject(X509ValidationLocation location, string configuredKey) { - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected subject cannot be blank"); + if (string.IsNullOrWhiteSpace(configuredKey)) + { + throw new ArgumentException("Configured key to retrieve expected subject cannot be blank", nameof(configuredKey)); + } return WithSubject(GetValidationLocationImplementation(location), configuredKey); } @@ -43,8 +45,14 @@ public CertificateAuthenticationConfigBuilder WithSubject(X509ValidationLocation /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithSubject(IX509ValidationLocation location, string configuredKey) { - Guard.NotNull(location, nameof(location), "Location implementation to retrieve the expected subject cannot be 'null'"); - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected subject cannot be blank"); + if (location is null) + { + throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected subject cannot be 'null'"); + } + if (configuredKey is null) + { + throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected subject cannot be blank"); + } return AddCertificateRequirement(X509ValidationRequirement.SubjectName, location, configuredKey); } @@ -57,7 +65,10 @@ public CertificateAuthenticationConfigBuilder WithSubject(IX509ValidationLocatio /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithIssuer(X509ValidationLocation location, string configuredKey) { - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected issuer cannot be blank"); + if (string.IsNullOrWhiteSpace(configuredKey)) + { + throw new ArgumentException("Configured key to retrieve expected issuer cannot be blank", nameof(configuredKey)); + } return WithIssuer(GetValidationLocationImplementation(location), configuredKey); } @@ -71,8 +82,14 @@ public CertificateAuthenticationConfigBuilder WithIssuer(X509ValidationLocation /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithIssuer(IX509ValidationLocation location, string configuredKey) { - Guard.NotNull(location, nameof(location), "Location implementation to retrieve the expected issuer cannot be 'null'"); - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected issuer cannot be blank"); + if (location is null) + { + throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected issuer cannot be 'null'"); + } + if (configuredKey is null) + { + throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected issuer cannot be blank"); + } return AddCertificateRequirement(X509ValidationRequirement.IssuerName, location, configuredKey); } @@ -85,7 +102,10 @@ public CertificateAuthenticationConfigBuilder WithIssuer(IX509ValidationLocation /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithThumbprint(X509ValidationLocation location, string configuredKey) { - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected thumbprint cannot be blank"); + if (string.IsNullOrWhiteSpace(configuredKey)) + { + throw new ArgumentException("Configured key to retrieve expected thumbprint cannot be blank", nameof(configuredKey)); + } return WithThumbprint(GetValidationLocationImplementation(location), configuredKey); } @@ -99,8 +119,14 @@ public CertificateAuthenticationConfigBuilder WithThumbprint(X509ValidationLocat /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithThumbprint(IX509ValidationLocation location, string configuredKey) { - Guard.NotNull(location, nameof(location), "Location implementation to retrieve the expected thumbprint cannot be 'null'"); - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key to retrieve expected thumbprint cannot be blank"); + if (location is null) + { + throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected thumbprint cannot be 'null'"); + } + if (configuredKey is null) + { + throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected thumbprint cannot be blank"); + } return AddCertificateRequirement(X509ValidationRequirement.Thumbprint, location, configuredKey); } @@ -110,8 +136,14 @@ private CertificateAuthenticationConfigBuilder AddCertificateRequirement( IX509ValidationLocation location, string configuredKey) { - Guard.NotNull(location, nameof(location), "Location cannot be 'null'"); - Guard.NotNullOrWhitespace(configuredKey, nameof(configuredKey), "Configured key cannot be blank"); + if (location is null) + { + throw new ArgumentNullException(nameof(location), "Location cannot be 'null'"); + } + if (string.IsNullOrWhiteSpace(configuredKey)) + { + throw new ArgumentException("Configured key cannot be blank", nameof(configuredKey)); + } // Overwrites existing requirements. _locationAndKeyByRequirement[requirement] = (location, configuredKey); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs index edab5b68..ce0a6b33 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs @@ -5,7 +5,6 @@ using System.Security.Cryptography.X509Certificates; using System.Text.RegularExpressions; using System.Threading.Tasks; -using GuardNet; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; @@ -47,8 +46,7 @@ public CertificateAuthenticationFilter() : this(new CertificateAuthenticationOpt [Obsolete("Use the new constructor with the certificate authentication filter instead")] public CertificateAuthenticationFilter(CertificateAuthenticationOptions options) { - Guard.NotNull(options, nameof(options), "Requires a set of additional consumer-configurable options to determine the behavior of the certificate authentication"); - _options = options; + _options = options ?? throw new ArgumentNullException(nameof(options), "Requires a set of additional consumer-configurable options to determine the behavior of the certificate authentication"); } /// @@ -61,11 +59,8 @@ public CertificateAuthenticationFilter( CertificateAuthenticationValidator validator, CertificateAuthenticationOptions options) { - Guard.NotNull(validator, nameof(validator), "Requires an instance to validate the incoming client certificate of the HTTP request"); - Guard.NotNull(options, nameof(options), "Requires a set of additional consumer-configurable options to determine the behavior of the certificate authentication"); - - _validator = validator; - _options = options; + _validator = validator ?? throw new ArgumentNullException(nameof(validator), "Requires an instance to validate the incoming client certificate of the HTTP request"); + _options = options ?? throw new ArgumentNullException(nameof(options), "Requires a set of additional consumer-configurable options to determine the behavior of the certificate authentication"); } /// @@ -74,10 +69,22 @@ public CertificateAuthenticationFilter( /// The . public async Task OnAuthorizationAsync(AuthorizationFilterContext context) { - Guard.NotNull(context, nameof(context)); - Guard.NotNull(context.HttpContext, nameof(context.HttpContext)); - Guard.For(() => context.HttpContext.Connection is null, "Invalid action context given without any HTTP connection"); - Guard.For(() => context.HttpContext.RequestServices is null, "Invalid action context given without any HTTP request services"); + if (context is null) + { + throw new ArgumentNullException(nameof(context)); + } + if (context.HttpContext is null) + { + throw new ArgumentNullException(nameof(context)); + } + if (context.HttpContext.Connection is null) + { + throw new ArgumentException("Invalid action context given without any HTTP connection"); + } + if (context.HttpContext.RequestServices is null) + { + throw new ArgumentException("Invalid action context given without any HTTP request services"); + } IServiceProvider services = context.HttpContext.RequestServices; ILogger logger = services.GetLoggerOrDefault(); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs index 41c290bd..547293fd 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs @@ -3,10 +3,10 @@ using System.Linq; using System.Security.Cryptography.X509Certificates; using System.Threading.Tasks; -using GuardNet; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; +using Newtonsoft.Json.Serialization; namespace Arcus.WebApi.Security.Authentication.Certificates { @@ -24,9 +24,7 @@ public class CertificateAuthenticationValidator /// When the is null. public CertificateAuthenticationValidator(CertificateAuthenticationConfig certificateAuthenticationConfig) { - Guard.NotNull(certificateAuthenticationConfig, nameof(certificateAuthenticationConfig), "Certificate authentication configuration cannot be 'null'"); - - _certificateAuthenticationConfig = certificateAuthenticationConfig; + _certificateAuthenticationConfig = certificateAuthenticationConfig ?? throw new ArgumentNullException(nameof(certificateAuthenticationConfig), "Certificate authentication configuration cannot be 'null'"); } /// @@ -42,8 +40,14 @@ public CertificateAuthenticationValidator(CertificateAuthenticationConfig certif /// Thrown when the is null. internal async Task IsCertificateAllowedAsync(X509Certificate2 clientCertificate, IServiceProvider services) { - Guard.NotNull(clientCertificate, nameof(clientCertificate), "Certificate authentication validation requires a client certificate"); - Guard.NotNull(services, nameof(services), "Certificate authentication validation requires a service object to retrieve registered services"); + if (clientCertificate is null) + { + throw new ArgumentNullException(nameof(clientCertificate), "Certificate authentication validation requires a client certificate"); + } + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Certificate authentication validation requires a service object to retrieve registered services"); + } ILogger logger = services.GetService>() diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs index 9e6de5b1..c98f4eb0 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs @@ -3,7 +3,6 @@ using System.Security.Cryptography.X509Certificates; using System.Threading.Tasks; using Arcus.WebApi.Security.Authentication.Certificates.Interfaces; -using GuardNet; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; @@ -34,8 +33,14 @@ public Task GetExpectedCertificateValueForConfiguredKeyAsync(string conf { try { - Guard.NotNullOrWhitespace(configurationKey, nameof(configurationKey), "Configured key cannot be blank"); - Guard.NotNull(services, nameof(services), "Registered services cannot be 'null'"); + if (string.IsNullOrWhiteSpace(configurationKey)) + { + throw new ArgumentException("Configured key cannot be blank", nameof(configurationKey)); + } + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Registered services cannot be 'null'"); + } var configuration = services.GetService(); if (configuration == null) diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs index 8c4fae08..2d8686c9 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs @@ -5,7 +5,6 @@ using Arcus.Security.Core; using Arcus.Security.Core.Caching; using Arcus.WebApi.Security.Authentication.Certificates.Interfaces; -using GuardNet; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Hosting; @@ -34,8 +33,15 @@ private SecretProviderValidationLocation() { } /// The services collections of the HTTP request pipeline to retrieve registered instances. public async Task GetExpectedCertificateValueForConfiguredKeyAsync(string configurationKey, IServiceProvider services) { - Guard.NotNullOrWhitespace(configurationKey, nameof(configurationKey), "Configured key cannot be blank"); - Guard.NotNull(services, nameof(services), "Registered services cannot be 'null'"); + if (string.IsNullOrWhiteSpace(configurationKey)) + { + throw new ArgumentException("Configured key cannot be blank", nameof(configurationKey)); + } + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Registered services cannot be 'null'"); + } + var userDefinedSecretProvider = services.GetService(); if (userDefinedSecretProvider is null) diff --git a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs index d47215c5..2fb230a7 100644 --- a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs +++ b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs @@ -1,7 +1,6 @@ using System; using System.Collections.Generic; using Arcus.WebApi.Security.Authorization.Jwt; -using GuardNet; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Microsoft.IdentityModel.Tokens; @@ -60,8 +59,14 @@ public JwtTokenAuthorizationOptions(IDictionary claimCheck) /// Thrown when the is blank. public JwtTokenAuthorizationOptions(IJwtTokenReader reader, string headerName) { - Guard.NotNull(reader, nameof(reader), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); - Guard.NotNullOrWhitespace(headerName, nameof(headerName), "Requires a non-blank request header name to look for the JWT token"); + if (reader is null) + { + throw new ArgumentNullException(nameof(reader), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); + } + if (string.IsNullOrWhiteSpace(headerName)) + { + throw new ArgumentException("Requires a non-blank request header name to look for the JWT token", nameof(headerName)); + } JwtTokenReader = reader; HeaderName = headerName; @@ -76,7 +81,10 @@ public string HeaderName get => _headerName; set { - Guard.NotNullOrWhitespace(value, nameof(value), "Requires an non-blank request header name to look for the JWT token"); + if (string.IsNullOrWhiteSpace(value)) + { + throw new ArgumentException("Requires a non-blank request header name to look for the JWT token", nameof(value)); + } _headerName = value; } } @@ -90,7 +98,10 @@ public IJwtTokenReader JwtTokenReader get => _jwtTokenReader; set { - Guard.NotNull(value, nameof(value), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); + if (value is null) + { + throw new ArgumentNullException(nameof(value), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); + } _jwtTokenReader = value; _createJwtTokenReader = null; } @@ -119,10 +130,8 @@ public void AddJwtTokenReader() where TImplementation : IJwtTok /// Thrown when the is null. public void AddJwtTokenReader(Func createReader) { - Guard.NotNull(createReader, nameof(createReader), $"Requires an implementation function to create an {nameof(IJwtTokenReader)} instance"); - _jwtTokenReader = null; - _createJwtTokenReader = createReader; + _createJwtTokenReader = createReader ?? throw new ArgumentNullException(nameof(createReader), $"Requires an implementation function to create an {nameof(IJwtTokenReader)} instance"); } /// @@ -136,7 +145,10 @@ internal IJwtTokenReader GetOrCreateJwtTokenReader(IServiceProvider serviceProvi { try { - Guard.NotNull(serviceProvider, nameof(serviceProvider), $"Requires an collection of services to create an {nameof(IJwtTokenReader)} instance"); + if (serviceProvider is null) + { + throw new ArgumentNullException(nameof(serviceProvider), $"Requires a collection of services to create an {nameof(IJwtTokenReader)} instance"); + } return _createJwtTokenReader(serviceProvider); } catch (Exception exception) From 3b364d53a9620c2b6f55c1ff0504c2b357c89a79 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 31 Dec 2024 13:37:54 +0100 Subject: [PATCH 10/31] remove guard.net from source code --- .../IServiceCollectionExtensions.cs | 11 +++- .../Extensions/MvcOptionsExtensions.cs | 31 ++++++--- .../AuthenticationBuilderExtensions.cs | 6 +- .../Extensions/MvcOptionsExtensions.cs | 51 +++++++++++---- .../SharedAccessKeyAuthenticationAttribute.cs | 13 ++-- .../SharedAccessKeyAuthenticationFilter.cs | 38 ++++++++--- .../Extensions/MvcOptionsExtensions.cs | 53 ++++++++++++---- .../Authorization/Jwt/JwtTokenReader.cs | 63 ++++++++++++++----- .../JwtTokenAuthorizationFilter .cs | 32 +++++++--- .../Extensions/IServiceProviderExtensions.cs | 6 +- 10 files changed, 228 insertions(+), 76 deletions(-) diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs index a9da81c7..8cbaff2c 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs @@ -1,6 +1,5 @@ using System; using Arcus.WebApi.Security.Authentication.Certificates; -using GuardNet; // ReSharper disable once CheckNamespace namespace Microsoft.Extensions.DependencyInjection @@ -33,8 +32,14 @@ public static IServiceCollection AddCertificateAuthenticationValidation( this IServiceCollection services, Action configureAuthentication) { - Guard.NotNull(services, nameof(services), "Requires a set of application services to register the certificate authentication validator"); - Guard.NotNull(configureAuthentication, nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Requires a set of application services to register the certificate authentication validator"); + } + if (configureAuthentication is null) + { + throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + } var builder = new CertificateAuthenticationConfigBuilder(); configureAuthentication(builder); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs index da937122..d5fcd208 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs @@ -1,6 +1,5 @@ using System; using Arcus.WebApi.Security.Authentication.Certificates; -using GuardNet; using Microsoft.AspNetCore.Mvc; // ReSharper disable once CheckNamespace @@ -24,7 +23,10 @@ public static partial class MvcOptionsExtensions [Obsolete("Use the " + nameof(AddCertificateAuthenticationFilter) + " overload where the certificate validation locations are configured directly")] public static MvcOptions AddCertificateAuthenticationFilter(this MvcOptions options) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + } return AddCertificateAuthenticationFilter(options, configureOptions: null); } @@ -46,7 +48,10 @@ public static MvcOptions AddCertificateAuthenticationFilter( this MvcOptions options, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + } var authOptions = new CertificateAuthenticationOptions(); configureOptions?.Invoke(authOptions); @@ -77,8 +82,14 @@ public static MvcOptions AddCertificateAuthenticationFilter( this MvcOptions options, Action configureAuthentication) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); - Guard.NotNull(configureAuthentication, nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + } + if (configureAuthentication is null) + { + throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + } return AddCertificateAuthenticationFilter(options, configureAuthentication, configureOptions: null); } @@ -109,8 +120,14 @@ public static MvcOptions AddCertificateAuthenticationFilter( Action configureAuthentication, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); - Guard.NotNull(configureAuthentication, nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); + } + if (configureAuthentication is null) + { + throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); + } var builder = new CertificateAuthenticationConfigBuilder(); configureAuthentication(builder); diff --git a/src/Arcus.WebApi.Security/Authentication/JwtBearer/Extensions/AuthenticationBuilderExtensions.cs b/src/Arcus.WebApi.Security/Authentication/JwtBearer/Extensions/AuthenticationBuilderExtensions.cs index d96bf796..2d7e0f4d 100644 --- a/src/Arcus.WebApi.Security/Authentication/JwtBearer/Extensions/AuthenticationBuilderExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/JwtBearer/Extensions/AuthenticationBuilderExtensions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; using Microsoft.AspNetCore.Authentication; using Microsoft.AspNetCore.Authentication.JwtBearer; using Microsoft.Extensions.Options; @@ -21,7 +20,10 @@ public static class AuthenticationBuilderExtensions /// A reference to builder after the operation has completed. public static AuthenticationBuilder AddJwtBearer(this AuthenticationBuilder builder, Action configureOptions) { - Guard.NotNull(builder, nameof(builder), "Requires an authentication builder instance to add the JWT Bearer authentication"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires an authentication builder instance to add the JWT Bearer authentication"); + } if (configureOptions != null) { diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs index 508cbec2..d545c7b7 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs @@ -1,7 +1,6 @@ using System; using Arcus.Security.Core; using Arcus.WebApi.Security.Authentication.SharedAccessKey; -using GuardNet; using Microsoft.AspNetCore.Mvc; // ReSharper disable once CheckNamespace @@ -26,8 +25,14 @@ public static partial class MvcOptionsExtensions /// Thrown when the or is blank. public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnHeader(this MvcOptions options, string headerName, string secretName) { - Guard.NotNull(options, nameof(options), "Requires an MVC options instance to add the shared access key authenctication filter"); - Guard.NotNullOrWhitespace(headerName, nameof(headerName), "Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires an MVC options instance to add the shared access key authentication filter"); + } + if (string.IsNullOrWhiteSpace(headerName)) + { + throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName); + } return AddSharedAccessKeyAuthenticationFilterOnHeader(options, headerName, secretName, configureOptions: null); } @@ -53,8 +58,14 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnHeader( string secretName, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires an MVC options instance to add the shared access key authenctication filter"); - Guard.NotNullOrWhitespace(headerName, nameof(headerName), "Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a MVC options instance to add the shared access key authentication filter"); + } + if (string.IsNullOrWhiteSpace(headerName)) + { + throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName)); + } var authOptions = new SharedAccessKeyAuthenticationOptions(); configureOptions?.Invoke(authOptions); @@ -80,9 +91,18 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnQuery( string parameterName, string secretName) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); - Guard.NotNullOrWhitespace(parameterName, nameof(parameterName), "Requires a non-blank HTTP request query parameter name name to match the stored secret during the shared access key authentication"); - Guard.NotNullOrWhitespace(secretName, nameof(secretName), "Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); + } + if (string.IsNullOrWhiteSpace(parameterName)) + { + throw new ArgumentException("Requires a non-blank HTTP request query parameter name to match the stored secret during the shared access key authentication", nameof(parameterName)); + } + if (string.IsNullOrWhiteSpace(secretName)) + { + throw new ArgumentException("Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication", nameof(secretName)); + } return AddSharedAccessKeyAuthenticationFilterOnQuery(options, parameterName, secretName, configureOptions: null); } @@ -108,9 +128,18 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnQuery( string secretName, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); - Guard.NotNullOrWhitespace(parameterName, nameof(parameterName), "Requires a non-blank HTTP request query parameter name name to match the stored secret during the shared access key authentication"); - Guard.NotNullOrWhitespace(secretName, nameof(secretName), "Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); + } + if (string.IsNullOrWhiteSpace(parameterName)) + { + throw new ArgumentException("Requires a non-blank HTTP request query parameter name to match the stored secret during the shared access key authentication", nameof(parameterName)); + } + if (string.IsNullOrWhiteSpace(secretName)) + { + throw new ArgumentException("Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication", nameof(secretName)); + } var authOptions = new SharedAccessKeyAuthenticationOptions(); configureOptions?.Invoke(authOptions); diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs index e5160971..b44de2f6 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs @@ -1,6 +1,5 @@ using System; using Arcus.Security.Core; -using GuardNet; using Microsoft.AspNetCore.Mvc; namespace Arcus.WebApi.Security.Authentication.SharedAccessKey @@ -28,10 +27,14 @@ public class SharedAccessKeyAuthenticationAttribute : TypeFilterAttribute public SharedAccessKeyAuthenticationAttribute(string secretName, string headerName = null, string queryParameterName = null) : base(typeof(SharedAccessKeyAuthenticationFilter)) { - Guard.NotNullOrWhitespace(secretName, nameof(secretName), "Secret name cannot be blank"); - Guard.For( - () => String.IsNullOrWhiteSpace(headerName) && String.IsNullOrWhiteSpace(queryParameterName), - "Requires either a non-blank header name or query parameter name"); + if (string.IsNullOrWhiteSpace(secretName)) + { + throw new ArgumentException(" Secret name cannot be blank", nameof(secretName)); + } + if (string.IsNullOrWhiteSpace(headerName) && string.IsNullOrWhiteSpace(queryParameterName)) + { + throw new ArgumentException("Requires either a non-blank header name or query parameter name"); + } _options = new SharedAccessKeyAuthenticationOptions(); Arguments = new object[] { headerName?? String.Empty, queryParameterName?? String.Empty, secretName, _options }; diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs index 20743355..c4c4854d 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs @@ -5,7 +5,6 @@ using System.Threading.Tasks; using Arcus.Security.Core; using Arcus.Security.Core.Caching; -using GuardNet; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; @@ -52,10 +51,14 @@ public SharedAccessKeyAuthenticationFilter(string headerName, string queryParame /// Thrown when the and are blank. public SharedAccessKeyAuthenticationFilter(string headerName, string queryParameterName, string secretName, SharedAccessKeyAuthenticationOptions options) { - Guard.NotNullOrWhitespace(secretName, nameof(secretName), "Requires a non-blank secret name"); - Guard.For( - () => String.IsNullOrWhiteSpace(headerName) && String.IsNullOrWhiteSpace(queryParameterName), - "Requires either a non-blank header name or query parameter name"); + if (string.IsNullOrWhiteSpace(secretName)) + { + throw new ArgumentException("Requires a non-blank secret name", nameof(secretName)); + } + if (string.IsNullOrWhiteSpace(headerName) && string.IsNullOrWhiteSpace(queryParameterName)) + { + throw new ArgumentException("Requires either a non-blank header name or query parameter name"); + } _headerName = headerName; _queryParameterName = queryParameterName; @@ -72,11 +75,26 @@ public SharedAccessKeyAuthenticationFilter(string headerName, string queryParame /// public async Task OnAuthorizationAsync(AuthorizationFilterContext context) { - Guard.NotNull(context, nameof(context)); - Guard.NotNull(context.HttpContext, nameof(context.HttpContext)); - Guard.For(() => context.HttpContext.Request is null, "Invalid action context given without any HTTP request"); - Guard.For(() => context.HttpContext.Request.Headers is null, "Invalid action context given without any HTTP request headers"); - Guard.For(() => context.HttpContext.RequestServices is null, "Invalid action context given without any HTTP request services"); + if (context is null) + { + throw new ArgumentNullException(nameof(context)); + } + if (context.HttpContext is null) + { + throw new ArgumentNullException(nameof(context.HttpContext)); + } + if (context.HttpContext.Request is null) + { + throw new ArgumentException("Invalid action context given without any HTTP request"); + } + if (context.HttpContext.Request.Headers is null) + { + throw new ArgumentException("Invalid action context given without any HTTP request headers"); + } + if (context.HttpContext.RequestServices is null) + { + throw new ArgumentException("Invalid action context given without any HTTP request services"); + } ILogger logger = context.HttpContext.RequestServices.GetLoggerOrDefault(); diff --git a/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs index 617cc1bb..07157d55 100644 --- a/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Linq; using Arcus.WebApi.Security.Authorization; -using GuardNet; using Microsoft.AspNetCore.Mvc; // ReSharper disable once CheckNamespace @@ -20,7 +19,10 @@ public static partial class MvcOptionsExtensions /// Thrown when the is null. public static MvcOptions AddJwtTokenAuthorizationFilter(this MvcOptions options) { - Guard.NotNull(options, nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + } return AddJwtTokenAuthorizationFilter(options, configureOptions: null); } @@ -35,7 +37,10 @@ public static MvcOptions AddJwtTokenAuthorizationFilter( this MvcOptions options, Action configureOptions) { - Guard.NotNull(options, nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + } var authOptions= new JwtTokenAuthorizationOptions(); configureOptions?.Invoke(authOptions); @@ -54,11 +59,22 @@ public static MvcOptions AddJwtTokenAuthorizationFilter( this MvcOptions options, IDictionary claimCheck) { - Guard.NotNull(options, nameof(options), "Requires a filter collection to add the JWT token authorization filter"); - Guard.NotNull(claimCheck, nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); - Guard.NotAny(claimCheck, nameof(claimCheck), "Requires at least one entry in the set of claim checks to verify the claims in the request JWT"); - Guard.For(() => claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value)), - "Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + } + if (claimCheck is null) + { + throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); + } + if (!claimCheck.Any()) + { + throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); + } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) + { + throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + } return AddJwtTokenAuthorizationFilter(options, configureOptions: null, claimCheck: claimCheck); } @@ -76,11 +92,22 @@ public static MvcOptions AddJwtTokenAuthorizationFilter( Action configureOptions, IDictionary claimCheck) { - Guard.NotNull(options, nameof(options), "Requires a filter collection to add the JWT token authorization filter"); - Guard.NotNull(claimCheck, nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); - Guard.NotAny(claimCheck, nameof(claimCheck), "Requires at least one entry in the set of claim checks to verify the claims in the request JWT"); - Guard.For(() => claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value)), - "Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); + } + if (claimCheck is null) + { + throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); + } + if (!claimCheck.Any()) + { + throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); + } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) + { + throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + } var authOptions = new JwtTokenAuthorizationOptions(claimCheck); configureOptions?.Invoke(authOptions); diff --git a/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs b/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs index debf9a42..8c97eed5 100644 --- a/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs +++ b/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs @@ -3,7 +3,6 @@ using System.IdentityModel.Tokens.Jwt; using System.Linq; using System.Threading.Tasks; -using GuardNet; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; using IdentityModel; @@ -35,7 +34,10 @@ public class JwtTokenReader : IJwtTokenReader /// Thrown when the is blank. public JwtTokenReader(string audience) : this(audience, NullLogger.Instance) { - Guard.NotNullOrWhitespace(audience, nameof(audience), "Requires an audience to validate against"); + if (string.IsNullOrWhiteSpace(audience)) + { + throw new ArgumentException("Requires an audience to validate against", nameof(audience)); + } } /// @@ -47,7 +49,10 @@ public JwtTokenReader(string audience) : this(audience, NullLoggerThrown when the is blank. public JwtTokenReader(string audience, ILogger logger) : this(new Dictionary {{ JwtClaimTypes.Audience, audience}}, logger) { - Guard.NotNullOrWhitespace(audience, nameof(audience), "Requires an audience to validate against"); + if (string.IsNullOrWhiteSpace(audience)) + { + throw new ArgumentException("Requires an audience to validate against", nameof(audience)); + } } /// @@ -71,10 +76,18 @@ public JwtTokenReader(IDictionary claimCheck) : this(claimCheck, /// Thrown when the doesn't have any entries or one of the entries has blank key/value inputs. public JwtTokenReader(IDictionary claimCheck, ILogger logger) { - Guard.NotNull(claimCheck, nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); - Guard.NotAny(claimCheck, nameof(claimCheck), "Requires at least one entry in the set of claim checks to verify the claims in the request JWT"); - Guard.For(() => claimCheck.Any(item => String.IsNullOrWhiteSpace(item.Key) || String.IsNullOrWhiteSpace(item.Value)), - "Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + if (claimCheck is null) + { + throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); + } + if (!claimCheck.Any()) + { + throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); + } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) + { + throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT", nameof(claimCheck)); + } _claimCheck = claimCheck; _logger = logger ?? NullLogger.Instance; @@ -153,8 +166,14 @@ public JwtTokenReader(TokenValidationParameters tokenValidationParameters, strin /// Thrown when the is blank. public JwtTokenReader(TokenValidationParameters tokenValidationParameters, string openIdConnectDiscoveryUri, ILogger logger) { - Guard.NotNull(tokenValidationParameters, nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); - Guard.NotNullOrWhitespace(openIdConnectDiscoveryUri, nameof(openIdConnectDiscoveryUri), "Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration"); + if (tokenValidationParameters is null) + { + throw new ArgumentNullException(nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); + } + if (string.IsNullOrWhiteSpace(openIdConnectDiscoveryUri)) + { + throw new ArgumentException("Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration", nameof(openIdConnectDiscoveryUri)); + } _tokenValidationParameters = tokenValidationParameters; _logger = logger ?? NullLogger.Instance; @@ -192,12 +211,26 @@ public JwtTokenReader( IDictionary claimCheck, ILogger logger) { - Guard.NotNullOrWhitespace(openIdConnectDiscoveryUri, nameof(openIdConnectDiscoveryUri), "Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration"); - Guard.NotNull(tokenValidationParameters, nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); - Guard.NotNull(claimCheck, nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); - Guard.NotAny(claimCheck, nameof(claimCheck), "Requires at least one entry in the set of claim checks to verify the claims in the request JWT"); - Guard.For(() => claimCheck.Any(item => String.IsNullOrWhiteSpace(item.Key) || String.IsNullOrWhiteSpace(item.Value)), - "Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); + if (string.IsNullOrWhiteSpace(openIdConnectDiscoveryUri)) + { + throw new ArgumentException("Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration", nameof(openIdConnectDiscoveryUri)); + } + if (tokenValidationParameters is null) + { + throw new ArgumentNullException(nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); + } + if (claimCheck is null) + { + throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); + } + if (!claimCheck.Any()) + { + throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); + } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) + { + throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT", nameof(claimCheck)); + } _tokenValidationParameters = tokenValidationParameters; _claimCheck = claimCheck; diff --git a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs index 4eff561e..e1739f44 100644 --- a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs +++ b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs @@ -5,7 +5,6 @@ using System.Linq; using System.Threading.Tasks; using Arcus.WebApi.Security.Authorization.Jwt; -using GuardNet; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc.Filters; @@ -32,8 +31,10 @@ public class JwtTokenAuthorizationFilter : IAsyncAuthorizationFilter /// Thrown when the is null. public JwtTokenAuthorizationFilter(JwtTokenAuthorizationOptions authorizationOptions) { - Guard.NotNull(authorizationOptions, nameof(authorizationOptions), - "Requires a set of options to configure how the JWT authorization filter should authorize requests"); + if (authorizationOptions is null) + { + throw new ArgumentNullException(nameof(authorizationOptions), "Requires a set of options to configure how the JWT authorization filter should authorize requests"); + } _authorizationOptions = authorizationOptions; } @@ -47,11 +48,26 @@ public JwtTokenAuthorizationFilter(JwtTokenAuthorizationOptions authorizationOpt /// public virtual async Task OnAuthorizationAsync(AuthorizationFilterContext context) { - Guard.NotNull(context, nameof(context)); - Guard.NotNull(context.HttpContext, nameof(context.HttpContext)); - Guard.For(() => context.HttpContext.Request is null, "Invalid action context given without any HTTP request"); - Guard.For(() => context.HttpContext.Request.Headers is null, "Invalid action context given without any HTTP request headers"); - Guard.For(() => context.HttpContext.RequestServices is null, "Invalid action context given without any HTTP request services"); + if (context is null) + { + throw new ArgumentNullException(nameof(context)); + } + if (context.HttpContext is null) + { + throw new ArgumentNullException(nameof(context.HttpContext)); + } + if (context.HttpContext.Request is null) + { + throw new ArgumentNullException(nameof(context), "INvalid action context given without any HTTP request"); + } + if (context.HttpContext.Request.Headers is null) + { + throw new ArgumentNullException(nameof(context), "Invalid action context given without any HTTP request headers"); + } + if (context.HttpContext.RequestServices is null) + { + throw new ArgumentNullException(nameof(context), "Invalid action context given without any HTTP request services"); + } ILogger logger = context.HttpContext.RequestServices.GetLoggerOrDefault(); diff --git a/src/Arcus.WebApi.Security/Extensions/IServiceProviderExtensions.cs b/src/Arcus.WebApi.Security/Extensions/IServiceProviderExtensions.cs index cd295456..625869a4 100644 --- a/src/Arcus.WebApi.Security/Extensions/IServiceProviderExtensions.cs +++ b/src/Arcus.WebApi.Security/Extensions/IServiceProviderExtensions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; @@ -23,7 +22,10 @@ public static class IServiceProviderExtensions /// Thrown when the is null. public static ILogger GetLoggerOrDefault(this IServiceProvider services) { - Guard.NotNull(services, nameof(services), "Requires a services collection to retrieve an logger instance"); + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Requires a services collection to retrieve a logger instance"); + } var loggerFactory = services.GetService(); ILogger logger = loggerFactory?.CreateLogger(); From 1c255691945b8bf58fabf595ce208ca7dc6d7611 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 31 Dec 2024 14:31:52 +0100 Subject: [PATCH 11/31] remove guard.net from test code --- .../Arcus.WebApi.Tests.Core.csproj | 1 - .../Logging/AzureFunctions/ReadOnceStream.cs | 11 +++-- .../Fixture/HttpRequestBuilder.cs | 44 ++++++++++++++----- .../Fixture/TestApiServer.cs | 21 ++++++--- .../Fixture/TestApiServerOptions.cs | 27 +++++++++--- .../Fixture/TestConfig.cs | 6 ++- .../Logging/Fixture/HttpAssert.cs | 16 +++++-- .../Logging/Fixture/HttpAssertProvider.cs | 26 +++++++---- .../HttpAssertServiceCollectionExtensions.cs | 16 +++++-- .../Fixture/TraceIdentifierMiddleware.cs | 8 +--- .../Fixture/CertificateConfiguration.cs | 5 +-- .../Fixture/SelfSignedCertificate.cs | 16 +++++-- .../HttpTriggerFunction.cs | 8 +--- .../DefaultHttpMessageHandlerBuilder.cs | 4 +- 14 files changed, 140 insertions(+), 69 deletions(-) diff --git a/src/Arcus.WebApi.Tests.Core/Arcus.WebApi.Tests.Core.csproj b/src/Arcus.WebApi.Tests.Core/Arcus.WebApi.Tests.Core.csproj index 77410530..4326c8cc 100644 --- a/src/Arcus.WebApi.Tests.Core/Arcus.WebApi.Tests.Core.csproj +++ b/src/Arcus.WebApi.Tests.Core/Arcus.WebApi.Tests.Core.csproj @@ -8,7 +8,6 @@ - diff --git a/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs b/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs index b2c9f4c4..c5502d8a 100644 --- a/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs +++ b/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs @@ -1,6 +1,5 @@ using System; using System.IO; -using GuardNet; namespace Arcus.WebApi.Tests.Unit.Logging.Fixture.AzureFunctions { @@ -14,8 +13,14 @@ public class ReadOnceStream : Stream /// public ReadOnceStream(Stream innerStream) { - Guard.NotNull(innerStream, nameof(innerStream)); - Guard.For(() => !innerStream.CanRead, "Requires a readable stream to represents a read-once stream"); + if (innerStream is null) + { + throw new ArgumentNullException(nameof(innerStream)); + } + if (!innerStream.CanRead) + { + throw new ArgumentException("Requires a readable stream to represent a read-once stream", nameof(innerStream)); + } _innerStream = innerStream; } diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs index 67c57cb5..ce7e71f0 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs @@ -4,7 +4,7 @@ using System.Linq; using System.Net.Http; using System.Text; -using GuardNet; +using System.Threading; namespace Arcus.WebApi.Tests.Integration.Fixture { @@ -33,7 +33,10 @@ private HttpRequestBuilder(HttpMethod method, string path) /// Thrown when the is blank. public static HttpRequestBuilder Get(string route) { - Guard.NotNullOrWhitespace(route, nameof(route), "Requires a non-blank HTTP relative route to create a HTTP GET request builder instance"); + if (string.IsNullOrWhiteSpace(route)) + { + throw new ArgumentException("Requires a non-blank HTTP relative route to create a HTTP GET request builder instance", nameof(route)); + } return new HttpRequestBuilder(HttpMethod.Get, route); } @@ -45,7 +48,10 @@ public static HttpRequestBuilder Get(string route) /// Thrown when the is blank. public static HttpRequestBuilder Post(string route) { - Guard.NotNullOrWhitespace(route, nameof(route), "Requires a non-blank HTTP relative route to create a HTTP POST request builder instance"); + if (string.IsNullOrWhiteSpace(route)) + { + throw new ArgumentException("Requires a non-blank HTTP relative route to create a HTTP POST request builder instance", nameof(route)); + } return new HttpRequestBuilder(HttpMethod.Post, route); } @@ -57,7 +63,10 @@ public static HttpRequestBuilder Post(string route) /// Thrown when the is blank. public HttpRequestBuilder WithHeader(string headerName, object headerValue) { - Guard.NotNullOrWhitespace(headerName, nameof(headerName), "Requires a non-blank header name to add the header to the HTTP request builder instance"); + if (string.IsNullOrWhiteSpace(headerName)) + { + throw new ArgumentException("Requires a non-blank header name to add the header to the HTTP request builder instance", nameof(headerName)); + } _headers.Add(new KeyValuePair(headerName, headerValue?.ToString())); return this; @@ -71,7 +80,10 @@ public HttpRequestBuilder WithHeader(string headerName, object headerValue) /// Thrown when the is blank. public HttpRequestBuilder WithParameter(string parameterName, object parameterValue) { - Guard.NotNullOrWhitespace(parameterName, nameof(parameterName), "Requires a non-blank query parameter name to add the parameter to the HTTP request builder instance"); + if (string.IsNullOrWhiteSpace(parameterName)) + { + throw new ArgumentException("Requires a non-blank query parameter to add the parameter to the HTTP request builder instance", nameof(parameterName)); + } _parameters.Add(new KeyValuePair(parameterName, parameterValue.ToString())); return this; @@ -85,12 +97,15 @@ public HttpRequestBuilder WithParameter(string parameterName, object parameterVa /// Thrown when the is blank. public HttpRequestBuilder WithJsonText(string text) { - Guard.NotNullOrWhitespace(text, nameof(text), "Requires non-blank JSON request text to add the content to the HTTP request builder instance"); + if (string.IsNullOrWhiteSpace(text)) + { + throw new ArgumentException("Requires a non-blank JSON request text to add the content to the HTTP request builder instance", nameof(text)); + } _createContent = () => new StringContent($"\"{text}\"", Encoding.UTF8, "application/json"); return this; } - + /// /// Adds a JSON json to the HTTP request. /// @@ -99,7 +114,10 @@ public HttpRequestBuilder WithJsonText(string text) /// Thrown when the is blank. public HttpRequestBuilder WithJsonBody(string json) { - Guard.NotNullOrWhitespace(json, nameof(json), "Requires non-blank JSON request body to add the content to the HTTP request builder instance"); + if (string.IsNullOrWhiteSpace(json)) + { + throw new ArgumentException("Requires a non-blank JSON request body to add the content to the HTTP request builder instance", nameof(json)); + } _createContent = () => new StringContent(json, Encoding.UTF8, "application/json"); return this; @@ -113,7 +131,10 @@ public HttpRequestBuilder WithJsonBody(string json) /// Thrown when the is blank. public HttpRequestBuilder WithTextBody(string text) { - Guard.NotNullOrWhitespace(text, nameof(text), "Requires a non-blank text input for the request body"); + if (string.IsNullOrWhiteSpace(text)) + { + throw new ArgumentException("Requires a non-blank text input for the request body", nameof(text)); + } _createContent = () => new StringContent(text, Encoding.UTF8, "text/plain"); return this; @@ -127,7 +148,10 @@ public HttpRequestBuilder WithTextBody(string text) /// Thrown when the is not in the correct HTTP format. internal HttpRequestMessage Build(string baseRoute) { - Guard.NotNullOrWhitespace(baseRoute, nameof(baseRoute), "Requires a non-blank base HTTP endpoint to create a HTTP request message from the HTTP request builder instance"); + if (string.IsNullOrWhiteSpace(baseRoute)) + { + throw new ArgumentException("Requires a non-blank base HTTP endpoint to create a HTTP request message from the HTTP request builder instance", nameof(baseRoute)); + } string parameters = ""; if (_parameters.Count > 0) diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs index c4ad7c15..e6e1baf2 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs @@ -4,12 +4,12 @@ using System.Security.Cryptography; using System.Threading.Tasks; using Arcus.Testing.Logging; -using GuardNet; using Microsoft.ApplicationInsights.AspNetCore.Extensions; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.ApplicationInsights; +using Microsoft.Extensions.Options; namespace Arcus.WebApi.Tests.Integration.Fixture { @@ -26,9 +26,7 @@ public class TestApiServer : IAsyncDisposable protected TestApiServer(IHost host, TestApiServerOptions options, ILogger logger) { - Guard.NotNull(host, nameof(host), "Requires a 'IHost' instance to start/stop the test API server"); - - _host = host; + _host = host ?? throw new ArgumentNullException(nameof(host), "Requires a 'IHost' instance to start/stop the test API server"); _options = options; _logger = logger; @@ -48,8 +46,14 @@ protected TestApiServer(IHost host, TestApiServerOptions options, ILogger logger /// Thrown when the or is null. public static async Task StartNewAsync(TestApiServerOptions options, ILogger logger) { - Guard.NotNull(options, nameof(options), "Requires a set of configurable options to control the behavior of the test API server"); - Guard.NotNull(logger, nameof(logger), "Requires a logger instance to write diagnostic messages during the lifetime of the test API server"); + if (options is null) + { + throw new ArgumentNullException(nameof(options), "Requires a set of configurable options to control the behavior of the test API server"); + } + if (logger is null) + { + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to write diagnostic messages during the lifetime of the test API server"); + } IHostBuilder builder = Host.CreateDefaultBuilder(); options.ApplyOptions(builder); @@ -88,7 +92,10 @@ public static async Task StartNewAsync(TestApiServerOptions optio /// Thrown when the is null. public async Task SendAsync(HttpRequestBuilder builder) { - Guard.NotNull(builder, nameof(builder), "Requires a HTTP request builder instance to create a HTTP request to the test API server"); + if (builder is null) + { + throw new ArgumentNullException(nameof(builder), "Requires a HTTP request builder instance to create a HTTP request to the test API server"); + } HttpRequestMessage request = builder.Build(_options.Url); diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs index 4172b1ec..3c1e8d75 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs @@ -3,7 +3,6 @@ using System.Collections.ObjectModel; using System.Linq; using Bogus; -using GuardNet; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; using Microsoft.Azure.Functions.Worker; @@ -47,7 +46,10 @@ public TestApiServerOptions() /// Thrown when the is null. public TestApiServerOptions ConfigureServices(Action configureServices) { - Guard.NotNull(configureServices, nameof(configureServices), "Requires a function to configure the dependency services on the test API server"); + if (configureServices is null) + { + throw new ArgumentNullException(nameof(configureServices), "Requires a function to configure the dependency services on the test API server"); + } _configureServices.Add(configureServices); return this; @@ -64,7 +66,10 @@ public TestApiServerOptions ConfigureServices(Action configu /// Thrown when the is null. public TestApiServerOptions PreConfigure(Action configure) { - Guard.NotNull(configure, nameof(configure), "Requires a function to configure the application on the test API server"); + if (configure is null) + { + throw new ArgumentNullException(nameof(configure), "Requires a function to configure the application on the test API server"); + } _preconfigures.Add(configure); return this; @@ -81,7 +86,10 @@ public TestApiServerOptions PreConfigure(Action configure) /// Thrown when the is null. public TestApiServerOptions Configure(Action configure) { - Guard.NotNull(configure, nameof(configure), "Requires a function to configure the application on the test API server"); + if (configure is null) + { + throw new ArgumentNullException(nameof(configure), "Requires a function to configure the application on the test API server"); + } _configures.Add(configure); return this; @@ -95,7 +103,10 @@ public TestApiServerOptions Configure(Action configure) /// Thrown when the is null. public TestApiServerOptions ConfigureHost(Action configure) { - Guard.NotNull(configure, nameof(configure), "Requires a function to configure the hosting configuration of the test API server"); + if (configure is null) + { + throw new ArgumentNullException(nameof(configure), "Requires a function to configure the hosting configuration of the test API server"); + } _hostingConfigures.Add(configure); return this; @@ -109,10 +120,12 @@ public TestApiServerOptions ConfigureHost(Action configure) /// Thrown when the is null. public TestApiServerOptions ConfigureAppConfiguration(Action configure) { - Guard.NotNull(configure, nameof(configure), "Requires a function to configure the application configuration of the test API server"); + if (configure is null) + { + throw new ArgumentNullException(nameof(configure), "Requires a function to configure the application configuration of the test API server"); + } _appConfigures.Add(configure); - return this; } /// diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestConfig.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestConfig.cs index 658cb152..574b8dbb 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestConfig.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestConfig.cs @@ -1,6 +1,5 @@ using System; using System.Collections.Generic; -using GuardNet; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.Primitives; @@ -15,7 +14,10 @@ public class TestConfig : IConfigurationRoot private TestConfig(IConfigurationRoot config) { - Guard.NotNull(config, nameof(config)); + if (config is null) + { + throw new ArgumentNullException(nameof(config)); + } _config = config; } diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssert.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssert.cs index 6584cdf0..2addf13a 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssert.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssert.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; using Microsoft.AspNetCore.Http; namespace Arcus.WebApi.Tests.Integration.Logging.Fixture @@ -13,7 +12,10 @@ public class HttpAssert private HttpAssert(Action assertion) { - Guard.NotNull(assertion, nameof(assertion), "Requires an assertion function to verify a HTTP context"); + if (assertion is null) + { + throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify a HTTP context"); + } _assertion = assertion; } @@ -24,7 +26,10 @@ private HttpAssert(Action assertion) /// Thrown when the is null. public static HttpAssert Create(Action assertion) { - Guard.NotNull(assertion, nameof(assertion), "Requires an assertion function to verify a HTTP context"); + if (assertion is null) + { + throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify a HTTP context"); + } return new HttpAssert(assertion); } @@ -35,7 +40,10 @@ public static HttpAssert Create(Action assertion) /// Thrown when the is null. public void Assert(HttpContext context) { - Guard.NotNull(context, nameof(context), "Requires a HTTP context to run an assertion function on it"); + if (context is null) + { + throw new ArgumentNullException(nameof(context), "Requires a HTTP context to run an assertion function to it"); + } _assertion(context); } } diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs index 73abd8fe..2c782f6c 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs @@ -1,7 +1,7 @@ -using System; +using Microsoft.AspNetCore.Mvc.ViewFeatures; +using System; using System.Collections.Generic; using System.Linq; -using GuardNet; using Xunit; using Xunit.Sdk; @@ -22,11 +22,18 @@ public class HttpAssertProvider /// Thrown when the contains null elements or has duplicate names. public HttpAssertProvider(IEnumerable> namedAssertions) { - Guard.NotNull(namedAssertions, nameof(namedAssertions), "Requires a series of named HTTP assertions to setup the HTTP assertion provider"); - Guard.For(() => namedAssertions.Any(item => item is null), - new ArgumentException("Requires a series of named HTTP assertions without any 'null' elements to setup the HTTP assertion provider", nameof(namedAssertions))); - Guard.For(() => namedAssertions.GroupBy(item => item.Item1).All(group => group.Count() != 1), - new ArgumentException("Requires a series of named HTTP assertions with unique names to setup the HTTP assertion provider", nameof(namedAssertions))); + if (namedAssertions is null) + { + throw new ArgumentNullException(nameof(namedAssertions), "Requires a series of named HTTP assertions to setup the HTTP assertion provider"); + } + if (namedAssertions.Any(item => item is null)) + { + throw new ArgumentException("Requires a series of named HTTP assertions without any 'null' elements to setup the HTTP assertion provider", nameof(namedAssertions); + } + if (namedAssertions.GroupBy(item => item.Item1).All(group => group.Count() != 1)) + { + throw new ArgumentException("Requires a series of named HTTP assertions with unique names to setup the HTTP assertion provider", nameof(namedAssertions)); + } _namedAssertions = namedAssertions.ToArray(); } @@ -39,7 +46,10 @@ public HttpAssertProvider(IEnumerable> namedAssertions /// Thrown when more than one was registered under the given . public HttpAssert GetAssertion(string name) { - Guard.NotNullOrWhitespace(name, nameof(name), "Requires a non-blank name to retrieve the HTTP assertion"); + if (string.IsNullOrWhiteSpace(name)) + { + throw new ArgumentException("Requires a non-blank name to retrieve the HTTP assertion", nameof(name)); + } return Assert.Single(_namedAssertions, item => item.Item1 == name).Item2; } } diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs index 90f6da5d..d57ac644 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs @@ -1,5 +1,4 @@ using System; -using GuardNet; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection.Extensions; @@ -22,9 +21,18 @@ public static class HttpAssertServiceCollectionExtensions /// Thrown when the is blank. public static IServiceCollection AddHttpAssert(this IServiceCollection services, string name, Action assertion) { - Guard.NotNull(services, nameof(services), "Requires a set of services to add the HTTP assertion to"); - Guard.NotNullOrWhitespace(name, nameof(name), "Requires a non-blank name to register the HTTP assertion"); - Guard.NotNull(assertion, nameof(assertion), "Requires an assertion function to verify the currently available HTTP context"); + if (services is null) + { + throw new ArgumentNullException(nameof(services), "Requires a set of services to add the HTTP assertion to"); + } + if (string.IsNullOrWhiteSpace(name)) + { + throw new ArgumentException("Requires a non-blank name to register the HTTP assertion", nameof(name)); + } + if (assertion is null) + { + throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify the currently available HTTP context"); + } services.TryAddSingleton(); return services.AddSingleton(Tuple.Create(name, HttpAssert.Create(assertion))); diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/TraceIdentifierMiddleware.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/TraceIdentifierMiddleware.cs index f3f3d17d..e93cd501 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/TraceIdentifierMiddleware.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/TraceIdentifierMiddleware.cs @@ -1,6 +1,5 @@ using System; using System.Threading.Tasks; -using GuardNet; using Microsoft.AspNetCore.Http; namespace Arcus.WebApi.Tests.Integration.Logging.Fixture @@ -20,11 +19,8 @@ public TraceIdentifierMiddleware( RequestDelegate next, TraceIdentifierOptions options) { - _next = next; - Guard.NotNull(next, nameof(next), "Requires a continuation delegate"); - Guard.NotNull(options, nameof(options), $"Requires a non-null '{nameof(TraceIdentifierOptions)}' options"); - - _options = options; + _next = next ?? throw new ArgumentNullException(nameof(next), "Requires a continuation delegate"); + _options = options ?? throw new ArgumentNullException(nameof(options), $"Requires a non-null '{nameof(TraceIdentifierOptions)}' options instance"); } /// Request handling method. diff --git a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/CertificateConfiguration.cs b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/CertificateConfiguration.cs index ba13080c..ab0850e9 100644 --- a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/CertificateConfiguration.cs +++ b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/CertificateConfiguration.cs @@ -1,7 +1,6 @@ using System; using System.Security.Cryptography.X509Certificates; using Arcus.WebApi.Tests.Integration.Fixture; -using GuardNet; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; @@ -21,9 +20,7 @@ internal class CertificateConfiguration : IStartupFilter /// When the is null. public CertificateConfiguration(X509Certificate2 clientCertificate) { - Guard.NotNull(clientCertificate, nameof(clientCertificate)); - - _clientCertificate = clientCertificate; + _clientCertificate = clientCertificate ?? throw new ArgumentNullException(nameof(clientCertificate)); } /// diff --git a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs index d79f14cc..7b4cd0c4 100644 --- a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs +++ b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs @@ -1,7 +1,6 @@ using System; using System.IO; using System.Security.Cryptography.X509Certificates; -using GuardNet; using Org.BouncyCastle.Asn1.X509; using Org.BouncyCastle.Crypto; using Org.BouncyCastle.Crypto.Generators; @@ -36,7 +35,10 @@ public static X509Certificate2 Create() /// When the is null. public static X509Certificate2 CreateWithSubject(string subjectName) { - Guard.NotNullOrWhitespace(subjectName, nameof(subjectName), "Subject name should not be blank"); + if (string.IsNullOrWhiteSpace(subjectName)) + { + throw new ArgumentException("Subject name should not be blank", nameof(subjectName)); + } return CreateWithIssuerAndSubjectName("TestCA", subjectName); } @@ -50,8 +52,14 @@ public static X509Certificate2 CreateWithSubject(string subjectName) /// When the is null. public static X509Certificate2 CreateWithIssuerAndSubjectName(string issuerName, string subjectName) { - Guard.NotNullOrWhitespace(subjectName, nameof(subjectName), "Subject name should not be blank"); - Guard.NotNullOrWhitespace(issuerName, nameof(issuerName), "Issuer name should not be blank"); + if (string.IsNullOrWhiteSpace(subjectName)) + { + throw new ArgumentException("Subject name should not be blank", nameof(subjectName)); + } + if (string.IsNullOrWhiteSpace(issuerName)) + { + throw new ArgumentException("Issuer name should not be blank", nameof(issuerName)); + } issuerName = issuerName.StartsWith("CN=") ? issuerName : "CN=" + issuerName; subjectName = subjectName.StartsWith("CN=") ? subjectName : "CN=" + subjectName; diff --git a/src/Arcus.WebApi.Tests.Runtimes.AzureFunction.Isolated/HttpTriggerFunction.cs b/src/Arcus.WebApi.Tests.Runtimes.AzureFunction.Isolated/HttpTriggerFunction.cs index ee26a47b..49cfee07 100644 --- a/src/Arcus.WebApi.Tests.Runtimes.AzureFunction.Isolated/HttpTriggerFunction.cs +++ b/src/Arcus.WebApi.Tests.Runtimes.AzureFunction.Isolated/HttpTriggerFunction.cs @@ -2,7 +2,6 @@ using Arcus.Observability.Correlation; using Arcus.WebApi.Logging.Core.Correlation; using Azure.Core.Serialization; -using GuardNet; using Microsoft.Azure.Functions.Worker; using Microsoft.Azure.Functions.Worker.Http; using Microsoft.Extensions.Logging; @@ -20,11 +19,8 @@ public HttpTriggerFunction( JsonObjectSerializer serializer, ILoggerFactory loggerFactory) { - Guard.NotNull(correlationAccessor, nameof(correlationAccessor)); - Guard.NotNull(serializer, nameof(serializer)); - - _correlationAccessor = correlationAccessor; - _serializer = serializer; + _correlationAccessor = correlationAccessor ?? throw new ArgumentNullException(nameof(correlationAccessor)); + _serializer = serializer ?? throw new ArgumentNullException(nameof(serializer)); _logger = loggerFactory.CreateLogger(); } diff --git a/src/Arcus.WebApi.Tests.Unit/Logging/Fixture/DefaultHttpMessageHandlerBuilder.cs b/src/Arcus.WebApi.Tests.Unit/Logging/Fixture/DefaultHttpMessageHandlerBuilder.cs index 54dc583e..ceba14ac 100644 --- a/src/Arcus.WebApi.Tests.Unit/Logging/Fixture/DefaultHttpMessageHandlerBuilder.cs +++ b/src/Arcus.WebApi.Tests.Unit/Logging/Fixture/DefaultHttpMessageHandlerBuilder.cs @@ -1,7 +1,6 @@ using System; using System.Collections.Generic; using System.Net.Http; -using GuardNet; using Microsoft.Extensions.Http; namespace Arcus.WebApi.Tests.Unit.Logging.Fixture @@ -18,8 +17,7 @@ public class DefaultHttpMessageHandlerBuilder : HttpMessageHandlerBuilder /// Thrown when the is null. public DefaultHttpMessageHandlerBuilder(IServiceProvider provider) { - Guard.NotNull(provider, nameof(provider), "Requires a service provider to retrieve the available application services when registering HTTP message handlers"); - Services = provider; + Services = provider ?? throw new ArgumentNullException(nameof(provider), "Requires a service provider to retrieve the available application services when registering HTTP message handlers"); } /// From 3925d90cf745851601dc85cbdea8e2726828ac46 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 31 Dec 2024 14:33:59 +0100 Subject: [PATCH 12/31] fix compilation error --- src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs index 3c10c42b..03ce6a5e 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs @@ -22,7 +22,7 @@ public RequestTrackingAttribute(Exclude filter) { if (!Enum.IsDefined(typeof(Exclude), filter) || filter is Exclude.None) { - throw new ArgumentOutOfRangeException(nameof(filter), $"Requires the exclusion filter to be within these bounds of the enumeration '{ExcludeFilterNames}'; 'None' is not allowed",); + throw new ArgumentOutOfRangeException(nameof(filter), $"Requires the exclusion filter to be within these bounds of the enumeration '{ExcludeFilterNames}'; 'None' is not allowed"); } Filter = filter; From e7370d3add22f9299d62a4d418bd496d8ece9f17 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 31 Dec 2024 14:46:28 +0100 Subject: [PATCH 13/31] fix compilation errors --- .../Certificates/CertificateAuthenticationValidator.cs | 1 - .../SharedAccessKey/Extensions/MvcOptionsExtensions.cs | 2 +- .../Fixture/TestApiServerOptions.cs | 1 + .../Logging/Fixture/HttpAssertProvider.cs | 2 +- 4 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs index 547293fd..270be901 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs @@ -6,7 +6,6 @@ using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Logging.Abstractions; -using Newtonsoft.Json.Serialization; namespace Arcus.WebApi.Security.Authentication.Certificates { diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs index d545c7b7..6038a57c 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs @@ -31,7 +31,7 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnHeader(this Mvc } if (string.IsNullOrWhiteSpace(headerName)) { - throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName); + throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName)); } return AddSharedAccessKeyAuthenticationFilterOnHeader(options, headerName, secretName, configureOptions: null); diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs index 3c1e8d75..1844785d 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs @@ -126,6 +126,7 @@ public TestApiServerOptions ConfigureAppConfiguration(Action diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs index 2c782f6c..5fdb547c 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs @@ -28,7 +28,7 @@ public HttpAssertProvider(IEnumerable> namedAssertions } if (namedAssertions.Any(item => item is null)) { - throw new ArgumentException("Requires a series of named HTTP assertions without any 'null' elements to setup the HTTP assertion provider", nameof(namedAssertions); + throw new ArgumentException("Requires a series of named HTTP assertions without any 'null' elements to setup the HTTP assertion provider", nameof(namedAssertions)); } if (namedAssertions.GroupBy(item => item.Item1).All(group => group.Count() != 1)) { From 8f1be5f217d626e4356bb9dedc4c23768557400e Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 31 Dec 2024 15:56:31 +0100 Subject: [PATCH 14/31] use StartsWith(char) instead of StarsWith(string) --- .../Fixture/HttpRequestBuilder.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs index ce7e71f0..9378e903 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs @@ -160,7 +160,7 @@ internal HttpRequestMessage Build(string baseRoute) } string path = _path; - if (path.StartsWith("/")) + if (path.StartsWith('/')) { path = path.TrimStart('/'); } From 14eab7881d2c86a0b4c87761497663e727fbee7a Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 09:00:48 +0100 Subject: [PATCH 15/31] Correct Hosting projects --- .../Formatting/AzureFunctionsJsonFormattingMiddleware.cs | 1 + .../Formatting/Extensions/IServiceCollectionExtensions.cs | 1 + .../Formatting/Extensions/MvcOptionsExtensions.cs | 1 + 3 files changed, 3 insertions(+) diff --git a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/AzureFunctionsJsonFormattingMiddleware.cs b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/AzureFunctionsJsonFormattingMiddleware.cs index 4610d8cd..7d481c2d 100644 --- a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/AzureFunctionsJsonFormattingMiddleware.cs +++ b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/AzureFunctionsJsonFormattingMiddleware.cs @@ -44,6 +44,7 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next private static async Task DetermineHttpRequestAsync(FunctionContext context) { HttpRequestData request = await context.GetHttpRequestDataAsync(); + if (request is null) { throw new InvalidOperationException( diff --git a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs index 66dd8edb..71dc1cf8 100644 --- a/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Hosting.AzureFunctions/Formatting/Extensions/IServiceCollectionExtensions.cs @@ -27,6 +27,7 @@ public static IServiceCollection ConfigureJsonFormatting( { throw new ArgumentNullException(nameof(builder), "Requires an Azure Functions application builder instance to add the JSON serializer to the application services"); } + if (configureOptions is null) { throw new ArgumentNullException(nameof(configureOptions), "Requires a function to configure the JSON serialization options to add the JSON serializer to the application services"); diff --git a/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs index 464672c2..ed08c869 100644 --- a/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Hosting/Formatting/Extensions/MvcOptionsExtensions.cs @@ -52,6 +52,7 @@ public static MvcOptions ConfigureJsonFormatting(this MvcOptions options, Action { throw new ArgumentNullException(nameof(options), "Requires MVC options to configure the JSON formatters"); } + if (configureOptions is null) { throw new ArgumentNullException(nameof(configureOptions), "Requires a function to configure the JSON formatters in the MVC options"); From 26520893381dff33685fad8fdae2251906ba36e3 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 09:10:08 +0100 Subject: [PATCH 16/31] Fix comments on .Logging --- .../Correlation/CorrelationMiddleware.cs | 3 +++ .../Extensions/IServiceCollectionExtensions.cs | 9 +-------- src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs | 2 ++ src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs | 4 ++++ src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs | 3 +++ 5 files changed, 13 insertions(+), 8 deletions(-) diff --git a/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs b/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs index 2a4caea3..645f30e2 100644 --- a/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs +++ b/src/Arcus.WebApi.Logging/Correlation/CorrelationMiddleware.cs @@ -44,14 +44,17 @@ public async Task Invoke(HttpContext httpContext, HttpCorrelation service) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context"); } + if (service is null) { throw new ArgumentNullException(nameof(service), "Requires the HTTP correlation service"); } + if (httpContext.Response is null) { throw new ArgumentException("Requires a 'Response'", nameof(httpContext)); } + if (httpContext.Response.Headers is null) { throw new ArgumentException("Requires a 'Response' object with headers", nameof(httpContext)); diff --git a/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs index 2918c3de..65e26d94 100644 --- a/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Logging/Extensions/IServiceCollectionExtensions.cs @@ -19,14 +19,7 @@ public static class IServiceCollectionExtensions /// /// The services collection containing the dependency injection services. public static IServiceCollection AddHttpCorrelation(this IServiceCollection services) - { - if (services == null) - { - throw new ArgumentNullException(nameof(services), "Requires a services collection to add the HTTP correlation services"); - } - - return AddHttpCorrelation(services, configureOptions: (HttpCorrelationInfoOptions options) => { }); - } + => AddHttpCorrelation(services, configureOptions: (HttpCorrelationInfoOptions options) => { }); /// /// Adds operation and transaction correlation to the application. diff --git a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs index 03ce6a5e..6fc5b192 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingAttribute.cs @@ -68,10 +68,12 @@ public RequestTrackingAttribute(int minimumStatusCode, int maximumStatusCode) { throw new ArgumentOutOfRangeException(nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold to not be less than 100"); } + if (maximumStatusCode > 599) { throw new ArgumentOutOfRangeException(nameof(maximumStatusCode), "Requires the maximum HTTP status code threshold to not be greater than 599"); } + if (minimumStatusCode >= maximumStatusCode) { throw new ArgumentOutOfRangeException(nameof(minimumStatusCode), "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); diff --git a/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs b/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs index f7e94052..f8813ee4 100644 --- a/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging/RequestTrackingMiddleware.cs @@ -37,10 +37,12 @@ public RequestTrackingMiddleware( { throw new ArgumentNullException(nameof(options), "Requires a set of options to control the behavior of the HTTP tracking middleware"); } + if (next is null) { throw new ArgumentNullException(nameof(next), "Requires a function pipeline to delegate the remainder of the request processing"); } + if (logger is null) { throw new ArgumentNullException(nameof(logger), "Requires a logger instance to write telemetry tracking during the request processing"); @@ -62,10 +64,12 @@ public async Task Invoke(HttpContext httpContext) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context instance to track the incoming request and outgoing response"); } + if (httpContext.Request is null) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP request in the context to track the request"); } + if (httpContext.Response is null) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP response in the context to track the request"); diff --git a/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs b/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs index da8d62f2..4a02d363 100644 --- a/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging/VersionTrackingMiddleware.cs @@ -38,10 +38,12 @@ public VersionTrackingMiddleware( { throw new ArgumentNullException(nameof(appVersion), "Requires an instance to retrieve the current application version to add the version to the response"); } + if (next is null) { throw new ArgumentNullException(nameof(next), "Requires a continuation delegate to move towards the next functionality in the request pipeline"); } + if (options is null) { throw new ArgumentNullException(nameof(options), "Requires version tracking options to specify how the application version should be tracked in the response"); @@ -65,6 +67,7 @@ public async Task Invoke(HttpContext context) { throw new ArgumentNullException(nameof(context), "Requires a HTTP context to add the application version to the response"); } + if (context.Response is null) { throw new ArgumentException("Requires a HTTP context with a response to add the application version", nameof(context)); From c0acdebf73c29fd8abd24d663c7f369398b01262 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 09:19:36 +0100 Subject: [PATCH 17/31] apply comments to Logging.AzureFunctions --- .../AzureFunctionsRequestTrackingMiddleware.cs | 8 ++------ .../AzureFunctionsCorrelationMiddleware.cs | 1 + .../Correlation/AzureFunctionsHttpCorrelation.cs | 2 ++ .../AzureFunctionsInProcessHttpCorrelation.cs | 3 +++ ...IFunctionsWorkerApplicationBuilderExtensions.cs | 14 +++++--------- .../Extensions/IServiceCollectionExtensions.cs | 9 +-------- .../FunctionContextMiddleware.cs | 1 + 7 files changed, 15 insertions(+), 23 deletions(-) diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs index b10be58d..5ad6631e 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs @@ -37,6 +37,7 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next ILogger logger = context.GetLogger(); HttpRequestData request = await context.GetHttpRequestDataAsync(); + if (request is null || IsRequestPathOmitted(PathString.FromUriComponent(request.Url), logger)) { await next(context); @@ -72,13 +73,8 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next private static async Task EnableHttpRequestBufferingAsync(FunctionContext context) { - BindingMetadata bindingMetadata = context.FunctionDefinition.InputBindings.Values.FirstOrDefault(a => a.Type == "httpTrigger"); - if (bindingMetadata is null) - { - throw new InvalidOperationException( + BindingMetadata bindingMetadata = context.FunctionDefinition.InputBindings.Values.FirstOrDefault(a => a.Type == "httpTrigger") ?? throw new InvalidOperationException( "Cannot enable HTTP request body buffering because it cannot find the Azure Functions' HTTP trigger input binding representing the HTTP request"); - } - InputBindingData bindingData = await context.BindInputAsync(bindingMetadata); bindingData.Value = new BufferedHttpRequestData(bindingData.Value); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs index 76a2ff21..272965cd 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsCorrelationMiddleware.cs @@ -28,6 +28,7 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next { throw new ArgumentNullException(nameof(context), "Requires a function context instance of the current Azure Function invocation to HTTP correlate the HTTP request"); } + if (next is null) { throw new ArgumentNullException(nameof(next), "Requires a 'next' function to chain this HTTP correlation middleware to the next action in the HTTP request pipeline"); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs index 2030d8d0..f2af1692 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsHttpCorrelation.cs @@ -134,10 +134,12 @@ protected override void SetHttpResponseHeader(HttpResponseData response, string { throw new ArgumentNullException(nameof(response), "Requires a HTTP response to set the HTTP correlation headers"); } + if (string.IsNullOrWhiteSpace(headerName)) { throw new ArgumentException("Requires a non-blank HTTP correlation header name to set the HTTP correlation header in the HTTP request", nameof(headerName)); } + if (string.IsNullOrWhiteSpace(headerValue)) { throw new ArgumentException("Requires a non-blank HTTP correlation header value to set the HTTP correlation header in the HTTP request", nameof(headerValue)); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs index f57e32a9..a3b0d0ac 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Correlation/AzureFunctionsInProcessHttpCorrelation.cs @@ -36,10 +36,12 @@ public AzureFunctionsInProcessHttpCorrelation( { throw new ArgumentNullException(nameof(options), "Requires a set of HTTP correlation options to determine where the correlation information should be added to the HTTP response headers"); } + if (correlationInfoAccessor is null) { throw new ArgumentNullException(nameof(correlationInfoAccessor), "Requires a HTTP correlation accessor to retrieve the current correlation information"); } + if (logger is null) { throw new ArgumentNullException(nameof(logger), "Requires a logging instance to write diagnostic trace messages while adding the correlation information to the HTTP response headers"); @@ -69,6 +71,7 @@ public void AddCorrelationResponseHeaders(HttpContext httpContext) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP context to add the correlation information to the response headers"); } + if (httpContext.Response is null) { throw new ArgumentNullException(nameof(httpContext), "Requires a HTTP response in the HTTP context to add the correlation information to the response headers"); diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs index 5e6cc5a9..eb2b02ce 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IFunctionsWorkerApplicationBuilderExtensions.cs @@ -44,14 +44,7 @@ public static IFunctionsWorkerApplicationBuilder UseFunctionContext( /// Thrown when the is null. public static IFunctionsWorkerApplicationBuilder UseHttpCorrelation( this IFunctionsWorkerApplicationBuilder builder) - { - if (builder is null) - { - throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP correlation middleware"); - } - - return UseHttpCorrelation(builder, options => { }); - } + => UseHttpCorrelation(builder, options => { }); /// /// Adds a middleware component that exposes the in a scoped service . @@ -110,7 +103,8 @@ public static IFunctionsWorkerApplicationBuilder UseExceptionHandling(this IFunc { throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP exception handling middleware"); } - return builder.UseMiddleware(); + + return builder.UseMiddleware(); } /// @@ -126,6 +120,7 @@ public static IFunctionsWorkerApplicationBuilder UseExceptionHandling(); } @@ -141,6 +136,7 @@ public static IFunctionsWorkerApplicationBuilder UseRequestTracking( { throw new ArgumentNullException(nameof(builder), "Requires a function worker builder instance to add the HTTP request tracking middleware"); } + return UseRequestTracking(builder, configureOptions: null); } diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs index 8a11d6bc..eca0ce13 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/Extensions/IServiceCollectionExtensions.cs @@ -23,14 +23,7 @@ public static class IServiceCollectionExtensions /// The functions host builder containing the dependency injection services. /// Thrown when the is null. public static IServiceCollection AddHttpCorrelation(this IFunctionsHostBuilder builder) - { - if (builder is null) - { - throw new ArgumentNullException(nameof(builder), "Requires a function host builder instance to add the HTTP correlation services"); - } - - return AddHttpCorrelation(builder, options => { }); - } + => AddHttpCorrelation(builder, options => { }); /// /// Adds operation and transaction correlation to the application. diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs index 5f3ab2df..5db35a18 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/FunctionContextMiddleware.cs @@ -34,6 +34,7 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next { throw new ArgumentNullException(nameof(context), "Requires a function context instance to assign the context to the function context accessor"); } + if (next is null) { throw new ArgumentNullException(nameof(next), "Requires a 'next' function to chain this middleware to the next action in the HTTP request pipeline"); From a6e0e09290cf5cae262e0f56158081abd59faa0f Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 10:55:02 +0100 Subject: [PATCH 18/31] Fixed PR comments in Logging.Core --- .../Correlation/HttpCorrelation.cs | 10 +++-- .../HttpCorrelationClientOptions.cs | 13 +++--- .../HttpCorrelationInfoAccessor.cs | 4 +- .../HttpCorrelationInfoOperationOptions.cs | 4 +- .../HttpCorrelationInfoTransactionOptions.cs | 4 +- .../HttpCorrelationMessageHandler.cs | 6 +-- .../Correlation/HttpCorrelationResult.cs | 11 ++--- .../Correlation/HttpCorrelationTemplate.cs | 11 ++--- .../Extensions/HttpClientExtensions.cs | 44 ++++++++++++------- .../HttpCorrelationEnricherExtensions.cs | 5 ++- .../Extensions/IDictionaryExtensions.cs | 5 ++- .../Extensions/IHeaderDictionaryExtensions.cs | 4 +- .../IHttpClientBuilderExtensions.cs | 5 ++- .../RequestTracking/RequestTrackingOptions.cs | 4 +- .../RequestTrackingTemplate.cs | 9 ++-- .../RequestTracking/StatusCodeRange.cs | 13 +++--- 16 files changed, 89 insertions(+), 63 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs index a66d37b2..3ebe2336 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs @@ -40,7 +40,7 @@ public HttpCorrelation( { if (options is null) { - throw new ArgumentNullException(paramName: nameof(options), message: "Requires a value in the set of options to configure the correlation process"); + throw new ArgumentNullException(nameof(options), "Requires a value in the set of options to configure the correlation process"); } _httpContextAccessor = httpContextAccessor ?? throw new ArgumentNullException(paramName: nameof(httpContextAccessor), message: "Requires a HTTP context accessor to get the current HTTP context"); @@ -68,8 +68,9 @@ public void SetCorrelationInfo(CorrelationInfo correlationInfo) { if (_correlationInfoAccessor is null) { - throw new ArgumentNullException(paramName: nameof(correlationInfo), message: "Requires a correlation info instance to set and retrieve the correlation information"); + throw new ArgumentNullException(nameof(correlationInfo), "Requires a correlation info instance to set and retrieve the correlation information"); } + _correlationInfoAccessor.SetCorrelationInfo(correlationInfo); } @@ -85,11 +86,12 @@ public HttpCorrelationResult CorrelateHttpRequest() if (httpContext.Response is null) { - throw new ArgumentException(message: "Requires a 'Response'", paramName: nameof(httpContext.Response)); + throw new ArgumentException("Requires a 'Response'", nameof(httpContext.Response)); } + if (httpContext.Response.Headers is null) { - throw new ArgumentException(message: "Requires a 'Response' object with headers", paramName: nameof(httpContext.Response.Headers)); + throw new ArgumentException("Requires a 'Response' object with headers", nameof(httpContext.Response.Headers)); } HttpCorrelationResult result = TrySettingCorrelationFromRequest(httpContext.Request, httpContext.TraceIdentifier); diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index 739a4516..1cc64ffd 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -26,8 +26,9 @@ public Func GenerateDependencyId { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), message: "Requires a function to generate the dependency ID used when tracking HTTP dependencies"); + throw new ArgumentNullException(nameof(value), "Requires a function to generate the dependency ID used when tracking HTTP dependencies"); } + _generateDependencyId = value; } } @@ -43,8 +44,9 @@ public string UpstreamServiceHeaderName { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), message: "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); + throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); } + _upstreamServiceHeaderName = value; } } @@ -60,8 +62,9 @@ public string TransactionIdHeaderName { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), message: "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); + throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); } + _transactionIdHeaderName = value; } } @@ -78,9 +81,9 @@ public string TransactionIdHeaderName /// Thrown when the is null. public void AddTelemetryContext(Dictionary telemetryContext) { - if (telemetryContext == null) + if (telemetryContext is null) { - throw new ArgumentNullException(paramName: nameof(telemetryContext), message: "Requires a telemetry context dictionary to add to the HTTP dependency tracking"); + throw new ArgumentNullException(nameof(telemetryContext), "Requires a telemetry context dictionary to add to the HTTP dependency tracking"); } foreach (KeyValuePair item in telemetryContext) { diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs index d4fbdd9b..f68acb79 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoAccessor.cs @@ -22,7 +22,7 @@ public HttpCorrelationInfoAccessor(IHttpContextAccessor contextAccessor) { if (contextAccessor is null) { - throw new ArgumentNullException(paramName: nameof(contextAccessor)); + throw new ArgumentNullException(nameof(contextAccessor)); } _httpContextAccessor = contextAccessor; @@ -46,7 +46,7 @@ public void SetCorrelationInfo(CorrelationInfo correlationInfo) { if (correlationInfo is null) { - throw new ArgumentNullException(paramName: nameof(correlationInfo)); + throw new ArgumentNullException(nameof(correlationInfo)); } _httpContextAccessor.HttpContext?.Features?.Set(correlationInfo); diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs index 5b4af2f9..efba65fe 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoOperationOptions.cs @@ -28,7 +28,7 @@ public string HeaderName { if (string.IsNullOrWhiteSpace(value)) { - throw new ArgumentException(message: "Correlation operation header cannot be blank", paramName: nameof(value)); + throw new ArgumentException("Correlation operation header cannot be blank", nameof(value)); } _headerName = value; @@ -48,7 +48,7 @@ public Func GenerateId { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), message: "Correlation function to generate an operation ID cannot be 'null'"); + throw new ArgumentNullException(nameof(value), "Correlation function to generate an operation ID cannot be 'null'"); } _generateId = value; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs index 2113e972..6b56078a 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationInfoTransactionOptions.cs @@ -44,7 +44,7 @@ public string HeaderName { if (string.IsNullOrWhiteSpace(value)) { - throw new ArgumentException(message: "Header name cannot be blank", paramName: nameof(value)); + throw new ArgumentException("Header name cannot be blank", nameof(value)); } _headerName = value; @@ -67,7 +67,7 @@ public Func GenerateId { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), message: "Correlation function to generate an transaction ID cannot be 'null'"); + throw new ArgumentNullException(nameof(value), "Correlation function to generate an transaction ID cannot be 'null'"); } _generateId = value; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs index 3bb72cc7..13bb04f6 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationMessageHandler.cs @@ -32,9 +32,9 @@ public HttpCorrelationMessageHandler( HttpCorrelationClientOptions options, ILogger logger) { - _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a HTTP context accessor to retrieve the current HTTP correlation"); - _options = options ?? throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of additional user-configurable options to influence the HTTP dependency tracking"); - _logger = logger ?? throw new ArgumentNullException(paramName: nameof(logger), message: "Requires a logger instance to write the HTTP dependency telemetry"); + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(nameof(correlationInfoAccessor), "Requires a HTTP context accessor to retrieve the current HTTP correlation"); + _options = options ?? throw new ArgumentNullException(nameof(options), "Requires a set of additional user-configurable options to influence the HTTP dependency tracking"); + _logger = logger ?? throw new ArgumentNullException(nameof(logger), "Requires a logger instance to write the HTTP dependency telemetry"); } /// diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs index 74f0992b..b3a0df12 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs @@ -22,6 +22,7 @@ private HttpCorrelationResult(bool isSuccess, string requestId, string errorMess { throw new ArgumentException("Cannot create a successful HTTP correlation result with an error user message", nameof(errorMessage)); } + if (!isSuccess && requestId != null) { throw new ArgumentException("Cannot create a failed HTTP correlation result with a request ID", nameof(requestId)); @@ -89,7 +90,7 @@ public static HttpCorrelationResult Failure(string errorMessage) { if (string.IsNullOrWhiteSpace(errorMessage)) { - throw new ArgumentException(paramName: nameof(errorMessage), message: "Requires an error user message that describes why the HTTP correlation process failed on the current HTTP request"); + throw new ArgumentException("Requires an error user message that describes why the HTTP correlation process failed on the current HTTP request", nameof(errorMessage)); } return new HttpCorrelationResult(isSuccess: false, requestId: null, errorMessage); @@ -106,11 +107,11 @@ public static HttpCorrelationResult Success(TelemetryClient client, string trans { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); + throw new ArgumentNullException(nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); } if (string.IsNullOrWhiteSpace(transactionId)) { - throw new ArgumentException(message: "Requires a non-blank transaction ID for the pending HTTP correlation", paramName: nameof(transactionId)); + throw new ArgumentException("Requires a non-blank transaction ID for the pending HTTP correlation", nameof(transactionId)); } return Success(client, transactionId, operationParentId: null, traceParent: null); @@ -131,11 +132,11 @@ public static HttpCorrelationResult Success(TelemetryClient client, string trans { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); + throw new ArgumentNullException(nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); } if (string.IsNullOrWhiteSpace(transactionId)) { - throw new ArgumentException(message: "Requires a non-blank transaction ID for the pending HTTP correlation", paramName: nameof(transactionId)); + throw new ArgumentException("Requires a non-blank transaction ID for the pending HTTP correlation", nameof(transactionId)); } var telemetry = new RequestTelemetry(); diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs index 19c8faf9..61d199be 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs @@ -37,8 +37,8 @@ protected HttpCorrelationTemplate( IHttpCorrelationInfoAccessor correlationInfoAccessor, ILogger logger) { - _options = options ?? throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of options to configure the correlation process"); ; - _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a correlation info instance to set and retrieve the correlation information"); + _options = options ?? throw new ArgumentNullException(nameof(options), "Requires a set of options to configure the correlation process"); + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(nameof(correlationInfoAccessor), "Requires a correlation info instance to set and retrieve the correlation information"); Logger = logger ?? NullLogger.Instance; } @@ -66,7 +66,7 @@ public HttpCorrelationResult TrySettingCorrelationFromRequest(THttpRequest reque { if (request is null) { - throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to determine the HTTP correlation of the application"); + throw new ArgumentNullException(nameof(request), "Requires a HTTP request to determine the HTTP correlation of the application"); } IHeaderDictionary requestHeaders = GetRequestHeaders(request); @@ -406,11 +406,12 @@ public void SetCorrelationHeadersInResponse(THttpResponse response, HttpCorrelat { if (response is null) { - throw new ArgumentNullException(paramName: nameof(response), message: "Requires a HTTP response to set the HTTP correlation headers"); + throw new ArgumentNullException(nameof(response), "Requires a HTTP response to set the HTTP correlation headers"); } + if (result is null) { - throw new ArgumentNullException(paramName: nameof(result), message: "Requires a HTTP correlation result to determine to set the HTTP correlation headers in the HTTP request"); + throw new ArgumentNullException(nameof(result), "Requires a HTTP correlation result to determine to set the HTTP correlation headers in the HTTP request"); } string requestId = result.RequestId; diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs index 27bf3289..eac1d8c5 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs @@ -38,19 +38,22 @@ public static async Task SendAsync( { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); } + if (request is null) { - throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); } + if (correlationAccessor is null) { - throw new ArgumentNullException(paramName: nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); + throw new ArgumentNullException(nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); } + if (logger is null) { - throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); } return await SendAsync(client, request, correlationAccessor, logger, configureOptions: null); @@ -83,19 +86,22 @@ public static async Task SendAsync( { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); } + if (request is null) { - throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); } + if (correlationAccessor is null) { - throw new ArgumentNullException(paramName: nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); + throw new ArgumentNullException(nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); } + if (logger is null) { - throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); } CorrelationInfo correlation = correlationAccessor.GetCorrelationInfo(); @@ -127,19 +133,22 @@ public static async Task SendAsync( { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); } + if (request is null) { - throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); } + if (correlationInfo is null) { - throw new ArgumentNullException(paramName: nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); + throw new ArgumentNullException(nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); } + if (logger is null) { - throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); } return await SendAsync(client, request, correlationInfo, logger, configureOptions: null); @@ -172,19 +181,22 @@ public static async Task SendAsync( { if (client is null) { - throw new ArgumentNullException(paramName: nameof(client), message: "Requires a HTTP client to track the HTTP request with HTTP correlation"); + throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); } + if (request is null) { - throw new ArgumentNullException(paramName: nameof(request), message: "Requires a HTTP request to enrich with HTTP correlation"); + throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); } + if (correlationInfo is null) { - throw new ArgumentNullException(paramName: nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); + throw new ArgumentNullException(nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); } + if (logger is null) { - throw new ArgumentNullException(paramName: nameof(logger), "Requires a logger instance to track the correlated HTTP request"); + throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); } var options = new HttpCorrelationClientOptions(); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs index 3cf1fd02..93a389f0 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/HttpCorrelationEnricherExtensions.cs @@ -28,11 +28,12 @@ public static LoggerConfiguration WithHttpCorrelationInfo(this LoggerEnrichmentC { if (enrichmentConfiguration is null) { - throw new ArgumentNullException(paramName: nameof(enrichmentConfiguration), message: "Requires a Serilog logger enrichment configuration to register the HTTP correlation as enrichment"); + throw new ArgumentNullException(nameof(enrichmentConfiguration), "Requires a Serilog logger enrichment configuration to register the HTTP correlation as enrichment"); } + if (serviceProvider is null) { - throw new ArgumentNullException(paramName: nameof(serviceProvider), message: "Requires a service provider to retrieve the HTTP correlation from the registered services when enriching the Serilog with the HTTP correlation"); + throw new ArgumentNullException(nameof(serviceProvider), "Requires a service provider to retrieve the HTTP correlation from the registered services when enriching the Serilog with the HTTP correlation"); } var correlationInfoAccessor = serviceProvider.GetService(); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs index c997c674..137fd248 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/IDictionaryExtensions.cs @@ -23,11 +23,12 @@ public static IDictionary Where(this IDictionary diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs index 65f8f0a4..39811dd6 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs @@ -34,7 +34,7 @@ public int? RequestBodyBufferSize { if (value < 0) { - throw new ArgumentOutOfRangeException(paramName: nameof(value), message: "Requires a request body buffer size greater than zero"); + throw new ArgumentOutOfRangeException(nameof(value), "Requires a request body buffer size greater than zero"); } _requestBodyBufferSize = value; } @@ -56,7 +56,7 @@ public int? ResponseBodyBufferSize { if (value < 0) { - throw new ArgumentOutOfRangeException(paramName: nameof(value), message: "Requires a response body buffer size greater than zero"); + throw new ArgumentOutOfRangeException(nameof(value), "Requires a response body buffer size greater than zero"); } _responseBodyBufferSize = value; } diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs index 28490da3..6b6d5a3d 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs @@ -26,7 +26,7 @@ protected RequestTrackingTemplate(RequestTrackingOptions options) { if (options is null) { - throw new ArgumentNullException(paramName: nameof(options), message: "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); + throw new ArgumentNullException(nameof(options), "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); } Options = options; } @@ -122,12 +122,12 @@ protected Dictionary CreateTelemetryContext(string requestBody, IDictionary headers = GetSanitizedRequestHeaders(requestHeaders, logger); Dictionary telemetryContext = headers.ToDictionary(header => header.Key, header => string.Join(",", header.Value)); - if (string.IsNullOrWhiteSpace(requestBody) == false) + if (!string.IsNullOrWhiteSpace(requestBody)) { telemetryContext.Add("RequestBody", requestBody); } - if (string.IsNullOrWhiteSpace(responseBody) == false) + if (!string.IsNullOrWhiteSpace(responseBody)) { telemetryContext.Add("ResponseBody", responseBody); } @@ -186,8 +186,9 @@ protected async Task GetBodyAsync(Stream body, int? maxLength, string ta { if (body is null) { - throw new ArgumentNullException(paramName: nameof(body), message: $"Requires a streamed body to read the string representation of the {targetName}"); + throw new ArgumentNullException(nameof(body), $"Requires a streamed body to read the string representation of the {targetName}"); } + logger = logger ?? NullLogger.Instance; logger.LogTrace("Prepare for {Target} body to be tracked...", targetName); diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs index c0f7fc4a..eb688caa 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/StatusCodeRange.cs @@ -31,15 +31,17 @@ public StatusCodeRange(int minimum, int maximum) { if (minimum < 100) { - throw new ArgumentOutOfRangeException(paramName: nameof(minimum), message: "Requires the minimum HTTP status code threshold to not be less than 100"); + throw new ArgumentOutOfRangeException(nameof(minimum), "Requires the minimum HTTP status code threshold to not be less than 100"); } + if (maximum > 599) { - throw new ArgumentOutOfRangeException(paramName: nameof(maximum), message: "Requires the maximum HTTP status code threshold to not be greater than 599"); + throw new ArgumentOutOfRangeException(nameof(maximum), "Requires the maximum HTTP status code threshold to not be greater than 599"); } + if (minimum > maximum) { - throw new ArgumentOutOfRangeException(paramName: nameof(minimum), message: "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); + throw new ArgumentOutOfRangeException(nameof(minimum), "Requires the minimum HTTP status code threshold to be less than the maximum HTTP status code threshold"); } Minimum = minimum; @@ -68,11 +70,12 @@ public bool IsWithinRange(int statusCode) { if (statusCode < 100) { - throw new ArgumentOutOfRangeException(paramName: nameof(statusCode), message: "Requires the response HTTP status code to not be less than 100"); + throw new ArgumentOutOfRangeException(nameof(statusCode), "Requires the response HTTP status code to not be less than 100"); } + if (statusCode > 599) { - throw new ArgumentOutOfRangeException(paramName: nameof(statusCode), message: "Requires the response HTTP status code to not be greater than 599"); + throw new ArgumentOutOfRangeException(nameof(statusCode), "Requires the response HTTP status code to not be greater than 599"); } return Minimum <= statusCode && statusCode <= Maximum; From 1162dd1efc4739760f2cec54f7287569bd3219e1 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 11:03:23 +0100 Subject: [PATCH 19/31] Fix comments on OpenApi.Extensions --- .../CertificateAuthenticationOperationFilter.cs | 4 ++-- .../OAuthAuthorizeOperationFilter.cs | 7 ++++--- .../SharedAccessKeyAuthenticationOperationFilter.cs | 7 ++++--- 3 files changed, 10 insertions(+), 8 deletions(-) diff --git a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs index 010d8e69..670b6e8d 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs @@ -43,12 +43,12 @@ public CertificateAuthenticationOperationFilter( #endif ) { - _securitySchemeName = securitySchemeName ?? throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Certificate security scheme"); + _securitySchemeName = securitySchemeName ?? throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Certificate security scheme"); #if !NETSTANDARD2_1 if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType)) { - throw new ArgumentException(paramName: nameof(securitySchemeType), message: "Requires a security scheme type for the Certificate authentication that is within the bounds of the enumeration"); + throw new ArgumentException("Requires a security scheme type for the Certificate authentication that is within the bounds of the enumeration", nameof(securitySchemeType)); } _securitySchemeType = securitySchemeType; diff --git a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs index ab057270..d69b6422 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs @@ -32,15 +32,16 @@ public OAuthAuthorizeOperationFilter(IEnumerable scopes, string security { if (scopes.Any(String.IsNullOrWhiteSpace)) { - throw new ArgumentException(message: "Requires a list of non-blank API scopes", paramName: nameof(scopes)); + throw new ArgumentException("Requires a list of non-blank API scopes", nameof(scopes)); } + if (string.IsNullOrWhiteSpace(securitySchemaName)) { - throw new ArgumentNullException(paramName: nameof(securitySchemaName), message: "Requires a name for the OAuth2 security scheme"); + throw new ArgumentNullException(nameof(securitySchemaName), "Requires a name for the OAuth2 security scheme"); } _securitySchemaName = securitySchemaName; - _scopes = scopes ?? throw new ArgumentNullException(paramName: nameof(scopes), message: "Requires a list of API scopes"); + _scopes = scopes ?? throw new ArgumentNullException(nameof(scopes), "Requires a list of API scopes"); } /// diff --git a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs index 1c9174f2..6f7bc955 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs @@ -36,11 +36,12 @@ public SharedAccessKeyAuthenticationOperationFilter( { if (string.IsNullOrWhiteSpace(securitySchemeName)) { - throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Shared Access Key security scheme"); + throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); } + if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType)) { - throw new ArgumentException(message: "Requires a security scheme type for the Shared Access Key authentication that is within the bounds of the enumeration", paramName: nameof(securitySchemeType)); + throw new ArgumentException("Requires a security scheme type for the Shared Access Key authentication that is within the bounds of the enumeration", nameof(securitySchemeType)); } _securitySchemeName = securitySchemeName; @@ -55,7 +56,7 @@ public SharedAccessKeyAuthenticationOperationFilter(string securitySchemeName) { if (string.IsNullOrWhiteSpace(securitySchemeName)) { - throw new ArgumentNullException(paramName: nameof(securitySchemeName), message: "Requires a name for the Shared Access Key security scheme"); + throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); } _securitySchemeName = securitySchemeName; From 60fd60aa618777c0418e90c1b3cf62c10af0b598 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 11:51:23 +0100 Subject: [PATCH 20/31] Fix comments .Security --- .../CertificateAuthenticationConfig.cs | 2 + .../CertificateAuthenticationConfigBuilder.cs | 67 ++----------------- .../CertificateAuthenticationFilter.cs | 3 + .../ConfigurationValidationLocation.cs | 8 +-- .../IServiceCollectionExtensions.cs | 1 + .../Extensions/MvcOptionsExtensions.cs | 23 +------ .../SecretProviderValidationLocation.cs | 1 + .../Extensions/MvcOptionsExtensions.cs | 32 ++------- .../SharedAccessKeyAuthenticationAttribute.cs | 3 +- .../SharedAccessKeyAuthenticationFilter.cs | 26 +++---- .../Extensions/MvcOptionsExtensions.cs | 33 ++------- .../Authorization/Jwt/JwtTokenReader.cs | 10 ++- .../JwtTokenAuthorizationFilter .cs | 4 ++ .../JwtTokenAuthorizationOptions.cs | 4 ++ 14 files changed, 55 insertions(+), 162 deletions(-) diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs index 67697747..173f9e3e 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfig.cs @@ -30,6 +30,7 @@ internal CertificateAuthenticationConfig( { throw new ArgumentNullException(nameof(locationAndKeyByRequirement), "Location and key by certificate requirement dictionary cannot be 'null'"); } + if (locationAndKeyByRequirement.Any(keyValue => keyValue.Value.location is null || keyValue.Value.configuredKey is null)) { throw new ArgumentException("All locations and configured keys by certificate requirement cannot be 'null'"); @@ -52,6 +53,7 @@ internal async Task> GetAllExpect { throw new ArgumentNullException(nameof(services), "Request services cannot be 'null'"); } + if (logger is null) { throw new ArgumentNullException(nameof(logger), "Logger cannot be 'null'"); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs index 778c56bc..139f878a 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationConfigBuilder.cs @@ -27,14 +27,7 @@ public CertificateAuthenticationConfigBuilder() /// The configured key that the requires to retrieve the expected subject name. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithSubject(X509ValidationLocation location, string configuredKey) - { - if (string.IsNullOrWhiteSpace(configuredKey)) - { - throw new ArgumentException("Configured key to retrieve expected subject cannot be blank", nameof(configuredKey)); - } - - return WithSubject(GetValidationLocationImplementation(location), configuredKey); - } + => WithSubject(GetValidationLocationImplementation(location), configuredKey); /// /// Configures the validation for the from a given using a specified . @@ -44,18 +37,7 @@ public CertificateAuthenticationConfigBuilder WithSubject(X509ValidationLocation /// Thrown when the is null. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithSubject(IX509ValidationLocation location, string configuredKey) - { - if (location is null) - { - throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected subject cannot be 'null'"); - } - if (configuredKey is null) - { - throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected subject cannot be blank"); - } - - return AddCertificateRequirement(X509ValidationRequirement.SubjectName, location, configuredKey); - } + => AddCertificateRequirement(X509ValidationRequirement.SubjectName, location, configuredKey); /// /// Configures the validation for the from a given using a specified . @@ -64,14 +46,7 @@ public CertificateAuthenticationConfigBuilder WithSubject(IX509ValidationLocatio /// The configured key that the requires to retrieve the expected issuer name. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithIssuer(X509ValidationLocation location, string configuredKey) - { - if (string.IsNullOrWhiteSpace(configuredKey)) - { - throw new ArgumentException("Configured key to retrieve expected issuer cannot be blank", nameof(configuredKey)); - } - - return WithIssuer(GetValidationLocationImplementation(location), configuredKey); - } + => WithIssuer(GetValidationLocationImplementation(location), configuredKey); /// /// Configures the validation for the from a given using a specified . @@ -81,18 +56,7 @@ public CertificateAuthenticationConfigBuilder WithIssuer(X509ValidationLocation /// Thrown when the is null. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithIssuer(IX509ValidationLocation location, string configuredKey) - { - if (location is null) - { - throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected issuer cannot be 'null'"); - } - if (configuredKey is null) - { - throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected issuer cannot be blank"); - } - - return AddCertificateRequirement(X509ValidationRequirement.IssuerName, location, configuredKey); - } + => AddCertificateRequirement(X509ValidationRequirement.IssuerName, location, configuredKey); /// /// Configures the validation for the from a given using a specified . @@ -101,14 +65,7 @@ public CertificateAuthenticationConfigBuilder WithIssuer(IX509ValidationLocation /// The configured key that the requires to retrieve the expected thumbprint. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithThumbprint(X509ValidationLocation location, string configuredKey) - { - if (string.IsNullOrWhiteSpace(configuredKey)) - { - throw new ArgumentException("Configured key to retrieve expected thumbprint cannot be blank", nameof(configuredKey)); - } - - return WithThumbprint(GetValidationLocationImplementation(location), configuredKey); - } + => WithThumbprint(GetValidationLocationImplementation(location), configuredKey); /// /// Configures the validation for the from a given using a specified . @@ -118,18 +75,7 @@ public CertificateAuthenticationConfigBuilder WithThumbprint(X509ValidationLocat /// Thrown when the is null. /// Thrown when the is blank. public CertificateAuthenticationConfigBuilder WithThumbprint(IX509ValidationLocation location, string configuredKey) - { - if (location is null) - { - throw new ArgumentNullException(nameof(location), "Location implementation to retrieve the expected thumbprint cannot be 'null'"); - } - if (configuredKey is null) - { - throw new ArgumentNullException(nameof(configuredKey), "Configured key to retrieve expected thumbprint cannot be blank"); - } - - return AddCertificateRequirement(X509ValidationRequirement.Thumbprint, location, configuredKey); - } + => AddCertificateRequirement(X509ValidationRequirement.Thumbprint, location, configuredKey); private CertificateAuthenticationConfigBuilder AddCertificateRequirement( X509ValidationRequirement requirement, @@ -140,6 +86,7 @@ private CertificateAuthenticationConfigBuilder AddCertificateRequirement( { throw new ArgumentNullException(nameof(location), "Location cannot be 'null'"); } + if (string.IsNullOrWhiteSpace(configuredKey)) { throw new ArgumentException("Configured key cannot be blank", nameof(configuredKey)); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs index ce0a6b33..d799292d 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationFilter.cs @@ -73,14 +73,17 @@ public async Task OnAuthorizationAsync(AuthorizationFilterContext context) { throw new ArgumentNullException(nameof(context)); } + if (context.HttpContext is null) { throw new ArgumentNullException(nameof(context)); } + if (context.HttpContext.Connection is null) { throw new ArgumentException("Invalid action context given without any HTTP connection"); } + if (context.HttpContext.RequestServices is null) { throw new ArgumentException("Invalid action context given without any HTTP request services"); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs index c98f4eb0..9d4bdea3 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs @@ -37,19 +37,15 @@ public Task GetExpectedCertificateValueForConfiguredKeyAsync(string conf { throw new ArgumentException("Configured key cannot be blank", nameof(configurationKey)); } + if (services is null) { throw new ArgumentNullException(nameof(services), "Registered services cannot be 'null'"); } - var configuration = services.GetService(); - if (configuration == null) - { - throw new KeyNotFoundException( + var configuration = services.GetService() ?? throw new KeyNotFoundException( $"No configured {nameof(IConfiguration)} implementation found in the request service container. " + "Please configure such an implementation (ex. in the Startup) of your application"); - } - string value = configuration[configurationKey]; return Task.FromResult(value); } diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs index 8cbaff2c..b975542f 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/IServiceCollectionExtensions.cs @@ -36,6 +36,7 @@ public static IServiceCollection AddCertificateAuthenticationValidation( { throw new ArgumentNullException(nameof(services), "Requires a set of application services to register the certificate authentication validator"); } + if (configureAuthentication is null) { throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs index d5fcd208..92a28b14 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/Extensions/MvcOptionsExtensions.cs @@ -22,15 +22,8 @@ public static partial class MvcOptionsExtensions /// Thrown when the is null. [Obsolete("Use the " + nameof(AddCertificateAuthenticationFilter) + " overload where the certificate validation locations are configured directly")] public static MvcOptions AddCertificateAuthenticationFilter(this MvcOptions options) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); - } + => AddCertificateAuthenticationFilter(options, configureOptions: null); - return AddCertificateAuthenticationFilter(options, configureOptions: null); - } - /// /// Adds an certificate authentication MVC filter to the given that authenticates the incoming HTTP request. /// @@ -81,18 +74,7 @@ public static MvcOptions AddCertificateAuthenticationFilter( public static MvcOptions AddCertificateAuthenticationFilter( this MvcOptions options, Action configureAuthentication) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); - } - if (configureAuthentication is null) - { - throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); - } - - return AddCertificateAuthenticationFilter(options, configureAuthentication, configureOptions: null); - } + => AddCertificateAuthenticationFilter(options, configureAuthentication, configureOptions: null); /// /// Adds an certificate authentication MVC filter to the given that authenticates the incoming HTTP request. @@ -124,6 +106,7 @@ public static MvcOptions AddCertificateAuthenticationFilter( { throw new ArgumentNullException(nameof(options), "Requires a set of MVC filters to add the certificate authentication MVC filter"); } + if (configureAuthentication is null) { throw new ArgumentNullException(nameof(configureAuthentication), "Requires a function to configure the certificate validation locations"); diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs index 2d8686c9..22fa90ff 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/SecretProviderValidationLocation.cs @@ -37,6 +37,7 @@ public async Task GetExpectedCertificateValueForConfiguredKeyAsync(strin { throw new ArgumentException("Configured key cannot be blank", nameof(configurationKey)); } + if (services is null) { throw new ArgumentNullException(nameof(services), "Registered services cannot be 'null'"); diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs index 6038a57c..cd326669 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs @@ -24,18 +24,7 @@ public static partial class MvcOptionsExtensions /// Thrown when the is null. /// Thrown when the or is blank. public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnHeader(this MvcOptions options, string headerName, string secretName) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires an MVC options instance to add the shared access key authentication filter"); - } - if (string.IsNullOrWhiteSpace(headerName)) - { - throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName)); - } - - return AddSharedAccessKeyAuthenticationFilterOnHeader(options, headerName, secretName, configureOptions: null); - } + => AddSharedAccessKeyAuthenticationFilterOnHeader(options, headerName, secretName, configureOptions: null); /// /// Adds an shared access key authentication MVC filter to the given that authenticates the incoming HTTP request on its header. @@ -90,22 +79,7 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnQuery( this MvcOptions options, string parameterName, string secretName) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); - } - if (string.IsNullOrWhiteSpace(parameterName)) - { - throw new ArgumentException("Requires a non-blank HTTP request query parameter name to match the stored secret during the shared access key authentication", nameof(parameterName)); - } - if (string.IsNullOrWhiteSpace(secretName)) - { - throw new ArgumentException("Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication", nameof(secretName)); - } - - return AddSharedAccessKeyAuthenticationFilterOnQuery(options, parameterName, secretName, configureOptions: null); - } + => AddSharedAccessKeyAuthenticationFilterOnQuery(options, parameterName, secretName, configureOptions: null); /// /// Adds an shared access key authentication MVC filter to the given that authenticates the incoming HTTP request on its query. @@ -132,10 +106,12 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnQuery( { throw new ArgumentNullException(nameof(options), "Requires a set of MVC options to add the shared access authentication MVC filter"); } + if (string.IsNullOrWhiteSpace(parameterName)) { throw new ArgumentException("Requires a non-blank HTTP request query parameter name to match the stored secret during the shared access key authentication", nameof(parameterName)); } + if (string.IsNullOrWhiteSpace(secretName)) { throw new ArgumentException("Requires a non-blank secret name to retrieve the stored access key in the secret store during the shared access key authentication", nameof(secretName)); diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs index b44de2f6..7eed9d77 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationAttribute.cs @@ -29,8 +29,9 @@ public SharedAccessKeyAuthenticationAttribute(string secretName, string headerNa { if (string.IsNullOrWhiteSpace(secretName)) { - throw new ArgumentException(" Secret name cannot be blank", nameof(secretName)); + throw new ArgumentException("Secret name cannot be blank", nameof(secretName)); } + if (string.IsNullOrWhiteSpace(headerName) && string.IsNullOrWhiteSpace(queryParameterName)) { throw new ArgumentException("Requires either a non-blank header name or query parameter name"); diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs index c4c4854d..998ccacb 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs @@ -55,6 +55,7 @@ public SharedAccessKeyAuthenticationFilter(string headerName, string queryParame { throw new ArgumentException("Requires a non-blank secret name", nameof(secretName)); } + if (string.IsNullOrWhiteSpace(headerName) && string.IsNullOrWhiteSpace(queryParameterName)) { throw new ArgumentException("Requires either a non-blank header name or query parameter name"); @@ -79,18 +80,22 @@ public async Task OnAuthorizationAsync(AuthorizationFilterContext context) { throw new ArgumentNullException(nameof(context)); } + if (context.HttpContext is null) { throw new ArgumentNullException(nameof(context.HttpContext)); } + if (context.HttpContext.Request is null) { throw new ArgumentException("Invalid action context given without any HTTP request"); } + if (context.HttpContext.Request.Headers is null) { throw new ArgumentException("Invalid action context given without any HTTP request headers"); } + if (context.HttpContext.RequestServices is null) { throw new ArgumentException("Invalid action context given without any HTTP request services"); @@ -124,29 +129,14 @@ public async Task OnAuthorizationAsync(AuthorizationFilterContext context) private async Task GetAuthorizationSecretAsync(AuthorizationFilterContext context) { - var userDefinedSecretProvider = context.HttpContext.RequestServices.GetService(); - if (userDefinedSecretProvider is null) - { - throw new InvalidOperationException( + var userDefinedSecretProvider = context.HttpContext.RequestServices.GetService() ?? throw new InvalidOperationException( "Cannot retrieve the shared access key to validate the HTTP request because no Arcus secret store was registered in the application," + $"please register the secret store with '{nameof(IHostBuilderExtensions.ConfigureSecretStore)}' on the '{nameof(IHostBuilder)}' or with 'AddSecretStore' on the '{nameof(IServiceCollection)}'," + "for more information on the Arcus secret store: https://security.arcus-azure.net/features/secret-store"); - } - - Task> rawSecretAsync = userDefinedSecretProvider.GetRawSecretsAsync(_secretName); - if (rawSecretAsync is null) - { - throw new InvalidOperationException( + Task> rawSecretAsync = userDefinedSecretProvider.GetRawSecretsAsync(_secretName) ?? throw new InvalidOperationException( $"Configured {nameof(ISecretProvider)} is not implemented correctly as it returns 'null' for a {nameof(Task)} value when calling {nameof(ISecretProvider.GetRawSecretAsync)}"); - } - IEnumerable foundSecrets = await rawSecretAsync; - if (foundSecrets is null) - { - throw new SecretNotFoundException(_secretName); - } - - return foundSecrets.ToArray(); + return foundSecrets is null ? throw new SecretNotFoundException(_secretName) : foundSecrets.ToArray(); } private void ValidateSharedAccessKeyInRequestHeader(AuthorizationFilterContext context, string[] foundSecrets, ILogger logger) diff --git a/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs index 07157d55..7c8cee38 100644 --- a/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authorization/Extensions/MvcOptionsExtensions.cs @@ -18,14 +18,7 @@ public static partial class MvcOptionsExtensions /// The options that are being applied to the request pipeline. /// Thrown when the is null. public static MvcOptions AddJwtTokenAuthorizationFilter(this MvcOptions options) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); - } - - return AddJwtTokenAuthorizationFilter(options, configureOptions: null); - } + => AddJwtTokenAuthorizationFilter(options, configureOptions: null); /// /// Adds JWT token authorization. @@ -58,26 +51,7 @@ public static MvcOptions AddJwtTokenAuthorizationFilter( public static MvcOptions AddJwtTokenAuthorizationFilter( this MvcOptions options, IDictionary claimCheck) - { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); - } - if (claimCheck is null) - { - throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); - } - if (!claimCheck.Any()) - { - throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); - } - if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) - { - throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); - } - - return AddJwtTokenAuthorizationFilter(options, configureOptions: null, claimCheck: claimCheck); - } + => AddJwtTokenAuthorizationFilter(options, configureOptions: null, claimCheck: claimCheck); /// /// Adds JWT token authorization. @@ -96,14 +70,17 @@ public static MvcOptions AddJwtTokenAuthorizationFilter( { throw new ArgumentNullException(nameof(options), "Requires a filter collection to add the JWT token authorization filter"); } + if (claimCheck is null) { throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); } + if (!claimCheck.Any()) { throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) { throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT"); diff --git a/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs b/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs index 8c97eed5..bf4e49b6 100644 --- a/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs +++ b/src/Arcus.WebApi.Security/Authorization/Jwt/JwtTokenReader.cs @@ -80,10 +80,12 @@ public JwtTokenReader(IDictionary claimCheck, ILogger string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) { throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT", nameof(claimCheck)); @@ -170,6 +172,7 @@ public JwtTokenReader(TokenValidationParameters tokenValidationParameters, strin { throw new ArgumentNullException(nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); } + if (string.IsNullOrWhiteSpace(openIdConnectDiscoveryUri)) { throw new ArgumentException("Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration", nameof(openIdConnectDiscoveryUri)); @@ -215,18 +218,23 @@ public JwtTokenReader( { throw new ArgumentException("Requires an non-blank OpenId URI connection endpoint for discovering the OpenId configuration", nameof(openIdConnectDiscoveryUri)); } + if (tokenValidationParameters is null) { throw new ArgumentNullException(nameof(tokenValidationParameters), "Requires a collection of parameters to influence how the token validation is done"); } + if (claimCheck is null) { throw new ArgumentNullException(nameof(claimCheck), "Requires a set of claim checks to verify the claims request JWT"); } + if (!claimCheck.Any()) { - throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the request JWT", nameof(claimCheck)); + throw new ArgumentException("Requires at least one entry in the set of claim checks to verify the claims in the " + + "request JWT", nameof(claimCheck)); } + if (claimCheck.Any(item => string.IsNullOrWhiteSpace(item.Key) || string.IsNullOrWhiteSpace(item.Value))) { throw new ArgumentException("Requires all entries in the set of claim checks to be non-blank to correctly verify the claims in the request JWT", nameof(claimCheck)); diff --git a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs index e1739f44..265c7574 100644 --- a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs +++ b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationFilter .cs @@ -52,18 +52,22 @@ public virtual async Task OnAuthorizationAsync(AuthorizationFilterContext contex { throw new ArgumentNullException(nameof(context)); } + if (context.HttpContext is null) { throw new ArgumentNullException(nameof(context.HttpContext)); } + if (context.HttpContext.Request is null) { throw new ArgumentNullException(nameof(context), "INvalid action context given without any HTTP request"); } + if (context.HttpContext.Request.Headers is null) { throw new ArgumentNullException(nameof(context), "Invalid action context given without any HTTP request headers"); } + if (context.HttpContext.RequestServices is null) { throw new ArgumentNullException(nameof(context), "Invalid action context given without any HTTP request services"); diff --git a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs index 2fb230a7..3f8a0244 100644 --- a/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs +++ b/src/Arcus.WebApi.Security/Authorization/JwtTokenAuthorizationOptions.cs @@ -63,6 +63,7 @@ public JwtTokenAuthorizationOptions(IJwtTokenReader reader, string headerName) { throw new ArgumentNullException(nameof(reader), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); } + if (string.IsNullOrWhiteSpace(headerName)) { throw new ArgumentException("Requires a non-blank request header name to look for the JWT token", nameof(headerName)); @@ -85,6 +86,7 @@ public string HeaderName { throw new ArgumentException("Requires a non-blank request header name to look for the JWT token", nameof(value)); } + _headerName = value; } } @@ -102,6 +104,7 @@ public IJwtTokenReader JwtTokenReader { throw new ArgumentNullException(nameof(value), $"Requires a valid {nameof(IJwtTokenReader)} to verify the JWT token"); } + _jwtTokenReader = value; _createJwtTokenReader = null; } @@ -149,6 +152,7 @@ internal IJwtTokenReader GetOrCreateJwtTokenReader(IServiceProvider serviceProvi { throw new ArgumentNullException(nameof(serviceProvider), $"Requires a collection of services to create an {nameof(IJwtTokenReader)} instance"); } + return _createJwtTokenReader(serviceProvider); } catch (Exception exception) From 53074a9fc38dd97dce8bfcfc813376dcf73b5f8c Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 11:55:45 +0100 Subject: [PATCH 21/31] Use char check instead of string check for single character --- .../Correlation/HttpCorrelationTemplate.cs | 2 +- .../RequestTracking/RequestTrackingTemplate.cs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs index 61d199be..9be97cd2 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationTemplate.cs @@ -365,7 +365,7 @@ private string ExtractLatestOperationParentIdFromHeader(string requestId) // returns: def Logger.LogTrace("Extracting operation parent ID from request ID '{RequestId}' from the upstream service according to W3C Trace-Context standard", requestId); - if (requestId.Contains(".")) + if (requestId.Contains('.')) { string[] ids = requestId.Split('.'); string operationParentId = ids.LastOrDefault(id => !string.IsNullOrWhiteSpace(id)); diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs index 6b6d5a3d..005f40fe 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs @@ -48,7 +48,7 @@ protected bool IsRequestPathOmitted(PathString requestPath, ILogger logger) IEnumerable allOmittedRoutes = Options.OmittedRoutes ?? new Collection(); string[] matchedOmittedRoutes = allOmittedRoutes - .Select(omittedRoute => omittedRoute?.StartsWith("/") == true ? omittedRoute : "/" + omittedRoute) + .Select(omittedRoute => omittedRoute?.StartsWith('/') == true ? omittedRoute : "/" + omittedRoute) .Where(omittedRoute => requestPath.StartsWithSegments(omittedRoute, StringComparison.OrdinalIgnoreCase)) .ToArray(); From 1366d0e76d9d4147aa798db287b9489125012f76 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 14:59:07 +0100 Subject: [PATCH 22/31] small polish --- .../CorrelationInfoUpstreamServiceOptions.cs | 2 + .../HttpCorrelationClientOptions.cs | 1 + .../Correlation/HttpCorrelationResult.cs | 2 + .../Extensions/HttpClientExtensions.cs | 63 +------------------ .../RequestTracking/RequestTrackingOptions.cs | 2 + .../RequestTrackingTemplate.cs | 6 +- 6 files changed, 10 insertions(+), 66 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs index 550b3e6f..42248599 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs @@ -42,6 +42,7 @@ public string HeaderName { throw new ArgumentException(message: "Requires a non-blank value for the operation parent ID request header name", paramName: nameof(value)); } + _headerName = value; } } @@ -62,6 +63,7 @@ public Func GenerateId { throw new ArgumentNullException(paramName: nameof(value), "Requires a function to generate the operation parent ID"); } + _generateId = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index 1cc64ffd..eb8b15c3 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -85,6 +85,7 @@ public void AddTelemetryContext(Dictionary telemetryContext) { throw new ArgumentNullException(nameof(telemetryContext), "Requires a telemetry context dictionary to add to the HTTP dependency tracking"); } + foreach (KeyValuePair item in telemetryContext) { TelemetryContext[item.Key] = item.Value; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs index b3a0df12..73fcab57 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationResult.cs @@ -109,6 +109,7 @@ public static HttpCorrelationResult Success(TelemetryClient client, string trans { throw new ArgumentNullException(nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); } + if (string.IsNullOrWhiteSpace(transactionId)) { throw new ArgumentException("Requires a non-blank transaction ID for the pending HTTP correlation", nameof(transactionId)); @@ -134,6 +135,7 @@ public static HttpCorrelationResult Success(TelemetryClient client, string trans { throw new ArgumentNullException(nameof(client), "Requires a telemetry client instance to automatically track built-in Microsoft dependencies"); } + if (string.IsNullOrWhiteSpace(transactionId)) { throw new ArgumentException("Requires a non-blank transaction ID for the pending HTTP correlation", nameof(transactionId)); diff --git a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs index eac1d8c5..7f0081a7 100644 --- a/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs +++ b/src/Arcus.WebApi.Logging.Core/Extensions/HttpClientExtensions.cs @@ -35,29 +35,7 @@ public static async Task SendAsync( HttpRequestMessage request, IHttpCorrelationInfoAccessor correlationAccessor, ILogger logger) - { - if (client is null) - { - throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - } - - if (request is null) - { - throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - } - - if (correlationAccessor is null) - { - throw new ArgumentNullException(nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); - } - - if (logger is null) - { - throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); - } - - return await SendAsync(client, request, correlationAccessor, logger, configureOptions: null); - } + => await SendAsync(client, request, correlationAccessor, logger, configureOptions: null); /// /// Sends an HTTP request as an asynchronous operation while tracking the HTTP correlation. @@ -84,25 +62,10 @@ public static async Task SendAsync( ILogger logger, Action configureOptions) { - if (client is null) - { - throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - } - - if (request is null) - { - throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - } - if (correlationAccessor is null) { throw new ArgumentNullException(nameof(correlationAccessor), "Requires a HTTP correlation accessor instance to retrieve the current correlation to include in the HTTP request"); } - - if (logger is null) - { - throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); - } CorrelationInfo correlation = correlationAccessor.GetCorrelationInfo(); return await SendAsync(client, request, correlation, logger, configureOptions); @@ -130,29 +93,7 @@ public static async Task SendAsync( HttpRequestMessage request, CorrelationInfo correlationInfo, ILogger logger) - { - if (client is null) - { - throw new ArgumentNullException(nameof(client), "Requires a HTTP client to track the HTTP request with HTTP correlation"); - } - - if (request is null) - { - throw new ArgumentNullException(nameof(request), "Requires a HTTP request to enrich with HTTP correlation"); - } - - if (correlationInfo is null) - { - throw new ArgumentNullException(nameof(correlationInfo), "Requires a HTTP correlation instance to include in the HTTP request"); - } - - if (logger is null) - { - throw new ArgumentNullException(nameof(logger), "Requires a logger instance to track the correlated HTTP request"); - } - - return await SendAsync(client, request, correlationInfo, logger, configureOptions: null); - } + => await SendAsync(client, request, correlationInfo, logger, configureOptions: null); /// /// Sends an HTTP request as an asynchronous operation while tracking the HTTP correlation. diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs index 39811dd6..9325c864 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingOptions.cs @@ -36,6 +36,7 @@ public int? RequestBodyBufferSize { throw new ArgumentOutOfRangeException(nameof(value), "Requires a request body buffer size greater than zero"); } + _requestBodyBufferSize = value; } } @@ -58,6 +59,7 @@ public int? ResponseBodyBufferSize { throw new ArgumentOutOfRangeException(nameof(value), "Requires a response body buffer size greater than zero"); } + _responseBodyBufferSize = value; } } diff --git a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs index 005f40fe..ab5518b4 100644 --- a/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs +++ b/src/Arcus.WebApi.Logging.Core/RequestTracking/RequestTrackingTemplate.cs @@ -24,11 +24,7 @@ public class RequestTrackingTemplate /// Thrown when the is null. protected RequestTrackingTemplate(RequestTrackingOptions options) { - if (options is null) - { - throw new ArgumentNullException(nameof(options), "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); - } - Options = options; + Options = options ?? throw new ArgumentNullException(nameof(options), "Requires a set of additional user-configurable options to influence the behavior of the HTTP request tracking"); } /// From cbb0c0d0ec0859c6058ded5f3148faeb8fcd36d2 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Mon, 6 Jan 2025 16:22:59 +0100 Subject: [PATCH 23/31] add whitelines --- src/Arcus.WebApi.Logging/VersionTrackingOptions.cs | 1 + .../Certificates/CertificateAuthenticationValidator.cs | 1 + .../SharedAccessKey/Extensions/MvcOptionsExtensions.cs | 1 + .../Logging/AzureFunctions/ReadOnceStream.cs | 1 + .../Fixture/HttpRequestBuilder.cs | 10 ++++++++++ .../Fixture/TestApiServer.cs | 1 + .../Fixture/TestApiServerOptions.cs | 1 + .../Fixture/TestConfig.cs | 1 + .../Logging/Fixture/HttpAssert.cs | 3 +++ .../Logging/Fixture/HttpAssertProvider.cs | 3 +++ .../Fixture/HttpAssertServiceCollectionExtensions.cs | 2 ++ .../Authentication/Fixture/SelfSignedCertificate.cs | 1 + 12 files changed, 26 insertions(+) diff --git a/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs b/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs index eb79e44d..a3d5aeb9 100644 --- a/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs +++ b/src/Arcus.WebApi.Logging/VersionTrackingOptions.cs @@ -21,6 +21,7 @@ public string HeaderName { throw new ArgumentException("Requires a non-blank header name to add the current application version to the response", nameof(value)); } + _headerName = value; } } diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs index 270be901..0e07a4c0 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/CertificateAuthenticationValidator.cs @@ -43,6 +43,7 @@ internal async Task IsCertificateAllowedAsync(X509Certificate2 clientCerti { throw new ArgumentNullException(nameof(clientCertificate), "Certificate authentication validation requires a client certificate"); } + if (services is null) { throw new ArgumentNullException(nameof(services), "Certificate authentication validation requires a service object to retrieve registered services"); diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs index cd326669..0de1b620 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/Extensions/MvcOptionsExtensions.cs @@ -51,6 +51,7 @@ public static MvcOptions AddSharedAccessKeyAuthenticationFilterOnHeader( { throw new ArgumentNullException(nameof(options), "Requires a MVC options instance to add the shared access key authentication filter"); } + if (string.IsNullOrWhiteSpace(headerName)) { throw new ArgumentException("Requires a non-blank HTTP request header name to match the stored secret during the shared access key authentication", nameof(headerName)); diff --git a/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs b/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs index c5502d8a..f7f5cf7c 100644 --- a/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs +++ b/src/Arcus.WebApi.Tests.Core/Logging/AzureFunctions/ReadOnceStream.cs @@ -17,6 +17,7 @@ public ReadOnceStream(Stream innerStream) { throw new ArgumentNullException(nameof(innerStream)); } + if (!innerStream.CanRead) { throw new ArgumentException("Requires a readable stream to represent a read-once stream", nameof(innerStream)); diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs index 9378e903..b1de99c7 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/HttpRequestBuilder.cs @@ -37,6 +37,7 @@ public static HttpRequestBuilder Get(string route) { throw new ArgumentException("Requires a non-blank HTTP relative route to create a HTTP GET request builder instance", nameof(route)); } + return new HttpRequestBuilder(HttpMethod.Get, route); } @@ -52,6 +53,7 @@ public static HttpRequestBuilder Post(string route) { throw new ArgumentException("Requires a non-blank HTTP relative route to create a HTTP POST request builder instance", nameof(route)); } + return new HttpRequestBuilder(HttpMethod.Post, route); } @@ -67,6 +69,7 @@ public HttpRequestBuilder WithHeader(string headerName, object headerValue) { throw new ArgumentException("Requires a non-blank header name to add the header to the HTTP request builder instance", nameof(headerName)); } + _headers.Add(new KeyValuePair(headerName, headerValue?.ToString())); return this; @@ -84,6 +87,7 @@ public HttpRequestBuilder WithParameter(string parameterName, object parameterVa { throw new ArgumentException("Requires a non-blank query parameter to add the parameter to the HTTP request builder instance", nameof(parameterName)); } + _parameters.Add(new KeyValuePair(parameterName, parameterValue.ToString())); return this; @@ -101,6 +105,7 @@ public HttpRequestBuilder WithJsonText(string text) { throw new ArgumentException("Requires a non-blank JSON request text to add the content to the HTTP request builder instance", nameof(text)); } + _createContent = () => new StringContent($"\"{text}\"", Encoding.UTF8, "application/json"); return this; @@ -118,6 +123,7 @@ public HttpRequestBuilder WithJsonBody(string json) { throw new ArgumentException("Requires a non-blank JSON request body to add the content to the HTTP request builder instance", nameof(json)); } + _createContent = () => new StringContent(json, Encoding.UTF8, "application/json"); return this; @@ -135,6 +141,7 @@ public HttpRequestBuilder WithTextBody(string text) { throw new ArgumentException("Requires a non-blank text input for the request body", nameof(text)); } + _createContent = () => new StringContent(text, Encoding.UTF8, "text/plain"); return this; @@ -153,13 +160,16 @@ internal HttpRequestMessage Build(string baseRoute) throw new ArgumentException("Requires a non-blank base HTTP endpoint to create a HTTP request message from the HTTP request builder instance", nameof(baseRoute)); } + string parameters = ""; + if (_parameters.Count > 0) { parameters = "?" + String.Join("&", _parameters.Select(p => $"{p.Key}={p.Value}")); } string path = _path; + if (path.StartsWith('/')) { path = path.TrimStart('/'); diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs index e6e1baf2..4ea971d8 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServer.cs @@ -50,6 +50,7 @@ public static async Task StartNewAsync(TestApiServerOptions optio { throw new ArgumentNullException(nameof(options), "Requires a set of configurable options to control the behavior of the test API server"); } + if (logger is null) { throw new ArgumentNullException(nameof(logger), "Requires a logger instance to write diagnostic messages during the lifetime of the test API server"); diff --git a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs index 1844785d..080f4264 100644 --- a/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs +++ b/src/Arcus.WebApi.Tests.Integration/Fixture/TestApiServerOptions.cs @@ -124,6 +124,7 @@ public TestApiServerOptions ConfigureAppConfiguration(Action assertion) { throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify a HTTP context"); } + _assertion = assertion; } @@ -30,6 +31,7 @@ public static HttpAssert Create(Action assertion) { throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify a HTTP context"); } + return new HttpAssert(assertion); } @@ -44,6 +46,7 @@ public void Assert(HttpContext context) { throw new ArgumentNullException(nameof(context), "Requires a HTTP context to run an assertion function to it"); } + _assertion(context); } } diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs index 5fdb547c..0120b499 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertProvider.cs @@ -26,10 +26,12 @@ public HttpAssertProvider(IEnumerable> namedAssertions { throw new ArgumentNullException(nameof(namedAssertions), "Requires a series of named HTTP assertions to setup the HTTP assertion provider"); } + if (namedAssertions.Any(item => item is null)) { throw new ArgumentException("Requires a series of named HTTP assertions without any 'null' elements to setup the HTTP assertion provider", nameof(namedAssertions)); } + if (namedAssertions.GroupBy(item => item.Item1).All(group => group.Count() != 1)) { throw new ArgumentException("Requires a series of named HTTP assertions with unique names to setup the HTTP assertion provider", nameof(namedAssertions)); @@ -50,6 +52,7 @@ public HttpAssert GetAssertion(string name) { throw new ArgumentException("Requires a non-blank name to retrieve the HTTP assertion", nameof(name)); } + return Assert.Single(_namedAssertions, item => item.Item1 == name).Item2; } } diff --git a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs index d57ac644..b91eb137 100644 --- a/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs +++ b/src/Arcus.WebApi.Tests.Integration/Logging/Fixture/HttpAssertServiceCollectionExtensions.cs @@ -25,10 +25,12 @@ public static IServiceCollection AddHttpAssert(this IServiceCollection services, { throw new ArgumentNullException(nameof(services), "Requires a set of services to add the HTTP assertion to"); } + if (string.IsNullOrWhiteSpace(name)) { throw new ArgumentException("Requires a non-blank name to register the HTTP assertion", nameof(name)); } + if (assertion is null) { throw new ArgumentNullException(nameof(assertion), "Requires an assertion function to verify the currently available HTTP context"); diff --git a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs index 7b4cd0c4..9605d780 100644 --- a/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs +++ b/src/Arcus.WebApi.Tests.Integration/Security/Authentication/Fixture/SelfSignedCertificate.cs @@ -56,6 +56,7 @@ public static X509Certificate2 CreateWithIssuerAndSubjectName(string issuerName, { throw new ArgumentException("Subject name should not be blank", nameof(subjectName)); } + if (string.IsNullOrWhiteSpace(issuerName)) { throw new ArgumentException("Issuer name should not be blank", nameof(issuerName)); From a2c4f49495ce8a47dc9ae69aeaabfc0aac466202 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Thu, 9 Jan 2025 10:32:27 +0100 Subject: [PATCH 24/31] Revert oneliner --- .../AzureFunctionsRequestTrackingMiddleware.cs | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs b/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs index 5ad6631e..ebed2409 100644 --- a/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs +++ b/src/Arcus.WebApi.Logging.AzureFunctions/AzureFunctionsRequestTrackingMiddleware.cs @@ -61,20 +61,25 @@ public async Task Invoke(FunctionContext context, FunctionExecutionDelegate next HttpResponseData response = context.GetHttpResponseData(); var attributeTrackedStatusCodes = Enumerable.Empty(); - if (response != null && AllowedToTrackStatusCode((int) response.StatusCode, attributeTrackedStatusCodes, logger)) + if (response != null && AllowedToTrackStatusCode((int)response.StatusCode, attributeTrackedStatusCodes, logger)) { string responseBody = await GetPotentialResponseBodyAsync(response, logger); - LogRequest(requestBody, responseBody, request, response, measurement, logger); + LogRequest(requestBody, responseBody, request, response, measurement, logger); } - } + } } } } private static async Task EnableHttpRequestBufferingAsync(FunctionContext context) { - BindingMetadata bindingMetadata = context.FunctionDefinition.InputBindings.Values.FirstOrDefault(a => a.Type == "httpTrigger") ?? throw new InvalidOperationException( + BindingMetadata bindingMetadata = context.FunctionDefinition.InputBindings.Values.FirstOrDefault(a => a.Type == "httpTrigger"); + if (bindingMetadata is null) + { + throw new InvalidOperationException( "Cannot enable HTTP request body buffering because it cannot find the Azure Functions' HTTP trigger input binding representing the HTTP request"); + } + InputBindingData bindingData = await context.BindInputAsync(bindingMetadata); bindingData.Value = new BufferedHttpRequestData(bindingData.Value); @@ -146,7 +151,7 @@ private void LogRequest(string requestBody, string responseBody, HttpRequestData request.Url.Host, request.Url.AbsolutePath, operationName: null, - (int) response.StatusCode, + (int)response.StatusCode, duration.StartTime, duration.Elapsed, logContext)); From f9b4671b6800819706a966544443ce9ad1906a21 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Thu, 9 Jan 2025 10:55:55 +0100 Subject: [PATCH 25/31] Adapt to PR comments --- .../Correlation/CorrelationInfoUpstreamServiceOptions.cs | 4 ++-- .../Correlation/HttpCorrelation.cs | 8 ++++---- .../Certificates/ConfigurationValidationLocation.cs | 9 +++++++-- .../SharedAccessKeyAuthenticationFilter.cs | 8 +++++++- 4 files changed, 20 insertions(+), 9 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs index 42248599..804b4ff8 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/CorrelationInfoUpstreamServiceOptions.cs @@ -40,7 +40,7 @@ public string HeaderName { if (string.IsNullOrWhiteSpace(value)) { - throw new ArgumentException(message: "Requires a non-blank value for the operation parent ID request header name", paramName: nameof(value)); + throw new ArgumentException("Requires a non-blank value for the operation parent ID request header name", nameof(value)); } _headerName = value; @@ -61,7 +61,7 @@ public Func GenerateId { if (value is null) { - throw new ArgumentNullException(paramName: nameof(value), "Requires a function to generate the operation parent ID"); + throw new ArgumentNullException(nameof(value), "Requires a function to generate the operation parent ID"); } _generateId = value; diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs index 3ebe2336..0ba67e67 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelation.cs @@ -43,9 +43,9 @@ public HttpCorrelation( throw new ArgumentNullException(nameof(options), "Requires a value in the set of options to configure the correlation process"); } - _httpContextAccessor = httpContextAccessor ?? throw new ArgumentNullException(paramName: nameof(httpContextAccessor), message: "Requires a HTTP context accessor to get the current HTTP context"); - _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(paramName: nameof(correlationInfoAccessor), message: "Requires a correlation info instance to set and retrieve the correlation information"); - _options = options.Value ?? throw new ArgumentNullException(paramName: nameof(options.Value), message: "Requires a value in the set of options to configure the correlation process"); + _httpContextAccessor = httpContextAccessor ?? throw new ArgumentNullException(nameof(httpContextAccessor), "Requires a HTTP context accessor to get the current HTTP context"); + _correlationInfoAccessor = correlationInfoAccessor ?? throw new ArgumentNullException(nameof(correlationInfoAccessor), "Requires a correlation info instance to set and retrieve the correlation information"); + _options = options.Value ?? throw new ArgumentNullException(nameof(options.Value), "Requires a value in the set of options to configure the correlation process"); _logger = logger ?? NullLogger.Instance; @@ -82,7 +82,7 @@ public void SetCorrelationInfo(CorrelationInfo correlationInfo) /// Thrown when the given doesn't have any response headers to set the correlation headers. public HttpCorrelationResult CorrelateHttpRequest() { - HttpContext httpContext = _httpContextAccessor.HttpContext ?? throw new ArgumentNullException(paramName: nameof(HttpContext), "Requires a HTTP context from the HTTP context accessor to start correlating the HTTP request"); + HttpContext httpContext = _httpContextAccessor.HttpContext ?? throw new ArgumentNullException(nameof(HttpContext), "Requires a HTTP context from the HTTP context accessor to start correlating the HTTP request"); if (httpContext.Response is null) { diff --git a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs index 9d4bdea3..cc5e6a70 100644 --- a/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs +++ b/src/Arcus.WebApi.Security/Authentication/Certificates/ConfigurationValidationLocation.cs @@ -6,7 +6,7 @@ using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; -namespace Arcus.WebApi.Security.Authentication.Certificates +namespace Arcus.WebApi.Security.Authentication.Certificates { /// /// Certificate location implementation to retrieve the expected value from an @@ -43,9 +43,14 @@ public Task GetExpectedCertificateValueForConfiguredKeyAsync(string conf throw new ArgumentNullException(nameof(services), "Registered services cannot be 'null'"); } - var configuration = services.GetService() ?? throw new KeyNotFoundException( + var configuration = services.GetService(); + if (configuration is null) + { + throw new KeyNotFoundException( $"No configured {nameof(IConfiguration)} implementation found in the request service container. " + "Please configure such an implementation (ex. in the Startup) of your application"); + } + string value = configuration[configurationKey]; return Task.FromResult(value); } diff --git a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs index 998ccacb..1e257b6f 100644 --- a/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs +++ b/src/Arcus.WebApi.Security/Authentication/SharedAccessKey/SharedAccessKeyAuthenticationFilter.cs @@ -136,7 +136,13 @@ private async Task GetAuthorizationSecretAsync(AuthorizationFilterCont Task> rawSecretAsync = userDefinedSecretProvider.GetRawSecretsAsync(_secretName) ?? throw new InvalidOperationException( $"Configured {nameof(ISecretProvider)} is not implemented correctly as it returns 'null' for a {nameof(Task)} value when calling {nameof(ISecretProvider.GetRawSecretAsync)}"); IEnumerable foundSecrets = await rawSecretAsync; - return foundSecrets is null ? throw new SecretNotFoundException(_secretName) : foundSecrets.ToArray(); + + if (foundSecrets is null) + { + throw new SecretNotFoundException(_secretName); + } + + return foundSecrets.ToArray(); } private void ValidateSharedAccessKeyInRequestHeader(AuthorizationFilterContext context, string[] foundSecrets, ILogger logger) From d73675f8008bcc75ba1f78e9dff3dfc180d916a5 Mon Sep 17 00:00:00 2001 From: Joachim Goris <23077318+joachimgoris@users.noreply.github.com> Date: Tue, 14 Jan 2025 14:52:16 +0100 Subject: [PATCH 26/31] fix unit tests --- .../Correlation/HttpCorrelationClientOptions.cs | 4 ++-- .../CertificateAuthenticationOperationFilter.cs | 7 ++++++- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index eb8b15c3..6ddd85d0 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -42,7 +42,7 @@ public string UpstreamServiceHeaderName get => _upstreamServiceHeaderName; set { - if (value is null) + if (string.IsNullOrWhiteSpace(value)) { throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); } @@ -60,7 +60,7 @@ public string TransactionIdHeaderName get => _transactionIdHeaderName; set { - if (value is null) + if (string.IsNullOrWhiteSpace(value)) { throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); } diff --git a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs index 670b6e8d..53697aa1 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/CertificateAuthenticationOperationFilter.cs @@ -43,7 +43,12 @@ public CertificateAuthenticationOperationFilter( #endif ) { - _securitySchemeName = securitySchemeName ?? throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Certificate security scheme"); + if (string.IsNullOrWhiteSpace(securitySchemeName)) + { + throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Certificate security scheme"); + } + + _securitySchemeName = securitySchemeName; #if !NETSTANDARD2_1 if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType)) From f6ff1185595e3184bf192a335efa35685ca7586d Mon Sep 17 00:00:00 2001 From: Frederik Gheysels Date: Thu, 16 Jan 2025 13:21:17 +0100 Subject: [PATCH 27/31] Update src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs Co-authored-by: Stijn Moreels <9039753+stijnmoreels@users.noreply.github.com> --- .../Correlation/HttpCorrelationClientOptions.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index 6ddd85d0..d3e522ec 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -44,7 +44,7 @@ public string UpstreamServiceHeaderName { if (string.IsNullOrWhiteSpace(value)) { - throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies"); + throw new ArgumentException("Requires a non-blank value for the HTTP request header where the dependency ID should be added when tracking HTTP dependencies", nameof(value)); } _upstreamServiceHeaderName = value; From 99cc4cd3f2d2bb94216d40ef0ef6a20a42b487a3 Mon Sep 17 00:00:00 2001 From: Frederik Gheysels Date: Thu, 16 Jan 2025 13:21:29 +0100 Subject: [PATCH 28/31] Update src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs Co-authored-by: Stijn Moreels <9039753+stijnmoreels@users.noreply.github.com> --- .../Correlation/HttpCorrelationClientOptions.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs index d3e522ec..68a26353 100644 --- a/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs +++ b/src/Arcus.WebApi.Logging.Core/Correlation/HttpCorrelationClientOptions.cs @@ -62,7 +62,7 @@ public string TransactionIdHeaderName { if (string.IsNullOrWhiteSpace(value)) { - throw new ArgumentNullException(nameof(value), "Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies"); + throw new ArgumentException("Requires a non-blank value for the HTTP request header where the transaction ID should be added when tracking HTTP dependencies", nameof(value)); } _transactionIdHeaderName = value; From 3705279016365f0e2bc19cc4b1ebf67a101e9454 Mon Sep 17 00:00:00 2001 From: Frederik Gheysels Date: Thu, 16 Jan 2025 13:21:52 +0100 Subject: [PATCH 29/31] Update src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs Co-authored-by: Stijn Moreels <9039753+stijnmoreels@users.noreply.github.com> --- .../OAuthAuthorizeOperationFilter.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs index d69b6422..6b8fd88f 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/OAuthAuthorizeOperationFilter.cs @@ -37,7 +37,7 @@ public OAuthAuthorizeOperationFilter(IEnumerable scopes, string security if (string.IsNullOrWhiteSpace(securitySchemaName)) { - throw new ArgumentNullException(nameof(securitySchemaName), "Requires a name for the OAuth2 security scheme"); + throw new ArgumentException("Requires a name for the OAuth2 security scheme", nameof(securitySchemaName)); } _securitySchemaName = securitySchemaName; From 7d17a691c63ae1f61e1abe8f869823f08f286cdc Mon Sep 17 00:00:00 2001 From: Frederik Gheysels Date: Thu, 16 Jan 2025 13:22:07 +0100 Subject: [PATCH 30/31] Update src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs Co-authored-by: Stijn Moreels <9039753+stijnmoreels@users.noreply.github.com> --- .../SharedAccessKeyAuthenticationOperationFilter.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs index 6f7bc955..b44d9b81 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs @@ -56,7 +56,7 @@ public SharedAccessKeyAuthenticationOperationFilter(string securitySchemeName) { if (string.IsNullOrWhiteSpace(securitySchemeName)) { - throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); + throw new ArgumentException("Requires a name for the Shared Access Key security scheme", nameof(securitySchemeName)); } _securitySchemeName = securitySchemeName; From bf4a3b53daac6b57b30d08b23910656788d23e5f Mon Sep 17 00:00:00 2001 From: Frederik Gheysels Date: Thu, 16 Jan 2025 13:22:17 +0100 Subject: [PATCH 31/31] Update src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs Co-authored-by: Stijn Moreels <9039753+stijnmoreels@users.noreply.github.com> --- .../SharedAccessKeyAuthenticationOperationFilter.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs index b44d9b81..4500eaef 100644 --- a/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs +++ b/src/Arcus.WebApi.OpenApi.Extensions/SharedAccessKeyAuthenticationOperationFilter.cs @@ -36,7 +36,7 @@ public SharedAccessKeyAuthenticationOperationFilter( { if (string.IsNullOrWhiteSpace(securitySchemeName)) { - throw new ArgumentNullException(nameof(securitySchemeName), "Requires a name for the Shared Access Key security scheme"); + throw new ArgumentException("Requires a name for the Shared Access Key security scheme", nameof(securitySchemeName)); } if (!Enum.IsDefined(typeof(SecuritySchemeType), securitySchemeType))