#region Copyright // /************************************************************************ // Copyright (c) 2016 Jamie Rees // File: BaseGenericRepository.cs // Created By: Jamie Rees // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // ************************************************************************/ #endregion using System; using System.Collections.Generic; using System.Data; using System.Linq; using System.Threading.Tasks; using Dapper.Contrib.Extensions; using Dapper; using Mono.Data.Sqlite; using NLog; using PlexRequests.Helpers; namespace PlexRequests.Store.Repository { public abstract class BaseGenericRepository where T : class { private const string CorruptMessage = "The database is corrupt, this could be due to the application exiting unexpectedly. See here to fix it: http://www.dosomethinghere.com/2013/02/20/fixing-the-sqlite-error-the-database-disk-image-is-malformed/"; protected BaseGenericRepository(ISqliteConfiguration config, ICacheProvider cache) { Config = config; Cache = cache; } protected ICacheProvider Cache { get; } protected ISqliteConfiguration Config { get; } private static readonly Logger Log = LogManager.GetCurrentClassLogger(); public abstract T Get(string id); public abstract Task GetAsync(int id); public abstract T Get(int id); public abstract Task GetAsync(string id); private IDbConnection Connection => Config.DbConnection(); public IEnumerable Custom(Func> func) { using (var cnn = Connection) { return func(cnn); } } public async Task> CustomAsync(Func>> func) { using (var cnn = Connection) { return await func(cnn); } } public long Insert(T entity) { try { ResetCache(); using (var cnn = Config.DbConnection()) { cnn.Open(); return cnn.Insert(entity); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public void Delete(T entity) { try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); db.Delete(entity); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public async Task DeleteAsync(T entity) { try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); await db.DeleteAsync(entity).ConfigureAwait(false); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public bool Update(T entity) { try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); return db.Update(entity); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public async Task UpdateAsync(T entity) { try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); return await db.UpdateAsync(entity).ConfigureAwait(false); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public bool UpdateAll(IEnumerable entity) { try { ResetCache(); var result = new HashSet(); using (var db = Config.DbConnection()) { db.Open(); foreach (var e in entity) { result.Add(db.Update(e)); } } return result.All(x => true); } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public async Task UpdateAllAsync(IEnumerable entity) { try { ResetCache(); var tasks = new List>(); using (var db = Config.DbConnection()) { db.Open(); foreach (var e in entity) { tasks.Add(db.UpdateAsync(e)); } } var result = await Task.WhenAll(tasks).ConfigureAwait(false); return result.All(x => true); } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public async Task InsertAsync(T entity) { try { ResetCache(); using (var cnn = Config.DbConnection()) { cnn.Open(); return await cnn.InsertAsync(entity).ConfigureAwait(false); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } private void ResetCache() { Cache.Remove("Get"); Cache.Remove("GetAll"); } public IEnumerable GetAll() { try { using (var db = Config.DbConnection()) { db.Open(); var result = db.GetAll(); return result; } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public async Task> GetAllAsync() { try { using (var db = Config.DbConnection()) { db.Open(); var result = await db.GetAllAsync().ConfigureAwait(false); return result; } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public bool BatchInsert(IEnumerable entities, string tableName, params string[] values) { // If we have nothing to update, then it didn't fail... var enumerable = entities as T[] ?? entities.ToArray(); if (!enumerable.Any()) { return true; } try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); using (var tran = db.BeginTransaction()) { var result = enumerable.Sum(e => db.Insert(e)); if (result != 0) { tran.Commit(); return true; } tran.Rollback(); return false; } } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } public void DeleteAll(string tableName) { try { ResetCache(); using (var db = Config.DbConnection()) { db.Open(); db.Execute($"delete from {tableName}"); } } catch (SqliteException e) when (e.ErrorCode == SQLiteErrorCode.Corrupt) { Log.Fatal(CorruptMessage); throw; } } } }