using System.Diagnostics; using System.IO; using System.Runtime.CompilerServices; using NLog; using NLog.Config; using NLog.Targets; using Flow.Launcher.Infrastructure.UserSettings; using NLog.Targets.Wrappers; using System.Runtime.ExceptionServices; namespace Flow.Launcher.Infrastructure.Logger { public static class Log { public const string DirectoryName = "Logs"; public static string CurrentLogDirectory { get; } static Log() { CurrentLogDirectory = Path.Combine(DataLocation.DataDirectory(), DirectoryName, Constant.Version); if (!Directory.Exists(CurrentLogDirectory)) { Directory.CreateDirectory(CurrentLogDirectory); } var configuration = new LoggingConfiguration(); const string layout = @"${date:format=HH\:mm\:ss.ffffK} - " + @"${level:uppercase=true:padding=-5} - ${logger} - ${message:l}" + @"${onexception:${newline}" + @"EXCEPTION OCCURS\: ${exception:format=tostring}${newline}}"; var fileTarget = new FileTarget { FileName = CurrentLogDirectory.Replace(@"\", "/") + "/${shortdate}.txt", Layout = layout }; var fileTargetASyncWrapper = new AsyncTargetWrapper(fileTarget); var debugTarget = new OutputDebugStringTarget { Layout = layout }; configuration.AddTarget("file", fileTargetASyncWrapper); configuration.AddTarget("debug", debugTarget); var fileRule = new LoggingRule("*", LogLevel.Debug, fileTargetASyncWrapper) { RuleName = "file" }; #if DEBUG var debugRule = new LoggingRule("*", LogLevel.Debug, debugTarget) { RuleName = "debug" }; configuration.LoggingRules.Add(debugRule); #endif configuration.LoggingRules.Add(fileRule); LogManager.Configuration = configuration; } public static void SetLogLevel(LOGLEVEL level) { switch (level) { case LOGLEVEL.DEBUG: UseDebugLogLevel(); break; default: UseInfoLogLevel(); break; } Info(nameof(Logger), $"Using log level: {level}."); } private static void UseDebugLogLevel() { LogManager.Configuration.FindRuleByName("file").SetLoggingLevels(LogLevel.Debug, LogLevel.Fatal); } private static void UseInfoLogLevel() { LogManager.Configuration.FindRuleByName("file").SetLoggingLevels(LogLevel.Info, LogLevel.Fatal); } private static void LogFaultyFormat(string message) { var logger = LogManager.GetLogger("FaultyLogger"); message = $"Wrong logger message format <{message}>"; logger.Fatal(message); } private static bool FormatValid(string message) { var parts = message.Split('|'); var valid = parts.Length == 3 && !string.IsNullOrWhiteSpace(parts[1]) && !string.IsNullOrWhiteSpace(parts[2]); return valid; } public static void Exception(string className, string message, System.Exception exception, [CallerMemberName] string methodName = "") { exception = exception.Demystify(); #if DEBUG ExceptionDispatchInfo.Capture(exception).Throw(); #else var classNameWithMethod = CheckClassAndMessageAndReturnFullClassWithMethod(className, message, methodName); ExceptionInternal(classNameWithMethod, message, exception); #endif } private static string CheckClassAndMessageAndReturnFullClassWithMethod(string className, string message, string methodName) { if (string.IsNullOrWhiteSpace(className)) { LogFaultyFormat($"Fail to specify a class name during logging of message: {message ?? "no message entered"}"); } if (string.IsNullOrWhiteSpace(message)) { // todo: not sure we really need that LogFaultyFormat($"Fail to specify a message during logging"); } if (!string.IsNullOrWhiteSpace(methodName)) { return className + "." + methodName; } return className; } private static void ExceptionInternal(string classAndMethod, string message, System.Exception e) { var logger = LogManager.GetLogger(classAndMethod); logger.Error(e, message); } private static void LogInternal(string message, LogLevel level) { if (FormatValid(message)) { var parts = message.Split('|'); var prefix = parts[1]; var unprefixed = parts[2]; var logger = LogManager.GetLogger(prefix); logger.Log(level, unprefixed); } else { LogFaultyFormat(message); } } /// Example: "|ClassName.MethodName|Message" /// Example: "|ClassName.MethodName|Message" /// Exception public static void Exception(string message, System.Exception e) { e = e.Demystify(); #if DEBUG ExceptionDispatchInfo.Capture(e).Throw(); #else if (FormatValid(message)) { var parts = message.Split('|'); var prefix = parts[1]; var unprefixed = parts[2]; ExceptionInternal(prefix, unprefixed, e); } else { LogFaultyFormat(message); } #endif } /// Example: "|ClassName.MethodName|Message" public static void Error(string message) { LogInternal(message, LogLevel.Error); } public static void Error(string className, string message, [CallerMemberName] string methodName = "") { LogInternal(LogLevel.Error, className, message, methodName); } private static void LogInternal(LogLevel level, string className, string message, [CallerMemberName] string methodName = "") { var classNameWithMethod = CheckClassAndMessageAndReturnFullClassWithMethod(className, message, methodName); var logger = LogManager.GetLogger(classNameWithMethod); logger.Log(level, message); } public static void Debug(string className, string message, [CallerMemberName] string methodName = "") { LogInternal(LogLevel.Debug, className, message, methodName); } /// Example: "|ClassName.MethodName|Message"" public static void Debug(string message) { LogInternal(message, LogLevel.Debug); } public static void Info(string className, string message, [CallerMemberName] string methodName = "") { LogInternal(LogLevel.Info, className, message, methodName); } /// Example: "|ClassName.MethodName|Message" public static void Info(string message) { LogInternal(message, LogLevel.Info); } public static void Warn(string className, string message, [CallerMemberName] string methodName = "") { LogInternal(LogLevel.Warn, className, message, methodName); } /// Example: "|ClassName.MethodName|Message" public static void Warn(string message) { LogInternal(message, LogLevel.Warn); } } public enum LOGLEVEL { DEBUG, INFO } }