diff --git a/src/Shared/LayoutRenderers/AspNetRequestUrlRenderer.cs b/src/Shared/LayoutRenderers/AspNetRequestUrlRenderer.cs index 3ba59e11..6c60aeca 100644 --- a/src/Shared/LayoutRenderers/AspNetRequestUrlRenderer.cs +++ b/src/Shared/LayoutRenderers/AspNetRequestUrlRenderer.cs @@ -5,6 +5,7 @@ using NLog.Web.Internal; #if ASP_NET_CORE using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Http.Features; #else using System.Collections.Specialized; @@ -24,7 +25,7 @@ namespace NLog.Web.LayoutRenderers /// ${aspnet-request-url:IncludePort=true} - produces http://www.exmaple.com:80/ /// ${aspnet-request-url:IncludePort=false} - produces http://www.exmaple.com/ /// ${aspnet-request-url:IncludeScheme=false} - produces www.exmaple.com/ - /// ${aspnet-request-url:IncludePort=true:IncludeQueryString=true} - produces http://www.exmaple.com:80/?t=1 + /// ${aspnet-request-url:IncludePort=true:IncludeQueryString=true} - produces http://www.exmaple.com:80/?t=1 /// /// [LayoutRenderer("aspnet-request-url")] @@ -37,7 +38,7 @@ public class AspNetRequestUrlRenderer : AspNetLayoutRendererBase public bool IncludeQueryString { get; set; } = false; /// - /// To specify whether to include /exclude the Port. Default is false. + /// To specify whether to include / exclude the Port. Default is false. /// public bool IncludePort { get; set; } = false; @@ -51,6 +52,16 @@ public class AspNetRequestUrlRenderer : AspNetLayoutRendererBase /// public bool IncludeScheme { get; set; } = true; +#if ASP_NET_CORE + + /// + /// To specify whether to use raw path and full query. Default is false. + /// See https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.http.features.ihttprequestfeature.rawtarget + /// + public bool UseRawTarget { get; set; } = false; + +#endif + /// /// Renders the Request URL from the HttpRequest /// @@ -68,6 +79,7 @@ protected override void DoAppend(StringBuilder builder, LogEventInfo logEvent) } #if !ASP_NET_CORE + private void RenderUrl(HttpRequestBase httpRequest, StringBuilder builder) { var url = httpRequest.Url; @@ -112,13 +124,22 @@ private void RenderUrl(HttpRequest httpRequest, StringBuilder builder) builder.Append(httpRequest.Host.Port.Value); } - builder.Append(httpRequest.PathBase.ToUriComponent()); - builder.Append(httpRequest.Path.ToUriComponent()); - if (IncludeQueryString) + IHttpRequestFeature httpRequestFeature; + if (UseRawTarget && (httpRequestFeature = httpRequest.HttpContext.Features.Get()) != null) + { + builder.Append(httpRequestFeature.RawTarget); + } + else { - builder.Append(httpRequest.QueryString.Value); + builder.Append(httpRequest.PathBase.ToUriComponent()); + builder.Append(httpRequest.Path.ToUriComponent()); + + if (IncludeQueryString) + { + builder.Append(httpRequest.QueryString.Value); + } } } #endif } -} \ No newline at end of file +} diff --git a/tests/Shared/LayoutRenderers/AspNetRequestUrlRendererTests.cs b/tests/Shared/LayoutRenderers/AspNetRequestUrlRendererTests.cs index 6752d829..524f9daf 100644 --- a/tests/Shared/LayoutRenderers/AspNetRequestUrlRendererTests.cs +++ b/tests/Shared/LayoutRenderers/AspNetRequestUrlRendererTests.cs @@ -10,11 +10,15 @@ using Microsoft.Extensions.Primitives; using HttpContextBase = Microsoft.AspNetCore.Http.HttpContext; using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Http.Features; #endif using NLog.Web.LayoutRenderers; using NSubstitute; + using Xunit; +using System.IO; + namespace NLog.Web.Tests.LayoutRenderers { public class AspNetRequestUrlRendererTests : LayoutRenderersTestBase @@ -146,8 +150,32 @@ public void UrlPresentRenderNonEmpty_ExcludeHost_IncludeQueryString() Assert.Equal("http:///Test.asp?t=1", result); } +#if ASP_NET_CORE + [Fact] + public void UrlPresentRenderNonEmpty_UseRawTarget() + { + var renderer = CreateRenderer("www.google.com:80", "?t=1", "http", "/Test.asp", rawTarget: "/rawTarget"); + renderer.UseRawTarget = true; - private static AspNetRequestUrlRenderer CreateRenderer(string hostBase, string queryString = "", string scheme = "http", string page = "/", string pathBase = "") + string result = renderer.Render(LogEventInfo.CreateNullEvent()); + + Assert.Equal("http://www.google.com/rawTarget", result); + } + + [Fact] + public void UrlPresentRenderNonEmpty_UseRawTarget_IncludeQueryString() + { + var renderer = CreateRenderer("www.google.com:80", "?t=1", "http", "/Test.asp", rawTarget: "/rawTarget"); + renderer.UseRawTarget = true; + renderer.IncludeQueryString = true; + + string result = renderer.Render(LogEventInfo.CreateNullEvent()); + + Assert.Equal("http://www.google.com/rawTarget", result); + } +#endif + + private static AspNetRequestUrlRenderer CreateRenderer(string hostBase, string queryString = "", string scheme = "http", string page = "/", string pathBase = "", string rawTarget = null) { var (renderer, httpContext) = CreateWithHttpContext(); @@ -160,8 +188,40 @@ private static AspNetRequestUrlRenderer CreateRenderer(string hostBase, string q httpContext.Request.QueryString.Returns(new QueryString(queryString)); httpContext.Request.Host.Returns(new HostString(hostBase)); httpContext.Request.Scheme.Returns(scheme); + + if (rawTarget != null) + { + httpContext.Request.HttpContext.Returns(httpContext); + + var httpRequestFeature = new HttpRequestFeatureMock(); + httpRequestFeature.RawTarget = rawTarget; + var collection = new FeatureCollection(); + collection.Set(httpRequestFeature); + httpContext.Features.Returns(collection); + } #endif return renderer; } + +#if ASP_NET_CORE + + private class HttpRequestFeatureMock : IHttpRequestFeature + { + #region Implementation of IHttpRequestFeature + + public Stream Body { get; set; } + public IHeaderDictionary Headers { get; set; } + public string Method { get; set; } + public string Path { get; set; } + public string PathBase { get; set; } + public string Protocol { get; set; } + public string QueryString { get; set; } + public string RawTarget { get; set; } + public string Scheme { get; set; } + + #endregion Implementation of IHttpRequestFeature + } + +#endif } }