SysConfigService.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. namespace Admin.NET.Core.Service;
  7. /// <summary>
  8. /// 系统参数配置服务 🧩
  9. /// </summary>
  10. [ApiDescriptionSettings(Order = 440)]
  11. public class SysConfigService : IDynamicApiController, ITransient
  12. {
  13. private readonly SysCacheService _sysCacheService;
  14. private readonly SqlSugarRepository<SysConfig> _sysConfigRep;
  15. public SysConfigService(SysCacheService sysCacheService,
  16. SqlSugarRepository<SysConfig> sysConfigRep)
  17. {
  18. _sysCacheService = sysCacheService;
  19. _sysConfigRep = sysConfigRep;
  20. }
  21. /// <summary>
  22. /// 获取参数配置分页列表 🔖
  23. /// </summary>
  24. /// <param name="input"></param>
  25. /// <returns></returns>
  26. [DisplayName("获取参数配置分页列表")]
  27. public async Task<SqlSugarPagedList<SysConfig>> Page(PageConfigInput input)
  28. {
  29. return await _sysConfigRep.AsQueryable()
  30. .Where(u => u.GroupCode != "WebConfig") // 不显示 WebConfig 分组
  31. .WhereIF(!string.IsNullOrWhiteSpace(input.Name?.Trim()), u => u.Name.Contains(input.Name))
  32. .WhereIF(!string.IsNullOrWhiteSpace(input.Code?.Trim()), u => u.Code.Contains(input.Code))
  33. .WhereIF(!string.IsNullOrWhiteSpace(input.GroupCode?.Trim()), u => u.GroupCode.Equals(input.GroupCode))
  34. .OrderBuilder(input)
  35. .ToPagedListAsync(input.Page, input.PageSize);
  36. }
  37. /// <summary>
  38. /// 获取参数配置列表 🔖
  39. /// </summary>
  40. /// <returns></returns>
  41. [DisplayName("获取参数配置列表")]
  42. public async Task<List<SysConfig>> List(PageConfigInput input)
  43. {
  44. return await _sysConfigRep.AsQueryable()
  45. .WhereIF(!string.IsNullOrWhiteSpace(input.GroupCode?.Trim()), u => u.GroupCode.Equals(input.GroupCode))
  46. .ToListAsync();
  47. }
  48. /// <summary>
  49. /// 增加参数配置 🔖
  50. /// </summary>
  51. /// <param name="input"></param>
  52. /// <returns></returns>
  53. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  54. [DisplayName("增加参数配置")]
  55. public async Task AddConfig(AddConfigInput input)
  56. {
  57. var isExist = await _sysConfigRep.IsAnyAsync(u => u.Name == input.Name || u.Code == input.Code);
  58. if (isExist)
  59. throw Oops.Oh(ErrorCodeEnum.D9000);
  60. await _sysConfigRep.InsertAsync(input.Adapt<SysConfig>());
  61. }
  62. /// <summary>
  63. /// 更新参数配置 🔖
  64. /// </summary>
  65. /// <param name="input"></param>
  66. /// <returns></returns>
  67. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  68. [DisplayName("更新参数配置")]
  69. public async Task UpdateConfig(UpdateConfigInput input)
  70. {
  71. var isExist = await _sysConfigRep.IsAnyAsync(u => (u.Name == input.Name || u.Code == input.Code) && u.Id != input.Id);
  72. if (isExist)
  73. throw Oops.Oh(ErrorCodeEnum.D9000);
  74. var config = input.Adapt<SysConfig>();
  75. await _sysConfigRep.AsUpdateable(config).IgnoreColumns(true).ExecuteCommandAsync();
  76. _sysCacheService.Remove($"{CacheConst.KeyConfig}{config.Code}");
  77. }
  78. /// <summary>
  79. /// 删除参数配置 🔖
  80. /// </summary>
  81. /// <param name="input"></param>
  82. /// <returns></returns>
  83. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  84. [DisplayName("删除参数配置")]
  85. public async Task DeleteConfig(DeleteConfigInput input)
  86. {
  87. var config = await _sysConfigRep.GetFirstAsync(u => u.Id == input.Id);
  88. if (config.SysFlag == YesNoEnum.Y) // 禁止删除系统参数
  89. throw Oops.Oh(ErrorCodeEnum.D9001);
  90. await _sysConfigRep.DeleteAsync(config);
  91. _sysCacheService.Remove($"{CacheConst.KeyConfig}{config.Code}");
  92. }
  93. /// <summary>
  94. /// 批量删除参数配置 🔖
  95. /// </summary>
  96. /// <param name="ids"></param>
  97. /// <returns></returns>
  98. [ApiDescriptionSettings(Name = "BatchDelete"), HttpPost]
  99. [DisplayName("批量删除参数配置")]
  100. public async Task BatchDeleteConfig(List<long> ids)
  101. {
  102. foreach (var id in ids)
  103. {
  104. var config = await _sysConfigRep.GetFirstAsync(u => u.Id == id);
  105. if (config.SysFlag == YesNoEnum.Y) // 禁止删除系统参数
  106. continue;
  107. await _sysConfigRep.DeleteAsync(config);
  108. _sysCacheService.Remove($"{CacheConst.KeyConfig}{config.Code}");
  109. }
  110. }
  111. /// <summary>
  112. /// 获取参数配置详情 🔖
  113. /// </summary>
  114. /// <param name="input"></param>
  115. /// <returns></returns>
  116. [DisplayName("获取参数配置详情")]
  117. public async Task<SysConfig> GetDetail([FromQuery] ConfigInput input)
  118. {
  119. return await _sysConfigRep.GetFirstAsync(u => u.Id == input.Id);
  120. }
  121. /// <summary>
  122. /// 获取参数配置值
  123. /// </summary>
  124. /// <param name="code"></param>
  125. /// <returns></returns>
  126. [NonAction]
  127. public async Task<T> GetConfigValue<T>(string code)
  128. {
  129. if (string.IsNullOrWhiteSpace(code)) return default;
  130. var value = _sysCacheService.Get<string>($"{CacheConst.KeyConfig}{code}");
  131. if (string.IsNullOrEmpty(value))
  132. {
  133. var config = await _sysConfigRep.CopyNew().GetFirstAsync(u => u.Code == code);
  134. value = config != null ? config.Value : default;
  135. _sysCacheService.Set($"{CacheConst.KeyConfig}{code}", value);
  136. }
  137. if (string.IsNullOrWhiteSpace(value)) return default;
  138. return (T)Convert.ChangeType(value, typeof(T));
  139. }
  140. /// <summary>
  141. /// 更新参数配置值
  142. /// </summary>
  143. /// <param name="code"></param>
  144. /// <param name="value"></param>
  145. /// <returns></returns>
  146. [NonAction]
  147. public async Task UpdateConfigValue(string code, string value)
  148. {
  149. var config = await _sysConfigRep.GetFirstAsync(u => u.Code == code);
  150. if (config == null) return;
  151. config.Value = value;
  152. await _sysConfigRep.AsUpdateable(config).ExecuteCommandAsync();
  153. _sysCacheService.Remove($"{CacheConst.KeyConfig}{config.Code}");
  154. }
  155. /// <summary>
  156. /// 获取分组列表 🔖
  157. /// </summary>
  158. /// <returns></returns>
  159. [DisplayName("获取分组列表")]
  160. public async Task<List<string>> GetGroupList()
  161. {
  162. return await _sysConfigRep.AsQueryable()
  163. .Where(u => u.GroupCode != "WebConfig") // 不显示 WebConfig 分组
  164. .GroupBy(u => u.GroupCode)
  165. .Select(u => u.GroupCode).ToListAsync();
  166. }
  167. /// <summary>
  168. /// 获取 Token 过期时间
  169. /// </summary>
  170. /// <returns></returns>
  171. [NonAction]
  172. public async Task<int> GetTokenExpire()
  173. {
  174. var tokenExpireStr = await GetConfigValue<string>(CommonConst.SysTokenExpire);
  175. _ = int.TryParse(tokenExpireStr, out var tokenExpire);
  176. return tokenExpire == 0 ? 20 : tokenExpire;
  177. }
  178. /// <summary>
  179. /// 获取 RefreshToken 过期时间
  180. /// </summary>
  181. /// <returns></returns>
  182. [NonAction]
  183. public async Task<int> GetRefreshTokenExpire()
  184. {
  185. var refreshTokenExpireStr = await GetConfigValue<string>(CommonConst.SysRefreshTokenExpire);
  186. _ = int.TryParse(refreshTokenExpireStr, out var refreshTokenExpire);
  187. return refreshTokenExpire == 0 ? 40 : refreshTokenExpire;
  188. }
  189. /// <summary>
  190. /// 批量更新参数配置值
  191. /// </summary>
  192. /// <param name="input"></param>
  193. /// <returns></returns>
  194. [ApiDescriptionSettings(Name = "BatchUpdate"), HttpPost]
  195. [DisplayName("批量更新参数配置值")]
  196. public async Task BatchUpdateConfig(List<BatchConfigInput> input)
  197. {
  198. foreach (var Config in input)
  199. {
  200. await _sysConfigRep.AsUpdateable().SetColumns(u => u.Value == Config.Value).Where(u => u.Code == Config.Code).ExecuteCommandAsync();
  201. _sysCacheService.Remove($"{CacheConst.KeyConfig}{Config.Code}");
  202. }
  203. }
  204. /// <summary>
  205. /// 获取系统信息 🔖
  206. /// </summary>
  207. /// <returns></returns>
  208. [SuppressMonitor]
  209. [AllowAnonymous]
  210. [DisplayName("获取系统信息")]
  211. public async Task<dynamic> GetSysInfo()
  212. {
  213. var sysLogo = await GetConfigValue<string>("sys_web_logo");
  214. var sysTitle = await GetConfigValue<string>("sys_web_title");
  215. var sysViceTitle = await GetConfigValue<string>("sys_web_viceTitle");
  216. var sysViceDesc = await GetConfigValue<string>("sys_web_viceDesc");
  217. var sysWatermark = await GetConfigValue<string>("sys_web_watermark");
  218. var sysCopyright = await GetConfigValue<string>("sys_web_copyright");
  219. var sysIcp = await GetConfigValue<string>("sys_web_icp");
  220. var sysIcpUrl = await GetConfigValue<string>("sys_web_icpUrl");
  221. return new
  222. {
  223. SysLogo = sysLogo,
  224. SysTitle = sysTitle,
  225. SysViceTitle = sysViceTitle,
  226. SysViceDesc = sysViceDesc,
  227. SysWatermark = sysWatermark,
  228. SysCopyright = sysCopyright,
  229. SysIcp = sysIcp,
  230. SysIcpUrl = sysIcpUrl
  231. };
  232. }
  233. /// <summary>
  234. /// 保存系统信息 🔖
  235. /// </summary>
  236. /// <returns></returns>
  237. [DisplayName("保存系统信息")]
  238. public async Task SaveSysInfo(InfoSaveInput input)
  239. {
  240. // logo 不为空才保存
  241. if (!string.IsNullOrEmpty(input.SysLogoBase64))
  242. {
  243. // 旧图标文件相对路径
  244. var oldSysLogoRelativeFilePath = await GetConfigValue<string>("sys_web_logo") ?? "";
  245. var oldSysLogoAbsoluteFilePath = Path.Combine(App.WebHostEnvironment.WebRootPath, oldSysLogoRelativeFilePath.TrimStart('/'));
  246. var groups = Regex.Match(input.SysLogoBase64, @"data:image/(?<type>.+?);base64,(?<data>.+)").Groups;
  247. var type = groups["type"].Value;
  248. var base64Data = groups["data"].Value;
  249. var binData = Convert.FromBase64String(base64Data);
  250. // 本地图标保存路径
  251. var path = "Upload";
  252. var absoluteFilePath = Path.Combine(App.WebHostEnvironment.WebRootPath, path, $"logo.{type}");
  253. // 删除已存在文件
  254. if (File.Exists(oldSysLogoAbsoluteFilePath))
  255. File.Delete(oldSysLogoAbsoluteFilePath);
  256. // 创建文件夹
  257. var absoluteFileDir = Path.GetDirectoryName(absoluteFilePath);
  258. if (!Directory.Exists(absoluteFileDir))
  259. Directory.CreateDirectory(absoluteFileDir);
  260. // 保存图标文件
  261. await File.WriteAllBytesAsync(absoluteFilePath, binData);
  262. // 保存图标配置
  263. var relativeUrl = $"/{path}/logo.{type}";
  264. await UpdateConfigValue("sys_web_logo", relativeUrl);
  265. }
  266. await UpdateConfigValue("sys_web_title", input.SysTitle);
  267. await UpdateConfigValue("sys_web_viceTitle", input.SysViceTitle);
  268. await UpdateConfigValue("sys_web_viceDesc", input.SysViceDesc);
  269. await UpdateConfigValue("sys_web_watermark", input.SysWatermark);
  270. await UpdateConfigValue("sys_web_copyright", input.SysCopyright);
  271. await UpdateConfigValue("sys_web_icp", input.SysIcp);
  272. await UpdateConfigValue("sys_web_icpUrl", input.SysIcpUrl);
  273. }
  274. }