createDeptLogic.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. package dept
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "time"
  7. "perms-system-server/internal/consts"
  8. authHelper "perms-system-server/internal/logic/auth"
  9. deptModel "perms-system-server/internal/model/dept"
  10. "perms-system-server/internal/response"
  11. "perms-system-server/internal/svc"
  12. "perms-system-server/internal/types"
  13. "github.com/zeromicro/go-zero/core/logx"
  14. "github.com/zeromicro/go-zero/core/stores/sqlx"
  15. )
  16. type CreateDeptLogic struct {
  17. logx.Logger
  18. ctx context.Context
  19. svcCtx *svc.ServiceContext
  20. }
  21. func NewCreateDeptLogic(ctx context.Context, svcCtx *svc.ServiceContext) *CreateDeptLogic {
  22. return &CreateDeptLogic{
  23. Logger: logx.WithContext(ctx),
  24. ctx: ctx,
  25. svcCtx: svcCtx,
  26. }
  27. }
  28. // CreateDept 创建部门。在指定父部门下新建子部门,自动继承路径层级。仅超管可调用。
  29. func (l *CreateDeptLogic) CreateDept(req *types.CreateDeptReq) (resp *types.IdResp, err error) {
  30. if err := authHelper.RequireSuperAdmin(l.ctx); err != nil {
  31. return nil, err
  32. }
  33. if len(req.Name) > 64 {
  34. return nil, response.ErrBadRequest("部门名称长度不能超过64个字符")
  35. }
  36. if len(req.Remark) > 255 {
  37. return nil, response.ErrBadRequest("备注长度不能超过255个字符")
  38. }
  39. parentPath := "/"
  40. if req.ParentId > 0 {
  41. parent, err := l.svcCtx.SysDeptModel.FindOne(l.ctx, req.ParentId)
  42. if err != nil {
  43. return nil, response.ErrNotFound("父部门不存在")
  44. }
  45. parentPath = parent.Path
  46. }
  47. now := time.Now().Unix()
  48. var deptId int64
  49. deptType := req.DeptType
  50. if deptType == "" {
  51. deptType = consts.DeptTypeNormal
  52. } else if deptType != consts.DeptTypeNormal && deptType != consts.DeptTypeDev {
  53. return nil, response.ErrBadRequest("无效的部门类型")
  54. }
  55. err = l.svcCtx.SysDeptModel.TransactCtx(l.ctx, func(ctx context.Context, session sqlx.Session) error {
  56. if req.ParentId > 0 {
  57. // 审计 L-R12-2:事务内用 FindOneForShareTx 重取父部门快照——同时拿到 status 与
  58. // path。此前只 SELECT `id` 无法感知"父部门在事务外 FindOne 之后被 UpdateDept 禁用"
  59. // 这条交错:读到 id 仍存在就放行,结果子部门会以 Enabled 挂在已 Disabled 的父部门下,
  60. // 让运营侧的"禁用整个子树"意图被静默绕过。改用事务内带 S 锁的完整行读取,同时把
  61. // parentPath 覆盖为事务内视图——与事务外 FindOne 理论上一致(UpdateDept 不改 path),
  62. // 但如果未来 UpdateDept 扩展支持 path 重写,这里已经内建 snapshot。
  63. parent, err := l.svcCtx.SysDeptModel.FindOneForShareTx(ctx, session, req.ParentId)
  64. if err != nil {
  65. if errors.Is(err, sqlx.ErrNotFound) {
  66. return response.ErrNotFound("父部门已被删除")
  67. }
  68. return err
  69. }
  70. if parent.Status != consts.StatusEnabled {
  71. return response.ErrBadRequest("父部门已被禁用,无法创建子部门")
  72. }
  73. parentPath = parent.Path
  74. }
  75. result, err := l.svcCtx.SysDeptModel.InsertWithTx(ctx, session, &deptModel.SysDept{
  76. ParentId: req.ParentId,
  77. Name: req.Name,
  78. Path: parentPath,
  79. Sort: req.Sort,
  80. DeptType: deptType,
  81. Remark: req.Remark,
  82. Status: consts.StatusEnabled,
  83. CreateTime: now,
  84. UpdateTime: now,
  85. })
  86. if err != nil {
  87. return err
  88. }
  89. deptId, _ = result.LastInsertId()
  90. d, err := l.svcCtx.SysDeptModel.FindOneWithTx(ctx, session, deptId)
  91. if err != nil {
  92. return err
  93. }
  94. d.Path = fmt.Sprintf("%s%d/", parentPath, deptId)
  95. return l.svcCtx.SysDeptModel.UpdateWithTx(ctx, session, d)
  96. })
  97. if err != nil {
  98. return nil, err
  99. }
  100. return &types.IdResp{Id: deptId}, nil
  101. }