3
0
2024-12-11 15:35:16 +01:00

232 lines
6.3 KiB
Go

package rest
import (
"context"
"github.com/cortezaproject/corteza/server/compose/rest/request"
"github.com/cortezaproject/corteza/server/compose/service"
"github.com/cortezaproject/corteza/server/compose/service/event"
"github.com/cortezaproject/corteza/server/compose/types"
"github.com/cortezaproject/corteza/server/pkg/api"
"github.com/cortezaproject/corteza/server/pkg/corredor"
"github.com/cortezaproject/corteza/server/pkg/filter"
)
type (
moduleSetPayload struct {
Filter types.ModuleFilter `json:"filter"`
Set []*modulePayload `json:"set"`
}
modulePayload struct {
*types.Module
Fields []*moduleFieldPayload `json:"fields"`
CanGrant bool `json:"canGrant"`
CanExport bool `json:"canExport"`
CanUpdateModule bool `json:"canUpdateModule"`
CanDeleteModule bool `json:"canDeleteModule"`
CanCreateRecord bool `json:"canCreateRecord"`
CanCreateOwnedRecord bool `json:"canCreateOwnedRecord"`
}
moduleFieldPayload struct {
*types.ModuleField
CanReadRecordValue bool `json:"canReadRecordValue"`
CanUpdateRecordValue bool `json:"canUpdateRecordValue"`
}
Module struct {
module service.ModuleService
locale service.ResourceTranslationsManagerService
namespace service.NamespaceService
ac moduleAccessController
}
moduleAccessController interface {
CanGrant(context.Context) bool
CanExportModule(context.Context, *types.Module) bool
CanUpdateModule(context.Context, *types.Module) bool
CanDeleteModule(context.Context, *types.Module) bool
CanCreateRecordOnModule(context.Context, *types.Module) bool
CanCreateOwnedRecordOnModule(context.Context, *types.Module) bool
CanReadRecord(context.Context, *types.Record) bool
CanReadRecordValueOnModuleField(context.Context, *types.ModuleField) bool
CanUpdateRecordValueOnModuleField(context.Context, *types.ModuleField) bool
}
)
func (Module) New() *Module {
return &Module{
module: service.DefaultModule,
namespace: service.DefaultNamespace,
ac: service.DefaultAccessControl,
locale: service.DefaultResourceTranslation,
}
}
func (ctrl *Module) List(ctx context.Context, r *request.ModuleList) (interface{}, error) {
var (
err error
f = types.ModuleFilter{
NamespaceID: r.NamespaceID,
Query: r.Query,
Name: r.Name,
Handle: r.Handle,
Labels: r.Labels,
}
)
if f.Paging, err = filter.NewPaging(r.Limit, r.PageCursor); err != nil {
return nil, err
}
f.IncTotal = r.IncTotal
if f.Sorting, err = filter.NewSorting(r.Sort); err != nil {
return nil, err
}
set, filter, err := ctrl.module.Find(ctx, f)
return ctrl.makeFilterPayload(ctx, set, filter, err)
}
func (ctrl *Module) Read(ctx context.Context, r *request.ModuleRead) (interface{}, error) {
mod, err := ctrl.module.FindByID(ctx, r.NamespaceID, r.ModuleID)
return ctrl.makePayload(ctx, mod, err)
}
func (ctrl *Module) ListTranslations(ctx context.Context, r *request.ModuleListTranslations) (interface{}, error) {
return ctrl.locale.Module(ctx, r.NamespaceID, r.ModuleID)
}
func (ctrl *Module) UpdateTranslations(ctx context.Context, r *request.ModuleUpdateTranslations) (interface{}, error) {
return api.OK(), ctrl.locale.Upsert(ctx, r.Translations)
}
func (ctrl *Module) Create(ctx context.Context, r *request.ModuleCreate) (interface{}, error) {
var (
err error
mod = &types.Module{
NamespaceID: r.NamespaceID,
Config: r.Config,
Name: r.Name,
Handle: r.Handle,
Fields: r.Fields,
Meta: r.Meta,
Labels: r.Labels,
}
)
mod, err = ctrl.module.Create(ctx, mod)
return ctrl.makePayload(ctx, mod, err)
}
func (ctrl *Module) Update(ctx context.Context, r *request.ModuleUpdate) (interface{}, error) {
var (
err error
mod = &types.Module{
ID: r.ModuleID,
NamespaceID: r.NamespaceID,
Config: r.Config,
Name: r.Name,
Handle: r.Handle,
Fields: r.Fields,
Meta: r.Meta,
Labels: r.Labels,
UpdatedAt: r.UpdatedAt,
}
)
mod, err = ctrl.module.Update(ctx, mod)
return ctrl.makePayload(ctx, mod, err)
}
func (ctrl *Module) Delete(ctx context.Context, r *request.ModuleDelete) (interface{}, error) {
_, err := ctrl.module.FindByID(ctx, r.NamespaceID, r.ModuleID)
if err != nil {
return nil, err
}
return api.OK(), ctrl.module.DeleteByID(ctx, r.NamespaceID, r.ModuleID)
}
func (ctrl *Module) TriggerScript(ctx context.Context, r *request.ModuleTriggerScript) (rsp interface{}, err error) {
var (
module *types.Module
namespace *types.Namespace
)
if module, err = ctrl.module.FindByID(ctx, r.NamespaceID, r.ModuleID); err != nil {
return
}
if namespace, err = ctrl.namespace.FindByID(ctx, r.NamespaceID); err != nil {
return
}
// @todo implement same behaviour as we have on record - module+oldModule
err = corredor.Service().Exec(ctx, r.Script, corredor.ExtendScriptArgs(event.ModuleOnManual(module, module, namespace), r.Args))
return ctrl.makePayload(ctx, module, err)
}
func (ctrl Module) makePayload(ctx context.Context, m *types.Module, err error) (*modulePayload, error) {
if err != nil || m == nil {
return nil, err
}
mfp, err := ctrl.makeFieldsPayload(ctx, m)
if err != nil {
return nil, err
}
return &modulePayload{
Module: m,
Fields: mfp,
CanGrant: ctrl.ac.CanGrant(ctx),
CanExport: ctrl.ac.CanExportModule(ctx, m),
CanUpdateModule: ctrl.ac.CanUpdateModule(ctx, m),
CanDeleteModule: ctrl.ac.CanDeleteModule(ctx, m),
CanCreateRecord: ctrl.ac.CanCreateRecordOnModule(ctx, m),
CanCreateOwnedRecord: ctrl.ac.CanCreateOwnedRecordOnModule(ctx, m),
}, nil
}
func (ctrl Module) makeFieldsPayload(ctx context.Context, m *types.Module) (out []*moduleFieldPayload, err error) {
out = make([]*moduleFieldPayload, len(m.Fields))
for i, f := range m.Fields {
out[i] = &moduleFieldPayload{
ModuleField: f,
CanReadRecordValue: ctrl.ac.CanReadRecordValueOnModuleField(ctx, f),
CanUpdateRecordValue: ctrl.ac.CanUpdateRecordValueOnModuleField(ctx, f),
}
}
return
}
func (ctrl Module) makeFilterPayload(ctx context.Context, nn types.ModuleSet, f types.ModuleFilter, err error) (*moduleSetPayload, error) {
if err != nil {
return nil, err
}
msp := &moduleSetPayload{Filter: f, Set: make([]*modulePayload, len(nn))}
for i := range nn {
msp.Set[i], _ = ctrl.makePayload(ctx, nn[i], nil)
}
return msp, nil
}