logoutLogic.go 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. // Code scaffolded by goctl. Safe to edit.
  2. // goctl 1.10.0
  3. package auth
  4. import (
  5. "context"
  6. "errors"
  7. "fmt"
  8. "perms-system-server/internal/middleware"
  9. userModel "perms-system-server/internal/model/user"
  10. "perms-system-server/internal/response"
  11. "perms-system-server/internal/svc"
  12. "github.com/zeromicro/go-zero/core/limit"
  13. "github.com/zeromicro/go-zero/core/logx"
  14. )
  15. type LogoutLogic struct {
  16. logx.Logger
  17. ctx context.Context
  18. svcCtx *svc.ServiceContext
  19. }
  20. func NewLogoutLogic(ctx context.Context, svcCtx *svc.ServiceContext) *LogoutLogic {
  21. return &LogoutLogic{
  22. Logger: logx.WithContext(ctx),
  23. ctx: ctx,
  24. svcCtx: svcCtx,
  25. }
  26. }
  27. // Logout 用户注销。递增当前用户的 tokenVersion 使所有已签发的 access/refresh 令牌立即失效,并清除用户缓存。
  28. func (l *LogoutLogic) Logout() error {
  29. // 审计 M-R11-2:从 middleware 的 UserDetails 直接取 userId/username,
  30. // 不再到 IncrementTokenVersion 内部再 FindOne 一次仅为构造缓存键。
  31. ud := middleware.GetUserDetails(l.ctx)
  32. if ud == nil || ud.UserId == 0 {
  33. return response.ErrUnauthorized("未登录")
  34. }
  35. userId := ud.UserId
  36. if l.svcCtx.TokenOpLimiter != nil {
  37. code, _ := l.svcCtx.TokenOpLimiter.Take(fmt.Sprintf("logout:%d", userId))
  38. if code == limit.OverQuota {
  39. return response.ErrTooManyRequests("操作过于频繁,请稍后再试")
  40. }
  41. }
  42. if _, err := l.svcCtx.SysUserModel.IncrementTokenVersion(l.ctx, userId, ud.Username); err != nil {
  43. // 审计 L-R10-3:IncrementTokenVersion 在目标用户已被并发删除时会返 ErrUpdateConflict。
  44. // Logout 的语义目标本就是"让该账号的旧令牌立即失效",用户已经消失等同语义已达成,
  45. // 按幂等成功处理并继续清缓存,不要让一次正常的注销因为极罕见的删号竞态回 500。
  46. if !errors.Is(err, userModel.ErrUpdateConflict) {
  47. return err
  48. }
  49. logx.WithContext(l.ctx).Infof("logout on already-deleted user userId=%d, treated as idempotent success", userId)
  50. }
  51. l.svcCtx.UserDetailsLoader.Clean(l.ctx, userId)
  52. return nil
  53. }