From 33f97d54f9a74d00b36becfa27bc261b20f4d570 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Fri, 20 Jul 2018 07:51:30 +1000 Subject: [PATCH 01/18] Bump the `dev` branch's version (This needs to be done manually after a release is merged to master.) --- .../Serilog.Extensions.Hosting.csproj | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index b72ded6..fd5779f 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -2,7 +2,7 @@ Serilog support for .NET Core logging in hosted services - 2.0.0 + 2.0.1 Microsoft;Serilog Contributors netstandard2.0 true From 8d116f926c0a0c00fc0a9b1d1e7c9356e3955dcf Mon Sep 17 00:00:00 2001 From: Andrew Lock Date: Sun, 5 Aug 2018 08:50:06 +0100 Subject: [PATCH 02/18] Fix incorrect urls --- README.md | 2 +- .../Serilog.Extensions.Hosting.csproj | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index efb5643..f99b283 100644 --- a/README.md +++ b/README.md @@ -54,7 +54,7 @@ public class Program } ``` -**Finally**, clean up by removing the rema `"Logging"` section from _appsettings.json_ files (this can be replaced with [Serilog configuration](https://github.com/serilog/serilog-settings-configuration) as shown in [this example](https://github.com/serilog/serilog-hosting/blob/dev/samples/SimpleServiceSample/Program.cs), if required) +**Finally**, clean up by removing the remaining `"Logging"` section from _appsettings.json_ files (this can be replaced with [Serilog configuration](https://github.com/serilog/serilog-settings-configuration) as shown in [this example](https://github.com/serilog/serilog-extensions-hosting/blob/dev/samples/SimpleServiceSample/Program.cs), if required) That's it! You will see log output like: diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index fd5779f..29dc34f 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -14,7 +14,7 @@ Serilog.Extensions.Hosting serilog;aspnet;aspnetcore;hosting http://serilog.net/images/serilog-extension-nuget.png - https://github.com/serilog/serilog-hosting + https://github.com/serilog/serilog-extensions-hosting http://www.apache.org/licenses/LICENSE-2.0 false Serilog From efad033aedbb99e9d7a41395f853d787ef4ebe4a Mon Sep 17 00:00:00 2001 From: Maxime Rouiller Date: Fri, 21 Sep 2018 11:14:52 -0400 Subject: [PATCH 03/18] fixing repository url --- .../Serilog.Extensions.Hosting.csproj | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index 29dc34f..c343413 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -16,6 +16,8 @@ http://serilog.net/images/serilog-extension-nuget.png https://github.com/serilog/serilog-extensions-hosting http://www.apache.org/licenses/LICENSE-2.0 + https://github.com/serilog/serilog-extensions-hosting + git false Serilog From a6610faa4f5704e641fb67441179a1dcb0e1fc99 Mon Sep 17 00:00:00 2001 From: Jon McGuire Date: Mon, 1 Oct 2018 09:26:00 -0400 Subject: [PATCH 04/18] only build packaging-related commits --- appveyor.yml | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/appveyor.yml b/appveyor.yml index 02761fd..9dc3dbe 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -7,6 +7,12 @@ build_script: - ps: ./Build.ps1 artifacts: - path: artifacts/Serilog.*.nupkg +only_commits: + files: + - serilog-extensions-hosting.sln + - src/Serilog.Extensions.Hosting/ + - Build.ps1 + - assets/ deploy: - provider: NuGet api_key: From 8ff6998c9d09a0158385d4a01a0875aac9bf7354 Mon Sep 17 00:00:00 2001 From: Jon McGuire Date: Mon, 8 Oct 2018 14:03:22 -0400 Subject: [PATCH 05/18] include unit-tests per discussion in tracking issue --- appveyor.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/appveyor.yml b/appveyor.yml index 9dc3dbe..ea3b06b 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -13,6 +13,7 @@ only_commits: - src/Serilog.Extensions.Hosting/ - Build.ps1 - assets/ + - test/Serilog.Extensions.Hosting.Tests/ deploy: - provider: NuGet api_key: From 0dac1ce0455e5a36ed8592a67d320f8de77848eb Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Sun, 23 Jun 2019 12:19:55 +1000 Subject: [PATCH 06/18] Pull changes from https://github.com/serilog/serilog-aspnetcore/pull/87 - basic provider support --- .../SimpleServiceSample/PrintTimeService.cs | 2 +- samples/SimpleServiceSample/Program.cs | 6 +- serilog-extensions-hosting.sln.DotSettings | 3 + .../Hosting/SerilogLoggerFactory.cs | 7 +- .../Serilog.Extensions.Hosting.csproj | 6 +- .../SerilogHostBuilderExtensions.cs | 89 +++++++++++++++---- 6 files changed, 87 insertions(+), 26 deletions(-) create mode 100644 serilog-extensions-hosting.sln.DotSettings rename src/Serilog.Extensions.Hosting/{ => Extensions}/Hosting/SerilogLoggerFactory.cs (89%) diff --git a/samples/SimpleServiceSample/PrintTimeService.cs b/samples/SimpleServiceSample/PrintTimeService.cs index aa59fd0..513cace 100644 --- a/samples/SimpleServiceSample/PrintTimeService.cs +++ b/samples/SimpleServiceSample/PrintTimeService.cs @@ -4,7 +4,7 @@ using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; -namespace SimpleWebSample +namespace SimpleServiceSample { public class PrintTimeService : IHostedService, IDisposable { diff --git a/samples/SimpleServiceSample/Program.cs b/samples/SimpleServiceSample/Program.cs index b357b10..4ff6e7e 100644 --- a/samples/SimpleServiceSample/Program.cs +++ b/samples/SimpleServiceSample/Program.cs @@ -1,11 +1,11 @@ using System; using System.IO; -using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Configuration; -using Serilog; using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using Serilog; -namespace SimpleWebSample +namespace SimpleServiceSample { public class Program { diff --git a/serilog-extensions-hosting.sln.DotSettings b/serilog-extensions-hosting.sln.DotSettings new file mode 100644 index 0000000..1753e50 --- /dev/null +++ b/serilog-extensions-hosting.sln.DotSettings @@ -0,0 +1,3 @@ + + True + True \ No newline at end of file diff --git a/src/Serilog.Extensions.Hosting/Hosting/SerilogLoggerFactory.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs similarity index 89% rename from src/Serilog.Extensions.Hosting/Hosting/SerilogLoggerFactory.cs rename to src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs index 1c91166..da5ddf7 100644 --- a/src/Serilog.Extensions.Hosting/Hosting/SerilogLoggerFactory.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs @@ -12,18 +12,23 @@ // See the License for the specific language governing permissions and // limitations under the License. +using System; using Microsoft.Extensions.Logging; using Serilog.Debugging; using Serilog.Extensions.Logging; +// To line up with the convention used elsewhere in the *.Extensions libraries, this +// should have been Serilog.Extensions.Hosting. +// ReSharper disable once CheckNamespace namespace Serilog.Hosting { /// /// Implements so that we can inject Serilog Logger. /// + [Obsolete("Replaced with Serilog.Extensions.Logging.SerilogLoggerFactory")] public class SerilogLoggerFactory : ILoggerFactory { - private readonly SerilogLoggerProvider _provider; + readonly SerilogLoggerProvider _provider; /// /// Initializes a new instance of the class. diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index c343413..5a96a0a 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -2,7 +2,7 @@ Serilog support for .NET Core logging in hosted services - 2.0.1 + 3.0.0 Microsoft;Serilog Contributors netstandard2.0 true @@ -23,8 +23,8 @@ - - + + diff --git a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs index 804b8c3..79533d2 100644 --- a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs +++ b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs @@ -1,4 +1,4 @@ -// Copyright 2018 Serilog Contributors +// Copyright 2019 Serilog Contributors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. @@ -15,8 +15,8 @@ using System; using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; -using Serilog.Hosting; using Microsoft.Extensions.DependencyInjection; +using Serilog.Extensions.Logging; namespace Serilog { @@ -30,21 +30,45 @@ public static class SerilogHostBuilderExtensions /// /// The host builder to configure. /// The Serilog logger; if not supplied, the static will be used. - /// When true, dispose when the framework disposes the provider. If the - /// logger is not specified but is true, the method will be + /// When true, dispose when the framework disposes the provider. If the + /// logger is not specified but is true, the method will be /// called on the static class instead. - /// The (generic) host builder. - public static IHostBuilder UseSerilog(this IHostBuilder builder, Serilog.ILogger logger = null, bool dispose = false) + /// A registered in the Serilog pipeline using the + /// WriteTo.Providers() configuration method, enabling other s to receive events. By + /// default, only Serilog sinks will receive events. + /// The host builder. + public static IHostBuilder UseSerilog( + this IHostBuilder builder, + ILogger logger = null, + bool dispose = false, + LoggerProviderCollection providers = null) { if (builder == null) throw new ArgumentNullException(nameof(builder)); - builder.ConfigureServices((context, collection) => - collection.AddSingleton(services => new SerilogLoggerFactory(logger, dispose))); + + builder.ConfigureServices((_, collection) => + { + if (providers != null) + { + collection.AddSingleton(services => + { + var factory = new SerilogLoggerFactory(logger, dispose, providers); + + foreach (var provider in services.GetServices()) + factory.AddProvider(provider); + + return factory; + }); + } + else + { + collection.AddSingleton(services => new SerilogLoggerFactory(logger, dispose)); + } + }); + return builder; } - /// - /// Sets Serilog as the logging provider. - /// + /// Sets Serilog as the logging provider. /// /// A is supplied so that configuration and hosting information can be used. /// The logger will be shut down when application services are disposed. @@ -52,27 +76,56 @@ public static IHostBuilder UseSerilog(this IHostBuilder builder, Serilog.ILogger /// The host builder to configure. /// The delegate for configuring the that will be used to construct a . /// Indicates whether to preserve the value of . - /// The (generic) host builder. - public static IHostBuilder UseSerilog(this IHostBuilder builder, Action configureLogger, bool preserveStaticLogger = false) + /// By default, Serilog does not write events to s registered through + /// the Microsoft.Extensions.Logging API. Normally, equivalent Serilog sinks are used in place of providers. Specify + /// true to write events to all providers. + /// The host builder. + public static IHostBuilder UseSerilog( + this IHostBuilder builder, + Action configureLogger, + bool preserveStaticLogger = false, + bool writeToProviders = false) { if (builder == null) throw new ArgumentNullException(nameof(builder)); if (configureLogger == null) throw new ArgumentNullException(nameof(configureLogger)); builder.ConfigureServices((context, collection) => { var loggerConfiguration = new LoggerConfiguration(); + + LoggerProviderCollection loggerProviders = null; + if (writeToProviders) + { + loggerProviders = new LoggerProviderCollection(); + loggerConfiguration.WriteTo.Providers(loggerProviders); + } + configureLogger(context, loggerConfiguration); var logger = loggerConfiguration.CreateLogger(); + + ILogger registeredLogger = null; if (preserveStaticLogger) - { - collection.AddSingleton(services => new SerilogLoggerFactory(logger, true)); - } - else { // Passing a `null` logger to `SerilogLoggerFactory` results in disposal via // `Log.CloseAndFlush()`, which additionally replaces the static logger with a no-op. Log.Logger = logger; - collection.AddSingleton(services => new SerilogLoggerFactory(null, true)); } + else + { + registeredLogger = logger; + } + + collection.AddSingleton(services => + { + var factory = new SerilogLoggerFactory(registeredLogger, true, loggerProviders); + + if (writeToProviders) + { + foreach (var provider in services.GetServices()) + factory.AddProvider(provider); + } + + return factory; + }); }); return builder; } From 1f401ac2e6afcb8b959a8c6e3d7c0062bc9dcbd8 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Sun, 23 Jun 2019 12:43:44 +1000 Subject: [PATCH 07/18] Fix flipped conditional --- .../SerilogHostBuilderExtensions.cs | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs index 79533d2..23fecea 100644 --- a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs +++ b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs @@ -88,6 +88,7 @@ public static IHostBuilder UseSerilog( { if (builder == null) throw new ArgumentNullException(nameof(builder)); if (configureLogger == null) throw new ArgumentNullException(nameof(configureLogger)); + builder.ConfigureServices((context, collection) => { var loggerConfiguration = new LoggerConfiguration(); @@ -105,13 +106,13 @@ public static IHostBuilder UseSerilog( ILogger registeredLogger = null; if (preserveStaticLogger) { - // Passing a `null` logger to `SerilogLoggerFactory` results in disposal via - // `Log.CloseAndFlush()`, which additionally replaces the static logger with a no-op. - Log.Logger = logger; + registeredLogger = logger; } else { - registeredLogger = logger; + // Passing a `null` logger to `SerilogLoggerFactory` results in disposal via + // `Log.CloseAndFlush()`, which additionally replaces the static logger with a no-op. + Log.Logger = logger; } collection.AddSingleton(services => From ac4f4cd4a7d2bd462881d3e44fc1d273ad302d2f Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Sun, 23 Jun 2019 16:17:35 +1000 Subject: [PATCH 08/18] Diagnostic context infra for downstream middleware --- global.json | 2 +- serilog-extensions-hosting.sln.DotSettings | 2 + .../AmbientDiagnosticContextCollector.cs | 32 ++++++++++ .../Extensions/Hosting/DiagnosticContext.cs | 59 +++++++++++++++++++ .../Hosting/DiagnosticContextCollector.cs | 51 ++++++++++++++++ .../Hosting/SerilogLoggerFactory.cs | 2 + .../IDiagnosticContext.cs | 31 ++++++++++ .../SerilogHostBuilderExtensions.cs | 9 +++ .../DiagnosticContextTests.cs | 38 ++++++++++++ .../Serilog.Extensions.Hosting.Tests.csproj | 11 ++-- .../SerilogHostBuilderExtensionsTests.cs | 16 ----- .../Support/Some.cs | 21 +++++++ 12 files changed, 253 insertions(+), 21 deletions(-) create mode 100644 src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs create mode 100644 src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs create mode 100644 src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs create mode 100644 src/Serilog.Extensions.Hosting/IDiagnosticContext.cs create mode 100644 test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs delete mode 100644 test/Serilog.Extensions.Hosting.Tests/SerilogHostBuilderExtensionsTests.cs create mode 100644 test/Serilog.Extensions.Hosting.Tests/Support/Some.cs diff --git a/global.json b/global.json index 6b51ddf..0a37afd 100644 --- a/global.json +++ b/global.json @@ -1,5 +1,5 @@ { "sdk": { - "version": "2.1.300-*" + "version": "2.2.105" } } diff --git a/serilog-extensions-hosting.sln.DotSettings b/serilog-extensions-hosting.sln.DotSettings index 1753e50..87d9b7b 100644 --- a/serilog-extensions-hosting.sln.DotSettings +++ b/serilog-extensions-hosting.sln.DotSettings @@ -1,3 +1,5 @@  True + True + True True \ No newline at end of file diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs new file mode 100644 index 0000000..acc1ccd --- /dev/null +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs @@ -0,0 +1,32 @@ +using System; +using System.Threading; + +namespace Serilog.Extensions.Hosting +{ + class AmbientDiagnosticContextCollector : IDisposable + { + static readonly AsyncLocal AmbientCollector = + new AsyncLocal(); + + // The indirection here ensures that completing collection cleans up the collector in all + // execution contexts. Via @benaadams' addition to `HttpContextAccessor` :-) + DiagnosticContextCollector _collector; + + public static DiagnosticContextCollector Current => AmbientCollector.Value?._collector; + + public static DiagnosticContextCollector Begin() + { + var value = new AmbientDiagnosticContextCollector(); + value._collector = new DiagnosticContextCollector(value); + AmbientCollector.Value = value; + return value._collector; + } + + public void Dispose() + { + _collector = null; + if (AmbientCollector.Value == this) + AmbientCollector.Value = null; + } + } +} diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs new file mode 100644 index 0000000..d6637a4 --- /dev/null +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs @@ -0,0 +1,59 @@ +// Copyright 2019 Serilog Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +using System; + +namespace Serilog.Extensions.Hosting +{ + /// + /// Implements an ambient/async-local diagnostic context. Consumers should + /// use in preference to this concrete type. + /// + public class DiagnosticContext : IDiagnosticContext + { + readonly ILogger _log; + + /// + /// Construct a . + /// + /// A logger for binding properties in the context, or null to use . + public DiagnosticContext(ILogger log) + { + _log = log; + } + + /// + /// Start collecting properties to associate with the current async context. This will replace + /// the active collector, if any. + /// + /// A collector that will receive events added in the current async context. + public DiagnosticContextCollector BeginCollection() + { + return AmbientDiagnosticContextCollector.Begin(); + } + + /// + public void Add(string propertyName, object value, bool destructureObjects = false) + { + if (propertyName == null) throw new ArgumentNullException(nameof(propertyName)); + + var collector = AmbientDiagnosticContextCollector.Current; + if (collector != null && + (_log ?? Log.Logger).BindProperty(propertyName, value, destructureObjects, out var property)) + { + collector.Add(property); + } + } + } +} diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs new file mode 100644 index 0000000..dd92af7 --- /dev/null +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs @@ -0,0 +1,51 @@ +using System; +using System.Collections.Generic; +using Serilog.Events; + +namespace Serilog.Extensions.Hosting +{ + /// + /// A container that receives properties added to a diagnostic context. + /// + public sealed class DiagnosticContextCollector : IDisposable + { + readonly AmbientDiagnosticContextCollector _ambientCollector; + List _properties = new List(); + + internal DiagnosticContextCollector(AmbientDiagnosticContextCollector ambientCollector) + { + _ambientCollector = ambientCollector ?? throw new ArgumentNullException(nameof(ambientCollector)); + } + + /// + /// Add the property to the context. + /// + /// The property to add. + public void Add(LogEventProperty property) + { + if (property == null) throw new ArgumentNullException(nameof(property)); + _properties?.Add(property); + } + + /// + /// Complete the context and retrieve the properties added to it, if any. This will + /// stop collection and remove the collector from the original execution context and + /// any of its children. + /// + /// The collected properties, or null if no collection is active. + /// True if properties could be collected. + public bool TryComplete(out List properties) + { + properties = _properties; + _properties = null; + Dispose(); + return properties != null; + } + + /// + public void Dispose() + { + _ambientCollector.Dispose(); + } + } +} diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs index da5ddf7..1b100b9 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/SerilogLoggerFactory.cs @@ -13,6 +13,7 @@ // limitations under the License. using System; +using System.ComponentModel; using Microsoft.Extensions.Logging; using Serilog.Debugging; using Serilog.Extensions.Logging; @@ -26,6 +27,7 @@ namespace Serilog.Hosting /// Implements so that we can inject Serilog Logger. /// [Obsolete("Replaced with Serilog.Extensions.Logging.SerilogLoggerFactory")] + [EditorBrowsable(EditorBrowsableState.Never)] public class SerilogLoggerFactory : ILoggerFactory { readonly SerilogLoggerProvider _provider; diff --git a/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs b/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs new file mode 100644 index 0000000..5b472c4 --- /dev/null +++ b/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs @@ -0,0 +1,31 @@ +// Copyright 2019 Serilog Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +namespace Serilog +{ + /// + /// Collects diagnostic information for packaging into wide events. + /// + public interface IDiagnosticContext + { + /// + /// Add the specified property to the request's diagnostic payload. + /// + /// The name of the property. Must be non-empty. + /// The property value. + /// If true, the value will be serialized as structured + /// data if possible; if false, the object will be recorded as a scalar or simple array. + void Add(string propertyName, object value, bool destructureObjects = false); + } +} diff --git a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs index 23fecea..805e4b8 100644 --- a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs +++ b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs @@ -63,6 +63,12 @@ public static IHostBuilder UseSerilog( { collection.AddSingleton(services => new SerilogLoggerFactory(logger, dispose)); } + + if (logger != null) + { + // This won't (and shouldn't) take ownership of the logger. + collection.AddSingleton(logger); + } }); return builder; @@ -103,6 +109,9 @@ public static IHostBuilder UseSerilog( configureLogger(context, loggerConfiguration); var logger = loggerConfiguration.CreateLogger(); + // This won't (and shouldn't) take ownership of the logger. + collection.AddSingleton(logger); + ILogger registeredLogger = null; if (preserveStaticLogger) { diff --git a/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs b/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs new file mode 100644 index 0000000..834492c --- /dev/null +++ b/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs @@ -0,0 +1,38 @@ +using System; +using System.Threading.Tasks; +using Serilog.Extensions.Hosting.Tests.Support; +using Xunit; + +namespace Serilog.Extensions.Hosting.Tests +{ + public class DiagnosticContextTests + { + [Fact] + public void AddIsSafeWhenNoContextIsActive() + { + var dc = new DiagnosticContext(Some.Logger()); + dc.Add(Some.String("name"), Some.Int32()); + } + + [Fact] + public async Task PropertiesAreCollectedInAnActiveContext() + { + var dc = new DiagnosticContext(Some.Logger()); + + var collector = dc.BeginCollection(); + dc.Add(Some.String("name"), Some.Int32()); + await Task.Delay(TimeSpan.FromMilliseconds(10)); + dc.Add(Some.String("name"), Some.Int32()); + + Assert.True(collector.TryComplete(out var properties)); + Assert.Equal(2, properties.Count); + + Assert.False(collector.TryComplete(out _)); + + collector.Dispose(); + + dc.Add(Some.String("name"), Some.Int32()); + Assert.False(collector.TryComplete(out _)); + } + } +} diff --git a/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj b/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj index bdbdbcc..f8952eb 100644 --- a/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj +++ b/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj @@ -1,7 +1,7 @@  - netcoreapp2.1 + netcoreapp2.2 Serilog.Extensions.Hosting.Tests ../../assets/Serilog.snk true @@ -14,9 +14,12 @@ - - - + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + diff --git a/test/Serilog.Extensions.Hosting.Tests/SerilogHostBuilderExtensionsTests.cs b/test/Serilog.Extensions.Hosting.Tests/SerilogHostBuilderExtensionsTests.cs deleted file mode 100644 index 3341f1d..0000000 --- a/test/Serilog.Extensions.Hosting.Tests/SerilogHostBuilderExtensionsTests.cs +++ /dev/null @@ -1,16 +0,0 @@ -// Copyright (c) .NET Foundation. All rights reserved. -// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. - -using Xunit; - -namespace Serilog.Extensions.Hosting.Tests -{ - public class SerilogHostBuilderExtensionsTests - { - [Fact] - public void Todo() - { - - } - } -} \ No newline at end of file diff --git a/test/Serilog.Extensions.Hosting.Tests/Support/Some.cs b/test/Serilog.Extensions.Hosting.Tests/Support/Some.cs new file mode 100644 index 0000000..5137cef --- /dev/null +++ b/test/Serilog.Extensions.Hosting.Tests/Support/Some.cs @@ -0,0 +1,21 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Threading; +using Serilog.Events; + +namespace Serilog.Extensions.Hosting.Tests.Support +{ + static class Some + { + static int _next; + + public static int Int32() => Interlocked.Increment(ref _next); + + public static string String(string tag = null) => $"s_{tag}{Int32()}"; + + public static LogEventProperty LogEventProperty() => new LogEventProperty(String("name"), new ScalarValue(Int32())); + + public static ILogger Logger() => new LoggerConfiguration().CreateLogger(); + } +} From c396419d12d3349f37d29e2c7aaf0f61f38e05c8 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Sun, 23 Jun 2019 16:47:54 +1000 Subject: [PATCH 09/18] Register diagnostic services; includes ILogger now in registered services since there's little reason not to provide it --- .../Extensions/Hosting/DiagnosticContext.cs | 10 +++--- .../SerilogHostBuilderExtensions.cs | 34 ++++++++++++++----- 2 files changed, 30 insertions(+), 14 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs index d6637a4..0e5e602 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs @@ -22,15 +22,15 @@ namespace Serilog.Extensions.Hosting /// public class DiagnosticContext : IDiagnosticContext { - readonly ILogger _log; + readonly ILogger _logger; /// /// Construct a . /// - /// A logger for binding properties in the context, or null to use . - public DiagnosticContext(ILogger log) + /// A logger for binding properties in the context, or null to use . + public DiagnosticContext(ILogger logger) { - _log = log; + _logger = logger; } /// @@ -50,7 +50,7 @@ public void Add(string propertyName, object value, bool destructureObjects = fal var collector = AmbientDiagnosticContextCollector.Current; if (collector != null && - (_log ?? Log.Logger).BindProperty(propertyName, value, destructureObjects, out var property)) + (_logger ?? Log.Logger).BindProperty(propertyName, value, destructureObjects, out var property)) { collector.Add(property); } diff --git a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs index 805e4b8..c0d1661 100644 --- a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs +++ b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs @@ -16,6 +16,7 @@ using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; using Microsoft.Extensions.DependencyInjection; +using Serilog.Extensions.Hosting; using Serilog.Extensions.Logging; namespace Serilog @@ -64,11 +65,7 @@ public static IHostBuilder UseSerilog( collection.AddSingleton(services => new SerilogLoggerFactory(logger, dispose)); } - if (logger != null) - { - // This won't (and shouldn't) take ownership of the logger. - collection.AddSingleton(logger); - } + ConfigureServices(collection, logger); }); return builder; @@ -80,7 +77,7 @@ public static IHostBuilder UseSerilog( /// The logger will be shut down when application services are disposed. /// /// The host builder to configure. - /// The delegate for configuring the that will be used to construct a . + /// The delegate for configuring the that will be used to construct a . /// Indicates whether to preserve the value of . /// By default, Serilog does not write events to s registered through /// the Microsoft.Extensions.Logging API. Normally, equivalent Serilog sinks are used in place of providers. Specify @@ -109,9 +106,6 @@ public static IHostBuilder UseSerilog( configureLogger(context, loggerConfiguration); var logger = loggerConfiguration.CreateLogger(); - // This won't (and shouldn't) take ownership of the logger. - collection.AddSingleton(logger); - ILogger registeredLogger = null; if (preserveStaticLogger) { @@ -136,8 +130,30 @@ public static IHostBuilder UseSerilog( return factory; }); + + ConfigureServices(collection, logger); }); return builder; } + + static void ConfigureServices(IServiceCollection collection, ILogger logger) + { + if (collection == null) throw new ArgumentNullException(nameof(collection)); + + if (logger != null) + { + // This won't (and shouldn't) take ownership of the logger. + collection.AddSingleton(logger); + } + + // Registered to provide two services... + var diagnosticContext = new DiagnosticContext(logger); + + // Consumed by e.g. middleware + collection.AddSingleton(diagnosticContext); + + // Consumed by user code + collection.AddSingleton(diagnosticContext); + } } } From 414a0098fa486b66469726a40e242ae9b7854b1f Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Sun, 23 Jun 2019 16:51:27 +1000 Subject: [PATCH 10/18] Reapply some doc improvements --- .../SerilogHostBuilderExtensions.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs index c0d1661..306eadd 100644 --- a/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs +++ b/src/Serilog.Extensions.Hosting/SerilogHostBuilderExtensions.cs @@ -31,8 +31,8 @@ public static class SerilogHostBuilderExtensions /// /// The host builder to configure. /// The Serilog logger; if not supplied, the static will be used. - /// When true, dispose when the framework disposes the provider. If the - /// logger is not specified but is true, the method will be + /// When true, dispose when the framework disposes the provider. If the + /// logger is not specified but is true, the method will be /// called on the static class instead. /// A registered in the Serilog pipeline using the /// WriteTo.Providers() configuration method, enabling other s to receive events. By From d691f3017fbdc7e46bf7a04b6bca0fa77562889b Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Tue, 25 Jun 2019 09:43:01 +1000 Subject: [PATCH 11/18] NuGet publishing key --- appveyor.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/appveyor.yml b/appveyor.yml index ea3b06b..70e4cbc 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -17,7 +17,7 @@ only_commits: deploy: - provider: NuGet api_key: - secure: bd9z4P73oltOXudAjPehwp9iDKsPtC+HbgshOrSgoyQKr5xVK+bxJQngrDJkHdY8 + secure: JAnOxPQMZRp8ousbAW4vYnVtd936AOMp4gDGucCQ1RZrDZW3J+X5QmbLO7OJKtBr skip_symbols: true on: branch: /^(master|dev)$/ From 3b88ae0558d33f52c98f67b1bb0701e1eaf1ca5d Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Tue, 25 Jun 2019 09:45:32 +1000 Subject: [PATCH 12/18] Fix commit filter so that appveyor.yml changes trigger a rebuild --- appveyor.yml | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/appveyor.yml b/appveyor.yml index 70e4cbc..9ca24f7 100644 --- a/appveyor.yml +++ b/appveyor.yml @@ -7,13 +7,9 @@ build_script: - ps: ./Build.ps1 artifacts: - path: artifacts/Serilog.*.nupkg -only_commits: +skip_commits: files: - - serilog-extensions-hosting.sln - - src/Serilog.Extensions.Hosting/ - - Build.ps1 - - assets/ - - test/Serilog.Extensions.Hosting.Tests/ + - README.md deploy: - provider: NuGet api_key: From cee6a1abb948c9752a2b0c20e0a2936aab7652a9 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Wed, 26 Jun 2019 10:51:27 +1000 Subject: [PATCH 13/18] Make DiagnosticContextCollector thread-safe --- .../Hosting/DiagnosticContextCollector.cs | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs index dd92af7..11bab25 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs @@ -10,6 +10,7 @@ namespace Serilog.Extensions.Hosting public sealed class DiagnosticContextCollector : IDisposable { readonly AmbientDiagnosticContextCollector _ambientCollector; + readonly object _propertiesLock = new object(); List _properties = new List(); internal DiagnosticContextCollector(AmbientDiagnosticContextCollector ambientCollector) @@ -24,7 +25,11 @@ internal DiagnosticContextCollector(AmbientDiagnosticContextCollector ambientCol public void Add(LogEventProperty property) { if (property == null) throw new ArgumentNullException(nameof(property)); - _properties?.Add(property); + + lock (_propertiesLock) + { + _properties?.Add(property); + } } /// @@ -36,10 +41,13 @@ public void Add(LogEventProperty property) /// True if properties could be collected. public bool TryComplete(out List properties) { - properties = _properties; - _properties = null; - Dispose(); - return properties != null; + lock (_propertiesLock) + { + properties = _properties; + _properties = null; + Dispose(); + return properties != null; + } } /// From 3a5f0795c3639d6ea45341b61dce6079216a031a Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Wed, 26 Jun 2019 14:25:07 +1000 Subject: [PATCH 14/18] `IDiagnosticContext.Add()` -> `IDiagnosticContext.Set()`; more small tweaks --- .../AmbientDiagnosticContextCollector.cs | 16 +++++++++- .../Extensions/Hosting/DiagnosticContext.cs | 17 +++++----- .../Hosting/DiagnosticContextCollector.cs | 28 +++++++++++++---- .../IDiagnosticContext.cs | 5 +-- .../DiagnosticContextTests.cs | 31 +++++++++++++++---- 5 files changed, 74 insertions(+), 23 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs index acc1ccd..68010ee 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/AmbientDiagnosticContextCollector.cs @@ -1,4 +1,18 @@ -using System; +// Copyright 2019 Serilog Contributors +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +using System; using System.Threading; namespace Serilog.Extensions.Hosting diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs index 0e5e602..7c12c19 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContext.cs @@ -13,14 +13,15 @@ // limitations under the License. using System; +using System.Threading; namespace Serilog.Extensions.Hosting { /// - /// Implements an ambient/async-local diagnostic context. Consumers should - /// use in preference to this concrete type. + /// Implements an ambient diagnostic context using . /// - public class DiagnosticContext : IDiagnosticContext + /// Consumers should use to set context properties. + public sealed class DiagnosticContext : IDiagnosticContext { readonly ILogger _logger; @@ -34,17 +35,17 @@ public DiagnosticContext(ILogger logger) } /// - /// Start collecting properties to associate with the current async context. This will replace + /// Start collecting properties to associate with the current diagnostic context. This will replace /// the active collector, if any. /// - /// A collector that will receive events added in the current async context. + /// A collector that will receive properties added in the current diagnostic context. public DiagnosticContextCollector BeginCollection() { return AmbientDiagnosticContextCollector.Begin(); } - /// - public void Add(string propertyName, object value, bool destructureObjects = false) + /// + public void Set(string propertyName, object value, bool destructureObjects = false) { if (propertyName == null) throw new ArgumentNullException(nameof(propertyName)); @@ -52,7 +53,7 @@ public void Add(string propertyName, object value, bool destructureObjects = fal if (collector != null && (_logger ?? Log.Logger).BindProperty(propertyName, value, destructureObjects, out var property)) { - collector.Add(property); + collector.AddOrUpdate(property); } } } diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs index 11bab25..5ebf216 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs @@ -9,26 +9,42 @@ namespace Serilog.Extensions.Hosting /// public sealed class DiagnosticContextCollector : IDisposable { - readonly AmbientDiagnosticContextCollector _ambientCollector; + readonly IDisposable _chainedDisposable; readonly object _propertiesLock = new object(); List _properties = new List(); - internal DiagnosticContextCollector(AmbientDiagnosticContextCollector ambientCollector) + /// + /// Construct a . + /// + /// An object that will be disposed to signal completion/disposal of + /// the collector. + public DiagnosticContextCollector(IDisposable chainedDisposable) { - _ambientCollector = ambientCollector ?? throw new ArgumentNullException(nameof(ambientCollector)); + _chainedDisposable = chainedDisposable ?? throw new ArgumentNullException(nameof(chainedDisposable)); } /// /// Add the property to the context. /// /// The property to add. - public void Add(LogEventProperty property) + public void AddOrUpdate(LogEventProperty property) { if (property == null) throw new ArgumentNullException(nameof(property)); lock (_propertiesLock) { - _properties?.Add(property); + if (_properties == null) return; + + for (var i = 0; i < _properties.Count; ++i) + { + if (_properties[i].Name == property.Name) + { + _properties[i] = property; + return; + } + } + + _properties.Add(property); } } @@ -53,7 +69,7 @@ public bool TryComplete(out List properties) /// public void Dispose() { - _ambientCollector.Dispose(); + _chainedDisposable.Dispose(); } } } diff --git a/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs b/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs index 5b472c4..468fa84 100644 --- a/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs +++ b/src/Serilog.Extensions.Hosting/IDiagnosticContext.cs @@ -20,12 +20,13 @@ namespace Serilog public interface IDiagnosticContext { /// - /// Add the specified property to the request's diagnostic payload. + /// Set the specified property on the current diagnostic context. The property will be collected + /// and attached to the event emitted at the completion of the context. /// /// The name of the property. Must be non-empty. /// The property value. /// If true, the value will be serialized as structured /// data if possible; if false, the object will be recorded as a scalar or simple array. - void Add(string propertyName, object value, bool destructureObjects = false); + void Set(string propertyName, object value, bool destructureObjects = false); } } diff --git a/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs b/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs index 834492c..c32f9ea 100644 --- a/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs +++ b/test/Serilog.Extensions.Hosting.Tests/DiagnosticContextTests.cs @@ -1,17 +1,21 @@ using System; +using System.Linq; using System.Threading.Tasks; +using Serilog.Events; using Serilog.Extensions.Hosting.Tests.Support; using Xunit; +// ReSharper disable PossibleNullReferenceException + namespace Serilog.Extensions.Hosting.Tests { public class DiagnosticContextTests { [Fact] - public void AddIsSafeWhenNoContextIsActive() + public void SetIsSafeWhenNoContextIsActive() { var dc = new DiagnosticContext(Some.Logger()); - dc.Add(Some.String("name"), Some.Int32()); + dc.Set(Some.String("name"), Some.Int32()); } [Fact] @@ -20,19 +24,34 @@ public async Task PropertiesAreCollectedInAnActiveContext() var dc = new DiagnosticContext(Some.Logger()); var collector = dc.BeginCollection(); - dc.Add(Some.String("name"), Some.Int32()); + dc.Set(Some.String("first"), Some.Int32()); await Task.Delay(TimeSpan.FromMilliseconds(10)); - dc.Add(Some.String("name"), Some.Int32()); + dc.Set(Some.String("second"), Some.Int32()); Assert.True(collector.TryComplete(out var properties)); - Assert.Equal(2, properties.Count); + Assert.Equal(2, properties.Count()); Assert.False(collector.TryComplete(out _)); collector.Dispose(); - dc.Add(Some.String("name"), Some.Int32()); + dc.Set(Some.String("third"), Some.Int32()); Assert.False(collector.TryComplete(out _)); } + + [Fact] + public void ExistingPropertiesCanBeUpdated() + { + var dc = new DiagnosticContext(Some.Logger()); + + var collector = dc.BeginCollection(); + dc.Set("name", 10); + dc.Set("name", 20); + + Assert.True(collector.TryComplete(out var properties)); + var prop = Assert.Single(properties); + var scalar = Assert.IsType(prop.Value); + Assert.Equal(20, scalar.Value); + } } } From df57744a852f96eb1af3c70808d4effa64fd0b2e Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Wed, 26 Jun 2019 14:35:55 +1000 Subject: [PATCH 15/18] Probably not paranoid here to trade some actual performance for better worst-case execution time --- .../Hosting/DiagnosticContextCollector.cs | 18 ++++-------------- 1 file changed, 4 insertions(+), 14 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs index 5ebf216..d1aba65 100644 --- a/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs +++ b/src/Serilog.Extensions.Hosting/Extensions/Hosting/DiagnosticContextCollector.cs @@ -11,7 +11,7 @@ public sealed class DiagnosticContextCollector : IDisposable { readonly IDisposable _chainedDisposable; readonly object _propertiesLock = new object(); - List _properties = new List(); + Dictionary _properties = new Dictionary(); /// /// Construct a . @@ -34,17 +34,7 @@ public void AddOrUpdate(LogEventProperty property) lock (_propertiesLock) { if (_properties == null) return; - - for (var i = 0; i < _properties.Count; ++i) - { - if (_properties[i].Name == property.Name) - { - _properties[i] = property; - return; - } - } - - _properties.Add(property); + _properties[property.Name] = property; } } @@ -55,11 +45,11 @@ public void AddOrUpdate(LogEventProperty property) /// /// The collected properties, or null if no collection is active. /// True if properties could be collected. - public bool TryComplete(out List properties) + public bool TryComplete(out IEnumerable properties) { lock (_propertiesLock) { - properties = _properties; + properties = _properties?.Values; _properties = null; Dispose(); return properties != null; From 55bb2a572ffb503bd81e0749ba7c56f31e552719 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Wed, 26 Jun 2019 14:51:32 +1000 Subject: [PATCH 16/18] README trivia --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index f99b283..259ae32 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ # Serilog.Extensions.Hosting [![Build status](https://ci.appveyor.com/api/projects/status/ue4s7htjwj88fulh?svg=true)](https://ci.appveyor.com/project/serilog/serilog-extensions-hosting) [![NuGet Version](http://img.shields.io/nuget/v/Serilog.Extensions.Hosting.svg?style=flat)](https://www.nuget.org/packages/Serilog.Extensions.Hosting/) -Serilog logging for Microsoft.Extensions.Hosting . This package routes Microsoft.Extensions.Hosting log messages through Serilog, so you can get information about the framework's internal operations logged to the same Serilog sinks as your application events. +Serilog logging for _Microsoft.Extensions.Hosting_. This package routes framework log messages through Serilog, so you can get information about the framework's internal operations written to the same Serilog sinks as your application events. ### Instructions From 8a77959ff291b85d75f5e267a7a1ec28203357ff Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Tue, 20 Aug 2019 14:13:54 +1000 Subject: [PATCH 17/18] Update to Serilog.Extensions.Logging 3.0; update other non-shipping dependencies --- README.md | 38 +++---------------- .../SimpleServiceSample.csproj | 12 +++--- .../Serilog.Extensions.Hosting.csproj | 2 +- .../Serilog.Extensions.Hosting.Tests.csproj | 2 +- 4 files changed, 14 insertions(+), 40 deletions(-) diff --git a/README.md b/README.md index 259ae32..eb17258 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,8 @@ Serilog logging for _Microsoft.Extensions.Hosting_. This package routes framework log messages through Serilog, so you can get information about the framework's internal operations written to the same Serilog sinks as your application events. +**ASP.NET Core** applications should consider [using _Serilog.AspNetCore_ instead](https://github.com/serilog/serilog-aspnetcore), which bundles this package and includes other ASP.NET Core-specific features. + ### Instructions **First**, install the _Serilog.Extensions.Hosting_ [NuGet package](https://www.nuget.org/packages/Serilog.Extensions.Hosting) into your app. You will need a way to view the log messages - _Serilog.Sinks.Console_ writes these to the console; there are [many more sinks available](https://www.nuget.org/packages?q=Tags%3A%22serilog%22) on NuGet. @@ -63,7 +65,7 @@ That's it! You will see log output like: [22:10:39 INF] The current time is: 12/05/2018 10:10:39 +00:00 ``` -A more complete example, showing _appsettings.json_ configuration, can be found in [the sample project here](https://github.com/serilog/serilog-hostinh/tree/dev/samples/SimpleServiceSample). +A more complete example, showing _appsettings.json_ configuration, can be found in [the sample project here](https://github.com/serilog/serilog-hosting/tree/dev/samples/SimpleServiceSample). ### Using the package @@ -78,39 +80,11 @@ You can alternatively configure Serilog using a delegate as shown below: ```csharp // dotnet add package Serilog.Settings.Configuration .UseSerilog((hostingContext, loggerConfiguration) => loggerConfiguration - .ReadFrom.Configuration(hostingContext.Configuration) - .Enrich.FromLogContext() - .WriteTo.Console()) + .ReadFrom.Configuration(hostingContext.Configuration) + .Enrich.FromLogContext() + .WriteTo.Console()) ``` This has the advantage of making the `hostingContext`'s `Configuration` object available for configuration of the logger, but at the expense of recording `Exception`s raised earlier in program startup. If this method is used, `Log.Logger` is assigned implicitly, and closed when the app is shut down. - -### Writing to the Azure Diagnostics Log Stream - -The Azure Diagnostic Log Stream ships events from any files in the `D:\home\LogFiles\` folder. To enable this for your app, first install the _Serilog.Sinks.File_ package: - -```powershell -Install-Package Serilog.Sinks.File -``` - -Then add a file sink to your `LoggerConfiguration`, taking care to set the `shared` and `flushToDiskInterval` parameters: - -```csharp - public static int Main(string[] args) - { - Log.Logger = new LoggerConfiguration() - .MinimumLevel.Debug() - .MinimumLevel.Override("Microsoft", LogEventLevel.Information) - .Enrich.FromLogContext() - .WriteTo.Console() - // Add this line: - .WriteTo.File( - @"D:\home\LogFiles\Application\myapp.txt", - fileSizeLimitBytes: 1_000_000, - rollOnFileSizeLimit: true, - shared: true, - flushToDiskInterval: TimeSpan.FromSeconds(1)) - .CreateLogger(); -``` diff --git a/samples/SimpleServiceSample/SimpleServiceSample.csproj b/samples/SimpleServiceSample/SimpleServiceSample.csproj index aadac55..36bbcf2 100644 --- a/samples/SimpleServiceSample/SimpleServiceSample.csproj +++ b/samples/SimpleServiceSample/SimpleServiceSample.csproj @@ -13,12 +13,12 @@ - - - - - - + + + + + + diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index 5a96a0a..1e9cf4d 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -24,7 +24,7 @@ - + diff --git a/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj b/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj index f8952eb..bb9c750 100644 --- a/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj +++ b/test/Serilog.Extensions.Hosting.Tests/Serilog.Extensions.Hosting.Tests.csproj @@ -14,7 +14,7 @@ - + all runtime; build; native; contentfiles; analyzers; buildtransitive From 463247162c395f3d9be614bf5dd5437d4d655a86 Mon Sep 17 00:00:00 2001 From: Nicholas Blumhardt Date: Thu, 22 Aug 2019 09:11:59 +1000 Subject: [PATCH 18/18] Play it safe, try to ensure the latest patch is installed --- .../Serilog.Extensions.Hosting.csproj | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj index 1e9cf4d..1c84fd6 100644 --- a/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj +++ b/src/Serilog.Extensions.Hosting/Serilog.Extensions.Hosting.csproj @@ -1,4 +1,4 @@ - + Serilog support for .NET Core logging in hosted services @@ -24,7 +24,7 @@ - +