sysProductMemberModel.go 3.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  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. CountActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string) (int64, error)
  15. CountOtherActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string, excludeId int64) (int64, error)
  16. FindOneForUpdateTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error)
  17. }
  18. customSysProductMemberModel struct {
  19. *defaultSysProductMemberModel
  20. }
  21. )
  22. func NewSysProductMemberModel(conn sqlx.SqlConn, c cache.CacheConf, cachePrefix string, opts ...cache.Option) SysProductMemberModel {
  23. return &customSysProductMemberModel{
  24. defaultSysProductMemberModel: newSysProductMemberModel(conn, c, cachePrefix, opts...),
  25. }
  26. }
  27. func (m *customSysProductMemberModel) FindListByProductCode(ctx context.Context, productCode string, page, pageSize int64) ([]*SysProductMember, int64, error) {
  28. var total int64
  29. countQuery := fmt.Sprintf("SELECT COUNT(*) FROM %s WHERE `productCode` = ?", m.table)
  30. if err := m.QueryRowNoCacheCtx(ctx, &total, countQuery, productCode); err != nil {
  31. return nil, 0, err
  32. }
  33. var list []*SysProductMember
  34. query := fmt.Sprintf("SELECT %s FROM %s WHERE `productCode` = ? ORDER BY id DESC LIMIT ?,?", sysProductMemberRows, m.table)
  35. if err := m.QueryRowsNoCacheCtx(ctx, &list, query, productCode, (page-1)*pageSize, pageSize); err != nil {
  36. return nil, 0, err
  37. }
  38. return list, total, nil
  39. }
  40. func (m *customSysProductMemberModel) CountActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string) (int64, error) {
  41. var ids []int64
  42. query := fmt.Sprintf("SELECT `id` FROM %s WHERE `productCode` = ? AND `memberType` = ? AND `status` = ? FOR UPDATE", m.table)
  43. if err := session.QueryRowsCtx(ctx, &ids, query, productCode, consts.MemberTypeAdmin, consts.StatusEnabled); err != nil {
  44. return 0, err
  45. }
  46. return int64(len(ids)), nil
  47. }
  48. // CountOtherActiveAdminsTx 统计"除 excludeId 这一行以外"的启用 ADMIN 数量。调用方一般把即将被删除
  49. // 或即将被降级的目标行 id 传进来;返回 0 即表示目标是最后一个 active admin,不能动。相比
  50. // CountActiveAdminsTx + adminCount <= 1 的反向推理,语义更贴合业务(见审计 L-5)。
  51. // 仍然使用 FOR UPDATE 锁住扫描范围,串行化与并发降级/删除的冲突。
  52. func (m *customSysProductMemberModel) CountOtherActiveAdminsTx(ctx context.Context, session sqlx.Session, productCode string, excludeId int64) (int64, error) {
  53. var ids []int64
  54. query := fmt.Sprintf("SELECT `id` FROM %s WHERE `productCode` = ? AND `memberType` = ? AND `status` = ? AND `id` != ? FOR UPDATE", m.table)
  55. if err := session.QueryRowsCtx(ctx, &ids, query, productCode, consts.MemberTypeAdmin, consts.StatusEnabled, excludeId); err != nil {
  56. return 0, err
  57. }
  58. return int64(len(ids)), nil
  59. }
  60. func (m *customSysProductMemberModel) FindOneForUpdateTx(ctx context.Context, session sqlx.Session, id int64) (*SysProductMember, error) {
  61. var data SysProductMember
  62. query := fmt.Sprintf("SELECT %s FROM %s WHERE `id` = ? FOR UPDATE", sysProductMemberRows, m.table)
  63. if err := session.QueryRowCtx(ctx, &data, query, id); err != nil {
  64. return nil, err
  65. }
  66. return &data, nil
  67. }