#region License Information /* HeuristicLab * Copyright (C) 2002-2008 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * HeuristicLab is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Linq; using System.Text; using HeuristicLab.Hive.Contracts.BusinessObjects; using System.Runtime.CompilerServices; namespace HeuristicLab.Hive.Server.ADODataAccess { abstract class CachedDataAdapter : DataAdapterBase, ICachedDataAdapter where CacheT: System.Data.TypedTableBase, new() where AdapterT : new() where RowT : System.Data.DataRow where ObjT : IHiveObject, new() { protected CacheT cache = new CacheT(); protected ICollection parentAdapters = new List(); protected CachedDataAdapter() { FillCache(); ServiceLocator.GetTransactionManager().OnUpdate += new EventHandler(CachedDataAdapter_OnUpdate); } protected virtual RowT FindSingleRow(Selector dbSelector, Selector cacheSelector) { RowT row = FindSingleRow(cacheSelector); //not in cache if (row == null) { row = FindSingleRow(dbSelector); } return row; } protected virtual IEnumerable FindMultipleRows(Selector dbSelector, Selector cacheSelector) { IList result = new List(cacheSelector()); IEnumerable result2 = dbSelector(); foreach (RowT row in result2) { if (!IsCached(row)) { result.Add(row); } } return result; } protected virtual ObjT FindSingle(Selector dbSelector, Selector cacheSelector) { RowT row = FindSingleRow(dbSelector, cacheSelector); if (row != null) { ObjT obj = new ObjT(); Convert(row, obj); return obj; } else { return default(ObjT); } } protected virtual ICollection FindMultiple(Selector dbSelector, Selector cacheSelector) { ICollection result = FindMultiple(cacheSelector); ICollection resultDb = FindMultiple(dbSelector); foreach (ObjT obj in resultDb) { if (!result.Contains(obj)) result.Add(obj); } return result; } protected abstract RowT InsertNewRowInCache(ObjT obj); protected abstract void FillCache(); public abstract void SyncWithDb(); protected abstract bool PutInCache(ObjT obj); protected abstract RowT FindCachedById(long id); void CachedDataAdapter_OnUpdate(object sender, EventArgs e) { foreach (ICachedDataAdapter parent in this.parentAdapters) { parent.SyncWithDb(); } this.SyncWithDb(); } protected virtual void RemoveRowFromCache(RowT row) { cache.Rows.Remove(row); } protected virtual bool IsCached(RowT row) { return cache.Contains(row); } protected override RowT GetRowById(long id) { RowT row = FindCachedById(id); if(row == null) row = FindSingleRow( delegate() { return FindById(id); }); return row; } [MethodImpl(MethodImplOptions.Synchronized)] public override void Update(ObjT obj) { if (obj != null) { RowT row = GetRowById(obj.Id); if (row == null) { if (PutInCache(obj)) { row = InsertNewRowInCache(obj); } else { row = InsertNewRow(obj); } UpdateRow(row); } Convert(obj, row); if (IsCached(row) && !PutInCache(obj)) { //remove from cache UpdateRow(row); RemoveRowFromCache(row); } obj.Id = (long)row[row.Table.PrimaryKey[0]]; if (!IsCached(row)) UpdateRow(row); } } } }