updateMemberLogic.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. package member
  2. import (
  3. "context"
  4. "time"
  5. "perms-system-server/internal/consts"
  6. "perms-system-server/internal/loaders"
  7. authHelper "perms-system-server/internal/logic/auth"
  8. "perms-system-server/internal/response"
  9. "perms-system-server/internal/svc"
  10. "perms-system-server/internal/types"
  11. "github.com/zeromicro/go-zero/core/logx"
  12. "github.com/zeromicro/go-zero/core/stores/sqlx"
  13. )
  14. type UpdateMemberLogic struct {
  15. logx.Logger
  16. ctx context.Context
  17. svcCtx *svc.ServiceContext
  18. }
  19. func NewUpdateMemberLogic(ctx context.Context, svcCtx *svc.ServiceContext) *UpdateMemberLogic {
  20. return &UpdateMemberLogic{
  21. Logger: logx.WithContext(ctx),
  22. ctx: ctx,
  23. svcCtx: svcCtx,
  24. }
  25. }
  26. // UpdateMember 更新产品成员。修改成员类型或启用/禁用状态。降级最后一个 ADMIN 时会被拒绝以保证产品始终有管理员。
  27. // 审计 L-R11-1:memberType / status 均为指针可选,nil 表示不改该字段;两者都为 nil 时直接 400。
  28. func (l *UpdateMemberLogic) UpdateMember(req *types.UpdateMemberReq) error {
  29. if req.MemberType == nil && req.Status == nil {
  30. return response.ErrBadRequest("请至少提供一个要更新的字段(memberType 或 status)")
  31. }
  32. member, err := l.svcCtx.SysProductMemberModel.FindOne(l.ctx, req.Id)
  33. if err != nil {
  34. return response.ErrNotFound("成员不存在")
  35. }
  36. nextType := member.MemberType
  37. if req.MemberType != nil {
  38. if *req.MemberType != consts.MemberTypeAdmin &&
  39. *req.MemberType != consts.MemberTypeDeveloper &&
  40. *req.MemberType != consts.MemberTypeMember {
  41. return response.ErrBadRequest("无效的成员类型")
  42. }
  43. nextType = *req.MemberType
  44. }
  45. if err := authHelper.CheckManageAccess(l.ctx, l.svcCtx, member.UserId, member.ProductCode); err != nil {
  46. return err
  47. }
  48. // 仅在 memberType 真的被改动时走 CheckMemberTypeAssignment:DEVELOPER 不得被普通 admin 分配,
  49. // 但"只改 status"的场景(已经是 DEVELOPER 的人冻结/启用)不应被该校验误拦。
  50. if req.MemberType != nil && nextType != member.MemberType {
  51. if err := authHelper.CheckMemberTypeAssignment(l.ctx, nextType); err != nil {
  52. return err
  53. }
  54. }
  55. nextStatus := member.Status
  56. if req.Status != nil {
  57. if *req.Status != consts.StatusEnabled && *req.Status != consts.StatusDisabled {
  58. return response.ErrBadRequest("状态值无效,仅支持 1(启用) 和 2(禁用)")
  59. }
  60. nextStatus = *req.Status
  61. }
  62. if nextType == member.MemberType && nextStatus == member.Status {
  63. return nil
  64. }
  65. if err := l.svcCtx.SysProductMemberModel.TransactCtx(l.ctx, func(ctx context.Context, session sqlx.Session) error {
  66. locked, err := l.svcCtx.SysProductMemberModel.FindOneForUpdateTx(ctx, session, req.Id)
  67. if err != nil {
  68. return response.ErrNotFound("成员不存在")
  69. }
  70. wasActiveAdmin := locked.MemberType == consts.MemberTypeAdmin && locked.Status == consts.StatusEnabled
  71. willBeActiveAdmin := nextType == consts.MemberTypeAdmin && nextStatus == consts.StatusEnabled
  72. if wasActiveAdmin && !willBeActiveAdmin {
  73. // 排除当前正在降级/禁用的这一行后还有几个 active admin;为 0 时即为最后一个(见审计 L-5)。
  74. otherAdminCount, err := l.svcCtx.SysProductMemberModel.CountOtherActiveAdminsTx(ctx, session, member.ProductCode, locked.Id)
  75. if err != nil {
  76. return err
  77. }
  78. if otherAdminCount == 0 {
  79. return response.ErrBadRequest("不能降级或禁用该产品的最后一个管理员")
  80. }
  81. }
  82. locked.MemberType = nextType
  83. locked.Status = nextStatus
  84. locked.UpdateTime = time.Now().Unix()
  85. return l.svcCtx.SysProductMemberModel.UpdateWithTx(ctx, session, locked)
  86. }); err != nil {
  87. return err
  88. }
  89. // 审计 L-R13-5 方案 A:memberType / status 变更直接改 loadPerms 的全权分支判定,
  90. // UD 失效脱离请求 ctx 防止 TTL 窗口内旧权限继续生效。
  91. cleanCtx, cancel := loaders.DetachCacheCleanCtx(l.ctx)
  92. defer cancel()
  93. l.svcCtx.UserDetailsLoader.Del(cleanCtx, member.UserId, member.ProductCode)
  94. return nil
  95. }