TypeInitializationException в NLog LoggerFactory - PullRequest
0 голосов
/ 11 июня 2018

Я разрабатываю приложение WPF, которое использует NLog.Оно было развернуто для нескольких потенциальных клиентов, и в одном из них приложение работало в течение недели, а теперь даже не открывается.То есть вы дважды щелкаете значок приложения, и ничего не происходит, буквально.Даже в журнале внутри условия AppDomain.CurrentDomain.UnhandledException catch.

Мне удалось идентифицировать событие в средстве просмотра событий Windows (см. Сообщение ниже).

Что меня беспокоит, так это внезапное появлениеэтой ошибки после недели безупречной работы и моей неспособности интерпретировать это сообщение или найти информацию о нем в Интернете.

Aplicativo: ForceViewer.exe
Versão do Framework: v4.0.30319
Descrição: O processo foi terminado devido a uma exceção sem tratamento.
Informações da Exceção: System.Xml.XmlException
em System.Xml.XmlTextReaderImpl.Throw(System.Exception)
em System.Xml.XmlTextReaderImpl.ParseDocumentContent()
em System.Xml.XmlTextReaderImpl.Read()
em System.Xml.XmlTextReader.Read()
em System.Configuration.XmlUtil..ctor(System.IO.Stream, System.String, Boolean, System.Configuration.ConfigurationSchemaErrors)
em System.Configuration.BaseConfigurationRecord.InitConfigFromFile()

Informações da Exceção: System.Configuration.ConfigurationErrorsException
em System.Configuration.ConfigurationSchemaErrors.ThrowIfErrors(Boolean)
em System.Configuration.BaseConfigurationRecord.ThrowIfParseErrors(System.Configuration.ConfigurationSchemaErrors)
em System.Configuration.BaseConfigurationRecord.ThrowIfInitErrors()
em System.Configuration.ClientConfigurationSystem.OnConfigRemoved(System.Object, System.Configuration.Internal.InternalConfigEventArgs)

Informações da Exceção: System.Configuration.ConfigurationErrorsException
em System.Configuration.ConfigurationManager.PrepareConfigSystem()
em System.Configuration.ConfigurationManager.get_AppSettings()
em NLog.Common.InternalLogger.GetSettingString(System.String, System.String)
em NLog.Common.InternalLogger.GetSetting[[System.Boolean, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]](System.String, System.String, Boolean)
em NLog.Common.InternalLogger.Reset()
em NLog.Common.InternalLogger..cctor()

Informações da Exceção: System.TypeInitializationException
em NLog.Common.InternalLogger.Log(System.Exception, NLog.LogLevel, System.String)
em NLog.Internal.ExceptionHelper.MustBeRethrown(System.Exception)
em NLog.LogFactory.get_Configuration()
em NLog.LogFactory.GetLogger(LoggerCacheKey)
em NLog.LogFactory.GetLogger(System.String)
em NLog.LogManager.GetLogger(System.String)
em Miotec.ForceViewer.App..cctor()

Informações da Exceção: System.TypeInitializationException
em Miotec.ForceViewer.App.Main(System.String[])

Вот мой файл App.xaml.cs:

public partial class App : Application
{
    static string AppName = "ForceViewer 1.1";

    static readonly Mutex mutex = new Mutex(true, AppName);

    // APPARENTLY the exception happens in the line below:
    static readonly Logger logger = LogManager.GetLogger(typeof(App).FullName);


    [STAThread]
    public static void Main(string[] args)
    {
        SplashScreen splashScreen = new SplashScreen("Splash.png");
        splashScreen.Show(true);

        if (mutex.WaitOne(TimeSpan.Zero, true))
        {
            var app = new App();
            app.InitializeComponent();
            app.Run();
            mutex.ReleaseMutex();
        }
        else
        {
            Extensions.EnviaMensagemPraAtivarOutraJanela();
        }
    }


    protected override void OnStartup(StartupEventArgs e)
    {
        base.OnStartup(e);
        WpfLauncher.Launch(this, new ForceViewerBootstrapper(AppName));
        logger.Info($"Aplicação {AppName} iniciada");
    }
}

ОБНОВЛЕНИЕ (с дополнительной, возможно, важной информацией):

Некоторые люди упоминали XML-файл конфигурации NLog, но я использую конфигурацию времени выполнения, как показано ниже:

        var dirname = Path.Combine(@"C:\\AppFolder", appName, "logs");

        Directory.CreateDirectory(dirname);

        var filename = Path.Combine(dirname, $"{appName}.log");

        var filetarget = new FileTarget("app")
        {
            FileName = filename,
            Encoding = Encoding.UTF8,
            Layout = "${longdate}|${level:uppercase=true}|${message} (${logger:shortName=true})",
            AutoFlush = true,
            MaxArchiveFiles = 8,
            ArchiveAboveSize = 1048576,
            ArchiveEvery = FileArchivePeriod.Friday,
            ConcurrentWrites = true
        };

        var asyncTarget = new AsyncTargetWrapper("app", filetarget);

        var config = new LoggingConfiguration();
        config.AddRuleForAllLevels(asyncTarget);
        config.AddTarget(asyncTarget);
        LogManager.Configuration = config;

Кроме того, стекtrace »(который в случае события Windows кажется напечатанным как бы задом наперед) предполагает, что NLog сам получает исключение из классов System.Configuration, как видно из декомпиляции InternalLogger:

  namespace NLog.Common
  {
      //...
      public static class InternalLogger
      {
          //...
          private static string GetSettingString(string configName, string envName)
          {
              // Line below seems to be throwing an exception
              string str = System.Configuration.ConfigurationManager.AppSettings[configName];
              //..

1 Ответ

0 голосов
/ 11 июня 2018

Возможно, в вашей конфигурации NLog (XML) есть ошибка конфигурации.

Так почему вы получаете TypeInitializationException, а не полезное сообщение?Это потому что вы инициализируете NLog перед запуском вашей программы.Строка:

static readonly Logger logger = LogManager.GetLogger(typeof(App).FullName);

будет запущена перед Main, потому что это статическое поле.К сожалению, NLog не может выдать лучшее исключение (см .: Лучшее TypeInitializationException (innerException также null) )

Рекомендация: в этом случае рекомендуется иметь нестатический регистратор или, static Lazy<Logger>:

static readonly Lazy<Logger> logger = new Lazy<Logger>(() => LogManager.GetLogger(typeof(App).FullName));
...