2
0

sys_casbin.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. package system
  2. import (
  3. "errors"
  4. "strconv"
  5. "sync"
  6. "gorm.io/gorm"
  7. "github.com/casbin/casbin/v2"
  8. "github.com/casbin/casbin/v2/model"
  9. gormadapter "github.com/casbin/gorm-adapter/v3"
  10. "github.com/flipped-aurora/gin-vue-admin/server/global"
  11. "github.com/flipped-aurora/gin-vue-admin/server/model/system/request"
  12. _ "github.com/go-sql-driver/mysql"
  13. "go.uber.org/zap"
  14. )
  15. //@author: [piexlmax](https://github.com/piexlmax)
  16. //@function: UpdateCasbin
  17. //@description: 更新casbin权限
  18. //@param: authorityId string, casbinInfos []request.CasbinInfo
  19. //@return: error
  20. type CasbinService struct{}
  21. var CasbinServiceApp = new(CasbinService)
  22. func (casbinService *CasbinService) UpdateCasbin(adminAuthorityID, AuthorityID uint, casbinInfos []request.CasbinInfo) error {
  23. err := AuthorityServiceApp.CheckAuthorityIDAuth(adminAuthorityID, AuthorityID)
  24. if err != nil {
  25. return err
  26. }
  27. if global.GVA_CONFIG.System.UseStrictAuth {
  28. apis, e := ApiServiceApp.GetAllApis(adminAuthorityID)
  29. if e != nil {
  30. return e
  31. }
  32. for i := range casbinInfos {
  33. hasApi := false
  34. for j := range apis {
  35. if apis[j].Path == casbinInfos[i].Path && apis[j].Method == casbinInfos[i].Method {
  36. hasApi = true
  37. break
  38. }
  39. }
  40. if !hasApi {
  41. return errors.New("存在api不在权限列表中")
  42. }
  43. }
  44. }
  45. authorityId := strconv.Itoa(int(AuthorityID))
  46. casbinService.ClearCasbin(0, authorityId)
  47. rules := [][]string{}
  48. //做权限去重处理
  49. deduplicateMap := make(map[string]bool)
  50. for _, v := range casbinInfos {
  51. key := authorityId + v.Path + v.Method
  52. if _, ok := deduplicateMap[key]; !ok {
  53. deduplicateMap[key] = true
  54. rules = append(rules, []string{authorityId, v.Path, v.Method})
  55. }
  56. }
  57. if len(rules) == 0 {
  58. return nil
  59. } // 设置空权限无需调用 AddPolicies 方法
  60. e := casbinService.Casbin()
  61. success, _ := e.AddPolicies(rules)
  62. if !success {
  63. return errors.New("存在相同api,添加失败,请联系管理员")
  64. }
  65. return nil
  66. }
  67. //@author: [piexlmax](https://github.com/piexlmax)
  68. //@function: UpdateCasbinApi
  69. //@description: API更新随动
  70. //@param: oldPath string, newPath string, oldMethod string, newMethod string
  71. //@return: error
  72. func (casbinService *CasbinService) UpdateCasbinApi(oldPath string, newPath string, oldMethod string, newMethod string) error {
  73. err := global.GVA_DB.Model(&gormadapter.CasbinRule{}).Where("v1 = ? AND v2 = ?", oldPath, oldMethod).Updates(map[string]interface{}{
  74. "v1": newPath,
  75. "v2": newMethod,
  76. }).Error
  77. if err != nil {
  78. return err
  79. }
  80. e := casbinService.Casbin()
  81. return e.LoadPolicy()
  82. }
  83. //@author: [piexlmax](https://github.com/piexlmax)
  84. //@function: GetPolicyPathByAuthorityId
  85. //@description: 获取权限列表
  86. //@param: authorityId string
  87. //@return: pathMaps []request.CasbinInfo
  88. func (casbinService *CasbinService) GetPolicyPathByAuthorityId(AuthorityID uint) (pathMaps []request.CasbinInfo) {
  89. e := casbinService.Casbin()
  90. authorityId := strconv.Itoa(int(AuthorityID))
  91. list, _ := e.GetFilteredPolicy(0, authorityId)
  92. for _, v := range list {
  93. pathMaps = append(pathMaps, request.CasbinInfo{
  94. Path: v[1],
  95. Method: v[2],
  96. })
  97. }
  98. return pathMaps
  99. }
  100. //@author: [piexlmax](https://github.com/piexlmax)
  101. //@function: ClearCasbin
  102. //@description: 清除匹配的权限
  103. //@param: v int, p ...string
  104. //@return: bool
  105. func (casbinService *CasbinService) ClearCasbin(v int, p ...string) bool {
  106. e := casbinService.Casbin()
  107. success, _ := e.RemoveFilteredPolicy(v, p...)
  108. return success
  109. }
  110. //@author: [piexlmax](https://github.com/piexlmax)
  111. //@function: RemoveFilteredPolicy
  112. //@description: 使用数据库方法清理筛选的politicy 此方法需要调用FreshCasbin方法才可以在系统中即刻生效
  113. //@param: db *gorm.DB, authorityId string
  114. //@return: error
  115. func (casbinService *CasbinService) RemoveFilteredPolicy(db *gorm.DB, authorityId string) error {
  116. return db.Delete(&gormadapter.CasbinRule{}, "v0 = ?", authorityId).Error
  117. }
  118. //@author: [piexlmax](https://github.com/piexlmax)
  119. //@function: SyncPolicy
  120. //@description: 同步目前数据库的policy 此方法需要调用FreshCasbin方法才可以在系统中即刻生效
  121. //@param: db *gorm.DB, authorityId string, rules [][]string
  122. //@return: error
  123. func (casbinService *CasbinService) SyncPolicy(db *gorm.DB, authorityId string, rules [][]string) error {
  124. err := casbinService.RemoveFilteredPolicy(db, authorityId)
  125. if err != nil {
  126. return err
  127. }
  128. return casbinService.AddPolicies(db, rules)
  129. }
  130. //@author: [piexlmax](https://github.com/piexlmax)
  131. //@function: AddPolicies
  132. //@description: 添加匹配的权限
  133. //@param: v int, p ...string
  134. //@return: bool
  135. func (casbinService *CasbinService) AddPolicies(db *gorm.DB, rules [][]string) error {
  136. var casbinRules []gormadapter.CasbinRule
  137. for i := range rules {
  138. casbinRules = append(casbinRules, gormadapter.CasbinRule{
  139. Ptype: "p",
  140. V0: rules[i][0],
  141. V1: rules[i][1],
  142. V2: rules[i][2],
  143. })
  144. }
  145. return db.Create(&casbinRules).Error
  146. }
  147. func (casbinService *CasbinService) FreshCasbin() (err error) {
  148. e := casbinService.Casbin()
  149. err = e.LoadPolicy()
  150. return err
  151. }
  152. //@author: [piexlmax](https://github.com/piexlmax)
  153. //@function: Casbin
  154. //@description: 持久化到数据库 引入自定义规则
  155. //@return: *casbin.Enforcer
  156. var (
  157. syncedCachedEnforcer *casbin.SyncedCachedEnforcer
  158. once sync.Once
  159. )
  160. func (casbinService *CasbinService) Casbin() *casbin.SyncedCachedEnforcer {
  161. once.Do(func() {
  162. a, err := gormadapter.NewAdapterByDB(global.GVA_DB)
  163. if err != nil {
  164. zap.L().Error("适配数据库失败请检查casbin表是否为InnoDB引擎!", zap.Error(err))
  165. return
  166. }
  167. text := `
  168. [request_definition]
  169. r = sub, obj, act
  170. [policy_definition]
  171. p = sub, obj, act
  172. [role_definition]
  173. g = _, _
  174. [policy_effect]
  175. e = some(where (p.eft == allow))
  176. [matchers]
  177. m = r.sub == p.sub && keyMatch2(r.obj,p.obj) && r.act == p.act
  178. `
  179. m, err := model.NewModelFromString(text)
  180. if err != nil {
  181. zap.L().Error("字符串加载模型失败!", zap.Error(err))
  182. return
  183. }
  184. syncedCachedEnforcer, _ = casbin.NewSyncedCachedEnforcer(m, a)
  185. syncedCachedEnforcer.SetExpireTime(60 * 60)
  186. _ = syncedCachedEnforcer.LoadPolicy()
  187. })
  188. return syncedCachedEnforcer
  189. }