MS Sql Database ile SignalR için Angularjs Optimizasyonu

Selamlar;

Bir önceki makalede,  Sql database üzerindeki tabloda değişiklik yapıldığında, WebApi sayesinde bir hub classına bağalanarak, ilgili client’lara signalR sayesinde oyunların listelendiği tüm partial view string’e çevirilip push edilmekte idi. Tüm sayfanın clientlara gönderilmesi büyük bir performans kaybına neden olmakta idi. Bunun için sayfada partial view yerine bir View Model kullanılıp sadece değişen data clientlara gönderilerek, var olan model den sadece bu değişen data replace edilerek clientlara gösterilmelidir. Bunun için Mvc View tarafında Partial View yerine AngularJs kullanıcaz ve sadece değişen datayı signalR ile clientlara push edicez. İsterseniz önce en başta ürünleri partial view yerine AngularJs ve signalR ile nasıl listeleyeceğimize bakalım.

Öncelikle NuGet’den alttaki dosyaların bir önceki versiyonlarını indiriyoruz. En son versiyonlarında angularJS ve signalR’ın birlikte çalışmısında sıkıntılar var arkadaşlar. Eminim bir süre sonra düzeltilecektir.

signal

angular

Aşağıda görüldüğü gibi HomeController Index view’daki  kodlar kaldırılmıştır. Bunun yerine GamesHub classında, client Connect olduğu zaman  ilgili game data model’i çekilip connect olan client’a push edilir. Yalnız burada isWebUser şeklinde bir kontrol vardır. Amaç client web’den mi yoksa webapi servisinden mi bağlanmaya çalıştığına bakılır. Eğer connection webapi servisin’den yapılıyor ise ilgili game data çekilip push edilmez. Denenir ise client bulunamıyacağı için timeout’a düşülür.

Aşağıda görüldüğü gibi view tarafında ilgili data signalR getAllProduct() client function’ı ile alınıp angularJS’de $scope.Games view modeline atanır. Bu işlem client’ın hub class’ına connect durumunda yapılır. Connection olunmadan önce dikkat ederseniz querystring’e yandaki değer atanmıştır. $.connection.hub.qs = { isWebUser: true }; . Amaç web sayfasından geldiğimizi belirtip server side’daki koşulu geçip ilgili datayı çekebilmektir. Daha sonra ng-repeat ile Games view model’i içinde dönülerek ilgili oyun tablosu doldurulur. Tüm operational işlemler {{ }} parentezler içinde yapılır. Ayrıca price alanında currency filter :| currency:’€’ şeklinde kullanılmaktadır.

Aşşağıda görüldüğü gibi connect durumunda tüm data çekilmiş , console’a json formatında log olarak yazılmış ve herbiri ng-repeat ile dönülerek ilgili tabloya basılmıştır.screen2Şimdi artık önceki makaleden de hatırlayacağınız gibi WebApi servisinden değişen oyunun ID’si ile birlikte tetiklenen RefreshData() methodu üzerinde konuşalım. Aşağıda görüldüğü gibi gelen değişen data ID’si ile ilgili data çekilir ve tüm clientlara signalR sayesinde push edilir.

View tarafında ilgili kod aşağıdaki gibidir. Tetiklenen refreshData() function’ı updateDataFromID() functionın’ı değişen oyun ID’si, ViewModel’deki Games dizisisi ve değişe data ile birlikte  call edilir. Değişen data’nın ID’si varolan viewModel’de gezilerek aranır. Bu şekilde bulunan data yeni gelen data ile değiştirilir. Ve $scope.$apply() ile bind edilir. Bu şekilde oyun datası ile dolan tablolar yenilenmiş olur.

Güncelenen datalara göre console’a basılan örnek loglar aşağıdadır.

log

Index.cshtml:

 HomeController.cs:

Yukarıda ki örnekte de görüldüğü gibi ilgili değişen datanın ID’si sql den ilgili WebApi servisine ordanda signalR methoduna gönerilmiştir. RefreshData() methodunda ilgili, yani sadece değişen data çekilerek json formatında view’a gönderilmiştir. AngularJS sayesinde var olan viewModel’de değişen data bulunup yenisi ile replace edilmiş ve clientlara gösterilmiştir. Böylece önceki makalede yaptığımız bütün modeli clientlara push etmek yerine sadece değişen data gönderilerek büyük bir performance kaybından kurtulunmuştur.

Bir sonraki makalede görüşmek üzere hoşçakalın.

Source Code: http://www.borakasmer.com/projects/SignalRSqlAngular.rar

Herkes Görsün:

Bunlar da hoşunuza gidebilir...

33 Cevaplar

  1. Cem dedi ki:

    Teşekkürler hocam. Ağzınıza sağlık. Çok açıklayıcı olmuş. Angular ile nasıl yapılacağı konusu baya kafama takılmış. Açıkçası harika bir teknoloji. Keşke herkez sizin gibi özgün olsa. Tabii bir de bu kadar bilgili ..

  2. Murat dedi ki:

    Cem;
    Bora hoca çok sağlamdır. Sonuçta doğrudan Sql ve SignalR le ilgili böyle yabancı kaynak bile nerde ise pek yok.
    İnan bu konuda çok şanslıyız. Gelsinler de adamlar türkçe öğrensinler şimdi :)

  3. Burak dedi ki:

    Selamlar;
    Genelde hiç yorum yazmam. Ama bu makaleden sonra dayanamadım. Hocam siz bunca zaman nerede idiniz. Bundan önceki makalenizi de okudum cidden muhteşem. Ne Mvp’ler var.Adı lazım değil tam da bu konuda seminerine katıldım. Yapılan örnek ile bunu kıyaslayınca tofaşla mercedes yok yok bugatti gibi kalıyor :)

    • borsoft dedi ki:

      Teşekkürler Burakcım. Yalnız bu konuda Türkiye’de tek bir Mvp var. Yani kimden bahsettiğin malesef belli. Hem arkadaşımızı rencide etmeme adına hemde kıyaslamanın hem bu işte hem de başka bir işte hiç doğru olmadığı kanaatindeyim. Sonuçta farklı konularda herkez birbirinden üstündür. O yüzden lütfen Microsoft’un MVP olarak kabul ettiği birine çamur atma. Hatta bilgi konusunda kimseye atma..

  4. Bertan dedi ki:

    Merhabalar

    Yazılarınızı ve videolarınızı zevk ile takip ediyorum. Benim size danışmak istediğim bir kaç senaryo var.

    Antrenman için bir üniversite otomasyonu hazırlıyorum. Bunu içinde öğrenci sistemi, yemekhane kart sistemi, kütüphane otomasyonu gibi bir üniversitenin tüm işlerini a’dan z’ye görecek şekilde bir otomasyon sistemi olarak düşünebilirsiniz.

    SignalR ve AngulaJS uyumu gerçekten muhteşem olmuş buda Mvc framework çatısı altına cuk diye oturmuş buraya kadar sorun yok. Benim danışmak istediğim konu;
    – Web üzerinden öğrencilerin ders seçtiğini düşünün ve 1000 öğrenci var her ders için kontenjan 50 kişi olarak belirlendi.
    -SignalR ile bir öğrenci A dersini seçtiğine anında (Diğer öğrencinin sayfayı yenilemeden görecek şekilde) kontenjan 49 a düşüyor. Buraya kadar sorun yok.
    -Sistem saat 00.00 da açılıyor ve 1000 kişi hucum ediyor anlık 1000 kişi A dersini seçmeye çalışıyor ve 50 kontenjanı olan derse tıklıyor. (Hocanın Bedavadan AA verdiğini farz ediyoruz :) ) Bu senaryoda sistem 51. kişiye izin verir mi ? Hata payı nedir. Bunun için ne kullanmalıyız?

    Teşekkürler.

    • borsoft dedi ki:

      Teşekkürler Bertan.
      Hayır gelen 51. kişi istersen sokmaz isin. Olay lock mekanizması. Önceki makalem olan mssql database ile signalr etkilesimleri‘de dikkat edersen orda bu konu üzerinde durdum. Aşşağıda görüldüğü gibi gerekli update islemlerini bu lock içinde yaparsan yapacağın kişi kontrolü filiter’ından 51. kişi kapı duvar olarak geri döner :)

      public static object _lock=new object();
      lock (_lock)
      {
      Database İşlemlerini burda yap.
      }

      İyi çalışmalar…

  5. Sertaç dedi ki:

    Hocam, güncel teknolojileri sıklıkla takip ediyorum.
    Signalr 2.1 + Angularjs 1.3 + Asp.Net MVC 6 + Azure SQL DB = E-ticaret Web Site + Admin Portal
    bu denklem mantıklı mıdır? Bazıları beta sürümünde bile ama böyle bir çılgınlığa siz kalkışırmıydınız? en modern ve en performancelı yapıyı kullanmak istiyorum. Tavsiyeleriniz nelerdir?

    • borsoft dedi ki:

      Selam Sertaç;
      Bunun neresi çılgınlık:) Bu konular zaten bir web developer’ın bilmesinde büyük fayda olan konular. Bunlardan hiçbiri beta değil:) Ben Mvc6, Angularjs, Azure DB ve TypeScript’i zaten iş hayatında bolca kullanıyorum. Yani hepsi kullanılıyor. Bu teknolojler gayet modern ama en performanslı yapı değiller. Eğer web teknolojileri ile uğraşacak isen aynen devam sertaç:)

  6. Mahmut dedi ki:

    Hocam Merhaba öncellikle emeğinize sağlık elleriniz dert görmesin :) böyle güzel ve sade anlatım için ayrıca teşekkürler. bir projede yukarıda çok güzel bir biçimde anlattığınız yöntemi kullamak istiyorum. sizin projenizi download ettim ve onun üzerinde öğrenmek için çalışıyorum . herşey çalışıyor fakat databasede bir değişiklik yaptığımda değişiklik clientlar’a push olmuyor sqlden calwebservices proceduresini execute ediyorum bir id ile response null dönüyor nerde yanlış yapıyorum 2 gündür bulamadım yardımcı olabilirseniz sevinirim. Saygılarımla. Çalışmalarınızda başarılar.

    • borsoft dedi ki:

      Teşekkürler Mahmut;

      Öncelikle web sayfası http://localhost:18852 sayfası olarak mı görünüyor ona bak. Eğer farklı bir url’den çalışıyor ise windows services signalR sınıfına erişemiyordur. id’nin null dönmesi CodeFirst poco nesnesi yani ilgili tablonun class’ı ile database’deki tablonun örtüşmemesinden olabilir. Şimdilik aklıma gelenler bunlar..

      İyi çalışmalar.

  7. mahmut dedi ki:

    Hocam merhaba, bir önceki sorumda olan problemleri düzelttim ve sizin projenizi çalıştırm. sizin projeniz üzerinden kendi projeme uyarlama yaptım. benim projemde updat değilde tabloya insert edildiğinde insert edilen kayıtı göstermek istiyorum.tüm işlemleri sizin projenizi baz olarak yaptım fakat şöyle bir kaç problem yaşıyorum bana yardımcı ollabilirseniz Çok sevinirim,
    1- projeme kullanıcı ve şifre ile giriliyor ve controllerde actionlarda authentication doğrulamısı yaptırıyorum kullanıcı giriş yapmamışsa giriş sayfasına yönlendiriyorum. apide trigger sınıfında
    ” stockTickerHubProxy.Invoke(“RefreshData”, ID);” satırında
    “tatusCode: 404, ReasonPhrase: ‘Not Found’, Version: 1.1, Content: System.Net.Http.StreamContent, Headers:

    {

    X-SourceFiles: =?UTF-8?B?”
    bu hata ile karşılaşıyorum bu authentication kontrolündenmi kaynaklanıyor. bu sorunu nasıl aşarım.

    2-callweservices store prosedürünü çalıştırdığımda authentication olmayan bir sayfada

    “Zaman uyumsuz bir modül veya işleyici, zaman uyumsuz işlem halen beklemedeyken tamamlandı.” Bu hatayı alıyorum ve şuan kilitli kalmış durumdayım :( ban buralarla ilgili yol göstermenizi istiyorum Saygılarımla. Verdiğim rahatsızlık için çok özür diler çalışmalarınızda başarılar dilerim

    • borsoft dedi ki:

      Selamlar Mahmut;

      Projeni Github’a koy. Bana da link’ini ver. Müsait olunca inceleyim. Bu şekilde sana daha çok yardımcı olabilirim.

      İyi çalışmalar…

      • Mahmut dedi ki:

        https://github.com/pisscom/DenemeSignalr.git
        Hocam yukarıdaki linkte benim kullanmam gereken sayfası aynen kodladım database bak dosyasınıda ekledim birde ufak açıklama dosyası koydum sizin makalenizi nasıl çalıştırmak istediğile ilgil şimdiden Çok teşekkür ederim değerli vaktinizden bana zaman ayırdığınız için .
        Saygılarımla.
        İyi çalışmalar..

        • borsoft dedi ki:

          Teşekkürler Mahmut;
          Gerçekten öğrenmişin. Çok mutlu oldum.. Birinİ daha SignalR’cı yaptık:)

          • Mahmut dedi ki:

            Asıl ben teşekkür ederim Hocam sizin sayenizde, inşallah iyi bir SignalR’cı olurum : ), ellerinize Yüreğinize Ağzınıza sağlık öyle güzel anlatıyorsunuz ki öğrenmemek ele değil ve bana kattığınız bu değer için tekrar çok teşekkür ederim . Hocam proje çalışıyormu bende yeni kayıt eklemede Güncelleme Gerçekleşmiyodu ama yorumunuzdan anladığım kadarıyla çalışıyor galiba. bana ayırdığınız vakit ve emek için minnettarım . Saygılarımla

          • Mahmut dedi ki:

            Hocam kendi projemdeki tüm hataları düzelttim Şuan sorunsuz Çalışıyor Bilgi vermek istedim. Saygılarımal

  8. Mahmut dedi ki:

    Hocam Merhaba, bu makale içiçn size nekadar teşekkür etsem azdır.Çok faydalı oldu ve bana çok iyi bi iş çıkartırdı. Size uFak bir Sorum olacak. Projemi kendi serverımız üzerinde yayınlayacağım webapi proje,birde html helper yazmıştım classlibrary olarak bu üçü aynı proje içinde , bu üç ünü tek bir seferde publish edebiliyormuyum yoksa web apiyi projeyi ayrı ayrı mı publish etmem ve çalıştırmam gerekiyor. Saygılarımla

  9. Ferdi İnan dedi ki:

    Merhabalar hocam, burdaki örneği kendi projemde kullandım database üzerinde direk ekleme yapınca sorunsuz olarak çalışıyor fakat bir uygulama dışarıdan kayıt attığında “belirtilen atama geçerli değil” diye hata veriyor triggeri iptal ettiğimde direk store producure çağırdığımda yine web servis tetikliniyor triggerin çaışmasını nasıl sağlayabilirim bu konuda yardımcı olurmusunuz.

    • borsoft dedi ki:

      Selamlar Ferdi,

      Öncelikle en sevdiğim makalemi bulmuşun. Tebrikler:) Senin sorunun WebApi de güvenlik. 2 yolun var. Kolay yol: Sql ile WebApi servisini aynı sunucuya koy. Ya da Webapi servisini Cross Domain’e aç. Onu da makalelerimden bulabilirsin.
      Türkçe olmasına ragmen bu makalem yurt dışından en çok ragbet gören, okunan ve soru sorulan makalemdir:)

      İyi çalışmalar.

  10. Seyfullah dedi ki:

    Merhaba Bora hocam yapmış olduğunuz makaleyi aynen aldım çalıştırdım. Gayet sağlıklı çalışıyor. Fakat ben baştan kendime göre uyarladığımda değişen kaydın bir önceki halini getiriyor. yani seyfullah yazısına seyfullah-1 yazıyorum “seyfullah” geliyor “seyfullah-2” yazdığımda “seyfullah-1” geliyor. kafam karıştı. Bunun sebebi nedir sizce

    • borsoft dedi ki:

      Selamlar,
      Büyük ihtimalle kaydın değişmemiş halini gönderiyorsunuz. Kayıdın değiştiğine emim olduktan sonra “signalR :Hub” methodunun tetiklenmesi gerekmektedir.

      İyi çalışmalar.

      • Seyfullah dedi ki:

        Merhaba Bora hocam
        public void RefreshData(int Id)
        {
        Thread.Sleep(1000); komutu ekleyince çalışıyor. Sizce durumun kaynağın ne olabilir. Komutlarınıza ek olarak sadece eklenen komut thread komutu.

        • borsoft dedi ki:

          Selamlar,

          Çekilmek istenen datanın sonucu belli bir zaman aldığı için bu şekilde bekleniyordur. Bundan dolayı çalışıyordur. Ama bu malesef doğru bir çözüm değil.
          Bu methodu asenkron çağırsan bu iş çözülür :)
          “var changeData = from prod in db.tblGames where prod.ID == ID select new { prod.Name, prod.Price, prod.ID, prod.ImageUrl };”

          İyi çalışmalar.

          • seyfullah dedi ki:

            Merhaba Bora hocam,

            “var changeData = from prod in db.tblGames where prod.ID == ID select new { prod.Name, prod.Price, prod.ID, prod.ImageUrl };” bu şekilde çağırıyorum. Fakat değişimden önceki kaydı gösteriyor. Ne yapacağımı çıkartamadım. yoğunluktan cevabınıza ancak cevap yazabilidim.

  11. seyfullah dedi ki:

    Merhaba Bora hocam,

    “var changeData = from prod in db.tblGames where prod.ID == ID select new { prod.Name, prod.Price, prod.ID, prod.ImageUrl };” bu şekilde çağırıyorum. Fakat değişimden önceki kaydı gösteriyor. Ne yapacağımı çıkartamadım. yoğunluktan cevabınıza ancak cevap yazabilidim.

  12. denemeyanılma dedi ki:

    Hocam,kolay gelsin bilgiler için çok teşekkürler. Angularjsi ben de neredeyse aynı amaç için kullanacağım ama hiçbir bilgim yok kendisi hakkında. Kaynak olarak nereden çalışmamı önerirsiniz ? Kendi sitesinden mi, online kurslardan mı, kitaplardan mı ? İlk adım çok zor geliyor, internet sitelerinde kayboluyorum boş boş :( teşekkür ederim

    • borsoft dedi ki:

      Selamlar,
      Öncelikle ben teşekkür ederim.

      AngularJs’mi Angular7 mi once ona karar vermek lazım:) Tavsiyem Angular7 tabiki. Öğrenmek için bloğumdan başlayıp, PluralSight’ı öneririm. Msdn subscription alınca ilk 3 ay sanırım ücretsiz oluyor.
      egghead.io da Angular için çok iyi bir eğitim portalı. Ama maalesef o da ücretli bir eğitim.

      İyi çalışmalar.

      • denemeyanılma dedi ki:

        Çok teşekkürler hocam, Allah her işinizi rast getirsin :) yol gösteren birinin olması çok güzel bir şey

  13. denemeyanılma dedi ki:

    Merhabalar hocam, bilgiler için çok teşekkürler.
    Ben de yeni bir kayıt eklendiğinde veri tabanına eklenen kayıtı il seferde yüklenen verilerin üzerine tek tek yazdırmak istiyorum. Refreshdata fonksiyonundaki IDyi direk sayfaya yazdırdığımda sıkıntı yok. Ama bu ID ile örnekteki gibi veritabanı işlemi yapınca değişen veri çok nadir geliyor , çoğunlukla boş geliyor . Yardım ederseniz çok sevinirim teşekkürler hocam.
    public async System.Threading.Tasks.Task RefreshData(int ID)
    {
    var changeData = await GetData(ID);
    Clients.All.refreshData(changeData);
    }

    public async Task GetData(int ID)
    {
    var data= await Task.FromResult( db.Canli.Where(t => t.Id == ID).ToString());
    return data;
    }
    Client kısmı :
    messagehub.client.refreshData = function (changedData) {
    console.log(“Değişen:” + changedData);
    $scope.deneme = changedData;
    $scope.$apply();
    };

Bir cevap yazın

E-posta hesabınız yayımlanmayacak.