Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ASP.NET Response Cookie Layout Renderer #789

Merged
merged 23 commits into from
Jun 6, 2022
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion src/NLog.Web.AspNetCore/NLog.Web.AspNetCore.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -75,14 +75,15 @@ NLog 5 release post: https://nlog-project.org/2021/08/25/nlog-5-0-preview1-ready
<DefineConstants>$(DefineConstants);ASP_NET_CORE;ASP_NET_CORE3</DefineConstants>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.Http.Extensions" Version="2.1.21" />
snakefoot marked this conversation as resolved.
Show resolved Hide resolved
<PackageReference Include="NLog.Extensions.Logging" Version="5.0.0" />
<PackageReference Include="System.Text.Encodings.Web" Version="4.5.1" />
snakefoot marked this conversation as resolved.
Show resolved Hide resolved
</ItemGroup>
<ItemGroup Condition=" '$(TargetFramework)' == 'netstandard2.0' or '$(TargetFramework)' == 'net461' ">
<!-- Fixed to 2.1.0 as 2.1 is Long Term Supported (LTS) and works with vanilla .NET Core 2.1 SDK -->
<PackageReference Include="Microsoft.AspNetCore.Http" Version="2.1.22" />
<PackageReference Include="Microsoft.AspNetCore.Hosting.Abstractions" Version="2.1.0" />
<PackageReference Include="Microsoft.AspNetCore.Routing.Abstractions" Version="2.1.0" />
<PackageReference Include="System.Text.Encodings.Web" Version="4.5.1" />
</ItemGroup>
<ItemGroup>
<PackageReference Include="Microsoft.SourceLink.GitHub" Version="1.1.1" PrivateAssets="All" />
Expand Down
36 changes: 0 additions & 36 deletions src/Shared/Internal/HttpContextExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -62,42 +62,6 @@ internal static HttpResponse TryGetResponse(this HttpContext context)
}
#endif

#if ASP_NET_CORE2
internal static string GetString(this ISession session, string key)
{
if (!session.TryGetValue(key, out var data))
{
return null;
}

if (data == null)
{
return null;
}

if (data.Length == 0)
{
return string.Empty;
}

return Encoding.UTF8.GetString(data);
}

public static int? GetInt32(this ISession session, string key)
{
if (!session.TryGetValue(key, out var data))
{
return null;
}

if (data == null || data.Length < 4)
{
return null;
}
return data[0] << 24 | data[1] << 16 | data[2] << 8 | data[3];
}
#endif

#if !ASP_NET_CORE
internal static HttpSessionStateBase TryGetSession(this HttpContextBase context)
{
Expand Down
185 changes: 185 additions & 0 deletions src/Shared/LayoutRenderers/AspNetResponseCookieLayoutRenderer.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using NLog.Config;
using NLog.LayoutRenderers;
using NLog.Web.Enums;
using NLog.Web.Internal;
#if !ASP_NET_CORE
using System.Collections.Specialized;
using System.Web;
using Cookies = System.Web.HttpCookieCollection;
#else
using Microsoft.Net.Http.Headers;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Http.Extensions;
#endif

namespace NLog.Web.LayoutRenderers
{
/// <summary>
/// ASP.NET Response Cookie
/// </summary>
/// <example>
/// <para>Example usage of ${aspnet-response-cookie}</para>
/// <code lang="NLog Layout Renderer">
/// ${aspnet-response-cookie:OutputFormat=Flat}
/// ${aspnet-response-cookie:OutputFormat=JsonArray}
/// ${aspnet-response-cookie:OutputFormat=JsonDictionary}
/// ${aspnet-response-cookie:OutputFormat=JsonDictionary:CookieNames=username}
/// ${aspnet-response-cookie:OutputFormat=JsonDictionary:Exclude=access_token}
/// </code>
/// </example>
[LayoutRenderer("aspnet-response-cookie")]
public class AspNetResponseCookieLayoutRenderer : AspNetLayoutMultiValueRendererBase
{
/// <summary>
/// Cookie names to be rendered.
/// If <c>null</c> or empty array, all cookies will be rendered.
/// </summary>
public List<string> CookieNames { get; set; }

/// <summary>
/// Gets or sets the keys to exclude from the output. If omitted, none are excluded.
/// </summary>
/// <docgen category='Rendering Options' order='10' />
#if ASP_NET_CORE
public ISet<string> Exclude { get; set; }
#else
public HashSet<string> Exclude { get; set; }
#endif

/// <summary>
/// Initializes a new instance of the <see cref="AspNetResponseCookieLayoutRenderer" /> class.
/// </summary>
public AspNetResponseCookieLayoutRenderer()
{
Exclude = new HashSet<string>(new[] { "AUTH", "SESS_ID" }, StringComparer.OrdinalIgnoreCase);
}

#if !ASP_NET_CORE
/// <summary>
/// Renders the ASP.NET Cookie appends it to the specified <see cref="StringBuilder" />.
/// </summary>
/// <param name="builder">The <see cref="StringBuilder" /> to append the rendered data to.</param>
/// <param name="logEvent">Logging event.</param>
protected override void DoAppend(StringBuilder builder, LogEventInfo logEvent)
{
var httpResponse = HttpContextAccessor.HttpContext.TryGetResponse();
if (httpResponse == null)
{
return;
}

var cookies = httpResponse.Cookies;

if (cookies?.Count > 0)
{
bool checkForExclude = (CookieNames == null || CookieNames.Count == 0) && Exclude?.Count > 0;
var cookieValues = GetCookieValues(cookies, checkForExclude);
SerializePairs(cookieValues, builder, logEvent);
}
}

private IEnumerable<KeyValuePair<string, string>> GetCookieValues(HttpCookieCollection cookies, bool checkForExclude)
{
var cookieNames = GetCookieNames(cookies);
foreach (var cookieName in cookieNames)
{
if (checkForExclude && Exclude.Contains(cookieName))
continue;

var httpCookie = cookies[cookieName];
if (httpCookie == null)
{
continue;
}

if (OutputFormat != AspNetRequestLayoutOutputFormat.Flat)
{
// Split multi-valued cookie, as allowed for in the HttpCookie API for backwards compatibility with classic ASP
var isFirst = true;
foreach (var multiValueKey in httpCookie.Values.AllKeys)
{
var cookieKey = multiValueKey;
if (isFirst)
{
cookieKey = cookieName;
isFirst = false;
}
yield return new KeyValuePair<string, string>(cookieKey, httpCookie.Values[multiValueKey]);
}
}
else
{
yield return new KeyValuePair<string, string>(cookieName, httpCookie.Value);
}
}
}

private List<string> GetCookieNames(HttpCookieCollection cookies)
{
return CookieNames?.Count > 0 ? CookieNames : cookies.Keys.Cast<string>().ToList();
}
#else
/// <summary>
/// Renders the ASP.NET Cookie appends it to the specified <see cref="StringBuilder" />.
/// </summary>
/// <param name="builder">The <see cref="StringBuilder" /> to append the rendered data to.</param>
/// <param name="logEvent">Logging event.</param>
protected override void DoAppend(StringBuilder builder, LogEventInfo logEvent)
{
var httpResponse = HttpContextAccessor.HttpContext.TryGetResponse();
if (httpResponse == null)
{
return;
}

IList<SetCookieHeaderValue> cookies = httpResponse.GetTypedHeaders().SetCookie;

if (cookies?.Count > 0)
{
bool checkForExclude = (CookieNames == null || CookieNames.Count == 0) && Exclude?.Count > 0;
var cookieValues = GetCookieValues(cookies, checkForExclude);
SerializePairs(cookieValues, builder, logEvent);
}
}

private List<string> GetCookieNames(IList<SetCookieHeaderValue> cookies)
{
if (CookieNames?.Count > 0)
{
return CookieNames;
}

var response = new List<string>();

foreach (var cookie in cookies)
{
response.Add(cookie.Name.ToString());
}

return response;
}

private IEnumerable<KeyValuePair<string, string>> GetCookieValues(IList<SetCookieHeaderValue> cookies, bool checkForExclude)
{
var cookieNames = GetCookieNames(cookies);
foreach (var cookieName in cookieNames)
{
if (checkForExclude && Exclude.Contains(cookieName))
continue;

var httpCookie = cookies.SingleOrDefault(cookie => cookie.Name.ToString() == cookieName);
if (httpCookie == null)
{
continue;
}

yield return new KeyValuePair<string, string>(cookieName, httpCookie.Value.ToString());
}
}
#endif
}
}
Loading