You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/host/plugin/host_set_member_test.go

251 lines
7.5 KiB

package plugin
import (
"context"
"testing"
"time"
"github.com/hashicorp/boundary/internal/db"
"github.com/hashicorp/boundary/internal/host/plugin/store"
"github.com/hashicorp/boundary/internal/iam"
"github.com/hashicorp/boundary/internal/kms"
"github.com/hashicorp/boundary/internal/oplog"
hostplg "github.com/hashicorp/boundary/internal/plugin/host"
"github.com/hashicorp/boundary/internal/scheduler"
plgpb "github.com/hashicorp/boundary/sdk/pbs/plugin"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
// TestHostSetMember_InsertDelete tests insertion and validates the lookup via set IDs
// function as well. After, we remove set memberships one at a time and validate
// the cleanup function removes the hosts.
func TestHostSetMember_InsertDelete(t *testing.T) {
ctx := context.Background()
conn, _ := db.TestSetup(t, "postgres")
rw := db.New(conn)
wrapper := db.TestWrapper(t)
kms := kms.TestKms(t, conn, wrapper)
sched := scheduler.TestScheduler(t, conn, wrapper)
_, prj := iam.TestScopes(t, iam.TestRepo(t, conn, wrapper))
plg := hostplg.TestPlugin(t, conn, "create")
plgm := map[string]plgpb.HostPluginServiceClient{
plg.GetPublicId(): NewWrappingPluginClient(&plgpb.UnimplementedHostPluginServiceServer{}),
}
repo, err := NewRepository(rw, rw, kms, sched, plgm)
require.NoError(t, err)
cats := TestCatalogs(t, conn, prj.PublicId, plg.PublicId, 2)
blueCat := cats[0]
blueSet1 := TestSet(t, conn, kms, sched, blueCat, plgm)
blueSet2 := TestSet(t, conn, kms, sched, blueCat, plgm)
blueSet3 := TestSet(t, conn, kms, sched, blueCat, plgm)
hostId, err := db.NewPublicId(HostPrefix)
require.NoError(t, err)
blueHost1 := NewHost(ctx, blueCat.PublicId, "blue1", withPluginId(plg.GetPublicId()))
blueHost1.PublicId = hostId
require.NoError(t, rw.Create(ctx, blueHost1))
hostId, err = db.NewPublicId(HostPrefix)
require.NoError(t, err)
blueHost2 := NewHost(ctx, blueCat.PublicId, "blue2", withPluginId(plg.GetPublicId()))
blueHost2.PublicId = hostId
require.NoError(t, rw.Create(ctx, blueHost2))
hostId, err = db.NewPublicId(HostPrefix)
require.NoError(t, err)
blueHost3 := NewHost(ctx, blueCat.PublicId, "blue3", withPluginId(plg.GetPublicId()))
blueHost3.PublicId = hostId
require.NoError(t, rw.Create(ctx, blueHost3))
hostId, err = db.NewPublicId(HostPrefix)
require.NoError(t, err)
blueHost4 := NewHost(ctx, blueCat.PublicId, "blue4", withPluginId(plg.GetPublicId()))
blueHost4.PublicId = hostId
require.NoError(t, rw.Create(ctx, blueHost4))
greenCat := cats[1]
greenSet := TestSet(t, conn, kms, sched, greenCat, plgm)
tests := []struct {
name string
set string
hosts []*Host
wantErr bool
direct bool
}{
{
name: "valid-host-in-set",
set: blueSet1.PublicId,
hosts: []*Host{blueHost1},
},
{
name: "valid-other-host-in-set",
set: blueSet2.PublicId,
hosts: []*Host{blueHost2},
},
{
name: "valid-two-hosts-in-set",
set: blueSet3.PublicId,
hosts: []*Host{blueHost3, blueHost4},
},
{
name: "invalid-diff-catalogs",
set: greenSet.PublicId,
hosts: []*Host{blueHost1},
wantErr: true,
},
{
name: "test-vet-for-write-no-set",
hosts: []*Host{blueHost1},
wantErr: true,
direct: true,
},
}
for _, tt := range tests {
tt := tt
t.Run(tt.name, func(t *testing.T) {
assert, require := assert.New(t), require.New(t)
var hostIds []string
for _, host := range tt.hosts {
var got *HostSetMember
var err error
hostIds = append(hostIds, host.PublicId)
if !tt.direct {
got, err = NewHostSetMember(ctx, tt.set, host.PublicId)
} else {
got = &HostSetMember{
HostSetMember: &store.HostSetMember{
SetId: tt.set,
},
}
if host != nil {
got.HostId = host.PublicId
}
}
require.NoError(err)
require.NotNil(got)
err2 := rw.Create(ctx, got)
if tt.wantErr {
assert.Error(err2)
return
}
assert.NoError(err2)
}
// Run a test on the aggregate to validate looking up sets
for _, host := range tt.hosts {
agg := &hostAgg{PublicId: host.PublicId}
require.NoError(rw.LookupByPublicId(ctx, agg))
h := agg.toHost()
assert.ElementsMatch(h.SetIds, []string{tt.set})
}
set, _, err := repo.LookupSet(ctx, tt.set)
require.NoError(err)
require.NotNil(set)
require.ElementsMatch(set.HostIds, hostIds)
})
}
hosts, err := repo.ListHostsBySetIds(ctx, []string{blueSet1.PublicId, blueSet2.PublicId})
require.NoError(t, err)
require.Len(t, hosts, 2)
// Base case the count by catalog ID
hosts, _, err = repo.ListHostsByCatalogId(ctx, blueCat.PublicId)
require.NoError(t, err)
assert.Len(t, hosts, 4)
j, err := newOrphanedHostCleanupJob(ctx, rw, rw, kms)
require.NoError(t, err)
// Delete first membership, validate host is gone
got, err := NewHostSetMember(ctx, blueSet1.PublicId, blueHost1.PublicId)
require.NoError(t, err)
require.NotNil(t, got)
num, err := rw.Delete(ctx, got)
require.NoError(t, err)
assert.Equal(t, 1, num)
count, err := j.deleteOrphanedHosts(ctx)
require.NoError(t, err)
assert.Equal(t, 1, count)
assert.NoError(t, db.TestVerifyOplog(t, rw, blueHost1.PublicId, db.WithOperation(oplog.OpType_OP_TYPE_DELETE), db.WithCreateNotBefore(10*time.Second)))
hosts, _, err = repo.ListHostsByCatalogId(ctx, blueCat.PublicId)
require.NoError(t, err)
require.Len(t, hosts, 3)
// Delete second, validate second host is gone
got, err = NewHostSetMember(ctx, blueSet2.PublicId, blueHost2.PublicId)
require.NoError(t, err)
require.NotNil(t, got)
num, err = rw.Delete(ctx, got)
require.NoError(t, err)
assert.Equal(t, 1, num)
count, err = j.deleteOrphanedHosts(ctx)
require.NoError(t, err)
assert.Equal(t, 1, count)
assert.NoError(t, db.TestVerifyOplog(t, rw, blueHost2.PublicId, db.WithOperation(oplog.OpType_OP_TYPE_DELETE), db.WithCreateNotBefore(10*time.Second)))
hosts, _, err = repo.ListHostsByCatalogId(ctx, blueCat.PublicId)
require.NoError(t, err)
require.Len(t, hosts, 2)
// Delete third set, validate remaining hosts are gone
gotSet, err := NewHostSet(ctx, blueCat.PublicId)
require.NoError(t, err)
require.NotNil(t, got)
gotSet.PublicId = blueSet3.PublicId
num, err = rw.Delete(ctx, gotSet)
require.NoError(t, err)
assert.Equal(t, 1, num)
count, err = j.deleteOrphanedHosts(ctx)
require.NoError(t, err)
assert.Equal(t, 2, count)
assert.NoError(t, db.TestVerifyOplog(t, rw, blueHost3.PublicId, db.WithOperation(oplog.OpType_OP_TYPE_DELETE), db.WithCreateNotBefore(10*time.Second)))
assert.NoError(t, db.TestVerifyOplog(t, rw, blueHost4.PublicId, db.WithOperation(oplog.OpType_OP_TYPE_DELETE), db.WithCreateNotBefore(10*time.Second)))
hosts, _, err = repo.ListHostsByCatalogId(ctx, blueCat.PublicId)
require.NoError(t, err)
require.Len(t, hosts, 0)
}
func TestHostSetMember_SetTableName(t *testing.T) {
defaultTableName := "host_plugin_set_member"
tests := []struct {
name string
initialName string
setNameTo string
want string
}{
{
name: "new-name",
initialName: "",
setNameTo: "new-name",
want: "new-name",
},
{
name: "reset to default",
initialName: "initial",
setNameTo: "",
want: defaultTableName,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
assert, require := assert.New(t), require.New(t)
def := &HostSetMember{
HostSetMember: &store.HostSetMember{},
}
require.Equal(defaultTableName, def.TableName())
s := &HostSetMember{
HostSetMember: &store.HostSetMember{},
tableName: tt.initialName,
}
s.SetTableName(tt.setNameTo)
assert.Equal(tt.want, s.TableName())
})
}
}