sys_auth_rule.go 9.2 KB

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