SysRoleService.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  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 = 480)]
  11. public class SysRoleService : IDynamicApiController, ITransient
  12. {
  13. private readonly UserManager _userManager;
  14. private readonly SqlSugarRepository<SysRole> _sysRoleRep;
  15. private readonly SysRoleMenuService _sysRoleMenuService;
  16. private readonly SysUserRoleService _sysUserRoleService;
  17. private readonly SysRoleOrgService _sysRoleOrgService;
  18. private readonly SysMenuService _sysMenuService;
  19. private readonly SysOrgService _sysOrgService;
  20. private readonly SysCacheService _sysCacheService;
  21. public SysRoleService(UserManager userManager,
  22. SysOrgService sysOrgService,
  23. SysMenuService sysMenuService,
  24. SysRoleOrgService sysRoleOrgService,
  25. SqlSugarRepository<SysRole> sysRoleRep,
  26. SysRoleMenuService sysRoleMenuService,
  27. SysUserRoleService sysUserRoleService,
  28. SysCacheService sysCacheService)
  29. {
  30. _userManager = userManager;
  31. _sysRoleRep = sysRoleRep;
  32. _sysOrgService = sysOrgService;
  33. _sysMenuService = sysMenuService;
  34. _sysRoleOrgService = sysRoleOrgService;
  35. _sysRoleMenuService = sysRoleMenuService;
  36. _sysUserRoleService = sysUserRoleService;
  37. _sysCacheService = sysCacheService;
  38. }
  39. /// <summary>
  40. /// 获取角色分页列表 🔖
  41. /// </summary>
  42. /// <param name="input"></param>
  43. /// <returns></returns>
  44. [DisplayName("获取角色分页列表")]
  45. public async Task<SqlSugarPagedList<SysRole>> Page(PageRoleInput input)
  46. {
  47. // 当前用户已拥有的角色集合
  48. var roleIdList = _userManager.SuperAdmin ? new List<long>() : await _sysUserRoleService.GetUserRoleIdList(_userManager.UserId);
  49. return await _sysRoleRep.AsQueryable()
  50. .WhereIF(_userManager.SuperAdmin && input.TenantId > 0, u => u.TenantId == input.TenantId)
  51. .WhereIF(!_userManager.SuperAdmin, u => u.TenantId == _userManager.TenantId) // 若非超管,则只能操作本租户的角色
  52. .WhereIF(!_userManager.SuperAdmin && !_userManager.SysAdmin, u => u.CreateUserId == _userManager.UserId || roleIdList.Contains(u.Id)) // 若非超管且非系统管理员,则只能操作自己创建的角色|自己拥有的角色
  53. .WhereIF(!string.IsNullOrWhiteSpace(input.Name), u => u.Name.Contains(input.Name))
  54. .WhereIF(!string.IsNullOrWhiteSpace(input.Code), u => u.Code.Contains(input.Code))
  55. .OrderBy(u => new { u.OrderNo, u.Id })
  56. .ToPagedListAsync(input.Page, input.PageSize);
  57. }
  58. /// <summary>
  59. /// 获取角色列表 🔖
  60. /// </summary>
  61. /// <returns></returns>
  62. [DisplayName("获取角色列表")]
  63. public async Task<List<RoleOutput>> GetList()
  64. {
  65. // 当前用户已拥有的角色集合
  66. var roleIdList = _userManager.SuperAdmin ? new List<long>() : await _sysUserRoleService.GetUserRoleIdList(_userManager.UserId);
  67. return await _sysRoleRep.AsQueryable()
  68. .WhereIF(!_userManager.SuperAdmin, u => u.TenantId == _userManager.TenantId) // 若非超管,则只能操作本租户的角色
  69. .WhereIF(!_userManager.SuperAdmin && !_userManager.SysAdmin, u => u.CreateUserId == _userManager.UserId || roleIdList.Contains(u.Id)) // 若非超管且非系统管理员,则只显示自己创建和已拥有的角色
  70. .Where(u => u.Status != StatusEnum.Disable) // 非禁用的
  71. .OrderBy(u => new { u.OrderNo, u.Id }).Select<RoleOutput>().ToListAsync();
  72. }
  73. /// <summary>
  74. /// 增加角色 🔖
  75. /// </summary>
  76. /// <param name="input"></param>
  77. /// <returns></returns>
  78. [ApiDescriptionSettings(Name = "Add"), HttpPost]
  79. [DisplayName("增加角色")]
  80. public async Task AddRole(AddRoleInput input)
  81. {
  82. if (await _sysRoleRep.IsAnyAsync(u => u.Name == input.Name && u.Code == input.Code))
  83. throw Oops.Oh(ErrorCodeEnum.D1006);
  84. var newRole = await _sysRoleRep.AsInsertable(input.Adapt<SysRole>()).ExecuteReturnEntityAsync();
  85. input.Id = newRole.Id;
  86. await UpdateRoleMenu(input);
  87. }
  88. /// <summary>
  89. /// 更新角色菜单权限
  90. /// </summary>
  91. /// <param name="input"></param>
  92. /// <returns></returns>
  93. private async Task UpdateRoleMenu(AddRoleInput input)
  94. {
  95. if (input.MenuIdList == null || input.MenuIdList.Count < 1) return;
  96. await GrantMenu(new RoleMenuInput()
  97. {
  98. Id = input.Id,
  99. MenuIdList = input.MenuIdList.ToList()
  100. });
  101. }
  102. /// <summary>
  103. /// 更新角色 🔖
  104. /// </summary>
  105. /// <param name="input"></param>
  106. /// <returns></returns>
  107. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  108. [DisplayName("更新角色")]
  109. public async Task UpdateRole(UpdateRoleInput input)
  110. {
  111. if (await _sysRoleRep.IsAnyAsync(u => u.Name == input.Name && u.Code == input.Code && u.Id != input.Id))
  112. throw Oops.Oh(ErrorCodeEnum.D1006);
  113. await _sysRoleRep.AsUpdateable(input.Adapt<SysRole>()).IgnoreColumns(true)
  114. .IgnoreColumns(u => new { u.DataScope }).ExecuteCommandAsync();
  115. await UpdateRoleMenu(input);
  116. }
  117. /// <summary>
  118. /// 删除角色 🔖
  119. /// </summary>
  120. /// <param name="input"></param>
  121. /// <returns></returns>
  122. [UnitOfWork]
  123. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  124. [DisplayName("删除角色")]
  125. public async Task DeleteRole(DeleteRoleInput input)
  126. {
  127. // 若角色有用户则禁止删除
  128. var userIds = await _sysUserRoleService.GetUserIdList(input.Id);
  129. if (userIds != null && userIds.Count > 0) throw Oops.Oh(ErrorCodeEnum.D1025);
  130. // 若有绑定注册方案则禁止删除
  131. var hasUserRegWay = await _sysRoleRep.Context.Queryable<SysUserRegWay>().AnyAsync(u => u.RoleId == input.Id);
  132. if (hasUserRegWay) throw Oops.Oh(ErrorCodeEnum.D1033);
  133. var sysRole = await _sysRoleRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002);
  134. await _sysRoleRep.DeleteAsync(sysRole);
  135. // 级联删除角色机构数据
  136. await _sysRoleOrgService.DeleteRoleOrgByRoleId(sysRole.Id);
  137. // 级联删除用户角色数据
  138. await _sysUserRoleService.DeleteUserRoleByRoleId(sysRole.Id);
  139. // 级联删除角色菜单数据
  140. await _sysRoleMenuService.DeleteRoleMenuByRoleId(sysRole.Id);
  141. }
  142. /// <summary>
  143. /// 授权角色菜单 🔖
  144. /// </summary>
  145. /// <param name="input"></param>
  146. /// <returns></returns>
  147. [UnitOfWork]
  148. [DisplayName("授权角色菜单")]
  149. public async Task GrantMenu(RoleMenuInput input)
  150. {
  151. if (input.MenuIdList == null || input.MenuIdList.Count < 1) return;
  152. await ClearUserApiCache(input.Id);
  153. await _sysRoleMenuService.GrantRoleMenu(input);
  154. }
  155. /// <summary>
  156. /// 授权角色数据范围 🔖
  157. /// </summary>
  158. /// <param name="input"></param>
  159. /// <returns></returns>
  160. [UnitOfWork]
  161. [DisplayName("授权角色数据范围")]
  162. public async Task GrantDataScope(RoleOrgInput input)
  163. {
  164. // 删除与该角色相关的用户机构缓存
  165. var userIdList = await _sysUserRoleService.GetUserIdList(input.Id);
  166. foreach (var userId in userIdList)
  167. {
  168. SqlSugarFilter.DeleteUserOrgCache(userId, _sysRoleRep.Context.CurrentConnectionConfig.ConfigId.ToString());
  169. }
  170. var role = await _sysRoleRep.GetFirstAsync(u => u.Id == input.Id);
  171. var dataScope = input.DataScope;
  172. if (!_userManager.SuperAdmin)
  173. {
  174. switch (dataScope)
  175. {
  176. // 非超级管理员没有全部数据范围权限
  177. case (int)DataScopeEnum.All: throw Oops.Oh(ErrorCodeEnum.D1016);
  178. // 若数据范围自定义,则判断授权数据范围是否有权限
  179. case (int)DataScopeEnum.Define:
  180. {
  181. var grantOrgIdList = input.OrgIdList;
  182. if (grantOrgIdList.Count > 0)
  183. {
  184. var orgIdList = await _sysOrgService.GetUserOrgIdList();
  185. if (orgIdList.Count < 1)
  186. throw Oops.Oh(ErrorCodeEnum.D1016);
  187. if (!grantOrgIdList.All(u => orgIdList.Any(c => c == u)))
  188. throw Oops.Oh(ErrorCodeEnum.D1016);
  189. }
  190. break;
  191. }
  192. }
  193. }
  194. role.DataScope = (DataScopeEnum)dataScope;
  195. await _sysRoleRep.AsUpdateable(role).UpdateColumns(u => new { u.DataScope }).ExecuteCommandAsync();
  196. await _sysRoleOrgService.GrantRoleOrg(input);
  197. }
  198. /// <summary>
  199. /// 根据角色Id获取菜单Id集合 🔖
  200. /// </summary>
  201. /// <param name="input"></param>
  202. /// <returns></returns>
  203. [DisplayName("根据角色Id获取菜单Id集合")]
  204. public async Task<List<long>> GetOwnMenuList([FromQuery] RoleInput input)
  205. {
  206. var menuIds = await _sysRoleMenuService.GetRoleMenuIdList(new List<long> { input.Id });
  207. return await _sysMenuService.ExcludeParentMenuOfFullySelected(menuIds);
  208. }
  209. /// <summary>
  210. /// 根据角色Id获取机构Id集合 🔖
  211. /// </summary>
  212. /// <param name="input"></param>
  213. /// <returns></returns>
  214. [DisplayName("根据角色Id获取机构Id集合")]
  215. public async Task<List<long>> GetOwnOrgList([FromQuery] RoleInput input)
  216. {
  217. return await _sysRoleOrgService.GetRoleOrgIdList(new List<long> { input.Id });
  218. }
  219. /// <summary>
  220. /// 设置角色状态 🔖
  221. /// </summary>
  222. /// <param name="input"></param>
  223. /// <returns></returns>
  224. [DisplayName("设置角色状态")]
  225. public async Task<int> SetStatus(RoleInput input)
  226. {
  227. if (!Enum.IsDefined(typeof(StatusEnum), input.Status)) throw Oops.Oh(ErrorCodeEnum.D3005);
  228. return await _sysRoleRep.AsUpdateable()
  229. .SetColumns(u => u.Status == input.Status)
  230. .Where(u => u.Id == input.Id)
  231. .ExecuteCommandAsync();
  232. }
  233. /// <summary>
  234. /// 删除与该角色相关的用户接口缓存
  235. /// </summary>
  236. /// <param name="roleId"></param>
  237. /// <returns></returns>
  238. [NonAction]
  239. public async Task ClearUserApiCache(long roleId)
  240. {
  241. var userIdList = await _sysUserRoleService.GetUserIdList(roleId);
  242. foreach (var userId in userIdList)
  243. {
  244. _sysCacheService.Remove($"{CacheConst.KeyUserButton}{userId}");
  245. }
  246. }
  247. }