From 046952c95e7aa279725c37484074c5688a3df6c6 Mon Sep 17 00:00:00 2001 From: Bill Menees Date: Sat, 18 Jun 2022 18:10:11 -0500 Subject: [PATCH] Note thread safety bug in ImmediateConsoleLogger.Log --- tests/Menees.Remoting.TestHost/LogManager.cs | 26 +++++++++++++------- 1 file changed, 17 insertions(+), 9 deletions(-) diff --git a/tests/Menees.Remoting.TestHost/LogManager.cs b/tests/Menees.Remoting.TestHost/LogManager.cs index e51b742..157fa51 100644 --- a/tests/Menees.Remoting.TestHost/LogManager.cs +++ b/tests/Menees.Remoting.TestHost/LogManager.cs @@ -30,15 +30,19 @@ public LogManager(LogLevel minimumLogLevel = LogLevel.Debug, LogLevel consoleLog // Make the messages show up in the debugger's Output window. builder.AddDebug(); - // The Microsoft.Extensions.Logging.Console provider behaves terribly in multi-threaded unit tests. - // Even when recreating the ILoggerFactory for each test, it shares a single logging scope for all - // ILogger instances of the same type. So two PipeServer instances share the same logger instance - // and the same scope instance! That means BeginScope calls pile up when N instances are active. :-( - // https://codeburst.io/unit-testing-with-net-core-ilogger-t-e8c16c503a80 - // So I wrote my own ImmediateConsoleLoggerProvider that's aware of async call contexts. - // But since it doesn't do any buffering, it is very slow on large tests! - this.consoleLoggerProvider = new ImmediateConsoleLoggerProvider(consoleLogLevel); - builder.AddProvider(this.consoleLoggerProvider); + if (Convert.ToBoolean(1)) + { + builder.AddConsole(); + } + else + { + // The Microsoft.Extensions.Logging.Console provider behaves weird in multi-threaded unit tests due to buffering. + // https://codeburst.io/unit-testing-with-net-core-ilogger-t-e8c16c503a80 + // So I wrote my own ImmediateConsoleLoggerProvider that's aware of async call contexts. + // But since it doesn't do any buffering, it is VERY slow on large tests! + this.consoleLoggerProvider = new ImmediateConsoleLoggerProvider(consoleLogLevel); + builder.AddProvider(this.consoleLoggerProvider); + } }); } @@ -157,6 +161,10 @@ public void Log(LogLevel logLevel, EventId eventId, TState state, Except Stack? contextStack = this.callContextStack.Value; if (contextStack != null) { + // TODO: This is not thread-safe! AsyncLocal values can be accessed by multiple Tasks at once! [Bill, 6/18/2022] + // This got InvalidOperationException "Collection was modified after the enumerator was instantiated." during a + // MultiServerLarge test on .NET Framework. + // https://stackoverflow.com/a/63732530/1882616 foreach (object? value in contextStack) { sb.Append(' ');