You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

LuEfCrudDataAccess.cs 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using Luticate2.Utils.Dbo.Basic;
  6. using Luticate2.Utils.Dbo.Filter;
  7. using Luticate2.Utils.Dbo.OrderBy;
  8. using Luticate2.Utils.Dbo.Result;
  9. using Luticate2.Utils.Interfaces;
  10. using Luticate2.Utils.Utils;
  11. using Microsoft.EntityFrameworkCore;
  12. namespace Luticate2.Utils.DataAccess
  13. {
  14. public abstract class LuEfCrudDataAccess<TModel, TDboCreate, TDboRead, TDboUpdate, TDbContext, TId> :
  15. LuEfDataAccess<TModel, TDbContext>,
  16. ILuCrudInterface<TDboCreate, TDboRead, TDboUpdate, TId>
  17. where TModel : class
  18. where TDboCreate : class
  19. where TDboRead : class
  20. where TDboUpdate : class
  21. where TDbContext : DbContext
  22. {
  23. protected LuEfCrudDataAccess(IServiceProvider serviceProvider) : base(serviceProvider)
  24. {
  25. }
  26. protected abstract TModel GetModelFromTCreate(TDboCreate obj);
  27. protected abstract void EditModelFromTUpdate(TDboUpdate obj, TModel model);
  28. protected abstract TDboRead GetDboFromModel(TModel model);
  29. protected virtual Expression<Func<TModel, object>> GetOrderByFieldExpression(string fieldName)
  30. {
  31. fieldName = fieldName.ToSnakeCase();
  32. if (!typeof(TModel).HasProperty(fieldName))
  33. {
  34. return null;
  35. }
  36. var param = Expression.Parameter(typeof(TModel), "x");
  37. var prop = Expression.Property(param, fieldName);
  38. var converted = Expression.Convert(prop, typeof(object));
  39. var exp = Expression.Lambda<Func<TModel, object>>(converted, param);
  40. return exp;
  41. }
  42. protected virtual Expression<Func<TModel, bool>> GetFilterExpression(LuFilterDbo filter)
  43. {
  44. return model => true;
  45. }
  46. protected virtual object GetId(TId id)
  47. {
  48. var s = id as string;
  49. if (s != null)
  50. {
  51. Guid guid;
  52. if (Guid.TryParse(s, out guid))
  53. {
  54. return guid;
  55. }
  56. return null;
  57. }
  58. return id;
  59. }
  60. protected virtual object GetModelId(object id)
  61. {
  62. var g = id as Guid?;
  63. if (g != null)
  64. {
  65. return g.ToString();
  66. }
  67. return id;
  68. }
  69. protected virtual IQueryable<TModel> GetGetQueryable(TDbContext db, DbSet<TModel> table)
  70. {
  71. return table;
  72. }
  73. protected virtual IQueryable<TModel> GetEditQueryable(TDbContext db, DbSet<TModel> table)
  74. {
  75. return table;
  76. }
  77. protected virtual IQueryable<TModel> GetDeleteQueryable(TDbContext db, DbSet<TModel> table)
  78. {
  79. return table;
  80. }
  81. protected virtual Func<TDboRead, T> GetIdFunc<T>()
  82. {
  83. return x => ((dynamic) x).Id;
  84. }
  85. protected virtual TModel GetModelFromTUpdate(TDboUpdate obj, TModel model)
  86. {
  87. EditModelFromTUpdate(obj, model);
  88. return model;
  89. }
  90. protected virtual LuResult<T> GetNotFoundResult<T>()
  91. {
  92. return LuResult<T>.Error(LuStatus.NotFound, typeof(TModel).Name + ": Value not found", "");
  93. }
  94. protected virtual IQueryable<TModel> GetFilterQuery(LuFilterDbo filter, TDbContext db, DbSet<TModel> table)
  95. {
  96. return GetGetQueryable(db, table).Where(GetFilterExpression(filter));
  97. }
  98. protected virtual LuResult<T> _Add<T>(TDboCreate obj, TDbContext db, DbSet<TModel> table)
  99. {
  100. return LuResult<T>.Ok(default(T));
  101. }
  102. public virtual LuResult<T> Add<T>(IEnumerable<TDboCreate> objs, Func<IEnumerable<TDboRead>, T> returnFunc)
  103. {
  104. var models = new List<TModel>();
  105. var addRes = Execute((db, table) =>
  106. {
  107. var transact = BeginTransaction(db);
  108. foreach (var dbo in objs)
  109. {
  110. var model = GetModelFromTCreate(dbo);
  111. table.Add(model);
  112. models.Add(model);
  113. var res = _Add<T>(dbo, db, table);
  114. if (!res)
  115. {
  116. return res;
  117. }
  118. }
  119. db.SaveChanges();
  120. foreach (var model in models)
  121. {
  122. db.Entry(model).State = EntityState.Detached;
  123. }
  124. CommitTransaction(transact);
  125. return LuResult<T>.Ok(default(T));
  126. });
  127. if (!addRes)
  128. {
  129. return addRes;
  130. }
  131. return GetMultiple(models, returnFunc);
  132. }
  133. public virtual LuResult<T> Add<T>(TDboCreate obj, Func<TDboRead, T> returnFunc)
  134. {
  135. return Add(new List<TDboCreate> {obj}, list => returnFunc(list.First()));
  136. }
  137. public virtual LuResult<IEnumerable<TId>> AddId(IEnumerable<TDboCreate> objs)
  138. {
  139. var func = GetIdFunc<TId>();
  140. return Add(objs, list => list.Select(func));
  141. }
  142. public virtual LuResult<TId> AddId(TDboCreate obj)
  143. {
  144. return AddId(new List<TDboCreate> {obj}).To(list => list.First());
  145. }
  146. public virtual LuResult<IEnumerable<TDboRead>> AddDbo(IEnumerable<TDboCreate> obj)
  147. {
  148. return Add(obj, read => read);
  149. }
  150. public virtual LuResult<TDboRead> AddDbo(TDboCreate obj)
  151. {
  152. return AddDbo(new List<TDboCreate> {obj}).To(list => list.First());
  153. }
  154. public virtual LuResult<TDboRead> GetSingle(Expression<Func<TModel, bool>> predicate)
  155. {
  156. return Execute((db, table) =>
  157. {
  158. var model = GetGetQueryable(db, table).AsNoTracking().FirstOrDefault(predicate);
  159. if (model == default(TModel))
  160. {
  161. return GetNotFoundResult<TDboRead>();
  162. }
  163. var dbo = GetDboFromModel(model);
  164. return LuResult<TDboRead>.Ok(dbo);
  165. });
  166. }
  167. public virtual LuResult<TDboRead> GetSingleByKeys(params KeyValuePair<string, object>[] keys)
  168. {
  169. return GetSingle(GetExpression(keys));
  170. }
  171. public virtual LuResult<TDboRead> GetSingleById(TId id)
  172. {
  173. var guid = GetId(id);
  174. if (guid == null)
  175. {
  176. return GetNotFoundResult<TDboRead>();
  177. }
  178. return GetSingleByKeys(new KeyValuePair<string, object>("id", guid));
  179. }
  180. protected virtual LuResult<T> GetMultiple<T>(IList<TModel> models, Func<IEnumerable<TDboRead>, T> returnFunc)
  181. {
  182. var dbos = new List<TDboRead>();
  183. foreach (var model in models)
  184. {
  185. var getRes = GetSingleById((TId) GetModelId(((dynamic)model).id));
  186. if (!getRes)
  187. {
  188. return getRes.To<T>();
  189. }
  190. dbos.Add(getRes.Data);
  191. }
  192. var res = returnFunc(dbos);
  193. return LuResult<T>.Ok(res);
  194. }
  195. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(Func<IQueryable<TModel>, IOrderedQueryable<TModel>> orderBy,
  196. Expression<Func<TModel, bool>> predicate, int page = 0, int perPage = int.MaxValue,
  197. params Func<IOrderedQueryable<TModel>, IOrderedQueryable<TModel>>[] otherOrderBy)
  198. {
  199. return Execute((db, table) =>
  200. {
  201. var queryable = GetGetQueryable(db, table);
  202. var count = queryable.Count(predicate);
  203. var ordered = orderBy(queryable);
  204. foreach (var func in otherOrderBy)
  205. {
  206. ordered = func(ordered);
  207. }
  208. var data = ordered.Where(predicate).Skip(page * perPage).Take(perPage).AsNoTracking().Select(GetDboFromModel).ToList();
  209. var result = new LuPaginatedDbo<TDboRead>
  210. {
  211. Count = count,
  212. Data = data
  213. };
  214. return LuResult<LuPaginatedDbo<TDboRead>>.Ok(result);
  215. });
  216. }
  217. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(Expression<Func<TModel, object>> orderBy,
  218. Expression<Func<TModel, bool>> predicate, int page = 0, int perPage = int.MaxValue,
  219. params Expression<Func<TModel, object>>[] otherOrderBy)
  220. {
  221. return GetMultiple(table => table.OrderBy(orderBy), predicate, page, perPage,
  222. otherOrderBy.Select<Expression<Func<TModel, object>>, Func<IOrderedQueryable<TModel>,
  223. IOrderedQueryable<TModel>>>(expression => (t => t.ThenBy(expression))).ToArray());
  224. }
  225. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(Func<IQueryable<TModel>, IOrderedQueryable<TModel>> orderBy,
  226. int page = 0, int perPage = int.MaxValue, params Func<IOrderedQueryable<TModel>, IOrderedQueryable<TModel>>[] otherOrderBy)
  227. {
  228. return GetMultiple(orderBy, x => true, page, perPage, otherOrderBy);
  229. }
  230. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(Expression<Func<TModel, object>> orderBy,
  231. int page = 0, int perPage = int.MaxValue, params Expression<Func<TModel, object>>[] otherOrderBy)
  232. {
  233. return GetMultiple(orderBy, x => true, page, perPage, otherOrderBy);
  234. }
  235. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(LuOrderByDbo orderBy, Func<TDbContext, DbSet<TModel>, IQueryable<TModel>> getQueryable,
  236. int page = 0, int perPage = int.MaxValue)
  237. {
  238. return Execute((db, table) =>
  239. {
  240. var queryable = getQueryable(db, table);
  241. var count = queryable.Count();
  242. IOrderedQueryable<TModel> ordered = null;
  243. foreach (var field in orderBy.Fields)
  244. {
  245. var exp = GetOrderByFieldExpression(field.Name);
  246. if (exp == null)
  247. {
  248. return LuResult<LuPaginatedDbo<TDboRead>>.Error(LuStatus.InputError,
  249. string.Format("LuEfCrudDataAccess: {0}", field.Name), "Invalid order by field");
  250. }
  251. if (ordered != null)
  252. {
  253. ordered = field.Asc ? ordered.ThenBy(exp) : ordered.ThenByDescending(exp);
  254. }
  255. else
  256. {
  257. ordered = field.Asc ? queryable.OrderBy(exp) : queryable.OrderByDescending(exp);
  258. }
  259. }
  260. var data = ordered.Skip(page * perPage).Take(perPage).AsNoTracking().Select(GetDboFromModel).ToList();
  261. var result = new LuPaginatedDbo<TDboRead>
  262. {
  263. Count = count,
  264. Data = data
  265. };
  266. return LuResult<LuPaginatedDbo<TDboRead>>.Ok(result);
  267. });
  268. }
  269. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(LuOrderByDbo orderBy, LuFilterDbo filter,
  270. int page = 0, int perPage = int.MaxValue)
  271. {
  272. return GetMultiple(orderBy, (db, table) => GetFilterQuery(filter, db, table), page, perPage);
  273. }
  274. public virtual LuResult<LuPaginatedDbo<TDboRead>> GetMultiple(LuOrderByDbo orderBy, int page = 0, int perPage = int.MaxValue)
  275. {
  276. return GetMultiple(orderBy, GetGetQueryable, page, perPage);
  277. }
  278. // public virtual LuResult<T> Edit<T>(Expression<Func<TModel, bool>> predicate, Action<TModel> update,
  279. // Func<IEnumerable<TDboRead>, T> returnFunc)
  280. // {
  281. // IList<TModel> models = null;
  282. // var editRes = Execute((db, table) =>
  283. // {
  284. // models = GetEditQueryable(db, table).Where(predicate).AsNoTracking().ToList();
  285. // foreach (var model in models)
  286. // {
  287. // update(model);
  288. // db.Entry(model).State = EntityState.Modified;
  289. // }
  290. // db.SaveChanges();
  291. // return LuResult<T>.Ok(default(T));
  292. // });
  293. // if (!editRes)
  294. // {
  295. // return editRes;
  296. // }
  297. // return GetMultiple(models, returnFunc);
  298. // }
  299. //
  300. // public virtual LuResult<IEnumerable<TId>> EditId(Expression<Func<TModel, bool>> predicate, Action<TModel> update)
  301. // {
  302. // var func = GetIdFunc<TId>();
  303. // return Edit(predicate, update, reads => reads.Select(func));
  304. // }
  305. //
  306. // public virtual LuResult<IEnumerable<TDboRead>> EditDbo(Expression<Func<TModel, bool>> predicate, Action<TModel> update)
  307. // {
  308. // return Edit(predicate, update, read => read);
  309. // }
  310. // public virtual LuResult<T> EditSingleById<T>(TId id, Action<TModel> update, Func<TDboRead, T> returnFunc)
  311. // {
  312. // var guid = GetId(id);
  313. // if (guid == null)
  314. // {
  315. // return GetNotFoundResult<T>();
  316. // }
  317. // return Edit(GetExpression(new KeyValuePair<string, object>("id", guid)), update,
  318. // list => returnFunc(list.FirstOrDefault()));
  319. // }
  320. //
  321. // public virtual LuResult<TId> EditSingleByIdId(TId id, Action<TModel> update)
  322. // {
  323. // return EditSingleById(id, update, GetIdFunc<TId>());
  324. // }
  325. //
  326. // public virtual LuResult<TDboRead> EditSingleByIdDbo(TId id, Action<TModel> update)
  327. // {
  328. // return EditSingleById(id, update, read => read);
  329. // }
  330. protected virtual LuResult<T> _EditSingleById<T>(TId id, TModel model, TDboUpdate obj, TDbContext db, DbSet<TModel> table)
  331. {
  332. return LuResult<T>.Ok(default(T));
  333. }
  334. public virtual LuResult<T> EditSingleById<T>(TId id, TDboUpdate update, Func<TDboRead, T> returnFunc)
  335. {
  336. var guid = GetId(id);
  337. if (guid == null)
  338. {
  339. return GetNotFoundResult<T>();
  340. }
  341. var transact = BeginTransaction(null);
  342. var editRes = Execute((db, table) =>
  343. {
  344. var model = GetEditQueryable(db, table).FirstOrDefault(GetExpression(new KeyValuePair<string, object>("id", guid)));
  345. EditModelFromTUpdate(update, model);
  346. var res = _EditSingleById<T>(id, model, update, db, table);
  347. if (!res)
  348. {
  349. RollbackTransaction(transact);
  350. return res;
  351. }
  352. db.Entry(model).State = EntityState.Modified;
  353. db.SaveChanges();
  354. return LuResult<T>.Ok(default(T));
  355. });
  356. if (!editRes)
  357. {
  358. RollbackTransaction(transact);
  359. return editRes;
  360. }
  361. var getRes = GetSingleById(id).To(returnFunc);
  362. CommitTransaction(transact);
  363. return getRes;
  364. }
  365. public virtual LuResult<TId> EditSingleByIdId(TId id, TDboUpdate update)
  366. {
  367. return EditSingleById(id, update, GetIdFunc<TId>());
  368. }
  369. public virtual LuResult<TDboRead> EditSingleByIdDbo(TId id, TDboUpdate update)
  370. {
  371. return EditSingleById(id, update, read => read);
  372. }
  373. public virtual LuResult<T> Delete<T>(Expression<Func<TModel, bool>> predicate,
  374. Func<IEnumerable<TDboRead>, T> returnFunc)
  375. {
  376. return Execute((db, table) =>
  377. {
  378. var transact = BeginTransaction(db);
  379. var models = GetDeleteQueryable(db, table).Where(predicate).AsNoTracking().ToList();
  380. var getRes = GetMultiple(models, returnFunc);
  381. if (!getRes)
  382. {
  383. RollbackTransaction(transact);
  384. return getRes;
  385. }
  386. table.RemoveRange(models);
  387. db.SaveChanges();
  388. CommitTransaction(transact);
  389. return getRes;
  390. });
  391. }
  392. public virtual LuResult<IEnumerable<TId>> DeleteId(Expression<Func<TModel, bool>> predicate)
  393. {
  394. var func = GetIdFunc<TId>();
  395. return Delete(predicate, reads => reads.Select(func));
  396. }
  397. public virtual LuResult<IEnumerable<TDboRead>> DeleteDbo(Expression<Func<TModel, bool>> predicate)
  398. {
  399. return Delete(predicate, read => read);
  400. }
  401. public virtual LuResult<T> DeleteSingleById<T>(TId id, Func<TDboRead, T> returnFunc)
  402. {
  403. var guid = GetId(id);
  404. if (guid == null)
  405. {
  406. return GetNotFoundResult<T>();
  407. }
  408. return Delete(GetExpression(new KeyValuePair<string, object>("id", guid)),
  409. reads => returnFunc(reads.FirstOrDefault()));
  410. }
  411. public virtual LuResult<TId> DeleteSingleByIdId(TId id)
  412. {
  413. var func = GetIdFunc<TId>();
  414. return DeleteSingleById(id, func);
  415. }
  416. public virtual LuResult<TDboRead> DeleteSingleByIdDbo(TId id)
  417. {
  418. return DeleteSingleById(id, read => read);
  419. }
  420. }
  421. }