sysProductMemberModel.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. package productmember
  2. import (
  3. "context"
  4. "fmt"
  5. "perms-system-server/internal/consts"
  6. "github.com/zeromicro/go-zero/core/stores/cache"
  7. "github.com/zeromicro/go-zero/core/stores/sqlx"
  8. )
  9. var _ SysProductMemberModel = (*customSysProductMemberModel)(nil)
  10. type (
  11. SysProductMemberModel interface {
  12. sysProductMemberModel
  13. FindListByProductCode(ctx context.Context, productCode string, page, pageSize int64) ([]*SysProductMember, int64, error)
  14. FindListByPage(ctx context.Context, page, pageSize int64) ([]*SysProductMember, int64, error)
  15. // CountOtherActiveAdminsTx 统计"除 excludeId 这一行以外"的启用 ADMIN 数量,是
  16. // "不能移除/降级最后一个 admin"这条不变式的唯一出口。之前同批引入的
  17. // CountActiveAdminsTx(不带 Other)业务层零调用(调用方反向推导更绕且易错),
  18. // 审计 L-2 要求直接删除以收敛接口 surface area、规避"应该用哪一个"的歧义。
  19. CountOtherActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string, excludeId int64) (int64, error)
  20. FindOneForUpdateTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error)
  21. // FindOneForShareTx 在当前事务里对 sys_product_member 目标行取 S 锁
  22. // (SELECT ... LOCK IN SHARE MODE)。用于"事务外读 memberType → 事务内 DeleteByUserIdForProductTx
  23. // + BatchInsertWithTx(DENY 行)"的 TOCTOU 闭环(审计 L-R13-2):UpdateMember / RemoveMember
  24. // 会对该行取 X 锁,被本 S 锁阻塞;本事务提交前 member.memberType 不会被并发改写,
  25. // DENY 脏行"能写永不生效"的数据污染被收敛。本方法不走缓存,必须在 TransactCtx / Session 下调用。
  26. FindOneForShareTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error)
  27. // FindActiveMemberUserIdsByProductCodeTx 在当前事务里返回某产品下所有启用成员的 userId 列表,
  28. // 供 UpdateProduct 禁用时批量递增对应 sys_user.tokenVersion 吊销 session(审计 L-R15-3)。
  29. // 使用 LOCK IN SHARE MODE:与并发 AddMember / UpdateMember / RemoveMember 的 X 锁互斥,
  30. // 保证在"拿到 userId 列表 → 批量 UPDATE sys_user"这段窗口里,member 行集合不会被并发改写
  31. // 导致新加入的成员 token 没被吊销、或已被移除的成员 token 被误吊销。按 `id` 排序保证锁获取
  32. // 顺序稳定,防止与其它按主键序扫描的事务互相死锁。
  33. FindActiveMemberUserIdsByProductCodeTx(ctx context.Context, session sqlx.Session, productCode string) ([]int64, error)
  34. FindByUserId(ctx context.Context, userId int64) ([]*SysProductMember, error)
  35. }
  36. customSysProductMemberModel struct {
  37. *defaultSysProductMemberModel
  38. }
  39. )
  40. func NewSysProductMemberModel(conn sqlx.SqlConn, c cache.CacheConf, cachePrefix string, opts ...cache.Option) SysProductMemberModel {
  41. return &customSysProductMemberModel{
  42. defaultSysProductMemberModel: newSysProductMemberModel(conn, c, cachePrefix, opts...),
  43. }
  44. }
  45. func (m *customSysProductMemberModel) FindListByProductCode(ctx context.Context, productCode string, page, pageSize int64) ([]*SysProductMember, int64, error) {
  46. var total int64
  47. countQuery := fmt.Sprintf("SELECT COUNT(*) FROM %s WHERE `productCode` = ?", m.table)
  48. if err := m.QueryRowNoCacheCtx(ctx, &total, countQuery, productCode); err != nil {
  49. return nil, 0, err
  50. }
  51. var list []*SysProductMember
  52. query := fmt.Sprintf("SELECT %s FROM %s WHERE `productCode` = ? ORDER BY id DESC LIMIT ?,?", sysProductMemberRows, m.table)
  53. if err := m.QueryRowsNoCacheCtx(ctx, &list, query, productCode, (page-1)*pageSize, pageSize); err != nil {
  54. return nil, 0, err
  55. }
  56. return list, total, nil
  57. }
  58. // CountOtherActiveAdminsTx 统计"除 excludeId 这一行以外"的启用 ADMIN 数量。调用方一般把即将被删除
  59. // 或即将被降级的目标行 id 传进来;返回 0 即表示目标是最后一个 active admin,不能动。相比
  60. // CountActiveAdminsTx + adminCount <= 1 的反向推理,语义更贴合业务(见审计 L-5 / L-2)。
  61. // 仍然使用 FOR UPDATE 锁住扫描范围,串行化与并发降级/删除的冲突。
  62. func (m *customSysProductMemberModel) CountOtherActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string, excludeId int64) (int64, error) {
  63. // 审计 L-R10-6:直接 SELECT COUNT(*) 即可,无需把匹配行的 id 全部回灌到应用层——后者对极端场景
  64. // (一产品 admin 数量异常多)会多一笔可避免的内存开销。FOR UPDATE 仍然保留,串行化"移除/降级最后
  65. // 一个 admin"的并发冲突;InnoDB 在 COUNT(*) ... FOR UPDATE 下会对匹配的索引/行同样加写锁。
  66. var count int64
  67. query := fmt.Sprintf("SELECT COUNT(*) FROM %s WHERE `productCode` = ? AND `memberType` = ? AND `status` = ? AND `id` != ? FOR UPDATE", m.table)
  68. if err := session.QueryRowCtx(ctx, &count, query, productCode, consts.MemberTypeAdmin, consts.StatusEnabled, excludeId); err != nil {
  69. return 0, err
  70. }
  71. return count, nil
  72. }
  73. func (m *customSysProductMemberModel) FindOneForUpdateTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error) {
  74. var data SysProductMember
  75. query := fmt.Sprintf("SELECT %s FROM %s WHERE `id` = ? FOR UPDATE", sysProductMemberRows, m.table)
  76. if err := session.QueryRowCtx(ctx, &data, query, id); err != nil {
  77. return nil, err
  78. }
  79. return &data, nil
  80. }
  81. // FindOneForShareTx 见接口注释(审计 L-R13-2)。
  82. func (m *customSysProductMemberModel) FindOneForShareTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error) {
  83. var data SysProductMember
  84. query := fmt.Sprintf("SELECT %s FROM %s WHERE `id` = ? LOCK IN SHARE MODE", sysProductMemberRows, m.table)
  85. if err := session.QueryRowCtx(ctx, &data, query, id); err != nil {
  86. return nil, err
  87. }
  88. return &data, nil
  89. }
  90. // FindByUserId 查询指定用户加入的所有产品成员记录,用于"用户产品列表"接口。
  91. func (m *customSysProductMemberModel) FindByUserId(ctx context.Context, userId int64) ([]*SysProductMember, error) {
  92. var list []*SysProductMember
  93. query := fmt.Sprintf("SELECT %s FROM %s WHERE `userId` = ? ORDER BY `id` DESC", sysProductMemberRows, m.table)
  94. if err := m.QueryRowsNoCacheCtx(ctx, &list, query, userId); err != nil {
  95. return nil, err
  96. }
  97. return list, nil
  98. }
  99. func (m *customSysProductMemberModel) FindListByPage(ctx context.Context, page, pageSize int64) ([]*SysProductMember, int64, error) {
  100. var total int64
  101. countQuery := fmt.Sprintf("SELECT COUNT(*) FROM %s", m.table)
  102. if err := m.QueryRowNoCacheCtx(ctx, &total, countQuery); err != nil {
  103. return nil, 0, err
  104. }
  105. var list []*SysProductMember
  106. query := fmt.Sprintf("SELECT %s FROM %s ORDER BY id DESC LIMIT ?,?", sysProductMemberRows, m.table)
  107. if err := m.QueryRowsNoCacheCtx(ctx, &list, query, (page-1)*pageSize, pageSize); err != nil {
  108. return nil, 0, err
  109. }
  110. return list, total, nil
  111. }
  112. // FindActiveMemberUserIdsByProductCodeTx 见接口注释(审计 L-R15-3)。
  113. func (m *customSysProductMemberModel) FindActiveMemberUserIdsByProductCodeTx(ctx context.Context, session sqlx.Session, productCode string) ([]int64, error) {
  114. var ids []int64
  115. query := fmt.Sprintf("SELECT `userId` FROM %s WHERE `productCode` = ? AND `status` = ? ORDER BY `id` LOCK IN SHARE MODE", m.table)
  116. if err := session.QueryRowsCtx(ctx, &ids, query, productCode, consts.StatusEnabled); err != nil {
  117. return nil, err
  118. }
  119. return ids, nil
  120. }