2013-03-24 05:16:00 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2013-02-05 05:07:07 +01:00
|
|
|
|
using System.Linq;
|
2013-03-24 05:16:00 +01:00
|
|
|
|
using System.Linq.Expressions;
|
2013-03-25 04:51:32 +01:00
|
|
|
|
using Marr.Data;
|
|
|
|
|
using Marr.Data.QGen;
|
2013-05-05 23:24:33 +02:00
|
|
|
|
using NzbDrone.Common.Messaging;
|
|
|
|
|
using NzbDrone.Core.Datastore.Events;
|
2013-06-02 08:41:30 +02:00
|
|
|
|
using NzbDrone.Common;
|
2013-03-25 04:51:32 +01:00
|
|
|
|
|
2013-02-05 05:07:07 +01:00
|
|
|
|
|
|
|
|
|
namespace NzbDrone.Core.Datastore
|
|
|
|
|
{
|
2013-03-02 19:25:39 +01:00
|
|
|
|
public interface IBasicRepository<TModel> where TModel : ModelBase, new()
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-02-19 07:56:02 +01:00
|
|
|
|
IEnumerable<TModel> All();
|
2013-02-23 20:38:25 +01:00
|
|
|
|
int Count();
|
2013-02-17 06:35:52 +01:00
|
|
|
|
TModel Get(int id);
|
2013-04-18 02:33:38 +02:00
|
|
|
|
IEnumerable<TModel> Get(IEnumerable<int> ids);
|
2013-03-06 22:20:33 +01:00
|
|
|
|
TModel SingleOrDefault();
|
2013-02-19 07:56:02 +01:00
|
|
|
|
TModel Insert(TModel model);
|
2013-02-18 08:59:43 +01:00
|
|
|
|
TModel Update(TModel model);
|
2013-03-24 05:16:00 +01:00
|
|
|
|
TModel Upsert(TModel model);
|
2013-02-17 06:35:52 +01:00
|
|
|
|
void Delete(int id);
|
2013-03-07 05:34:56 +01:00
|
|
|
|
void Delete(TModel model);
|
2013-03-24 05:16:00 +01:00
|
|
|
|
void InsertMany(IList<TModel> model);
|
|
|
|
|
void UpdateMany(IList<TModel> model);
|
2013-03-06 22:20:33 +01:00
|
|
|
|
void DeleteMany(List<TModel> model);
|
2013-02-23 20:38:25 +01:00
|
|
|
|
void Purge();
|
2013-03-02 19:25:39 +01:00
|
|
|
|
bool HasItems();
|
2013-03-24 05:16:00 +01:00
|
|
|
|
void DeleteMany(IEnumerable<int> ids);
|
2013-03-27 07:16:55 +01:00
|
|
|
|
void SetFields(TModel model, params Expression<Func<TModel, object>>[] properties);
|
2013-04-25 06:27:49 +02:00
|
|
|
|
TModel Single();
|
2013-06-05 02:49:53 +02:00
|
|
|
|
PagingSpec<TModel> GetPaged(PagingSpec<TModel> pagingSpec);
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-05-31 01:32:50 +02:00
|
|
|
|
|
2013-02-18 04:18:25 +01:00
|
|
|
|
public class BasicRepository<TModel> : IBasicRepository<TModel> where TModel : ModelBase, new()
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
private readonly IDatabase _database;
|
2013-05-05 23:24:33 +02:00
|
|
|
|
private readonly IMessageAggregator _messageAggregator;
|
2013-04-25 06:27:49 +02:00
|
|
|
|
|
2013-05-11 22:06:57 +02:00
|
|
|
|
private IDataMapper DataMapper
|
|
|
|
|
{
|
2013-07-16 02:45:49 +02:00
|
|
|
|
get { return _database.GetDataMapper(); }
|
2013-05-11 22:06:57 +02:00
|
|
|
|
}
|
2013-03-24 05:25:16 +01:00
|
|
|
|
|
2013-05-05 23:24:33 +02:00
|
|
|
|
public BasicRepository(IDatabase database, IMessageAggregator messageAggregator)
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
_database = database;
|
2013-05-05 23:24:33 +02:00
|
|
|
|
_messageAggregator = messageAggregator;
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-27 04:44:52 +01:00
|
|
|
|
protected QueryBuilder<TModel> Query
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
get { return DataMapper.Query<TModel>(); }
|
2013-03-27 04:44:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void Delete(Expression<Func<TModel, bool>> filter)
|
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Delete(filter);
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-25 04:51:32 +01:00
|
|
|
|
public IEnumerable<TModel> All()
|
2013-02-23 20:38:25 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
return DataMapper.Query<TModel>().ToList();
|
2013-03-24 05:25:16 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-25 04:51:32 +01:00
|
|
|
|
public int Count()
|
2013-03-24 05:25:16 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
return DataMapper.Query<TModel>().GetRowCount();
|
2013-02-23 20:38:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-02-16 04:50:22 +01:00
|
|
|
|
public TModel Get(int id)
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-07-10 00:06:30 +02:00
|
|
|
|
var model = DataMapper.Query<TModel>().SingleOrDefault(c => c.Id == id);
|
|
|
|
|
|
|
|
|
|
if (model == null)
|
|
|
|
|
{
|
|
|
|
|
throw new ModelNotFoundException(typeof(TModel), id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return model;
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-04-18 02:33:38 +02:00
|
|
|
|
public IEnumerable<TModel> Get(IEnumerable<int> ids)
|
|
|
|
|
{
|
2013-06-02 08:41:30 +02:00
|
|
|
|
var query = String.Format("Id IN ({0})", String.Join(",", ids));
|
2013-04-18 02:33:38 +02:00
|
|
|
|
|
2013-06-02 08:41:30 +02:00
|
|
|
|
var result = Query.Where(query).ToList();
|
|
|
|
|
|
|
|
|
|
if (result.Count != ids.Count())
|
|
|
|
|
{
|
|
|
|
|
throw new ApplicationException("Expected query to return {0} rows but returned {1}".Inject(ids.Count(), result.Count));
|
|
|
|
|
}
|
2013-04-18 02:33:38 +02:00
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-06 22:20:33 +01:00
|
|
|
|
public TModel SingleOrDefault()
|
2013-04-25 06:27:49 +02:00
|
|
|
|
{
|
|
|
|
|
return All().SingleOrDefault();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public TModel Single()
|
2013-03-06 22:20:33 +01:00
|
|
|
|
{
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return All().Single();
|
|
|
|
|
}
|
|
|
|
|
|
2013-02-19 07:56:02 +01:00
|
|
|
|
public TModel Insert(TModel model)
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-03-24 20:56:51 +01:00
|
|
|
|
if (model.Id != 0)
|
|
|
|
|
{
|
2013-06-03 05:44:31 +02:00
|
|
|
|
throw new InvalidOperationException("Can't insert model with existing ID " + model.Id);
|
2013-03-24 20:56:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Insert(model);
|
2013-05-21 06:10:04 +02:00
|
|
|
|
PublishModelEvent(model, RepositoryAction.Created);
|
2013-05-05 23:24:33 +02:00
|
|
|
|
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return model;
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-02-18 08:59:43 +01:00
|
|
|
|
public TModel Update(TModel model)
|
|
|
|
|
{
|
2013-03-24 20:56:51 +01:00
|
|
|
|
if (model.Id == 0)
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException("Can't update model with ID 0");
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Update(model, c => c.Id == model.Id);
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return model;
|
2013-02-18 08:59:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-07 05:34:56 +01:00
|
|
|
|
public void Delete(TModel model)
|
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Delete<TModel>(c => c.Id == model.Id);
|
2013-03-07 05:34:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-24 05:16:00 +01:00
|
|
|
|
public void InsertMany(IList<TModel> models)
|
2013-02-20 03:05:15 +01:00
|
|
|
|
{
|
2013-03-25 04:51:32 +01:00
|
|
|
|
foreach (var model in models)
|
|
|
|
|
{
|
|
|
|
|
Insert(model);
|
|
|
|
|
}
|
2013-02-20 03:05:15 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-24 05:16:00 +01:00
|
|
|
|
public void UpdateMany(IList<TModel> models)
|
2013-02-20 03:05:15 +01:00
|
|
|
|
{
|
2013-03-25 04:51:32 +01:00
|
|
|
|
foreach (var model in models)
|
|
|
|
|
{
|
|
|
|
|
Update(model);
|
|
|
|
|
}
|
2013-02-20 03:05:15 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-24 05:16:00 +01:00
|
|
|
|
public void DeleteMany(List<TModel> models)
|
2013-03-05 20:49:34 +01:00
|
|
|
|
{
|
2013-03-25 04:51:32 +01:00
|
|
|
|
models.ForEach(Delete);
|
2013-03-05 20:49:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-03-24 05:16:00 +01:00
|
|
|
|
public TModel Upsert(TModel model)
|
2013-02-18 08:59:43 +01:00
|
|
|
|
{
|
2013-02-26 04:58:57 +01:00
|
|
|
|
if (model.Id == 0)
|
2013-02-23 20:38:25 +01:00
|
|
|
|
{
|
2013-03-25 04:51:32 +01:00
|
|
|
|
Insert(model);
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return model;
|
2013-02-23 20:38:25 +01:00
|
|
|
|
}
|
2013-03-25 04:51:32 +01:00
|
|
|
|
Update(model);
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return model;
|
2013-02-18 08:59:43 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-02-16 04:50:22 +01:00
|
|
|
|
public void Delete(int id)
|
2013-02-05 05:07:07 +01:00
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Delete<TModel>(c => c.Id == id);
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
2013-02-23 20:38:25 +01:00
|
|
|
|
|
|
|
|
|
public void DeleteMany(IEnumerable<int> ids)
|
|
|
|
|
{
|
2013-03-25 04:51:32 +01:00
|
|
|
|
ids.ToList().ForEach(Delete);
|
2013-02-23 20:38:25 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Purge()
|
|
|
|
|
{
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Delete<TModel>(c => c.Id > -1);
|
2013-02-23 20:38:25 +01:00
|
|
|
|
}
|
2013-03-02 19:25:39 +01:00
|
|
|
|
|
|
|
|
|
public bool HasItems()
|
|
|
|
|
{
|
2013-03-24 05:16:00 +01:00
|
|
|
|
return Count() > 0;
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-27 07:16:55 +01:00
|
|
|
|
public void SetFields(TModel model, params Expression<Func<TModel, object>>[] properties)
|
2013-03-24 05:16:00 +01:00
|
|
|
|
{
|
2013-03-24 05:25:16 +01:00
|
|
|
|
if (model.Id == 0)
|
|
|
|
|
{
|
|
|
|
|
throw new InvalidOperationException("Attempted to updated model without ID");
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-11 22:06:57 +02:00
|
|
|
|
DataMapper.Update<TModel>()
|
2013-03-27 07:16:55 +01:00
|
|
|
|
.Where(c => c.Id == model.Id)
|
|
|
|
|
.ColumnsIncluding(properties)
|
|
|
|
|
.Entity(model)
|
|
|
|
|
.Execute();
|
2013-03-02 19:25:39 +01:00
|
|
|
|
}
|
2013-05-11 01:53:50 +02:00
|
|
|
|
|
2013-05-21 06:10:04 +02:00
|
|
|
|
|
2013-06-05 02:49:53 +02:00
|
|
|
|
public virtual PagingSpec<TModel> GetPaged(PagingSpec<TModel> pagingSpec)
|
|
|
|
|
{
|
|
|
|
|
var pagingQuery = Query.OrderBy(pagingSpec.OrderByClause(), pagingSpec.ToSortDirection())
|
|
|
|
|
.Skip(pagingSpec.PagingOffset())
|
|
|
|
|
.Take(pagingSpec.PageSize);
|
|
|
|
|
|
|
|
|
|
pagingSpec.Records = pagingQuery.ToList();
|
|
|
|
|
|
|
|
|
|
//TODO: Use the same query for count and records
|
|
|
|
|
pagingSpec.TotalRecords = Count();
|
|
|
|
|
|
|
|
|
|
return pagingSpec;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-05-22 07:31:29 +02:00
|
|
|
|
private void PublishModelEvent(TModel model, RepositoryAction action)
|
2013-05-21 06:10:04 +02:00
|
|
|
|
{
|
2013-05-22 07:31:29 +02:00
|
|
|
|
if (PublishModelEvents)
|
|
|
|
|
{
|
|
|
|
|
_messageAggregator.PublishEvent(new ModelEvent<TModel>(model, action));
|
|
|
|
|
}
|
2013-05-21 06:10:04 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-05-31 01:32:50 +02:00
|
|
|
|
protected virtual void OnModelChanged(IEnumerable<TModel> models)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected virtual void OnModelDeleted(IEnumerable<TModel> models)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-22 07:31:29 +02:00
|
|
|
|
protected virtual bool PublishModelEvents
|
|
|
|
|
{
|
|
|
|
|
get { return false; }
|
|
|
|
|
}
|
2013-02-05 05:07:07 +01:00
|
|
|
|
}
|
|
|
|
|
}
|