Bu yazımızda, veri merkezli(data centric) uygulamalarda alışılagelmiş bir yöntem olan Repository kullanımının, birden fazla veri kaynağının bulunduğu durumlarda nasıl yapılması gerektiği konusunu incelemeye çalışacağız.
Repository mantığında öncelikle tüm alt sınıfları hiyerarşik bir düzene sokmak adına tanımlanan bir interface vardır. Örnek olarak basit bir interface tanımlamasını şu şekilde ifade ediyor olalım.
public interface IRepository<T> { void Add(T entity); IList<T> GetAll(); }
Bu interface altında veritabanına ekleme ve veritabanından veri getirme gibi iki temel işlem vardır. Eğer bir SQL veri tabanına veri yazmayı amaçlıyorsak bu işlemi gerçekleştiren somut sınıfın oluşturulması gerekmektedir.
public class SqlRepository<T> : IRepository<T> { public void Add(T entity) { // Veritabanına ekle.. } public IList<T> GetAll() { // Veritananından oku.. } }
Buraya kadar işler normal akışında seyrediyor. Ancak SQL veri tabanına yazılamadığı durumlarda ne olacağı bir soru işareti. Ya da başka bir sebepten dolayı ilişkisel veri tabanına yazmak yerine kayıtları XML bir dosyaya yazmak gerektiğinde XML dosya ile çalışabilen somut bir repository oluşturulması gerekmektedir. Bu durumda yeni bir sınıfa ihtiyaç duyulmaktadır.
public class XmlRepository<T> : IRepository<T> { public void Add(T entity) { // XML dosyaya ekle... } public IList<T> GetAll() { // XML dosyadan oku... } }
Ancak bu iki farklı Repository tipinin, çalışma zamanında seçilebilir ve değiştirilebilir olması gerekmektedir. Aksi taktirde kodun değiştirilebilir olması istenmeyen bir durumdur. Bu değişime ayak uydurabilmek adına, değişikliği otomatik olarak gerçekleştirebilecek bir sınıfa ihtiyaç duyulmaktadır. Bu sınıf, çalışma zamanında dependency injection mekanizması ile yüklenebilen bir modül halinde tasarlanabilir.
public class ApplicationRepositorySelectionModule : NinjectModule { public override void Load() { switch (ApplicationConfiguration.StorageType) { case PersistenceStorageType.SqlRepository: Bind(typeof(IRepository<>)).To(typeof(SqlRepository<>)); break; case PersistenceStorageType.XmlRepository: Bind(typeof(IRepository<>)).To(typeof(SqlRepository<>)); break; default: throw new ApplicationException("Not supported storage type!"); } } }
NinjectModule sınıfı, Ninject dependenjy injector aracına ait bir sınıftır. Yaptığı işlem ise Interface türlerine karşılık hangi somut sınıfın oluşturulacağını belirlemektir. Zamanı geldiğinde ilgili nesneyi oluşturup programa sağlamak yine Ninject tarafından gerçekleştirilmektedir.
Modül içerisinde kullanılan PersistenceStorageType ve ApplicationConfiguration sınıfları ise şu şekildedir.
public class ApplicationConfiguration { public static PersistenceStorageType StorageType { get; set; } } public enum PersistenceStorageType { SqlRepository = 0, XmlRepository = 1 }
Bu şekilde birden fazla repository sınıfı ile konfigurasyonu sağlababilir şekilde çalışmak mümkün olmaktadır. Yazılımda çevikliği yani yazılımın değişime ayak uydurabilme yeteneğini sağlayabilmek açısından uygulanabilecek olan küçük bir detaydan bahsetmiş olduk.
Bir sonraki yazıda görüşmek dileğiyle.