sysUserRoleModel.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. package userrole
  2. import (
  3. "context"
  4. "database/sql"
  5. "fmt"
  6. "strings"
  7. "perms-system-server/internal/consts"
  8. "github.com/zeromicro/go-zero/core/stores/cache"
  9. "github.com/zeromicro/go-zero/core/stores/sqlx"
  10. )
  11. var _ SysUserRoleModel = (*customSysUserRoleModel)(nil)
  12. type (
  13. SysUserRoleModel interface {
  14. sysUserRoleModel
  15. FindRoleIdsByUserId(ctx context.Context, userId int64) ([]int64, error)
  16. FindRoleIdsByUserIdForProduct(ctx context.Context, userId int64, productCode string) ([]int64, error)
  17. FindUserIdsByRoleId(ctx context.Context, roleId int64) ([]int64, error)
  18. FindUserIdsByRoleIdForUpdateTx(ctx context.Context, session sqlx.Session, roleId int64) ([]int64, error)
  19. DeleteByRoleIdTx(ctx context.Context, session sqlx.Session, roleId int64) error
  20. DeleteByUserIdForProductTx(ctx context.Context, session sqlx.Session, userId int64, productCode string) error
  21. DeleteByUserIdAndRoleIdsTx(ctx context.Context, session sqlx.Session, userId int64, roleIds []int64) error
  22. }
  23. customSysUserRoleModel struct {
  24. *defaultSysUserRoleModel
  25. }
  26. )
  27. func NewSysUserRoleModel(conn sqlx.SqlConn, c cache.CacheConf, cachePrefix string, opts ...cache.Option) SysUserRoleModel {
  28. return &customSysUserRoleModel{
  29. defaultSysUserRoleModel: newSysUserRoleModel(conn, c, cachePrefix, opts...),
  30. }
  31. }
  32. // FindRoleIdsByUserId 查询用户关联的所有角色 ID(跨全部产品聚合)。
  33. // 仅在超管未带产品上下文时通过 UserDetail 调用,返回结果不区分产品归属。
  34. func (m *customSysUserRoleModel) FindRoleIdsByUserId(ctx context.Context, userId int64) ([]int64, error) {
  35. var ids []int64
  36. query := fmt.Sprintf("SELECT `roleId` FROM %s WHERE `userId` = ?", m.table)
  37. if err := m.QueryRowsNoCacheCtx(ctx, &ids, query, userId); err != nil {
  38. return nil, err
  39. }
  40. return ids, nil
  41. }
  42. func (m *customSysUserRoleModel) FindRoleIdsByUserIdForProduct(ctx context.Context, userId int64, productCode string) ([]int64, error) {
  43. var ids []int64
  44. // status 走占位参数 + consts.StatusEnabled,避免把"启用"语义钉死成字面量 1;未来新增
  45. // status=3/已归档 之类时只要改 consts 就行,这里不会漏掉(见审计 L-4)。
  46. query := fmt.Sprintf("SELECT ur.`roleId` FROM %s ur INNER JOIN `sys_role` r ON ur.`roleId` = r.`id` WHERE ur.`userId` = ? AND r.`productCode` = ? AND r.`status` = ?", m.table)
  47. if err := m.QueryRowsNoCacheCtx(ctx, &ids, query, userId, productCode, consts.StatusEnabled); err != nil {
  48. return nil, err
  49. }
  50. return ids, nil
  51. }
  52. func (m *customSysUserRoleModel) FindUserIdsByRoleId(ctx context.Context, roleId int64) ([]int64, error) {
  53. var ids []int64
  54. query := fmt.Sprintf("SELECT `userId` FROM %s WHERE `roleId` = ?", m.table)
  55. if err := m.QueryRowsNoCacheCtx(ctx, &ids, query, roleId); err != nil {
  56. return nil, err
  57. }
  58. return ids, nil
  59. }
  60. func (m *customSysUserRoleModel) FindUserIdsByRoleIdForUpdateTx(ctx context.Context, session sqlx.Session, roleId int64) ([]int64, error) {
  61. var ids []int64
  62. query := fmt.Sprintf("SELECT `userId` FROM %s WHERE `roleId` = ? FOR UPDATE", m.table)
  63. if err := session.QueryRowsCtx(ctx, &ids, query, roleId); err != nil {
  64. return nil, err
  65. }
  66. return ids, nil
  67. }
  68. func (m *customSysUserRoleModel) buildCacheKeys(list []*SysUserRole) []string {
  69. keys := make([]string, 0, len(list)*2)
  70. for _, data := range list {
  71. keys = append(keys,
  72. fmt.Sprintf("%s%v", cacheSysUserRoleIdPrefix, data.Id),
  73. fmt.Sprintf("%s%v:%v", cacheSysUserRoleUserIdRoleIdPrefix, data.UserId, data.RoleId),
  74. )
  75. }
  76. return keys
  77. }
  78. func (m *customSysUserRoleModel) DeleteByRoleIdTx(ctx context.Context, session sqlx.Session, roleId int64) error {
  79. var list []*SysUserRole
  80. findQuery := fmt.Sprintf("SELECT %s FROM %s WHERE `roleId` = ? FOR UPDATE", sysUserRoleRows, m.table)
  81. if err := session.QueryRowsCtx(ctx, &list, findQuery, roleId); err != nil {
  82. return err
  83. }
  84. if len(list) == 0 {
  85. return nil
  86. }
  87. keys := m.buildCacheKeys(list)
  88. _, err := m.ExecCtx(ctx, func(ctx context.Context, conn sqlx.SqlConn) (sql.Result, error) {
  89. query := fmt.Sprintf("DELETE FROM %s WHERE `roleId` = ?", m.table)
  90. return session.ExecCtx(ctx, query, roleId)
  91. }, keys...)
  92. return err
  93. }
  94. func (m *customSysUserRoleModel) DeleteByUserIdForProductTx(ctx context.Context, session sqlx.Session, userId int64, productCode string) error {
  95. var list []*SysUserRole
  96. findQuery := fmt.Sprintf("SELECT %s FROM %s WHERE `userId` = ? AND `roleId` IN (SELECT `id` FROM `sys_role` WHERE `productCode` = ?) FOR UPDATE", sysUserRoleRows, m.table)
  97. if err := session.QueryRowsCtx(ctx, &list, findQuery, userId, productCode); err != nil {
  98. return err
  99. }
  100. if len(list) == 0 {
  101. return nil
  102. }
  103. keys := m.buildCacheKeys(list)
  104. _, err := m.ExecCtx(ctx, func(ctx context.Context, conn sqlx.SqlConn) (sql.Result, error) {
  105. query := fmt.Sprintf("DELETE FROM %s WHERE `userId` = ? AND `roleId` IN (SELECT `id` FROM `sys_role` WHERE `productCode` = ?)", m.table)
  106. return session.ExecCtx(ctx, query, userId, productCode)
  107. }, keys...)
  108. return err
  109. }
  110. func (m *customSysUserRoleModel) DeleteByUserIdAndRoleIdsTx(ctx context.Context, session sqlx.Session, userId int64, roleIds []int64) error {
  111. if len(roleIds) == 0 {
  112. return nil
  113. }
  114. placeholders := make([]string, len(roleIds))
  115. args := make([]interface{}, 0, len(roleIds)+1)
  116. args = append(args, userId)
  117. for i, id := range roleIds {
  118. placeholders[i] = "?"
  119. args = append(args, id)
  120. }
  121. inClause := strings.Join(placeholders, ",")
  122. var list []*SysUserRole
  123. findQuery := fmt.Sprintf("SELECT %s FROM %s WHERE `userId` = ? AND `roleId` IN (%s) FOR UPDATE", sysUserRoleRows, m.table, inClause)
  124. if err := session.QueryRowsCtx(ctx, &list, findQuery, args...); err != nil {
  125. return err
  126. }
  127. if len(list) == 0 {
  128. return nil
  129. }
  130. keys := m.buildCacheKeys(list)
  131. _, err := m.ExecCtx(ctx, func(ctx context.Context, conn sqlx.SqlConn) (sql.Result, error) {
  132. query := fmt.Sprintf("DELETE FROM %s WHERE `userId` = ? AND `roleId` IN (%s)", m.table, inClause)
  133. return session.ExecCtx(ctx, query, args...)
  134. }, keys...)
  135. return err
  136. }