createRoleLogic.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. package role
  2. import (
  3. "context"
  4. "time"
  5. "perms-system-server/internal/consts"
  6. authHelper "perms-system-server/internal/logic/auth"
  7. "perms-system-server/internal/middleware"
  8. roleModel "perms-system-server/internal/model/role"
  9. "perms-system-server/internal/response"
  10. "perms-system-server/internal/svc"
  11. "perms-system-server/internal/types"
  12. "perms-system-server/internal/util"
  13. "github.com/zeromicro/go-zero/core/logx"
  14. )
  15. type CreateRoleLogic struct {
  16. logx.Logger
  17. ctx context.Context
  18. svcCtx *svc.ServiceContext
  19. }
  20. func NewCreateRoleLogic(ctx context.Context, svcCtx *svc.ServiceContext) *CreateRoleLogic {
  21. return &CreateRoleLogic{
  22. Logger: logx.WithContext(ctx),
  23. ctx: ctx,
  24. svcCtx: svcCtx,
  25. }
  26. }
  27. // CreateRole 创建角色。在指定产品下新建角色并设置权限级别,需产品 ADMIN 或超管权限。产品必须存在且已启用。
  28. func (l *CreateRoleLogic) CreateRole(req *types.CreateRoleReq) (resp *types.IdResp, err error) {
  29. if err := authHelper.RequireProductAdminFor(l.ctx, req.ProductCode); err != nil {
  30. return nil, err
  31. }
  32. product, err := l.svcCtx.SysProductModel.FindOneByCode(l.ctx, req.ProductCode)
  33. if err != nil {
  34. return nil, response.ErrNotFound("产品不存在")
  35. }
  36. if product.Status != consts.StatusEnabled {
  37. return nil, response.ErrBadRequest("产品已被禁用,无法创建角色")
  38. }
  39. if len(req.Name) > 64 {
  40. return nil, response.ErrBadRequest("角色名长度不能超过64个字符")
  41. }
  42. if len(req.Remark) > 255 {
  43. return nil, response.ErrBadRequest("备注长度不能超过255个字符")
  44. }
  45. if req.PermsLevel < 1 || req.PermsLevel > 999 {
  46. return nil, response.ErrBadRequest("权限级别必须在 1-999 之间")
  47. }
  48. // 审计 H-R17-3:镜像 UpdateRole 的 L-R12-3 防提权("非超管不能提升角色的权限级别"),
  49. // 但 CreateRole 的口径比 UpdateRole 更严——UpdateRole 限制的是"已有角色往更高权限挪",
  50. // CreateRole 如果不限制,product ADMIN 可以直接造出 permsLevel=1 的顶格角色,再走 BindRoles
  51. // 把下属 MEMBER/DEVELOPER 顶到 ADMIN 线(下一次 UD 重建时 MinPermsLevel=1),绕过 BindRoles
  52. // 的 GuardRoleLevelAssignable "同级拦截"。GuardCreateRolePermsLevel 把这条短路堵死:
  53. // - SuperAdmin 不受限;
  54. // - 非超管(含 product ADMIN / DEVELOPER)创建时 permsLevel 必须 >= 2,permsLevel=1
  55. // 作为"顶格语义"只允许 SuperAdmin 所生。
  56. caller := middleware.GetUserDetails(l.ctx)
  57. if err := authHelper.GuardCreateRolePermsLevel(l.ctx, l.svcCtx, caller, req.PermsLevel); err != nil {
  58. return nil, err
  59. }
  60. now := time.Now().Unix()
  61. result, err := l.svcCtx.SysRoleModel.Insert(l.ctx, &roleModel.SysRole{
  62. ProductCode: req.ProductCode,
  63. Name: req.Name,
  64. Remark: req.Remark,
  65. Status: consts.StatusEnabled,
  66. PermsLevel: req.PermsLevel,
  67. CreateTime: now,
  68. UpdateTime: now,
  69. })
  70. if err != nil {
  71. if util.IsDuplicateEntryErr(err) {
  72. return nil, response.ErrConflict("该产品下角色名已存在")
  73. }
  74. return nil, err
  75. }
  76. id, _ := result.LastInsertId()
  77. return &types.IdResp{Id: id}, nil
  78. }