Skip to content

Loglama Yapılarında Correlation ID ile Traceability’i Sağlamak

Merhaba arkadaşlar.

Bu makale konusunda sizlerle uygulamalarımızda Correlation ID aracılığı ile consistent bir şekilde nasıl log’lama işlemlerini gerçekleştirebiliriz konusunu ele almaya çalışacağım.

blog-arrows

Çoğu zaman geliştirmekte olduğumuz uygulamalarda, arkaplanda neyin olup bittiğini anlayabilmek için genelde çeşitli log’lama işlemlerini gerçekleştiririz. Özellikle büyümekte ve büyük çapta olan projelerde, monolithic architecture’ların yerine distributed architecture’ların tercih edilmesiyle beraber, distributed dünya sağlamış olduğu bir çok faydanın yanında business process yönetimi ve consistency gibi bazı sorunları da beraberinde getirmektedir.

Senaryo

payment-logging-flow

Yukarıda ki gibi bir örnekle açıklamak gerekirse eğer:

Örneğin X request’i ilgili business işlemini gerçekleştirebilmesi için, -n tane service ile çalışması gerekmektedir. Gerçekleştirecek olduğu bu service call’lar sonucunda ise, oluşturacağı bir result bulunmaktadır.

Peki, bu service call’ları sırasında herhangi bir noktada kopma/network split yaşanırsa? Hangi log işlemi, hangi request içerisinde gerçekleşti? İşte bu gibi problemlerde Correlation ID yaklaşımı imdadımıza koşmaktadır.

Yaklaşım

payment-logging-with-correlation-id-flow

Service’ler arasında her bir request için unique identifier görevini görecek olan bir Correlation ID gezdirilir. Bu sayede hangi request hangi koşullarda ve nasıl hata alındığının log’larını takip edebilmek, oldukça kolaylaşacaktır.

Distributed yaklaşımların dışında bazı durumlarda ise uygulamalarımızda ki bottleneck’leri ortaya koyabilmek için farklı senaryolarda test case’leri ve log işlemleri gerçekleştiririz.

Örneğin X request’i, ilgili result set’i oluşturabilmek için hangi business process’lerden/gateway’lerden geçiyor? Hangi parametreler ile ilerliyor? Nerede ne kadar vakit harcıyor?

Bunlar gibi soruları cevaplayabilmek için, log’lama işlemini gerçekleştirirken Correlation ID çözümü ile ilerleyebiliriz. Bu sayede istediğimiz contextual bilgileri log’larken, Correlation ID ile de log’lamadaki traceability’i arttırarak, bottleneck’leri hızlı bir şekilde ortaya koyabiliriz.

Loglama

Gerçekleştireceğimiz örnekte log’lama için NLog kütüphanesini kullanacağım. Loglama yaparken herhangi bir aspect kütüphanesi kullanmayacağım. Sizler dilerseniz lisanslı olarak PostSharp veya open-source olarak Fody gibi kütüphaneler aracılığı ile intercept işlemlerini gerçekleştirerek log’lamaları yapabilirsiniz.

LogTraceWithCorrelationId” isminde yeni bir console application oluşturalım. İçerisine aşağıdaki resimde olduğu gibi NuGet üzerinden “NLog” ve “NLog.Config” kütüphanelerini kuralım.

nlog

Kütüphanelerin kurulumu ardından öncelikle “NLog.config” dosyasını aşağıdaki gibi düzenleyelim.

Bu config dosyası ile log’ların database’e kaydedilmesi gerektiğini, parameter’lar ile de “commandText” üzerindeki parametreleri set ediyoruz. “rules” kısmında ise “minlevel” parametresi ile “debug” seviyesinde log’lanmasını ve target olarak da “database” target’ını kullanmasını söylüyoruz.

NLog başlıca aşağıdaki log level’larını desteklemektedir:

  • Trace: bu level genelde development sırasında detaylı olarak hight-volume bilgilere erişebilmek için kullanılmaktadır.
  • Debug: bu level’da production ortamında kullanılmaz ve trace level’ından daha az bilgi içermektedir.
  • Info: genelde information bilgileri için kullanılır ve production ortamlarında da kullanılabilinir.

Bize örneğimizde debug level’ı yeterli olacaktır.

Config dosyasının ayarlanması ardından yukarıdaki parametrelere denk gelecek olan SQL tablosunu da aşağıdaki script ile oluşturalım.

Tablo field’larına dikkat ettiyseniz “CorrelationId” field’ı dışındakiler, NLog‘un sahip olduğu default field’lardır. Bu bölümde ise “CorrelationId” field’ını nasıl kullanabileceğimize bakacağız.

NLog‘un 4.1 versiyonundan sonra “${activityid}” parametresi ile, log mesajlarını trace edebilmek için correlation id desteği getirilmiştir. Bundan önceki versiyonlarında ise bu tarz bir işlemler için custom olarak bir layout renderer yazmak gerekiyordu.

Not: NLog‘un ${activityid} parametresi correlation id’yi “System.Diagnostics” namespace’i altında bulunan, “Trace.CorrelationManager.ActivityId” property’sinden okumaktadır.

Log kütüphanesi hazır olduğuna göre “Program.cs” i açalım ve aşağıdaki gibi kodlayalım.

Dikkat edersek iç içe sequentially olarak ilerleyen 2 adet method bulunmakta.

Main” method’u içerisinde “System.Diagnostics.Trace.CorrelationManager.ActivityId” property’sine yeni bir guid set ediyoruz. Ardından ilk method olan “DoWork” method’unu call ediyoruz. Method’lar içerisinde ise “LogManager.GetCurrentClassLogger” method’u ile elde etmiş olduğumuz NLog instance’ı üzerinden, “Debug” method’unu kullanarak bir log mesajı atıyoruz.

NLog bu işlemlerin ardından bu process bitene kadar set etmiş olduğumuz “ActivityId” parametresini, otomatik olarak log’lar arasında taşıyacaktır. Bu sayede herhangi bir process/request için tüm akışı trace edebilmek kolay bir hale gelecektir.

Yukarıdaki console uygulamasını çalıştıralım ve çıktısına SQL Server Management Studio üzerinden bir bakalım.

logs

Gördüğümüz gibi atmış olduğumuz 2 adet “Debug” log’unu, CorrelationId’leri ile birlikte SQL üzerine insert etmiş durumdadır.

Burada log’lamak istediğiniz parametreleri, ilgili business value’larınıza göre sizler belirleyebilirsiniz.

Main method’unda set etmiş olduğumuz “ActivityId” parametresini, web ortamında ise handler’lar aracılığı ile “BeginRequest” event’larında request header’ına set edebilirsiniz.

Her ne kadar basit bir konu gibi görünse de, birçok case karşısında oldukça fayda sağlamaktadır/sağlayacaktır.

Bu makalede sizlere Correlation ID kullanımının, özellikle distributed ortamlarda önemini anlatabilmeye/aktarabilmeye çalıştım. Sizlerde içerisinde bulunduğunuz ortamlarda, farklı kütüphaneler aracılığı ile bu işlemi gerçekleştirebilirsiniz.

Umarım faydalı bir yazı olmuştur. Örnek uygulamaya aşağıdan ulaşabilirsiniz.

Takipte kalın.

logtracewithcorrelationid

Bu makale toplam (3547) kez okunmuştur.

28
0



Published in.NET

4 Comments

  1. Mehmet Yazıcı Mehmet Yazıcı

    ${threadid} özelliği direkt bu işe yaramıyor mu?

    • Merhaba, evet onunla da thread id’yi alabilirsiniz fakat, gerçekleştirmiş olduğunuz business flow’unda multithreading bir iş yapıyorsanız thread id’ler farklı olacağı için, flow’u correlate edemezsiniz. İyi günler dilerim.

  2. Ünal Ünal

    Merhaba,

    Web uygulamalarında BeginRequest içinde “System.Diagnostics.Trace.CorrelationManager.ActivityId = Guid.NewGuid();” satırını eklediğimizde tek kullanıcı için bu sorun olmayabilir ancak birden fazla kullanıcının site üzerinde istek oluşturması durumunda “ActivityId – CorrelationId” diğer thread’ler tarafından ezilmeyecek midir? Bu işlem Thread Safe olarak çalımışyor sanıyorum yanılıyor muyum? Örneğin Ali kullanıcısı için X değeri CorrelationId olarak atandı aynı anda Veli kullanıcısı istek oluşturdu ve Y CorrelationId değeri atandı. Bu durumda Ali’nin CorrelationId değeri Y olarak değişmez mi?

Leave a Reply

Your email address will not be published. Required fields are marked *

*