3
0
Files
corteza/compose/service/module_test.go
2022-07-27 16:53:05 +02:00

291 lines
7.1 KiB
Go

package service
import (
"context"
"testing"
"github.com/cortezaproject/corteza-server/pkg/dal"
"github.com/cortezaproject/corteza-server/pkg/logger"
"github.com/cortezaproject/corteza-server/compose/types"
"github.com/cortezaproject/corteza-server/pkg/eventbus"
"github.com/cortezaproject/corteza-server/pkg/rbac"
"github.com/cortezaproject/corteza-server/store"
"github.com/cortezaproject/corteza-server/store/adapters/rdbms/drivers/sqlite"
"github.com/stretchr/testify/require"
"go.uber.org/zap"
)
func makeTestModuleService(t *testing.T, mods ...any) *module {
var (
err error
req = require.New(t)
log = zap.NewNop()
)
for _, m := range mods {
switch c := m.(type) {
case *zap.Logger:
t.Log("using custom Logger to initialize Module service")
log = c
}
}
var (
ctx = logger.ContextWithValue(context.Background(), log)
svc = &module{
eventbus: eventbus.New(),
}
)
for _, m := range mods {
switch c := m.(type) {
case rbacService:
t.Log("using custom RBAC to initialize Module service")
svc.ac = &accessControl{rbac: c}
case store.Storer:
t.Log("using custom Store to initialize Module service")
svc.store = c
case dalService:
t.Log("using custom DAL to initialize Module service")
t.Log("make sure you manually reload models!")
svc.dal = c
}
}
if svc.ac == nil {
svc.ac = &accessControl{rbac: rbac.NewService(log, nil)}
}
if svc.store == nil {
t.Log("using SQLite in-memory Store")
svc.store, err = sqlite.ConnectInMemoryWithDebug(ctx)
req.NoError(err)
t.Log("upgrading store")
req.NoError(store.Upgrade(ctx, log, svc.store))
t.Log("data cleanup")
req.NoError(store.TruncateUsers(ctx, svc.store))
req.NoError(store.TruncateRoles(ctx, svc.store))
req.NoError(store.TruncateComposeNamespaces(ctx, svc.store))
req.NoError(store.TruncateComposeModules(ctx, svc.store))
req.NoError(store.TruncateComposeModuleFields(ctx, svc.store))
req.NoError(store.TruncateRbacRules(ctx, svc.store))
req.NoError(store.TruncateLabels(ctx, svc.store))
}
resourceMaker(ctx, t, svc.store, mods...)
if svc.dal == nil {
dalAux, err := dal.New(zap.NewNop(), true)
req.NoError(err)
const (
recordsTable = "compose_record"
)
req.NoError(
dalAux.ReplaceConnection(
ctx,
dal.MakeConnection(1, svc.store.ToDalConn(),
dal.ConnectionParams{},
dal.ConnectionMeta{DefaultModelIdent: recordsTable, DefaultAttributeIdent: "values"},
),
true,
),
)
svc.dal = dalAux
t.Log("reloading DAL models")
req.NoError(DalModelReload(ctx, svc.store, dalAux))
}
return svc
}
func TestModules(t *testing.T) {
var (
ctx = context.Background()
ns = &types.Namespace{Name: "testing", ID: nextID(), CreatedAt: *now()}
)
t.Run("crud", func(t *testing.T) {
req := require.New(t)
svc := makeTestModuleService(t,
ns,
&rbac.ServiceAllowAll{},
)
res, err := svc.Create(ctx, &types.Module{Name: "My first module", NamespaceID: ns.ID})
req.NoError(err)
req.NotNil(res)
res, err = svc.FindByID(ctx, ns.ID, res.ID)
req.NoError(err)
req.NotNil(res)
res, err = svc.FindByHandle(ctx, ns.ID, res.Handle)
req.NoError(err)
req.NotNil(res)
res.Name = "Changed"
res, err = svc.Update(ctx, res)
req.NoError(err)
req.NotNil(res)
req.NotNil(res.UpdatedAt)
req.Equal(res.Name, "Changed")
res, err = svc.FindByID(ctx, ns.ID, res.ID)
req.NoError(err)
req.NotNil(res)
req.Equal(res.Name, "Changed")
err = svc.DeleteByID(ctx, ns.ID, res.ID)
req.NoError(err)
req.NotNil(res)
// this works because we're allowed to do everything
res, err = svc.FindByID(ctx, ns.ID, res.ID)
req.NoError(err)
req.NotNil(res)
req.NotNil(res.DeletedAt)
})
}
func TestModule_LabelSearch(t *testing.T) {
var (
ns = &types.Namespace{Name: "testing", ID: nextID(), CreatedAt: *now()}
req = require.New(t)
svc = makeTestModuleService(t,
ns,
&rbac.ServiceAllowAll{},
)
ctx = context.Background()
//ctx = logger.ContextWithValue(context.Background(), logger.MakeDebugLogger())
makeModule = func(n ...string) *types.Module {
mod := &types.Module{
NamespaceID: ns.ID,
Name: n[0],
Labels: map[string]string{},
}
for i := 1; i < len(n); i += 2 {
mod.Labels[n[i]] = n[i+1]
}
out, err := svc.Create(ctx, mod)
req.NoError(err)
return out
}
findModules = func(labels map[string]string, IDs []uint64) types.ModuleSet {
f := types.ModuleFilter{NamespaceID: ns.ID, Labels: labels, ModuleID: IDs}
set, _, err := svc.Find(ctx, f)
req.NoError(err)
return set
}
)
makeModule("labeled module 1", "label1", "value1", "label2", "value2")
m2 := makeModule("labeled module 2", "label1", "value1")
m3 := makeModule("labeled module 3")
//return all -- no label/ID filter, return all
req.Len(findModules(nil, nil), 3)
// return 2 - both that have label1=valu1
req.Len(findModules(map[string]string{"label1": "value1"}, nil), 2)
// return 0 - none have foo=foo
req.Len(findModules(map[string]string{"missing": "missing"}, nil), 0)
// one has label2=value2
req.Len(findModules(map[string]string{"label2": "value2"}, nil), 1)
// explicit by ID and label
req.Len(findModules(map[string]string{"label1": "value1"}, []uint64{m2.ID}), 1)
// none with this combo
req.Len(findModules(map[string]string{"foo": "foo"}, []uint64{m3.ID}), 0)
// one with explicit ID (regression) and nil for label filter
req.Len(findModules(nil, []uint64{m3.ID}), 1)
// one with explicit ID (regression) and empty map for label filter
req.Len(findModules(map[string]string{}, []uint64{m3.ID}), 1)
}
func TestModule_LabelCRUD(t *testing.T) {
var (
ctx = context.Background()
//ctx = logger.ContextWithValue(context.Background(), logger.MakeDebugLogger())
ns = &types.Namespace{Name: "testing", ID: nextID(), CreatedAt: *now()}
req = require.New(t)
svc = makeTestModuleService(t,
ns,
&rbac.ServiceAllowAll{},
)
findAndReturnLabel = func(id uint64) map[string]string {
res, err := svc.FindByID(ctx, ns.ID, id)
req.NoError(err)
req.NotNil(res)
return res.Labels
}
)
// create unlabeled module
res, err := svc.Create(ctx, &types.Module{Name: "unLabeledIDs", NamespaceID: ns.ID})
req.NoError(err)
req.NotNil(res)
req.Nil(res.Labels)
// no labels should be present
req.Nil(findAndReturnLabel(res.ID))
// update the module with labels
res.Labels = map[string]string{"label1": "1st"}
res, err = svc.Update(ctx, res)
req.NoError(err)
req.NotNil(res)
req.Contains(res.Labels, "label1")
// must contain the added label
req.Contains(findAndReturnLabel(res.ID), "label1")
res, err = svc.Create(ctx, &types.Module{Name: "LabeledIDs", NamespaceID: ns.ID, Labels: map[string]string{"label2": "2nd"}})
req.NoError(err)
req.NotNil(res)
req.Contains(res.Labels, "label2")
// must contain the added label
req.Contains(findAndReturnLabel(res.ID), "label2")
// update with Labels:nil (should keep labels intact)
res.Labels = nil
res, err = svc.Update(ctx, res)
req.NoError(err)
req.Contains(findAndReturnLabel(res.ID), "label2")
// update with Labels:empty-map (should remove all labels)
res.Labels = map[string]string{}
res, err = svc.Update(ctx, res)
req.NoError(err)
req.Empty(findAndReturnLabel(res.ID))
}