RepositoryExtension.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using MapsterMapper;
  7. namespace Admin.NET.Core;
  8. public static class RepositoryExtension
  9. {
  10. /// <summary>
  11. /// 实体假删除 _rep.FakeDelete(entity)
  12. /// </summary>
  13. /// <typeparam name="T"></typeparam>
  14. /// <param name="repository"></param>
  15. /// <param name="entity"></param>
  16. /// <returns></returns>
  17. public static int FakeDelete<T>(this ISugarRepository repository, T entity) where T : EntityBaseDel, new()
  18. {
  19. return repository.Context.FakeDelete(entity);
  20. }
  21. /// <summary>
  22. /// 实体假删除 db.FakeDelete(entity)
  23. /// </summary>
  24. /// <typeparam name="T"></typeparam>
  25. /// <param name="db"></param>
  26. /// <param name="entity"></param>
  27. /// <returns></returns>
  28. public static int FakeDelete<T>(this ISqlSugarClient db, T entity) where T : EntityBaseDel, new()
  29. {
  30. return db.Updateable(entity).AS().ReSetValue(x => { x.IsDelete = true; })
  31. .IgnoreColumns(ignoreAllNullColumns: true)
  32. .EnableDiffLogEvent() // 记录差异日志
  33. .UpdateColumns(x => new { x.IsDelete, x.UpdateTime, x.UpdateUserId }) // 允许更新的字段-AOP拦截自动设置UpdateTime、UpdateUserId
  34. .ExecuteCommand();
  35. }
  36. /// <summary>
  37. /// 实体集合批量假删除 _rep.FakeDelete(entity)
  38. /// </summary>
  39. /// <typeparam name="T"></typeparam>
  40. /// <param name="repository"></param>
  41. /// <param name="entity"></param>
  42. /// <returns></returns>
  43. public static int FakeDelete<T>(this ISugarRepository repository, List<T> entity) where T : EntityBaseDel, new()
  44. {
  45. return repository.Context.FakeDelete(entity);
  46. }
  47. /// <summary>
  48. /// 实体集合批量假删除 db.FakeDelete(entity)
  49. /// </summary>
  50. /// <typeparam name="T"></typeparam>
  51. /// <param name="db"></param>
  52. /// <param name="entity"></param>
  53. /// <returns></returns>
  54. public static int FakeDelete<T>(this ISqlSugarClient db, List<T> entity) where T : EntityBaseDel, new()
  55. {
  56. return db.Updateable(entity).AS().ReSetValue(x => { x.IsDelete = true; })
  57. .IgnoreColumns(ignoreAllNullColumns: true)
  58. .EnableDiffLogEvent() // 记录差异日志
  59. .UpdateColumns(x => new { x.IsDelete, x.UpdateTime, x.UpdateUserId }) // 允许更新的字段-AOP拦截自动设置UpdateTime、UpdateUserId
  60. .ExecuteCommand();
  61. }
  62. /// <summary>
  63. /// 实体假删除异步 _rep.FakeDeleteAsync(entity)
  64. /// </summary>
  65. /// <typeparam name="T"></typeparam>
  66. /// <param name="repository"></param>
  67. /// <param name="entity"></param>
  68. /// <returns></returns>
  69. public static Task<int> FakeDeleteAsync<T>(this ISugarRepository repository, T entity) where T : EntityBaseDel, new()
  70. {
  71. return repository.Context.FakeDeleteAsync(entity);
  72. }
  73. /// <summary>
  74. /// 实体假删除 db.FakeDelete(entity)
  75. /// </summary>
  76. /// <typeparam name="T"></typeparam>
  77. /// <param name="db"></param>
  78. /// <param name="entity"></param>
  79. /// <returns></returns>
  80. public static Task<int> FakeDeleteAsync<T>(this ISqlSugarClient db, T entity) where T : EntityBaseDel, new()
  81. {
  82. return db.Updateable(entity).AS().ReSetValue(x => { x.IsDelete = true; })
  83. .IgnoreColumns(ignoreAllNullColumns: true)
  84. .EnableDiffLogEvent() // 记录差异日志
  85. .UpdateColumns(x => new { x.IsDelete, x.UpdateTime, x.UpdateUserId }) // 允许更新的字段-AOP拦截自动设置UpdateTime、UpdateUserId
  86. .ExecuteCommandAsync();
  87. }
  88. /// <summary>
  89. /// 实体集合批量假删除异步 _rep.FakeDeleteAsync(entity)
  90. /// </summary>
  91. /// <typeparam name="T"></typeparam>
  92. /// <param name="repository"></param>
  93. /// <param name="entity"></param>
  94. /// <returns></returns>
  95. public static Task<int> FakeDeleteAsync<T>(this ISugarRepository repository, List<T> entity) where T : EntityBaseDel, new()
  96. {
  97. return repository.Context.FakeDeleteAsync(entity);
  98. }
  99. /// <summary>
  100. /// 实体集合批量假删除 db.FakeDelete(entity)
  101. /// </summary>
  102. /// <typeparam name="T"></typeparam>
  103. /// <param name="db"></param>
  104. /// <param name="entity"></param>
  105. /// <returns></returns>
  106. public static Task<int> FakeDeleteAsync<T>(this ISqlSugarClient db, List<T> entity) where T : EntityBaseDel, new()
  107. {
  108. return db.Updateable(entity).AS().ReSetValue(x => { x.IsDelete = true; })
  109. .IgnoreColumns(ignoreAllNullColumns: true)
  110. .EnableDiffLogEvent() // 记录差异日志
  111. .UpdateColumns(x => new { x.IsDelete, x.UpdateTime, x.UpdateUserId }) // 允许更新的字段-AOP拦截自动设置UpdateTime、UpdateUserId
  112. .ExecuteCommandAsync();
  113. }
  114. /// <summary>
  115. /// 排序方式(默认降序)
  116. /// </summary>
  117. /// <param name="queryable"></param>
  118. /// <param name="pageInput"> </param>
  119. /// <param name="prefix"> </param>
  120. /// <param name="defaultSortField"> 默认排序字段 </param>
  121. /// <param name="descSort"> 是否降序 </param>
  122. /// <returns> </returns>
  123. public static ISugarQueryable<T> OrderBuilder<T>(this ISugarQueryable<T> queryable, BasePageInput pageInput, string prefix = "", string defaultSortField = "Id", bool descSort = true)
  124. {
  125. var iSqlBuilder = InstanceFactory.GetSqlBuilderWithContext(queryable.Context);
  126. // 约定默认每张表都有Id排序
  127. var orderStr = string.IsNullOrWhiteSpace(defaultSortField) ? "" : $"{prefix}{iSqlBuilder.GetTranslationColumnName(defaultSortField)}" + (descSort ? " Desc" : " Asc");
  128. TypeAdapterConfig typeAdapterConfig = new();
  129. typeAdapterConfig.ForType<T, BasePageInput>().IgnoreNullValues(true);
  130. Mapper mapper = new(typeAdapterConfig); // 务必将mapper设为单实例
  131. var nowPagerInput = mapper.Map<BasePageInput>(pageInput);
  132. // 排序是否可用-排序字段和排序顺序都为非空才启用排序
  133. if (!string.IsNullOrEmpty(nowPagerInput.Field) && !string.IsNullOrEmpty(nowPagerInput.Order))
  134. {
  135. nowPagerInput.Field = Regex.Replace(nowPagerInput.Field, @"[\s;()\-'@=/%]", ""); //过滤掉一些关键字符防止构造特殊SQL语句注入
  136. var col = queryable.Context.EntityMaintenance.GetEntityInfo<T>().Columns.FirstOrDefault(u => u.PropertyName.Equals(nowPagerInput.Field, StringComparison.CurrentCultureIgnoreCase));
  137. var dbColumnName = col != null ? col.DbColumnName : nowPagerInput.Field;
  138. orderStr = $"{prefix}{iSqlBuilder.GetTranslationColumnName(dbColumnName)} {(nowPagerInput.Order == nowPagerInput.DescStr ? "Desc" : "Asc")}";
  139. }
  140. return queryable.OrderByIF(!string.IsNullOrWhiteSpace(orderStr), orderStr);
  141. }
  142. /// <summary>
  143. /// 更新实体并记录差异日志 _rep.UpdateWithDiffLog(entity)
  144. /// </summary>
  145. /// <typeparam name="T"></typeparam>
  146. /// <param name="repository"></param>
  147. /// <param name="entity"></param>
  148. /// <param name="ignoreAllNullColumns"></param>
  149. /// <returns></returns>
  150. public static int UpdateWithDiffLog<T>(this ISugarRepository repository, T entity, bool ignoreAllNullColumns = true) where T : EntityBase, new()
  151. {
  152. return repository.Context.UpdateWithDiffLog(entity, ignoreAllNullColumns);
  153. }
  154. /// <summary>
  155. /// 更新实体并记录差异日志 _rep.UpdateWithDiffLog(entity)
  156. /// </summary>
  157. /// <typeparam name="T"></typeparam>
  158. /// <param name="db"></param>
  159. /// <param name="entity"></param>
  160. /// <param name="ignoreAllNullColumns"></param>
  161. /// <returns></returns>
  162. public static int UpdateWithDiffLog<T>(this ISqlSugarClient db, T entity, bool ignoreAllNullColumns = true) where T : EntityBase, new()
  163. {
  164. return db.Updateable(entity).AS()
  165. .IgnoreColumns(ignoreAllNullColumns: ignoreAllNullColumns)
  166. .EnableDiffLogEvent()
  167. .ExecuteCommand();
  168. }
  169. /// <summary>
  170. /// 更新实体并记录差异日志 _rep.UpdateWithDiffLogAsync(entity)
  171. /// </summary>
  172. /// <typeparam name="T"></typeparam>
  173. /// <param name="repository"></param>
  174. /// <param name="entity"></param>
  175. /// <param name="ignoreAllNullColumns"></param>
  176. /// <returns></returns>
  177. public static Task<int> UpdateWithDiffLogAsync<T>(this ISugarRepository repository, T entity, bool ignoreAllNullColumns = true) where T : EntityBase, new()
  178. {
  179. return repository.Context.UpdateWithDiffLogAsync(entity, ignoreAllNullColumns);
  180. }
  181. /// <summary>
  182. /// 更新实体并记录差异日志 _rep.UpdateWithDiffLogAsync(entity)
  183. /// </summary>
  184. /// <typeparam name="T"></typeparam>
  185. /// <param name="db"></param>
  186. /// <param name="entity"></param>
  187. /// <param name="ignoreAllNullColumns"></param>
  188. /// <returns></returns>
  189. public static Task<int> UpdateWithDiffLogAsync<T>(this ISqlSugarClient db, T entity, bool ignoreAllNullColumns = true) where T : EntityBase, new()
  190. {
  191. return db.Updateable(entity)
  192. .IgnoreColumns(ignoreAllNullColumns: ignoreAllNullColumns)
  193. .EnableDiffLogEvent()
  194. .ExecuteCommandAsync();
  195. }
  196. /// <summary>
  197. /// 新增实体并记录差异日志 _rep.InsertWithDiffLog(entity)
  198. /// </summary>
  199. /// <typeparam name="T"></typeparam>
  200. /// <param name="repository"></param>
  201. /// <param name="entity"></param>
  202. /// <returns></returns>
  203. public static int InsertWithDiffLog<T>(this ISugarRepository repository, T entity) where T : EntityBase, new()
  204. {
  205. return repository.Context.InsertWithDiffLog(entity);
  206. }
  207. /// <summary>
  208. /// 新增实体并记录差异日志 _rep.InsertWithDiffLog(entity)
  209. /// </summary>
  210. /// <typeparam name="T"></typeparam>
  211. /// <param name="db"></param>
  212. /// <param name="entity"></param>
  213. /// <returns></returns>
  214. public static int InsertWithDiffLog<T>(this ISqlSugarClient db, T entity) where T : EntityBase, new()
  215. {
  216. return db.Insertable(entity).AS().EnableDiffLogEvent().ExecuteCommand();
  217. }
  218. /// <summary>
  219. /// 新增实体并记录差异日志 _rep.InsertWithDiffLogAsync(entity)
  220. /// </summary>
  221. /// <typeparam name="T"></typeparam>
  222. /// <param name="repository"></param>
  223. /// <param name="entity"></param>
  224. /// <returns></returns>
  225. public static Task<int> InsertWithDiffLogAsync<T>(this ISugarRepository repository, T entity) where T : EntityBase, new()
  226. {
  227. return repository.Context.InsertWithDiffLogAsync(entity);
  228. }
  229. /// <summary>
  230. /// 新增实体并记录差异日志 _rep.InsertWithDiffLog(entity)
  231. /// </summary>
  232. /// <typeparam name="T"></typeparam>
  233. /// <param name="db"></param>
  234. /// <param name="entity"></param>
  235. /// <returns></returns>
  236. public static Task<int> InsertWithDiffLogAsync<T>(this ISqlSugarClient db, T entity) where T : EntityBase, new()
  237. {
  238. return db.Insertable(entity).AS().EnableDiffLogEvent().ExecuteCommandAsync();
  239. }
  240. /// <summary>
  241. /// 多库查询
  242. /// </summary>
  243. /// <param name="queryable"></param>
  244. /// <returns> </returns>
  245. public static ISugarQueryable<T> AS<T>(this ISugarQueryable<T> queryable)
  246. {
  247. var info = GetTableInfo<T>();
  248. return queryable.AS<T>($"{info.Item1}.{info.Item2}");
  249. }
  250. /// <summary>
  251. /// 多库查询
  252. /// </summary>
  253. /// <typeparam name="T"></typeparam>
  254. /// <typeparam name="T2"></typeparam>
  255. /// <param name="queryable"></param>
  256. /// <returns></returns>
  257. public static ISugarQueryable<T, T2> AS<T, T2>(this ISugarQueryable<T, T2> queryable)
  258. {
  259. var info = GetTableInfo<T2>();
  260. return queryable.AS<T2>($"{info.Item1}.{info.Item2}");
  261. }
  262. /// <summary>
  263. /// 多库更新
  264. /// </summary>
  265. /// <param name="updateable"></param>
  266. /// <returns> </returns>
  267. public static IUpdateable<T> AS<T>(this IUpdateable<T> updateable) where T : EntityBase, new()
  268. {
  269. var info = GetTableInfo<T>();
  270. return updateable.AS($"{info.Item1}.{info.Item2}");
  271. }
  272. /// <summary>
  273. /// 多库新增
  274. /// </summary>
  275. /// <param name="insertable"></param>
  276. /// <returns> </returns>
  277. public static IInsertable<T> AS<T>(this IInsertable<T> insertable) where T : EntityBase, new()
  278. {
  279. var info = GetTableInfo<T>();
  280. return insertable.AS($"{info.Item1}.{info.Item2}");
  281. }
  282. /// <summary>
  283. /// 多库删除
  284. /// </summary>
  285. /// <param name="deleteable"></param>
  286. /// <returns> </returns>
  287. public static IDeleteable<T> AS<T>(this IDeleteable<T> deleteable) where T : EntityBase, new()
  288. {
  289. var info = GetTableInfo<T>();
  290. return deleteable.AS($"{info.Item1}.{info.Item2}");
  291. }
  292. /// <summary>
  293. /// 根据实体类型获取表信息
  294. /// </summary>
  295. /// <typeparam name="T"></typeparam>
  296. /// <returns></returns>
  297. private static Tuple<string, string> GetTableInfo<T>()
  298. {
  299. var entityType = typeof(T);
  300. var attr = entityType.GetCustomAttribute<TenantAttribute>();
  301. var configId = attr == null ? SqlSugarConst.MainConfigId : attr.configId.ToString();
  302. var tableName = entityType.GetCustomAttribute<SugarTable>().TableName;
  303. return new Tuple<string, string>(configId, tableName);
  304. }
  305. /// <summary>
  306. /// 禁用过滤器-适用于更新和删除操作(只对当前请求有效,禁止使用异步)
  307. /// </summary>
  308. /// <param name="repository"></param>
  309. /// <param name="action">禁止异步</param>
  310. /// <returns></returns>
  311. public static void RunWithoutFilter(this ISugarRepository repository, Action action)
  312. {
  313. repository.Context.QueryFilter.ClearAndBackup(); // 清空并备份过滤器
  314. action.Invoke();
  315. repository.Context.QueryFilter.Restore(); // 还原过滤器
  316. // 用例
  317. //_rep.RunWithoutFilter(() =>
  318. //{
  319. // 执行更新或者删除
  320. // 禁止使用异步函数
  321. //});
  322. }
  323. /// <summary>
  324. /// 忽略租户
  325. /// </summary>
  326. /// <param name="queryable"></param>
  327. /// <param name="ignore">是否忽略 默认true</param>
  328. /// <returns> </returns>
  329. public static ISugarQueryable<T> IgnoreTenant<T>(this ISugarQueryable<T> queryable, bool ignore = true)
  330. {
  331. return ignore ? queryable.ClearFilter<ITenantIdFilter>() : queryable;
  332. }
  333. /// <summary>
  334. /// 只更新某些列
  335. /// </summary>
  336. /// <typeparam name="T"></typeparam>
  337. /// <typeparam name="R"></typeparam>
  338. /// <param name="updateable"></param>
  339. /// <returns></returns>
  340. public static IUpdateable<T> OnlyUpdateColumn<T, R>(this IUpdateable<T> updateable) where T : EntityBase, new() where R : class, new()
  341. {
  342. if (updateable.UpdateBuilder.UpdateColumns == null)
  343. updateable.UpdateBuilder.UpdateColumns = new List<string>();
  344. foreach (PropertyInfo info in typeof(R).GetProperties())
  345. {
  346. // 判断是否是相同属性
  347. if (typeof(T).GetProperty(info.Name) != null)
  348. updateable.UpdateBuilder.UpdateColumns.Add(info.Name);
  349. }
  350. return updateable;
  351. }
  352. /// <summary>
  353. /// 导航只更新(主表)某些列
  354. /// </summary>
  355. /// <typeparam name="T"></typeparam>
  356. /// <typeparam name="R"></typeparam>
  357. /// <param name="t"></param>
  358. /// <param name="r"></param>
  359. /// <returns></returns>
  360. public static UpdateNavRootOptions OnlyNavUpdateColumn<T, R>(this T t, R r)
  361. {
  362. UpdateNavRootOptions uNOption = new UpdateNavRootOptions();
  363. var updateColumns = new List<string>();
  364. foreach (PropertyInfo info in r.GetType().GetProperties())
  365. {
  366. //判断是否是相同属性
  367. PropertyInfo pro = t.GetType().GetProperty(info.Name);
  368. var attr = pro.GetCustomAttribute<SugarColumn>();
  369. if (pro != null && attr != null && !attr.IsPrimaryKey)
  370. updateColumns.Add(info.Name);
  371. }
  372. uNOption.UpdateColumns = updateColumns.ToArray();
  373. return uNOption;
  374. }
  375. /// <summary>
  376. /// 批量列表in查询
  377. /// </summary>
  378. /// <typeparam name="T1"></typeparam>
  379. /// <typeparam name="T2"></typeparam>
  380. /// <param name="queryable"></param>
  381. /// <param name="exp"></param>
  382. /// <param name="queryList"></param>
  383. /// <param name="stoppingToken"></param>
  384. /// <returns></returns>
  385. public static async Task<List<T1>> BulkListQuery<T1, T2>(this ISugarQueryable<T1> queryable,
  386. Expression<Func<T1, SingleColumnEntity<T2>, bool>> exp,
  387. IEnumerable<T2> queryList,
  388. CancellationToken stoppingToken) where T1 : class, new()
  389. {
  390. // 创建临时表 (用真表兼容性好,表名随机)
  391. var tableName = "Temp" + SnowFlakeSingle.Instance.NextId();
  392. try
  393. {
  394. var type = queryable.Context.DynamicBuilder().CreateClass(tableName, new SugarTable())
  395. .CreateProperty("ColumnName", typeof(string), new SugarColumn() { IsPrimaryKey = true }) // 主键不要自增
  396. .BuilderType();
  397. // 创建表
  398. queryable.Context.CodeFirst.InitTables(type);
  399. var insertData = queryList.Select(it => new SingleColumnEntity<T2>() { ColumnName = it }).ToList();
  400. // 插入临时表
  401. queryable.Context.Fastest<SingleColumnEntity<T2>>()
  402. .AS(tableName)
  403. .BulkCopy(insertData);
  404. var queryTemp = queryable.Context.Queryable<SingleColumnEntity<T2>>()
  405. .AS(tableName);
  406. var systemData = await queryable
  407. .InnerJoin(queryTemp, exp)
  408. .ToListAsync(stoppingToken);
  409. queryable.Context.DbMaintenance.DropTable(tableName);
  410. return systemData;
  411. }
  412. catch (Exception error)
  413. {
  414. queryable.Context.DbMaintenance.DropTable(tableName);
  415. throw Oops.Oh(error);
  416. }
  417. }
  418. }