Konfiguracja Serilog dla całej solucji

Konfiguracja Serilog dla całej solucji
AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

Witam.
Potrzebuje nakierowania jak uwspólnić jedną konfigurację Serilog w obrębie wszystkich aplikacji w solucji. Mam w solucji 3 projekty - worker service (usługa windows), class library (logika biznesowa), winforms (okno do konfiguracji uslugi).

Po stronie usługi mam konfigurację serilog z użyciem Seq

Kopiuj
.UseSerilog((ctx, lc) => lc.WriteTo.Console().WriteTo.Seq("http://localhost:5341"))

Po stronie okna mam

Kopiuj
services.AddLogging(logging =>
{
  logging.AddSerilog();
});

Kiedy zadanie wykona usługa to dane są logowane w Seq, ale kiedy zrobimy to samo zadanie oknem, to już nie ma tego wpisu. Moim zdaniem dzieje się tak, ponieważ "okno" nie korzysta z tej samej konfiguracji Seriloga. Metoda, która jest wykonywana jest ta sama metoda i oba projekty korzystają z tego samego co jest wpisane w trzecim projekcie (class library).

W WinForms nie mam opcji UseSerilog(), mam tylko AddLogging() oraz AddSerilog(). Jak skonfigurować to raz, aby oba programy poprawnie logowały błędy.

edytowany 1x, ostatnio: Riddle
Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0

Kiedy zadanie wykona usługa to dane są logowane w Seq, ale kiedy zrobimy to samo zadanie oknem, to już nie ma tego wpisu. Moim zdaniem dzieje się tak, ponieważ "okno" nie korzysta z tej samej konfiguracji Seriloga. Metoda, która jest wykonywana jest ta sama metoda i oba projekty korzystają z tego samego co jest wpisane w trzecim projekcie (class library).

potrzebna jest jeszcze informacja w jaki sposób to coś czego używa zarówno usługa jak i okno, loguje informacje czyli jak ma dostarczony interfejs logowania, który interfejs logowania (ILogger<?>) i jak go używa.

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

A więc tak... Konfiguracja po stronie usługi jest taka sama jak w ASP. Problemem jest całe depedency injection po stronie winforms.

USŁUGA - plik Program.cs

Kopiuj
using HotelSync;
using HotelSyncLib.Jobs;
using HotelSyncLib.Models;
using HotelSyncLib.Services.Enova;
using HotelSyncLib.Services.Enova.Enova;
using HotelSyncLib.Services.Wubook;
using Quartz;
using Serilog;

IHost host = Host.CreateDefaultBuilder(args)
    .UseWindowsService(options =>
    {
        options.ServiceName = "HotelSync";
    })
    .ConfigureServices((host, services) =>
    {
        services.AddScoped<EnovaService>();
        services.AddScoped<HotelReader>();
        services.AddScoped<SqlService>();

        services.Configure<SyncConfiguration>(host.Configuration.GetSection("ServiceSettings"));

        services.AddQuartz(q =>
        {
            q.UseMicrosoftDependencyInjectionJobFactory();
            var reservationsJobKey = new JobKey("Reservations");
            //var runNowReservations = new JobKey("RunNow");

            q.AddJob<ReservationsJob>(x => x.WithIdentity(reservationsJobKey));
            //q.AddJob<RunNowReservationJob>(x=>x.WithIdentity(runNowReservations));

            q.AddTrigger(x => x.ForJob(reservationsJobKey).WithCronSchedule(host.Configuration["ServiceSettings:Cron"]));
        });

        services.AddQuartzHostedService(options =>
        {
            options.WaitForJobsToComplete = false;
        });

        services.AddHostedService<Worker>();
    })
    .UseSerilog((ctx, lc) => lc.WriteTo.Console().WriteTo.Seq("http://localhost:5341"))
    .Build();

await host.RunAsync();

OKNO - plik Program.cs

Kopiuj
using DevExpress.LookAndFeel;
using DevExpress.Skins;
using DevExpress.UserSkins;
using DevExpress.Xpo.Logger.Transport;
using HotelSyncLib.Models;
using HotelSyncLib.Services.Enova;
using HotelSyncLib.Services.Enova.Enova;
using HotelSyncLib.Services.Wubook;
using Microsoft.Extensions.Configuration;
using Serilog.Extensions.Logging;
using Microsoft.Extensions.DependencyInjection;
using Serilog;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Versioning;
using System.Windows.Forms;

namespace HotelSyncConfiguration
{
    [SupportedOSPlatform("windows")]
    internal static class Program
    {
        /// <summary>
        /// The main entry point for the application.
        /// </summary>
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetHighDpiMode(HighDpiMode.SystemAware);
            Application.SetCompatibleTextRenderingDefault(false);

            var services = new ServiceCollection();
        
            ConfigureServices(services);

            using (ServiceProvider serviceProvider = services.BuildServiceProvider())
            {
                var mainForm = serviceProvider.GetRequiredService<MainForm>();
                Application.Run(mainForm);
            }
        }

        private static void ConfigureServices(ServiceCollection services) 
        {
            services.AddLogging(logging =>
            {
                logging.AddSerilog();
            });
            Configuration = new ConfigurationBuilder().AddJsonFile("appsettings.json").Build();
            services.Configure<SyncConfiguration>(Configuration.GetSection("ServiceSettings"));
            services.AddSingleton<IConfiguration>(Configuration);
            services.AddScoped<SqlService>();
            services.AddScoped<EnovaService>();
            services.AddScoped<HotelReader>();
            services.AddScoped<MainForm>();
        }

        public static IConfigurationRoot Configuration { get; set; }
    }
}

W bibliotece mam całą logikę i tam jest AddDocument(), z którego korzysta usługa jak wczytuje rezerwacje do Enova365 oraz okno gdy wczytujemy rezerwację "ręcznie".

Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0

NIe chodziło o okno i o usługę, tylko o to miejsce które ostatecznie ma coś zalogować. Chcę popatrzeć co i jak tam jest wstrzykiwane.

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

ILogger<EnovaService> jest wstrzykiwane

Kopiuj
    public class EnovaService
    {
        readonly ILogger<EnovaService> _logger;
        readonly SyncConfiguration _configuration;

        public EnovaService(ILogger<EnovaService> logger, IOptions<SyncConfiguration> options)
        {
            _logger = logger;
            _configuration = options.Value;
        }

        public void AddDocument(Reservation reservation, Session session, Booker booker, Rooms rooms, string company)
        {
            _logger.LogInformation($"DODAJE DOKUMENT | REZERWACJA: {reservation.IdHuman} | FIRMA: {company}");
            // ...
        }
    }
edytowany 1x, ostatnio: AdamWox
Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
1

Mi to wygląda tak że nigdzie nie ma konfiguracji seriloga (podpięcia sinków). W usłudze robi to .UseSerilog, w aplikacji desktopowej trzeba ręcznie

Kopiuj

        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetHighDpiMode(HighDpiMode.SystemAware);
            Application.SetCompatibleTextRenderingDefault(false);

            var services = new ServiceCollection();        
            ConfigureServices(services);

            // to musi być dodane
            Log.Logger = new LoggerConfiguration()
                  .MinimumLevel.Debug()
                  .WriteTo.Console().WriteTo.Seq("http://localhost:5341")
                  .CreateLogger();
            
            using (ServiceProvider serviceProvider = services.BuildServiceProvider())
            {
                var mainForm = serviceProvider.GetRequiredService<MainForm>();
                Application.Run(mainForm);
            }
        }
        
AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

No właśnie nie byłem pewny jak mam to skonfigurować. Sprawdzam rozwiązanie

EDIT

Niestety, ale nie poszło. Żaden sposób nie chce logować błędów w projekcie WinForms.

OPCJA 1 - dorzuciłem do projektu okienkowego wszystko co potrzebnę, aby stworzyć IHostBuilder, nie zadziałało.

Kopiuj
IHost host = Host.CreateDefaultBuilder(args)
    .UseSerilog((ctx, lc) => lc.WriteTo.Console().WriteTo.Seq("http://localhost:5341"))
    .Build()

OPCJA 2 - dopisałem stworzony logger do konfiguracji w AddSerilog(), też nic.

Kopiuj
services.AddLogging(logging =>
{
  logging.AddSerilog(Log.Logger, false);
});

Tak jakby konfiguracja dependency injection seriloga działała tylko w obrębie usługi z jakiś powodów. Czyli jeśli metoda AddDocument() wywoła się w z usługi do DI działa, jeśli ta sama metoda zostanie uruchomiona z okna, to już nie działa i nie loguje nic.

edytowany 1x, ostatnio: AdamWox
Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0

Niestety, ale nie poszło. Żaden sposób nie chce logować błędów w projekcie WinForms.

To znaczy że coś innego jest na rzeczy, na minimalnej aplikacji winforms sprawdziłem i to działa, z okienka loguje (co prawda podpiąłem logger plikowy ale to detal).

https://filebin.net/2njgb9t9zky94rnv/WinFormsApp1.zip

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

No tak, sink nie ma znaczenia. Moim zdaniem problemem jest dependency injection. Konfiguracja AddLogging() w winformsach narzuca domyślne logowanie jakie zapewnia Microsoft w .NET 6. Nie nadpisuje tego żadna inna konfiguracja, nawet AddSerilog()

somekind
Trochę nie rozumiem tego zdania. Serilog może być przecież implementacją microsoftowego loggera.
AdamWox
Początkowo podejrzewałem, że Serilog się w ogóle nie ładuje i oknem wstrzykuje do serwisu domyślny logger od Microsoftu, a nie Serilog.
Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0
AdamWox napisał(a):

No tak, sink nie ma znaczenia. Moim zdaniem problemem jest dependency injection. Konfiguracja AddLogging() w winformsach narzuca domyślne logowanie jakie zapewnia Microsoft w .NET 6. Nie nadpisuje tego żadna inna konfiguracja, nawet AddSerilog()

W tej przykładowej aplikacji którą podlinkowałem jest tak jak tu: do formularza trafia wstrzyknięte ILogger<T> który konfigurację czerpie z tego co się ustawi wcześniej.

Ja bym spróbował podpiąć tam jakiś logger plikowy, w mojej wersji zdaje się tak było i plik z logiem się tworzył.

Z resztą, w tym miejscu kodu w którym jest _logger.LogInformation można się zatrzymać pułapką na tym, zawisnąć myszą na _logger i rozwijać w debuggerze "Non-public members" tak długo aż się dokopie do listy sinks i wtedy widać że jest podpięty jeden sink plikowy i do jakiego pliku zapisuje (ścieżkę widać).

Jeżeli to działa w tej przykładowej aplikacji a nie działa w faktycznej, to znaczy że jeszcze jakiś inny czynnik jest, którego w tym "Minimal-reproducible-example" nie uwzględniono.

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

Debug pokazuje, że faktycznie Serilog jest załadowany poprawnie do ILogger, ale problem jest z sink'iem. Pokazuje Serilog.Core.Logger. Czy nie powinno być Serilog.Sinks.Seq?

logger.png

Doleciałem głębiej. ConsoleSink

logger2.png

edytowany 1x, ostatnio: AdamWox
Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0
AdamWox napisał(a):

Debug pokazuje, że faktycznie Serilog jest załadowany poprawnie do ILogger, ale problem jest z sink'iem. Pokazuje Serilog.Core.Logger. Czy nie powinno być Serilog.Sinks.Seq?

Na pierwszy rzut oka wygląda dobrze. Jak jest podpięty plikowy to zapisuje do pliku?

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
0

Nie jest plikowy, ponieważ jest wybrane ConsoleSink. Nawet jeśli byłby plikowy na co debug nie wskazuje, to żadnego pliku nie ma.

Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
0

Hmm, ja mówię "proszę spróbować plikowy" i "w mojej załączonej przykładowej aplikacji działa" a zwrotnie mam że jest podpięty Console. Console w aplikacji okienkowej przecież ma niewielki sens, tam domyślnie nie ma żadnej konsoli przecież.

Trzeba go wymienić na plikowy i spróbować.

Skoro w załączonej przeze mnie aplikacji działa, to wydaje się że to jest wystarczające żeby porównać jedno z drugim.

AdamWox
  • Rejestracja:ponad 7 lat
  • Ostatnio:około 7 godzin
  • Lokalizacja:Jastrzębie-Zdrój
  • Postów:2157
1

Skonfigurowałem do pliku i działa, zapisuje wszystko. Poszukałem w internetach o co chodzi z tym Seq. I sugerują, żeby ręcznie zrobić Serilog.Log.CloseAndFlush();. Wrzuciłem to do finally, faktycznie zaczął logować błędy i to co powinien. Usunąłem linię Serilog.Log.CloseAndFlush(); z finally i działa... Nic z tego już nie rozumiem 🤷‍♂️

Może się Seq "zawiesiło"? Może w przypadku UseSerilog() to całe CloseAndFlush() działa inaczej. Teraz pytanie czy to zadziała u klienta. Dzięki za pomoc 🤘

Wiktor Zychla
  • Rejestracja:prawie 7 lat
  • Ostatnio:około 12 godzin
  • Postów:77
1
AdamWox napisał(a):

Skonfigurowałem do pliku i działa, zapisuje wszystko. Poszukałem w internetach o co chodzi z tym Seq. I sugerują, żeby ręcznie zrobić Serilog.Log.CloseAndFlush();. Wrzuciłem to do finally, faktycznie zaczął logować błędy i to co powinien. Usunąłem linię Serilog.Log.CloseAndFlush(); z finally i działa... Nic z tego już nie rozumiem 🤷‍♂️

Może się Seq "zawiesiło"? Może w przypadku UseSerilog() to całe CloseAndFlush() działa inaczej. Teraz pytanie czy to zadziała u klienta. Dzięki za pomoc 🤘

Plikowy jest jakiś dziwny tam, bo ma domyślnie flush nie natychmiastowy tylko z opóźnieniem. Na pewno taki

Kopiuj
   ....WriteTo.File("foo.log", flushToDiskInterval: TimeSpan.FromSeconds(1))

działa tak jakby człowiek się spodziewał, czyli flushuje dość często.

Nie używam serilog na codzień i jak dla mnie to jest jakiś quirk, dokumentacja mówi że jak się tego nie ustawi to zapisuje natychmiast, a obserwacyjnie wygląda na to że jednak nie i trzeba nadpisać interval żeby zapisywał. Może ktoś kto się zna na serilogu lepiej potrafi to jakoś objaśnić.

Kliknij, aby dodać treść...

Pomoc 1.18.8

Typografia

Edytor obsługuje składnie Markdown, w której pojedynczy akcent *kursywa* oraz _kursywa_ to pochylenie. Z kolei podwójny akcent **pogrubienie** oraz __pogrubienie__ to pogrubienie. Dodanie znaczników ~~strike~~ to przekreślenie.

Możesz dodać formatowanie komendami , , oraz .

Ponieważ dekoracja podkreślenia jest przeznaczona na linki, markdown nie zawiera specjalnej składni dla podkreślenia. Dlatego by dodać podkreślenie, użyj <u>underline</u>.

Komendy formatujące reagują na skróty klawiszowe: Ctrl+B, Ctrl+I, Ctrl+U oraz Ctrl+S.

Linki

By dodać link w edytorze użyj komendy lub użyj składni [title](link). URL umieszczony w linku lub nawet URL umieszczony bezpośrednio w tekście będzie aktywny i klikalny.

Jeżeli chcesz, możesz samodzielnie dodać link: <a href="link">title</a>.

Wewnętrzne odnośniki

Możesz umieścić odnośnik do wewnętrznej podstrony, używając następującej składni: [[Delphi/Kompendium]] lub [[Delphi/Kompendium|kliknij, aby przejść do kompendium]]. Odnośniki mogą prowadzić do Forum 4programmers.net lub np. do Kompendium.

Wspomnienia użytkowników

By wspomnieć użytkownika forum, wpisz w formularzu znak @. Zobaczysz okienko samouzupełniające nazwy użytkowników. Samouzupełnienie dobierze odpowiedni format wspomnienia, zależnie od tego czy w nazwie użytkownika znajduje się spacja.

Znaczniki HTML

Dozwolone jest używanie niektórych znaczników HTML: <a>, <b>, <i>, <kbd>, <del>, <strong>, <dfn>, <pre>, <blockquote>, <hr/>, <sub>, <sup> oraz <img/>.

Skróty klawiszowe

Dodaj kombinację klawiszy komendą notacji klawiszy lub skrótem klawiszowym Alt+K.

Reprezentuj kombinacje klawiszowe używając taga <kbd>. Oddziel od siebie klawisze znakiem plus, np <kbd>Alt+Tab</kbd>.

Indeks górny oraz dolny

Przykład: wpisując H<sub>2</sub>O i m<sup>2</sup> otrzymasz: H2O i m2.

Składnia Tex

By precyzyjnie wyrazić działanie matematyczne, użyj składni Tex.

<tex>arcctg(x) = argtan(\frac{1}{x}) = arcsin(\frac{1}{\sqrt{1+x^2}})</tex>

Kod źródłowy

Krótkie fragmenty kodu

Wszelkie jednolinijkowe instrukcje języka programowania powinny być zawarte pomiędzy obróconymi apostrofami: `kod instrukcji` lub ``console.log(`string`);``.

Kod wielolinijkowy

Dodaj fragment kodu komendą . Fragmenty kodu zajmujące całą lub więcej linijek powinny być umieszczone w wielolinijkowym fragmencie kodu. Znaczniki ``` lub ~~~ umożliwiają kolorowanie różnych języków programowania. Możemy nadać nazwę języka programowania używając auto-uzupełnienia, kod został pokolorowany używając konkretnych ustawień kolorowania składni:

```javascript
document.write('Hello World');
```

Możesz zaznaczyć również już wklejony kod w edytorze, i użyć komendy  by zamienić go w kod. Użyj kombinacji Ctrl+`, by dodać fragment kodu bez oznaczników języka.

Tabelki

Dodaj przykładową tabelkę używając komendy . Przykładowa tabelka składa się z dwóch kolumn, nagłówka i jednego wiersza.

Wygeneruj tabelkę na podstawie szablonu. Oddziel komórki separatorem ; lub |, a następnie zaznacz szablonu.

nazwisko;dziedzina;odkrycie
Pitagoras;mathematics;Pythagorean Theorem
Albert Einstein;physics;General Relativity
Marie Curie, Pierre Curie;chemistry;Radium, Polonium

Użyj komendy by zamienić zaznaczony szablon na tabelkę Markdown.

Lista uporządkowana i nieuporządkowana

Możliwe jest tworzenie listy numerowanych oraz wypunktowanych. Wystarczy, że pierwszym znakiem linii będzie * lub - dla listy nieuporządkowanej oraz 1. dla listy uporządkowanej.

Użyj komendy by dodać listę uporządkowaną.

1. Lista numerowana
2. Lista numerowana

Użyj komendy by dodać listę nieuporządkowaną.

* Lista wypunktowana
* Lista wypunktowana
** Lista wypunktowana (drugi poziom)

Składnia Markdown

Edytor obsługuje składnię Markdown, która składa się ze znaków specjalnych. Dostępne komendy, jak formatowanie , dodanie tabelki lub fragmentu kodu są w pewnym sensie świadome otaczającej jej składni, i postarają się unikać uszkodzenia jej.

Dla przykładu, używając tylko dostępnych komend, nie możemy dodać formatowania pogrubienia do kodu wielolinijkowego, albo dodać listy do tabelki - mogłoby to doprowadzić do uszkodzenia składni.

W pewnych odosobnionych przypadkach brak nowej linii przed elementami markdown również mógłby uszkodzić składnie, dlatego edytor dodaje brakujące nowe linie. Dla przykładu, dodanie formatowania pochylenia zaraz po tabelce, mogłoby zostać błędne zinterpretowane, więc edytor doda oddzielającą nową linię pomiędzy tabelką, a pochyleniem.

Skróty klawiszowe

Skróty formatujące, kiedy w edytorze znajduje się pojedynczy kursor, wstawiają sformatowany tekst przykładowy. Jeśli w edytorze znajduje się zaznaczenie (słowo, linijka, paragraf), wtedy zaznaczenie zostaje sformatowane.

  • Ctrl+B - dodaj pogrubienie lub pogrub zaznaczenie
  • Ctrl+I - dodaj pochylenie lub pochyl zaznaczenie
  • Ctrl+U - dodaj podkreślenie lub podkreśl zaznaczenie
  • Ctrl+S - dodaj przekreślenie lub przekreśl zaznaczenie

Notacja Klawiszy

  • Alt+K - dodaj notację klawiszy

Fragment kodu bez oznacznika

  • Alt+C - dodaj pusty fragment kodu

Skróty operujące na kodzie i linijkach:

  • Alt+L - zaznaczenie całej linii
  • Alt+, Alt+ - przeniesienie linijki w której znajduje się kursor w górę/dół.
  • Tab/⌘+] - dodaj wcięcie (wcięcie w prawo)
  • Shit+Tab/⌘+[ - usunięcie wcięcia (wycięcie w lewo)

Dodawanie postów:

  • Ctrl+Enter - dodaj post
  • ⌘+Enter - dodaj post (MacOS)