Serilog не записывает журнал в файл при использовании модели хостинга InProcess в ASP.NET Core 2.2 - PullRequest
0 голосов
/ 19 декабря 2018

Если я использую недавно введенную InProcess модель хостинга в ASP.NET Core 2.2 следующим образом:

<PropertyGroup>
  <TargetFramework>netcoreapp2.2</TargetFramework>
  <AspNetCoreHostingModel>InProcess</AspNetCoreHostingModel>
</PropertyGroup>

Serilog не записывает журнал в файл.но если я удаляю <AspNetCoreHostingModel>InProcess</AspNetCoreHostingModel> из .csproj, все работает должным образом.

Моя Serilog конфигурация в классе Program выглядит следующим образом:

public class Program
{
    public static void Main(string[] args)
    {
        Log.Logger = new LoggerConfiguration()
            .MinimumLevel.Information() // Set the minimun log level
            .WriteTo.File("Logs\\log-.txt", rollingInterval: RollingInterval.Day, retainedFileCountLimit: 7) // this is for logging into file system
            .CreateLogger();

        try
        {
            Log.Information("Starting web host");
            CreateWebHostBuilder(args).Build().Run();
        }
        catch (Exception ex)
        {
            Log.Fatal(ex, "Host terminated unexpectedly");
        }
        finally
        {
            Log.CloseAndFlush();
        }

    }

    public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
            .UseStartup<Startup>()
            .ConfigureLogging(logging => { logging.ClearProviders(); }) // clearing all other logging providers
            .UseSerilog(); // Using serilog 
}

Любая мысль экспертапожалуйста!

Ответы [ 2 ]

0 голосов
/ 02 июня 2019

Попробуйте обновить версию .Net Core.Эта проблема, кажется, исправлена ​​в 2.2.3.

0 голосов
/ 20 декабря 2018

Как указано в комментариях к вашему вопросу, при запуске с использованием модели хостинга InProcess текущий каталог приложения отличается от модели хостинга OutOfProcess.Для InProcess этот каталог является местоположением самого IIS - например, C:\Program Files\IIS Express, что означает, что ваши файлы журнала записываются в C:\Program Files\IIS Express\Logs\log-.txt (при условии, что установлены соответствующие разрешения).

Обходной путьэто подробно описано в этой проблеме GitHub , которая предоставляет вспомогательный класс (CurrentDirectoryHelpers) для установки правильного текущего каталога.Статический метод SetCurrentDirectory использует PInvoke, определяя, выполняется ли приложение из IIS, и, если это так, устанавливает текущий каталог в соответствии с полным путем приложения.Использование этого подхода выглядит следующим образом:

public class Program
{
    public static void Main(string[] args)
    {
        CurrentDirectoryHelpers.SetCurrentDirectory();

        Log.Logger = new LoggerConfiguration()
            .MinimumLevel.Information() // Set the minimun log level
            .WriteTo.File("Logs\\log-.txt", rollingInterval: RollingInterval.Day, retainedFileCountLimit: 7) // this is for logging into file system
            .CreateLogger();

        ...
    }
}

Вот CurrentDirectoryHelpers для полноты:

using System;

namespace SampleApp
{
    internal class CurrentDirectoryHelpers
    {
        internal const string AspNetCoreModuleDll = "aspnetcorev2_inprocess.dll";

        [System.Runtime.InteropServices.DllImport("kernel32.dll")]
        private static extern IntPtr GetModuleHandle(string lpModuleName);

        [System.Runtime.InteropServices.DllImport(AspNetCoreModuleDll)]
        private static extern int http_get_application_properties(ref IISConfigurationData iiConfigData);

        [System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
        private struct IISConfigurationData
        {
            public IntPtr pNativeApplication;
            [System.Runtime.InteropServices.MarshalAs(System.Runtime.InteropServices.UnmanagedType.BStr)]
            public string pwzFullApplicationPath;
            [System.Runtime.InteropServices.MarshalAs(System.Runtime.InteropServices.UnmanagedType.BStr)]
            public string pwzVirtualApplicationPath;
            public bool fWindowsAuthEnabled;
            public bool fBasicAuthEnabled;
            public bool fAnonymousAuthEnable;
        }

        public static void SetCurrentDirectory()
        {
            try
            {
                // Check if physical path was provided by ANCM
                var sitePhysicalPath = Environment.GetEnvironmentVariable("ASPNETCORE_IIS_PHYSICAL_PATH");
                if (string.IsNullOrEmpty(sitePhysicalPath))
                {
                    // Skip if not running ANCM InProcess
                    if (GetModuleHandle(AspNetCoreModuleDll) == IntPtr.Zero)
                    {
                        return;
                    }

                    IISConfigurationData configurationData = default(IISConfigurationData);
                    if (http_get_application_properties(ref configurationData) != 0)
                    {
                        return;
                    }

                    sitePhysicalPath = configurationData.pwzFullApplicationPath;
                }

                Environment.CurrentDirectory = sitePhysicalPath;
            }
            catch
            {
                // ignore
            }
        }
    }
}
...