180 lines
4.5 KiB
Go
180 lines
4.5 KiB
Go
package biz
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/guxuan/hailin_service/internal/config"
|
|
"github.com/guxuan/hailin_service/internal/mods/rbac/dal"
|
|
"github.com/guxuan/hailin_service/internal/mods/rbac/schema"
|
|
"github.com/guxuan/hailin_service/pkg/cachex"
|
|
"github.com/guxuan/hailin_service/pkg/errors"
|
|
"github.com/guxuan/hailin_service/pkg/util"
|
|
)
|
|
|
|
// Role management for RBAC
|
|
type Role struct {
|
|
Cache cachex.Cacher
|
|
Trans *util.Trans
|
|
RoleDAL *dal.Role
|
|
RoleMenuDAL *dal.RoleMenu
|
|
UserRoleDAL *dal.UserRole
|
|
}
|
|
|
|
// Query roles from the data access object based on the provided parameters and options.
|
|
func (a *Role) Query(ctx context.Context, params schema.RoleQueryParam) (*schema.RoleQueryResult, error) {
|
|
params.Pagination = true
|
|
|
|
var selectFields []string
|
|
if params.ResultType == schema.RoleResultTypeSelect {
|
|
params.Pagination = false
|
|
selectFields = []string{"id", "name"}
|
|
}
|
|
|
|
result, err := a.RoleDAL.Query(ctx, params, schema.RoleQueryOptions{
|
|
QueryOptions: util.QueryOptions{
|
|
OrderFields: []util.OrderByParam{
|
|
{Field: "sequence", Direction: util.DESC},
|
|
{Field: "created_at", Direction: util.DESC},
|
|
},
|
|
SelectFields: selectFields,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
// Get the specified role from the data access object.
|
|
func (a *Role) Get(ctx context.Context, id string) (*schema.Role, error) {
|
|
role, err := a.RoleDAL.Get(ctx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
} else if role == nil {
|
|
return nil, errors.NotFound("", "Role not found")
|
|
}
|
|
|
|
roleMenuResult, err := a.RoleMenuDAL.Query(ctx, schema.RoleMenuQueryParam{
|
|
RoleID: id,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
role.Menus = roleMenuResult.Data
|
|
|
|
return role, nil
|
|
}
|
|
|
|
// Create a new role in the data access object.
|
|
func (a *Role) Create(ctx context.Context, formItem *schema.RoleForm) (*schema.Role, error) {
|
|
if exists, err := a.RoleDAL.ExistsCode(ctx, formItem.Code); err != nil {
|
|
return nil, err
|
|
} else if exists {
|
|
return nil, errors.BadRequest("", "Role code already exists")
|
|
}
|
|
|
|
role := &schema.Role{
|
|
ID: util.NewXID(),
|
|
CreatedAt: time.Now(),
|
|
}
|
|
if err := formItem.FillTo(role); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err := a.Trans.Exec(ctx, func(ctx context.Context) error {
|
|
if err := a.RoleDAL.Create(ctx, role); err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, roleMenu := range formItem.Menus {
|
|
roleMenu.ID = util.NewXID()
|
|
roleMenu.RoleID = role.ID
|
|
roleMenu.CreatedAt = time.Now()
|
|
if err := a.RoleMenuDAL.Create(ctx, roleMenu); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return a.syncToCasbin(ctx)
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
role.Menus = formItem.Menus
|
|
|
|
return role, nil
|
|
}
|
|
|
|
// Update the specified role in the data access object.
|
|
func (a *Role) Update(ctx context.Context, id string, formItem *schema.RoleForm) error {
|
|
role, err := a.RoleDAL.Get(ctx, id)
|
|
if err != nil {
|
|
return err
|
|
} else if role == nil {
|
|
return errors.NotFound("", "Role not found")
|
|
} else if role.Code != formItem.Code {
|
|
if exists, err := a.RoleDAL.ExistsCode(ctx, formItem.Code); err != nil {
|
|
return err
|
|
} else if exists {
|
|
return errors.BadRequest("", "Role code already exists")
|
|
}
|
|
}
|
|
|
|
if err := formItem.FillTo(role); err != nil {
|
|
return err
|
|
}
|
|
role.UpdatedAt = time.Now()
|
|
|
|
return a.Trans.Exec(ctx, func(ctx context.Context) error {
|
|
if err := a.RoleDAL.Update(ctx, role); err != nil {
|
|
return err
|
|
}
|
|
if err := a.RoleMenuDAL.DeleteByRoleID(ctx, id); err != nil {
|
|
return err
|
|
}
|
|
for _, roleMenu := range formItem.Menus {
|
|
if roleMenu.ID == "" {
|
|
roleMenu.ID = util.NewXID()
|
|
}
|
|
roleMenu.RoleID = role.ID
|
|
if roleMenu.CreatedAt.IsZero() {
|
|
roleMenu.CreatedAt = time.Now()
|
|
}
|
|
roleMenu.UpdatedAt = time.Now()
|
|
if err := a.RoleMenuDAL.Create(ctx, roleMenu); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return a.syncToCasbin(ctx)
|
|
})
|
|
}
|
|
|
|
// Delete the specified role from the data access object.
|
|
func (a *Role) Delete(ctx context.Context, id string) error {
|
|
exists, err := a.RoleDAL.Exists(ctx, id)
|
|
if err != nil {
|
|
return err
|
|
} else if !exists {
|
|
return errors.NotFound("", "Role not found")
|
|
}
|
|
|
|
return a.Trans.Exec(ctx, func(ctx context.Context) error {
|
|
if err := a.RoleDAL.Delete(ctx, id); err != nil {
|
|
return err
|
|
}
|
|
if err := a.RoleMenuDAL.DeleteByRoleID(ctx, id); err != nil {
|
|
return err
|
|
}
|
|
if err := a.UserRoleDAL.DeleteByRoleID(ctx, id); err != nil {
|
|
return err
|
|
}
|
|
|
|
return a.syncToCasbin(ctx)
|
|
})
|
|
}
|
|
|
|
func (a *Role) syncToCasbin(ctx context.Context) error {
|
|
return a.Cache.Set(ctx, config.CacheNSForRole, config.CacheKeyForSyncToCasbin, fmt.Sprintf("%d", time.Now().Unix()))
|
|
}
|