addMemberLogic.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  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/middleware"
  9. "perms-system-server/internal/model/productmember"
  10. "perms-system-server/internal/response"
  11. "perms-system-server/internal/svc"
  12. "perms-system-server/internal/types"
  13. "perms-system-server/internal/util"
  14. "github.com/zeromicro/go-zero/core/logx"
  15. )
  16. type AddMemberLogic struct {
  17. logx.Logger
  18. ctx context.Context
  19. svcCtx *svc.ServiceContext
  20. }
  21. func NewAddMemberLogic(ctx context.Context, svcCtx *svc.ServiceContext) *AddMemberLogic {
  22. return &AddMemberLogic{
  23. Logger: logx.WithContext(ctx),
  24. ctx: ctx,
  25. svcCtx: svcCtx,
  26. }
  27. }
  28. // AddMember 添加产品成员。将已有用户加入指定产品并设置成员类型(ADMIN/DEVELOPER/MEMBER),需产品 ADMIN 或超管权限。产品必须已启用。
  29. func (l *AddMemberLogic) AddMember(req *types.AddMemberReq) (resp *types.IdResp, err error) {
  30. caller := middleware.GetUserDetails(l.ctx)
  31. if caller == nil {
  32. return nil, response.ErrUnauthorized("未登录")
  33. }
  34. var productCode string
  35. if caller.IsSuperAdmin {
  36. if req.ProductCode == "" {
  37. return nil, response.ErrBadRequest("必须指定产品编码")
  38. }
  39. productCode = req.ProductCode
  40. } else {
  41. productCode = middleware.GetProductCode(l.ctx)
  42. if productCode == "" {
  43. return nil, response.ErrForbidden("缺少产品上下文")
  44. }
  45. }
  46. if err := authHelper.RequireProductAdminFor(l.ctx, productCode); err != nil {
  47. return nil, err
  48. }
  49. // 字面校验在 DB 读之前一并做掉——对非法 memberType 的请求直接 400,无需耗费 DB/缓存。
  50. if req.MemberType != consts.MemberTypeAdmin &&
  51. req.MemberType != consts.MemberTypeDeveloper &&
  52. req.MemberType != consts.MemberTypeMember {
  53. return nil, response.ErrBadRequest("无效的成员类型")
  54. }
  55. if err := authHelper.CheckMemberTypeAssignment(l.ctx, req.MemberType); err != nil {
  56. return nil, err
  57. }
  58. product, err := l.svcCtx.SysProductModel.FindOneByCode(l.ctx, productCode)
  59. if err != nil {
  60. return nil, response.ErrNotFound("产品不存在")
  61. }
  62. if product.Status != consts.StatusEnabled {
  63. return nil, response.ErrBadRequest("产品已被禁用,无法添加成员")
  64. }
  65. targetUser, err := l.svcCtx.SysUserModel.FindOne(l.ctx, req.UserId)
  66. if err != nil {
  67. return nil, response.ErrNotFound("用户不存在")
  68. }
  69. if targetUser.Status != consts.StatusEnabled {
  70. return nil, response.ErrBadRequest("用户已被冻结,无法添加为成员")
  71. }
  72. // 显式拒绝把超管拉入具体产品:loadMembership 虽然会把超管的 MemberType 固定为 SuperAdmin
  73. // 让实际权限不受影响,但 sys_product_member 里会留下一条"product_admin 纳管了 super_admin"
  74. // 的假成员关系,污染审计日志 / 权限推理工具(见审计 H-3)。
  75. if targetUser.IsSuperAdmin == consts.IsSuperAdminYes {
  76. return nil, response.ErrForbidden("无法将超级管理员加入具体产品")
  77. }
  78. // 补齐目标侧部门链授权:原先只做 RequireProductAdminFor(caller 侧),产品 ADMIN 就能把任意
  79. // 部门树外的用户(HR、财务、其他 BU)强行拉进自己的产品,叠加 H-2 PII 暴露后可以"随便拉人 →
  80. // 读全员 PII"。这里用 CheckAddMemberAccess 而不是 CheckManageAccess:
  81. // 1. target 还不是成员,checkPermLevel 对它必定落空报 403,会整体打穿 product-ADMIN 的添加流程;
  82. // 2. product-ADMIN 在 CheckManageAccess 中本身就会短路 checkDeptHierarchy,无法真正拦住跨
  83. // 部门拉人。CheckAddMemberAccess 专门为 AddMember 这类"target 尚未进入成员池"的前置流程
  84. // 设计,对 product ADMIN 也强制执行部门链校验(见审计 H-3)。
  85. if err := authHelper.CheckAddMemberAccess(l.ctx, l.svcCtx, targetUser); err != nil {
  86. return nil, err
  87. }
  88. _, findErr := l.svcCtx.SysProductMemberModel.FindOneByProductCodeUserId(l.ctx, productCode, req.UserId)
  89. if findErr == nil {
  90. return nil, response.ErrConflict("该用户已是该产品成员")
  91. }
  92. now := time.Now().Unix()
  93. result, err := l.svcCtx.SysProductMemberModel.Insert(l.ctx, &productmember.SysProductMember{
  94. ProductCode: productCode,
  95. UserId: req.UserId,
  96. MemberType: req.MemberType,
  97. Status: consts.StatusEnabled,
  98. CreateTime: now,
  99. UpdateTime: now,
  100. })
  101. if err != nil {
  102. if util.IsDuplicateEntryErr(err) {
  103. return nil, response.ErrConflict("该用户已是该产品成员")
  104. }
  105. return nil, err
  106. }
  107. // 审计 L-R13-5 方案 A:新成员插入后旧的"非成员"负缓存语义必须立刻失效——用 detached ctx
  108. // 防止 HTTP 层取消把 UD 旧状态悬挂到 TTL 结束。
  109. cleanCtx, cancel := loaders.DetachCacheCleanCtx(l.ctx)
  110. defer cancel()
  111. l.svcCtx.UserDetailsLoader.Del(cleanCtx, req.UserId, productCode)
  112. id, _ := result.LastInsertId()
  113. return &types.IdResp{Id: id}, nil
  114. }