SysLangTextService.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using AngleSharp.Dom;
  7. namespace Admin.NET.Core.Service;
  8. /// <summary>
  9. /// 翻译服务 🧩
  10. /// </summary>
  11. [ApiDescriptionSettings(Order = 100, Description = "翻译服务")]
  12. public partial class SysLangTextService : IDynamicApiController, ITransient
  13. {
  14. private readonly SqlSugarRepository<SysLangText> _sysLangTextRep;
  15. private readonly ISqlSugarClient _sqlSugarClient;
  16. private readonly SysLangTextCacheService _sysLangTextCacheService;
  17. public SysLangTextService(
  18. SqlSugarRepository<SysLangText> sysLangTextRep,
  19. SysLangTextCacheService sysLangTextCacheService,
  20. ISqlSugarClient sqlSugarClient)
  21. {
  22. _sysLangTextRep = sysLangTextRep;
  23. _sqlSugarClient = sqlSugarClient;
  24. _sysLangTextCacheService = sysLangTextCacheService;
  25. }
  26. /// <summary>
  27. /// 分页查询翻译表 🔖
  28. /// </summary>
  29. /// <param name="input"></param>
  30. /// <returns></returns>
  31. [DisplayName("分页查询翻译表")]
  32. [ApiDescriptionSettings(Name = "Page"), HttpPost]
  33. public async Task<SqlSugarPagedList<SysLangTextOutput>> Page(PageSysLangTextInput input)
  34. {
  35. input.Keyword = input.Keyword?.Trim();
  36. var query = _sysLangTextRep.AsQueryable()
  37. .WhereIF(!string.IsNullOrWhiteSpace(input.Keyword), u => u.EntityName.Contains(input.Keyword) || u.FieldName.Contains(input.Keyword) || u.LangCode.Contains(input.Keyword) || u.Content.Contains(input.Keyword))
  38. .WhereIF(!string.IsNullOrWhiteSpace(input.EntityName), u => u.EntityName.Contains(input.EntityName.Trim()))
  39. .WhereIF(!string.IsNullOrWhiteSpace(input.FieldName), u => u.FieldName.Contains(input.FieldName.Trim()))
  40. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode.Contains(input.LangCode.Trim()))
  41. .WhereIF(!string.IsNullOrWhiteSpace(input.Content), u => u.Content.Contains(input.Content.Trim()))
  42. .WhereIF(input.EntityId != null, u => u.EntityId == input.EntityId)
  43. .Select<SysLangTextOutput>();
  44. return await query.OrderBuilder(input).ToPagedListAsync(input.Page, input.PageSize);
  45. }
  46. [DisplayName("获取翻译表")]
  47. [ApiDescriptionSettings(Name = "List"), HttpPost]
  48. public async Task<List<SysLangTextOutput>> List(ListSysLangTextInput input)
  49. {
  50. var query = _sysLangTextRep.AsQueryable()
  51. .Where(u => u.EntityName == input.EntityName.Trim() && u.FieldName == input.FieldName.Trim() && u.EntityId == input.EntityId)
  52. .WhereIF(!string.IsNullOrWhiteSpace(input.LangCode), u => u.LangCode == input.LangCode.Trim())
  53. .Select<SysLangTextOutput>();
  54. return await query.ToListAsync();
  55. }
  56. /// <summary>
  57. /// 获取翻译表详情 ℹ️
  58. /// </summary>
  59. /// <param name="input"></param>
  60. /// <returns></returns>
  61. [DisplayName("获取翻译表详情")]
  62. [ApiDescriptionSettings(Name = "Detail"), HttpGet]
  63. public async Task<SysLangText> Detail([FromQuery] QueryByIdSysLangTextInput input)
  64. {
  65. return await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id);
  66. }
  67. /// <summary>
  68. /// 增加翻译表 ➕
  69. /// </summary>
  70. /// <param name="input"></param>
  71. /// <returns></returns>
  72. [DisplayName("增加翻译表")]
  73. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  74. public async Task<long> Add(AddSysLangTextInput input)
  75. {
  76. var entity = input.Adapt<SysLangText>();
  77. return await _sysLangTextRep.InsertAsync(entity) ? entity.Id : 0;
  78. }
  79. /// <summary>
  80. /// 更新翻译表 ✏️
  81. /// </summary>
  82. /// <param name="input"></param>
  83. /// <returns></returns>
  84. [DisplayName("更新翻译表")]
  85. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  86. public async Task Update(UpdateSysLangTextInput input)
  87. {
  88. var entity = input.Adapt<SysLangText>();
  89. await _sysLangTextRep.AsUpdateable(entity)
  90. .ExecuteCommandAsync();
  91. _sysLangTextCacheService.UpdateCache(entity.EntityName,entity.FieldName,entity.EntityId,entity.LangCode,entity.Content);
  92. }
  93. /// <summary>
  94. /// 删除翻译表 ❌
  95. /// </summary>
  96. /// <param name="input"></param>
  97. /// <returns></returns>
  98. [DisplayName("删除翻译表")]
  99. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  100. public async Task Delete(DeleteSysLangTextInput input)
  101. {
  102. var entity = await _sysLangTextRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  103. await _sysLangTextRep.DeleteAsync(entity); //真删除
  104. _sysLangTextCacheService.DeleteCache(entity.EntityName, entity.FieldName, entity.EntityId, entity.LangCode);
  105. }
  106. /// <summary>
  107. /// 批量删除翻译表 ❌
  108. /// </summary>
  109. /// <param name="input"></param>
  110. /// <returns></returns>
  111. [DisplayName("批量删除翻译表")]
  112. [ApiDescriptionSettings(Name = "BatchDelete"), HttpPost]
  113. public async Task BatchDelete([Required(ErrorMessage = "主键列表不能为空")] List<DeleteSysLangTextInput> input)
  114. {
  115. var exp = Expressionable.Create<SysLangText>();
  116. foreach (var row in input) exp = exp.Or(it => it.Id == row.Id);
  117. var list = await _sysLangTextRep.AsQueryable().Where(exp.ToExpression()).ToListAsync();
  118. await _sysLangTextRep.DeleteAsync(list); //真删除
  119. foreach (var item in list)
  120. {
  121. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  122. }
  123. }
  124. private static readonly object _sysLangTextBatchSaveLock = new object();
  125. /// <summary>
  126. /// 批量保存翻译表 ✏️
  127. /// </summary>
  128. /// <param name="input"></param>
  129. /// <returns></returns>
  130. [DisplayName("批量保存翻译表")]
  131. [ApiDescriptionSettings(Name = "BatchSave"), HttpPost]
  132. public void BatchSave([Required(ErrorMessage = "列表不能为空")] List<ImportSysLangTextInput> input)
  133. {
  134. lock (_sysLangTextBatchSaveLock)
  135. {
  136. // 校验并过滤必填基本类型为null的字段
  137. var rows = input.Where(x =>
  138. {
  139. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  140. if (x.EntityId == null)
  141. {
  142. x.Error = "所属实体ID不能为空";
  143. return false;
  144. }
  145. return true;
  146. }).Adapt<List<SysLangText>>();
  147. var storageable = _sysLangTextRep.Context.Storageable(rows)
  148. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  149. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  150. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  151. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  152. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  153. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  154. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  155. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  156. .SplitInsert(it => it.NotAny())
  157. .SplitUpdate(it => it.Any())
  158. .ToStorage();
  159. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  160. storageable.AsUpdateable.UpdateColumns(it => new
  161. {
  162. it.EntityName,
  163. it.EntityId,
  164. it.FieldName,
  165. it.LangCode,
  166. it.Content,
  167. }).ExecuteCommand();// 存在更新
  168. foreach (var item in rows)
  169. {
  170. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  171. }
  172. if (storageable.ErrorList.Any())
  173. {
  174. throw Oops.Oh($"处理过程中出现以下错误:{string.Join(";", storageable.ErrorList.Distinct())}");
  175. }
  176. }
  177. }
  178. /// <summary>
  179. /// 导出翻译表记录 🔖
  180. /// </summary>
  181. /// <param name="input"></param>
  182. /// <returns></returns>
  183. [DisplayName("导出翻译表记录")]
  184. [ApiDescriptionSettings(Name = "Export"), HttpPost, NonUnify]
  185. public async Task<IActionResult> Export(PageSysLangTextInput input)
  186. {
  187. var list = (await Page(input)).Items?.Adapt<List<ExportSysLangTextOutput>>() ?? new();
  188. if (input.SelectKeyList?.Count > 0) list = list.Where(x => input.SelectKeyList.Contains(x.Id)).ToList();
  189. return ExcelHelper.ExportTemplate(list, "翻译表导出记录");
  190. }
  191. /// <summary>
  192. /// 下载翻译表数据导入模板 ⬇️
  193. /// </summary>
  194. /// <returns></returns>
  195. [DisplayName("下载翻译表数据导入模板")]
  196. [ApiDescriptionSettings(Name = "Import"), HttpGet, NonUnify]
  197. public IActionResult DownloadTemplate()
  198. {
  199. return ExcelHelper.ExportTemplate(new List<ExportSysLangTextOutput>(), "翻译表导入模板");
  200. }
  201. private static readonly object _sysLangTextImportLock = new object();
  202. /// <summary>
  203. /// 导入翻译表记录 💾
  204. /// </summary>
  205. /// <returns></returns>
  206. [DisplayName("导入翻译表记录")]
  207. [ApiDescriptionSettings(Name = "Import"), HttpPost, NonUnify, UnitOfWork]
  208. public IActionResult ImportData([Required] IFormFile file)
  209. {
  210. lock (_sysLangTextImportLock)
  211. {
  212. var stream = ExcelHelper.ImportData<ImportSysLangTextInput, SysLangText>(file, (list, markerErrorAction) =>
  213. {
  214. _sqlSugarClient.Utilities.PageEach(list, 2048, pageItems =>
  215. {
  216. // 校验并过滤必填基本类型为null的字段
  217. var rows = pageItems.Where(x =>
  218. {
  219. if (!string.IsNullOrWhiteSpace(x.Error)) return false;
  220. if (x.EntityId == null)
  221. {
  222. x.Error = "所属实体ID不能为空";
  223. return false;
  224. }
  225. return true;
  226. }).Adapt<List<SysLangText>>();
  227. var storageable = _sysLangTextRep.Context.Storageable(rows)
  228. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.EntityName), "所属实体名不能为空")
  229. .SplitError(it => it.Item.EntityName?.Length > 255, "所属实体名长度不能超过255个字符")
  230. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.FieldName), "字段名不能为空")
  231. .SplitError(it => it.Item.FieldName?.Length > 255, "字段名长度不能超过255个字符")
  232. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.LangCode), "语言代码不能为空")
  233. .SplitError(it => it.Item.LangCode?.Length > 255, "语言代码长度不能超过255个字符")
  234. .SplitError(it => string.IsNullOrWhiteSpace(it.Item.Content), "翻译内容不能为空")
  235. .SplitError(it => it.Item.Content?.Length > 255, "翻译内容长度不能超过255个字符")
  236. .WhereColumns(it => new { it.EntityId, it.EntityName, it.FieldName, it.LangCode })
  237. .SplitInsert(it => it.NotAny())
  238. .SplitUpdate(it => it.Any())
  239. .ToStorage();
  240. storageable.AsInsertable.ExecuteCommand();// 不存在插入
  241. storageable.AsUpdateable.UpdateColumns(it => new
  242. {
  243. it.EntityName,
  244. it.EntityId,
  245. it.FieldName,
  246. it.LangCode,
  247. it.Content,
  248. }).ExecuteCommand();// 存在更新
  249. foreach (var item in rows)
  250. {
  251. _sysLangTextCacheService.DeleteCache(item.EntityName, item.FieldName, item.EntityId, item.LangCode);
  252. }
  253. // 标记错误信息
  254. markerErrorAction.Invoke(storageable, pageItems, rows);
  255. });
  256. });
  257. return stream;
  258. }
  259. }
  260. }