sys_auth_rule.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. /*
  2. * @desc:菜单处理
  3. * @company:云南奇讯科技有限公司
  4. * @Author: yixiaohu
  5. * @Date: 2022/3/11 15:07
  6. */
  7. package service
  8. import (
  9. "context"
  10. "fmt"
  11. "github.com/gogf/gf/v2/database/gdb"
  12. "github.com/gogf/gf/v2/errors/gerror"
  13. "github.com/gogf/gf/v2/frame/g"
  14. "github.com/gogf/gf/v2/util/gconv"
  15. "github.com/tiger1103/gfast/v3/api/v1/system"
  16. commonService "github.com/tiger1103/gfast/v3/internal/app/common/service"
  17. "github.com/tiger1103/gfast/v3/internal/app/system/consts"
  18. "github.com/tiger1103/gfast/v3/internal/app/system/model"
  19. "github.com/tiger1103/gfast/v3/internal/app/system/model/entity"
  20. "github.com/tiger1103/gfast/v3/internal/app/system/service/internal/dao"
  21. "github.com/tiger1103/gfast/v3/internal/app/system/service/internal/do"
  22. "github.com/tiger1103/gfast/v3/library/liberr"
  23. )
  24. type IRule interface {
  25. GetIsMenuList(ctx context.Context) ([]*model.SysAuthRuleInfoRes, error)
  26. GetMenuList(ctx context.Context) (list []*model.SysAuthRuleInfoRes, err error)
  27. GetIsButtonList(ctx context.Context) ([]*model.SysAuthRuleInfoRes, error)
  28. Add(ctx context.Context, req *system.RuleAddReq) (err error)
  29. Get(ctx context.Context, id uint) (rule *entity.SysAuthRule, err error)
  30. GetMenuRoles(ctx context.Context, id uint) (roleIds []uint, err error)
  31. Update(ctx context.Context, req *system.RuleUpdateReq) (err error)
  32. GetMenuListSearch(ctx context.Context, req *system.RuleSearchReq) (res []*model.SysAuthRuleInfoRes, err error)
  33. GetMenuListTree(pid uint, list []*model.SysAuthRuleInfoRes) []*model.SysAuthRuleTreeRes
  34. DeleteMenuByIds(ctx context.Context, ids []int) (err error)
  35. }
  36. type ruleImpl struct {
  37. }
  38. var ruleService = ruleImpl{}
  39. func Rule() IRule {
  40. return &ruleService
  41. }
  42. func (s *ruleImpl) GetMenuListSearch(ctx context.Context, req *system.RuleSearchReq) (res []*model.SysAuthRuleInfoRes, err error) {
  43. err = g.Try(func() {
  44. m := dao.SysAuthRule.Ctx(ctx)
  45. if req.Title != "" {
  46. m = m.Where("title like ?", "%"+req.Title+"%")
  47. }
  48. if req.Component != "" {
  49. m = m.Where("component like ?", "%"+req.Component+"%")
  50. }
  51. err = m.Fields(model.SysAuthRuleInfoRes{}).Order("weigh desc,id asc").Scan(&res)
  52. liberr.ErrIsNil(ctx, err, "获取菜单失败")
  53. })
  54. return
  55. }
  56. // GetIsMenuList 获取isMenu=0|1
  57. func (s *ruleImpl) GetIsMenuList(ctx context.Context) ([]*model.SysAuthRuleInfoRes, error) {
  58. list, err := s.GetMenuList(ctx)
  59. if err != nil {
  60. return nil, err
  61. }
  62. var gList = make([]*model.SysAuthRuleInfoRes, 0, len(list))
  63. for _, v := range list {
  64. if v.MenuType == 0 || v.MenuType == 1 {
  65. gList = append(gList, v)
  66. }
  67. }
  68. return gList, nil
  69. }
  70. // GetMenuList 获取所有菜单
  71. func (s *ruleImpl) GetMenuList(ctx context.Context) (list []*model.SysAuthRuleInfoRes, err error) {
  72. cache := commonService.Cache()
  73. //从缓存获取
  74. iList := cache.GetOrSetFuncLock(ctx, consts.CacheSysAuthMenu, s.getMenuListFromDb, 0, consts.CacheSysAuthTag)
  75. if iList != nil {
  76. err = gconv.Struct(iList, &list)
  77. liberr.ErrIsNil(ctx, err)
  78. }
  79. return
  80. }
  81. // 从数据库获取所有菜单
  82. func (s *ruleImpl) getMenuListFromDb(ctx context.Context) (value interface{}, err error) {
  83. err = g.Try(func() {
  84. var v []*model.SysAuthRuleInfoRes
  85. //从数据库获取
  86. err = dao.SysAuthRule.Ctx(ctx).
  87. Fields(model.SysAuthRuleInfoRes{}).Order("weigh desc,id asc").Scan(&v)
  88. liberr.ErrIsNil(ctx, err, "获取菜单数据失败")
  89. value = v
  90. })
  91. return
  92. }
  93. // GetIsButtonList 获取所有按钮isMenu=2 菜单列表
  94. func (s *ruleImpl) GetIsButtonList(ctx context.Context) ([]*model.SysAuthRuleInfoRes, error) {
  95. list, err := s.GetMenuList(ctx)
  96. if err != nil {
  97. return nil, err
  98. }
  99. var gList = make([]*model.SysAuthRuleInfoRes, 0, len(list))
  100. for _, v := range list {
  101. if v.MenuType == 2 {
  102. gList = append(gList, v)
  103. }
  104. }
  105. return gList, nil
  106. }
  107. // Add 添加菜单
  108. func (s *ruleImpl) Add(ctx context.Context, req *system.RuleAddReq) (err error) {
  109. if s.menuNameExists(ctx, req.Name, 0) {
  110. err = gerror.New("接口规则已经存在")
  111. return
  112. }
  113. err = g.DB().Transaction(ctx, func(ctx context.Context, tx *gdb.TX) error {
  114. err = g.Try(func() {
  115. //菜单数据
  116. data := do.SysAuthRule{
  117. Pid: req.Pid,
  118. Name: req.Name,
  119. Title: req.Title,
  120. Icon: req.Icon,
  121. Condition: req.Condition,
  122. Remark: req.Remark,
  123. MenuType: req.MenuType,
  124. Weigh: req.Weigh,
  125. IsHide: req.IsHide,
  126. Path: req.Path,
  127. Component: req.Component,
  128. IsLink: req.IsLink,
  129. IsIframe: req.IsIframe,
  130. IsCached: req.IsCached,
  131. Redirect: req.Redirect,
  132. IsAffix: req.IsAffix,
  133. LinkUrl: req.LinkUrl,
  134. }
  135. ruleId, e := dao.SysAuthRule.Ctx(ctx).TX(tx).InsertAndGetId(data)
  136. liberr.ErrIsNil(ctx, e, "添加菜单失败")
  137. e = s.BindRoleRule(ctx, ruleId, req.Roles)
  138. liberr.ErrIsNil(ctx, e, "添加菜单失败")
  139. })
  140. return err
  141. })
  142. if err == nil {
  143. // 删除相关缓存
  144. commonService.Cache().Remove(ctx, consts.CacheSysAuthMenu)
  145. }
  146. return
  147. }
  148. //检查菜单规则是否存在
  149. func (s *ruleImpl) menuNameExists(ctx context.Context, name string, id uint) bool {
  150. m := dao.SysAuthRule.Ctx(ctx).Where("name=?", name)
  151. if id != 0 {
  152. m = m.Where("id!=?", id)
  153. }
  154. c, err := m.Fields(dao.SysAuthRule.Columns().Id).Limit(1).One()
  155. if err != nil {
  156. g.Log().Error(ctx, err)
  157. return false
  158. }
  159. return !c.IsEmpty()
  160. }
  161. // BindRoleRule 绑定角色权限
  162. func (s *ruleImpl) BindRoleRule(ctx context.Context, ruleId interface{}, roleIds []uint) (err error) {
  163. err = g.Try(func() {
  164. enforcer, e := commonService.CasbinEnforcer(ctx)
  165. liberr.ErrIsNil(ctx, e)
  166. for _, roleId := range roleIds {
  167. _, err = enforcer.AddPolicy(fmt.Sprintf("%d", roleId), fmt.Sprintf("%d", ruleId), "All")
  168. liberr.ErrIsNil(ctx, err)
  169. }
  170. })
  171. return
  172. }
  173. func (s *ruleImpl) Get(ctx context.Context, id uint) (rule *entity.SysAuthRule, err error) {
  174. err = g.Try(func() {
  175. err = dao.SysAuthRule.Ctx(ctx).WherePri(id).Scan(&rule)
  176. liberr.ErrIsNil(ctx, err, "获取菜单失败")
  177. })
  178. return
  179. }
  180. func (s *ruleImpl) GetMenuRoles(ctx context.Context, id uint) (roleIds []uint, err error) {
  181. err = g.Try(func() {
  182. enforcer, e := commonService.CasbinEnforcer(ctx)
  183. liberr.ErrIsNil(ctx, e)
  184. policies := enforcer.GetFilteredNamedPolicy("p", 1, gconv.String(id))
  185. for _, policy := range policies {
  186. roleIds = append(roleIds, gconv.Uint(policy[0]))
  187. }
  188. })
  189. return
  190. }
  191. func (s *ruleImpl) Update(ctx context.Context, req *system.RuleUpdateReq) (err error) {
  192. if s.menuNameExists(ctx, req.Name, req.Id) {
  193. err = gerror.New("接口规则已经存在")
  194. return
  195. }
  196. err = g.DB().Transaction(ctx, func(ctx context.Context, tx *gdb.TX) error {
  197. err = g.Try(func() {
  198. //菜单数据
  199. data := do.SysAuthRule{
  200. Pid: req.Pid,
  201. Name: req.Name,
  202. Title: req.Title,
  203. Icon: req.Icon,
  204. Condition: req.Condition,
  205. Remark: req.Remark,
  206. MenuType: req.MenuType,
  207. Weigh: req.Weigh,
  208. IsHide: req.IsHide,
  209. Path: req.Path,
  210. Component: req.Component,
  211. IsLink: req.IsLink,
  212. IsIframe: req.IsIframe,
  213. IsCached: req.IsCached,
  214. Redirect: req.Redirect,
  215. IsAffix: req.IsAffix,
  216. LinkUrl: req.LinkUrl,
  217. }
  218. _, e := dao.SysAuthRule.Ctx(ctx).TX(tx).WherePri(req.Id).Update(data)
  219. liberr.ErrIsNil(ctx, e, "添加菜单失败")
  220. e = s.UpdateRoleRule(ctx, req.Id, req.Roles)
  221. liberr.ErrIsNil(ctx, e, "添加菜单失败")
  222. })
  223. return err
  224. })
  225. if err == nil {
  226. // 删除相关缓存
  227. commonService.Cache().Remove(ctx, consts.CacheSysAuthMenu)
  228. }
  229. return
  230. }
  231. func (s *ruleImpl) UpdateRoleRule(ctx context.Context, ruleId uint, roleIds []uint) (err error) {
  232. err = g.Try(func() {
  233. enforcer, e := commonService.CasbinEnforcer(ctx)
  234. liberr.ErrIsNil(ctx, e)
  235. //删除旧权限
  236. _, e = enforcer.RemoveFilteredPolicy(1, gconv.String(ruleId))
  237. liberr.ErrIsNil(ctx, e)
  238. // 添加新权限
  239. roleIdsStrArr := gconv.Strings(roleIds)
  240. for _, v := range roleIdsStrArr {
  241. _, e = enforcer.AddPolicy(v, gconv.String(ruleId), "All")
  242. liberr.ErrIsNil(ctx, e)
  243. }
  244. })
  245. return
  246. }
  247. func (s *ruleImpl) GetMenuListTree(pid uint, list []*model.SysAuthRuleInfoRes) []*model.SysAuthRuleTreeRes {
  248. tree := make([]*model.SysAuthRuleTreeRes, 0, len(list))
  249. for _, menu := range list {
  250. if menu.Pid == pid {
  251. t := &model.SysAuthRuleTreeRes{
  252. SysAuthRuleInfoRes: menu,
  253. }
  254. child := s.GetMenuListTree(menu.Id, list)
  255. if child != nil {
  256. t.Children = child
  257. }
  258. tree = append(tree, t)
  259. }
  260. }
  261. return tree
  262. }
  263. // DeleteMenuByIds 删除菜单
  264. func (s *ruleImpl) DeleteMenuByIds(ctx context.Context, ids []int) (err error) {
  265. var list []*model.SysAuthRuleInfoRes
  266. list, err = s.GetMenuList(ctx)
  267. if err != nil {
  268. return
  269. }
  270. childrenIds := make([]int, 0, len(list))
  271. for _, id := range ids {
  272. rules := s.FindSonByParentId(list, gconv.Uint(id))
  273. for _, child := range rules {
  274. childrenIds = append(childrenIds, gconv.Int(child.Id))
  275. }
  276. }
  277. ids = append(ids, childrenIds...)
  278. err = g.DB().Transaction(ctx, func(ctx context.Context, tx *gdb.TX) error {
  279. return g.Try(func() {
  280. _, err = dao.SysAuthRule.Ctx(ctx).Where("id in (?)", ids).Delete()
  281. liberr.ErrIsNil(ctx, err, "删除失败")
  282. //删除权限
  283. enforcer, err := commonService.CasbinEnforcer(ctx)
  284. liberr.ErrIsNil(ctx, err)
  285. for _, v := range ids {
  286. _, err = enforcer.RemoveFilteredPolicy(1, gconv.String(v))
  287. liberr.ErrIsNil(ctx, err)
  288. }
  289. // 删除相关缓存
  290. commonService.Cache().Remove(ctx, consts.CacheSysAuthMenu)
  291. })
  292. })
  293. return
  294. }
  295. func (s *ruleImpl) FindSonByParentId(list []*model.SysAuthRuleInfoRes, pid uint) []*model.SysAuthRuleInfoRes {
  296. children := make([]*model.SysAuthRuleInfoRes, 0, len(list))
  297. for _, v := range list {
  298. if v.Pid == pid {
  299. children = append(children, v)
  300. fChildren := s.FindSonByParentId(list, v.Id)
  301. children = append(children, fChildren...)
  302. }
  303. }
  304. return children
  305. }