resource: block default namespace deletion + test refactorings (#19822)

pull/19827/head
Semir Patel 2023-12-05 13:00:06 -06:00 committed by GitHub
parent aca8a185ca
commit c1bbda8128
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
42 changed files with 978 additions and 761 deletions

View File

@ -7,29 +7,29 @@ import (
"context" "context"
"testing" "testing"
"github.com/hashicorp/go-hclog"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"google.golang.org/grpc/codes" "google.golang.org/grpc/codes"
"google.golang.org/grpc/status" "google.golang.org/grpc/status"
"google.golang.org/protobuf/types/known/structpb" "google.golang.org/protobuf/types/known/structpb"
"github.com/hashicorp/go-hclog"
"github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/acl/resolver"
external "github.com/hashicorp/consul/agent/grpc-external"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/internal/catalog" "github.com/hashicorp/consul/internal/catalog"
"github.com/hashicorp/consul/internal/mesh" "github.com/hashicorp/consul/internal/mesh"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/resourcetest" "github.com/hashicorp/consul/internal/resource/resourcetest"
pbcatalog "github.com/hashicorp/consul/proto-public/pbcatalog/v2beta1" pbcatalog "github.com/hashicorp/consul/proto-public/pbcatalog/v2beta1"
"github.com/hashicorp/consul/proto-public/pbdataplane"
pbmesh "github.com/hashicorp/consul/proto-public/pbmesh/v2beta1" pbmesh "github.com/hashicorp/consul/proto-public/pbmesh/v2beta1"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
"github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/acl/resolver"
external "github.com/hashicorp/consul/agent/grpc-external"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/proto-public/pbdataplane"
) )
const ( const (
@ -262,7 +262,9 @@ func TestGetEnvoyBootstrapParams_Success_EnableV2(t *testing.T) {
} }
run := func(t *testing.T, tc testCase) { run := func(t *testing.T, tc testCase) {
resourceClient := svctest.RunResourceService(t, catalog.RegisterTypes, mesh.RegisterTypes) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(catalog.RegisterTypes, mesh.RegisterTypes).
Run(t)
options := structs.QueryOptions{Token: testToken} options := structs.QueryOptions{Token: testToken}
ctx, err := external.ContextWithQueryOptions(context.Background(), options) ctx, err := external.ContextWithQueryOptions(context.Background(), options)
@ -490,7 +492,9 @@ func TestGetEnvoyBootstrapParams_Error_EnableV2(t *testing.T) {
} }
run := func(t *testing.T, tc testCase) { run := func(t *testing.T, tc testCase) {
resourceClient := svctest.RunResourceService(t, catalog.RegisterTypes, mesh.RegisterTypes) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(catalog.RegisterTypes, mesh.RegisterTypes).
Run(t)
options := structs.QueryOptions{Token: testToken} options := structs.QueryOptions{Token: testToken}
ctx, err := external.ContextWithQueryOptions(context.Background(), options) ctx, err := external.ContextWithQueryOptions(context.Background(), options)

View File

@ -19,6 +19,7 @@ import (
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/storage" "github.com/hashicorp/consul/internal/storage"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
pbtenancy "github.com/hashicorp/consul/proto-public/pbtenancy/v2beta1"
) )
// Delete deletes a resource. // Delete deletes a resource.
@ -188,16 +189,13 @@ func (s *Server) ensureDeleteRequestValid(req *pbresource.DeleteRequest) (*resou
return nil, err return nil, err
} }
// Check scope if err := validateScopedTenancy(reg.Scope, reg.Type, req.Id.Tenancy); err != nil {
if reg.Scope == resource.ScopePartition && req.Id.Tenancy.Namespace != "" { return nil, err
return nil, status.Errorf(
codes.InvalidArgument,
"partition scoped resource %s cannot have a namespace. got: %s",
resource.ToGVK(req.Id.Type),
req.Id.Tenancy.Namespace,
)
} }
if err := blockBuiltinsDeletion(reg.Type, req.Id); err != nil {
return nil, err
}
return reg, nil return reg, nil
} }
@ -207,3 +205,12 @@ func TombstoneNameFor(deleteId *pbresource.ID) string {
// deleteId.Name is just included for easier identification // deleteId.Name is just included for easier identification
return fmt.Sprintf("tombstone-%v-%v", deleteId.Name, strings.ToLower(deleteId.Uid)) return fmt.Sprintf("tombstone-%v-%v", deleteId.Name, strings.ToLower(deleteId.Uid))
} }
func blockDefaultNamespaceDeletion(rtype *pbresource.Type, id *pbresource.ID) error {
if id.Name == resource.DefaultNamespaceName &&
id.Tenancy.Partition == resource.DefaultPartitionName &&
resource.EqualType(rtype, pbtenancy.NamespaceType) {
return status.Errorf(codes.InvalidArgument, "cannot delete default namespace")
}
return nil
}

View File

@ -0,0 +1,15 @@
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
//go:build !consulent
package resource
import "github.com/hashicorp/consul/proto-public/pbresource"
func blockBuiltinsDeletion(rtype *pbresource.Type, id *pbresource.ID) error {
if err := blockDefaultNamespaceDeletion(rtype, id); err != nil {
return err
}
return nil
}

View File

@ -1,10 +1,11 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
"fmt"
"strings" "strings"
"testing" "testing"
@ -15,123 +16,158 @@ import (
"google.golang.org/protobuf/proto" "google.golang.org/protobuf/proto"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest" rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
pbtenancy "github.com/hashicorp/consul/proto-public/pbtenancy/v2beta1"
pbdemo "github.com/hashicorp/consul/proto/private/pbdemo/v1" pbdemo "github.com/hashicorp/consul/proto/private/pbdemo/v1"
) )
func TestDelete_InputValidation(t *testing.T) { func TestDelete_InputValidation(t *testing.T) {
server := testServer(t)
client := testClient(t, server)
demo.RegisterTypes(server.Registry)
type testCase struct { type testCase struct {
modFn func(artistId, recordLabelId *pbresource.ID) *pbresource.ID modFn func(artistId, recordLabelId, executiveId *pbresource.ID) *pbresource.ID
errContains string errContains string
} }
run := func(t *testing.T, client pbresource.ResourceServiceClient, tc testCase) {
executive, err := demo.GenerateV1Executive("marvin", "CEO")
require.NoError(t, err)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err)
artist, err := demo.GenerateV2Artist()
require.NoError(t, err)
req := &pbresource.DeleteRequest{Id: tc.modFn(artist.Id, recordLabel.Id, executive.Id), Version: ""}
_, err = client.Delete(context.Background(), req)
require.Error(t, err)
require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String())
require.ErrorContains(t, err, tc.errContains)
}
testCases := map[string]testCase{ testCases := map[string]testCase{
"no id": { "no id": {
modFn: func(_, _ *pbresource.ID) *pbresource.ID { modFn: func(_, _, _ *pbresource.ID) *pbresource.ID {
return nil return nil
}, },
errContains: "id is required", errContains: "id is required",
}, },
"no type": { "no type": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Type = nil artistId.Type = nil
return artistId return artistId
}, },
errContains: "id.type is required", errContains: "id.type is required",
}, },
"no name": { "no name": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Name = "" artistId.Name = ""
return artistId return artistId
}, },
errContains: "id.name invalid", errContains: "id.name invalid",
}, },
"mixed case name": { "mixed case name": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Name = "DepecheMode" artistId.Name = "DepecheMode"
return artistId return artistId
}, },
errContains: "id.name invalid", errContains: "id.name invalid",
}, },
"name too long": { "name too long": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Name = strings.Repeat("n", resource.MaxNameLength+1) artistId.Name = strings.Repeat("n", resource.MaxNameLength+1)
return artistId return artistId
}, },
errContains: "id.name invalid", errContains: "id.name invalid",
}, },
"partition mixed case": { "partition mixed case": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Tenancy.Partition = "Default" artistId.Tenancy.Partition = "Default"
return artistId return artistId
}, },
errContains: "id.tenancy.partition invalid", errContains: "id.tenancy.partition invalid",
}, },
"partition name too long": { "partition name too long": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Tenancy.Partition = strings.Repeat("p", resource.MaxNameLength+1) artistId.Tenancy.Partition = strings.Repeat("p", resource.MaxNameLength+1)
return artistId return artistId
}, },
errContains: "id.tenancy.partition invalid", errContains: "id.tenancy.partition invalid",
}, },
"namespace mixed case": { "namespace mixed case": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Tenancy.Namespace = "Default" artistId.Tenancy.Namespace = "Default"
return artistId return artistId
}, },
errContains: "id.tenancy.namespace invalid", errContains: "id.tenancy.namespace invalid",
}, },
"namespace name too long": { "namespace name too long": {
modFn: func(artistId, _ *pbresource.ID) *pbresource.ID { modFn: func(artistId, _, _ *pbresource.ID) *pbresource.ID {
artistId.Tenancy.Namespace = strings.Repeat("n", resource.MaxNameLength+1) artistId.Tenancy.Namespace = strings.Repeat("n", resource.MaxNameLength+1)
return artistId return artistId
}, },
errContains: "id.tenancy.namespace invalid", errContains: "id.tenancy.namespace invalid",
}, },
"partition scoped resource with namespace": { "partition scoped resource with namespace": {
modFn: func(_, recordLabelId *pbresource.ID) *pbresource.ID { modFn: func(_, recordLabelId, _ *pbresource.ID) *pbresource.ID {
recordLabelId.Tenancy.Namespace = "ishouldnothaveanamespace" recordLabelId.Tenancy.Namespace = "ishouldnothaveanamespace"
return recordLabelId return recordLabelId
}, },
errContains: "cannot have a namespace", errContains: "cannot have a namespace",
}, },
"cluster scoped resource with partition": {
modFn: func(_, _, executiveId *pbresource.ID) *pbresource.ID {
executiveId.Tenancy.Partition = "ishouldnothaveapartition"
executiveId.Tenancy.Namespace = ""
return executiveId
},
errContains: "cannot have a partition",
},
"cluster scoped resource with namespace": {
modFn: func(_, _, executiveId *pbresource.ID) *pbresource.ID {
executiveId.Tenancy.Partition = ""
executiveId.Tenancy.Namespace = "ishouldnothaveanamespace"
return executiveId
},
errContains: "cannot have a namespace",
},
} }
for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) {
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err)
artist, err := demo.GenerateV2Artist() for _, useV2Tenancy := range []bool{false, true} {
require.NoError(t, err) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
client := svctest.NewResourceServiceBuilder().
WithV2Tenancy(useV2Tenancy).
WithRegisterFns(demo.RegisterTypes).
Run(t)
req := &pbresource.DeleteRequest{Id: tc.modFn(artist.Id, recordLabel.Id), Version: ""} for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) {
_, err = client.Delete(testContext(t), req) run(t, client, tc)
require.Error(t, err) })
require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String()) }
require.ErrorContains(t, err, tc.errContains)
}) })
} }
} }
func TestDelete_TypeNotRegistered(t *testing.T) { func TestDelete_TypeNotRegistered(t *testing.T) {
t.Parallel() for _, useV2Tenancy := range []bool{false, true} {
t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
client := svctest.NewResourceServiceBuilder().WithV2Tenancy(useV2Tenancy).Run(t)
_, client, ctx := testDeps(t) artist, err := demo.GenerateV2Artist()
artist, err := demo.GenerateV2Artist() require.NoError(t, err)
require.NoError(t, err)
// delete artist with unregistered type // delete artist with unregistered type
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: artist.Id, Version: ""}) _, err = client.Delete(context.Background(), &pbresource.DeleteRequest{Id: artist.Id, Version: ""})
require.Error(t, err) require.Error(t, err)
require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String()) require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String())
require.ErrorContains(t, err, "not registered")
})
}
} }
func TestDelete_ACLs(t *testing.T) { func TestDelete_ACLs(t *testing.T) {
@ -157,9 +193,8 @@ func TestDelete_ACLs(t *testing.T) {
for desc, tc := range testcases { for desc, tc := range testcases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) builder := svctest.NewResourceServiceBuilder().WithRegisterFns(demo.RegisterTypes)
client := testClient(t, server) client := builder.Run(t)
demo.RegisterTypes(server.Registry)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -169,10 +204,10 @@ func TestDelete_ACLs(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
// Mock is put in place after the above "write" since the "write" must also pass the ACL check. // Mock is put in place after the above "write" since the "write" must also pass the ACL check.
mockACLResolver := &MockACLResolver{} mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(tc.authz, nil) Return(tc.authz, nil)
server.ACLResolver = mockACLResolver builder.ServiceImpl().Config.ACLResolver = mockACLResolver
// Exercise ACL. // Exercise ACL.
_, err = client.Delete(testContext(t), &pbresource.DeleteRequest{Id: rsp.Resource.Id}) _, err = client.Delete(testContext(t), &pbresource.DeleteRequest{Id: rsp.Resource.Id})
@ -184,59 +219,70 @@ func TestDelete_ACLs(t *testing.T) {
func TestDelete_Success(t *testing.T) { func TestDelete_Success(t *testing.T) {
t.Parallel() t.Parallel()
run := func(t *testing.T, client pbresource.ResourceServiceClient, tc deleteTestCase, modFn func(artistId, recordlabelId *pbresource.ID) *pbresource.ID) {
ctx := context.Background()
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err)
writeRsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: recordLabel})
require.NoError(t, err)
recordLabel = writeRsp.Resource
originalRecordLabelId := clone(recordLabel.Id)
artist, err := demo.GenerateV2Artist()
require.NoError(t, err)
writeRsp, err = client.Write(ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err)
artist = writeRsp.Resource
originalArtistId := clone(artist.Id)
// Pick the resource to be deleted based on type's scope and mod tenancy
// based on the tenancy test case.
deleteId := modFn(artist.Id, recordLabel.Id)
deleteReq := tc.deleteReqFn(recordLabel)
if proto.Equal(deleteId.Type, demo.TypeV2Artist) {
deleteReq = tc.deleteReqFn(artist)
}
// Delete
_, err = client.Delete(ctx, deleteReq)
require.NoError(t, err)
// Verify deleted
_, err = client.Read(ctx, &pbresource.ReadRequest{Id: deleteId})
require.Error(t, err)
require.Equal(t, codes.NotFound.String(), status.Code(err).String())
// Derive tombstone name from resource that was deleted.
tname := svc.TombstoneNameFor(originalRecordLabelId)
if proto.Equal(deleteId.Type, demo.TypeV2Artist) {
tname = svc.TombstoneNameFor(originalArtistId)
}
// Verify tombstone created
_, err = client.Read(ctx, &pbresource.ReadRequest{
Id: &pbresource.ID{
Name: tname,
Type: resource.TypeV1Tombstone,
Tenancy: deleteReq.Id.Tenancy,
},
})
require.NoError(t, err, "expected tombstone to be found")
}
for desc, tc := range deleteTestCases() { for desc, tc := range deleteTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
for tenancyDesc, modFn := range tenancyCases() { for tenancyDesc, modFn := range tenancyCases() {
t.Run(tenancyDesc, func(t *testing.T) { t.Run(tenancyDesc, func(t *testing.T) {
server, client, ctx := testDeps(t) for _, useV2Tenancy := range []bool{false, true} {
demo.RegisterTypes(server.Registry) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
client := svctest.NewResourceServiceBuilder().
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") WithV2Tenancy(useV2Tenancy).
require.NoError(t, err) WithRegisterFns(demo.RegisterTypes).
writeRsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: recordLabel}) Run(t)
require.NoError(t, err) run(t, client, tc, modFn)
recordLabel = writeRsp.Resource })
originalRecordLabelId := clone(recordLabel.Id)
artist, err := demo.GenerateV2Artist()
require.NoError(t, err)
writeRsp, err = client.Write(ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err)
artist = writeRsp.Resource
originalArtistId := clone(artist.Id)
// Pick the resource to be deleted based on type's scope and mod tenancy
// based on the tenancy test case.
deleteId := modFn(artist.Id, recordLabel.Id)
deleteReq := tc.deleteReqFn(recordLabel)
if proto.Equal(deleteId.Type, demo.TypeV2Artist) {
deleteReq = tc.deleteReqFn(artist)
} }
// Delete
_, err = client.Delete(ctx, deleteReq)
require.NoError(t, err)
// Verify deleted
_, err = client.Read(ctx, &pbresource.ReadRequest{Id: deleteId})
require.Error(t, err)
require.Equal(t, codes.NotFound.String(), status.Code(err).String())
// Derive tombstone name from resource that was deleted.
tname := TombstoneNameFor(originalRecordLabelId)
if proto.Equal(deleteId.Type, demo.TypeV2Artist) {
tname = TombstoneNameFor(originalArtistId)
}
// Verify tombstone created
_, err = client.Read(ctx, &pbresource.ReadRequest{
Id: &pbresource.ID{
Name: tname,
Type: resource.TypeV1Tombstone,
Tenancy: deleteReq.Id.Tenancy,
},
})
require.NoError(t, err, "expected tombstone to be found")
}) })
} }
}) })
@ -246,54 +292,72 @@ func TestDelete_Success(t *testing.T) {
func TestDelete_TombstoneDeletionDoesNotCreateNewTombstone(t *testing.T) { func TestDelete_TombstoneDeletionDoesNotCreateNewTombstone(t *testing.T) {
t.Parallel() t.Parallel()
server, client, ctx := testDeps(t) for _, useV2Tenancy := range []bool{false, true} {
demo.RegisterTypes(server.Registry) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
ctx := context.Background()
client := svctest.NewResourceServiceBuilder().
WithV2Tenancy(useV2Tenancy).
WithRegisterFns(demo.RegisterTypes).
Run(t)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
rsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: artist}) rsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
artist = rsp.Resource artist = rsp.Resource
// delete artist // delete artist
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: artist.Id, Version: ""}) _, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: artist.Id, Version: ""})
require.NoError(t, err) require.NoError(t, err)
// verify artist's tombstone created // verify artist's tombstone created
rsp2, err := client.Read(ctx, &pbresource.ReadRequest{ rsp2, err := client.Read(ctx, &pbresource.ReadRequest{
Id: &pbresource.ID{ Id: &pbresource.ID{
Name: TombstoneNameFor(artist.Id), Name: svc.TombstoneNameFor(artist.Id),
Type: resource.TypeV1Tombstone, Type: resource.TypeV1Tombstone,
Tenancy: artist.Id.Tenancy, Tenancy: artist.Id.Tenancy,
}, },
}) })
require.NoError(t, err) require.NoError(t, err)
tombstone := rsp2.Resource tombstone := rsp2.Resource
// delete artist's tombstone // delete artist's tombstone
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: tombstone.Id, Version: tombstone.Version}) _, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: tombstone.Id, Version: tombstone.Version})
require.NoError(t, err) require.NoError(t, err)
// verify no new tombstones created and artist's existing tombstone deleted // verify no new tombstones created and artist's existing tombstone deleted
rsp3, err := client.List(ctx, &pbresource.ListRequest{Type: resource.TypeV1Tombstone, Tenancy: artist.Id.Tenancy}) rsp3, err := client.List(ctx, &pbresource.ListRequest{Type: resource.TypeV1Tombstone, Tenancy: artist.Id.Tenancy})
require.NoError(t, err) require.NoError(t, err)
require.Empty(t, rsp3.Resources) require.Empty(t, rsp3.Resources)
})
}
} }
func TestDelete_NotFound(t *testing.T) { func TestDelete_NotFound(t *testing.T) {
t.Parallel() t.Parallel()
for desc, tc := range deleteTestCases() { run := func(t *testing.T, client pbresource.ResourceServiceClient, tc deleteTestCase) {
t.Run(desc, func(t *testing.T) { artist, err := demo.GenerateV2Artist()
server, client, ctx := testDeps(t) require.NoError(t, err)
demo.RegisterTypes(server.Registry)
artist, err := demo.GenerateV2Artist()
require.NoError(t, err)
// verify delete of non-existant or already deleted resource is a no-op // verify delete of non-existant or already deleted resource is a no-op
_, err = client.Delete(ctx, tc.deleteReqFn(artist)) _, err = client.Delete(context.Background(), tc.deleteReqFn(artist))
require.NoError(t, err) require.NoError(t, err)
}
for _, useV2Tenancy := range []bool{false, true} {
t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
client := svctest.NewResourceServiceBuilder().
WithV2Tenancy(useV2Tenancy).
WithRegisterFns(demo.RegisterTypes).
Run(t)
for desc, tc := range deleteTestCases() {
t.Run(desc, func(t *testing.T) {
run(t, client, tc)
})
}
}) })
} }
} }
@ -301,86 +365,115 @@ func TestDelete_NotFound(t *testing.T) {
func TestDelete_VersionMismatch(t *testing.T) { func TestDelete_VersionMismatch(t *testing.T) {
t.Parallel() t.Parallel()
server, client, ctx := testDeps(t) for _, useV2Tenancy := range []bool{false, true} {
demo.RegisterTypes(server.Registry) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
artist, err := demo.GenerateV2Artist() client := svctest.NewResourceServiceBuilder().
require.NoError(t, err) WithV2Tenancy(useV2Tenancy).
rsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: artist}) WithRegisterFns(demo.RegisterTypes).
require.NoError(t, err) Run(t)
// delete with a version that is different from the stored version artist, err := demo.GenerateV2Artist()
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: rsp.Resource.Id, Version: "non-existent-version"}) require.NoError(t, err)
require.Error(t, err) rsp, err := client.Write(context.Background(), &pbresource.WriteRequest{Resource: artist})
require.Equal(t, codes.Aborted.String(), status.Code(err).String()) require.NoError(t, err)
require.ErrorContains(t, err, "CAS operation failed")
// delete with a version that is different from the stored version
_, err = client.Delete(context.Background(), &pbresource.DeleteRequest{Id: rsp.Resource.Id, Version: "non-existent-version"})
require.Error(t, err)
require.Equal(t, codes.Aborted.String(), status.Code(err).String())
require.ErrorContains(t, err, "CAS operation failed")
})
}
} }
func TestDelete_MarkedForDeletionWhenFinalizersPresent(t *testing.T) { func TestDelete_MarkedForDeletionWhenFinalizersPresent(t *testing.T) {
server, client, ctx := testDeps(t) for _, useV2Tenancy := range []bool{false, true} {
demo.RegisterTypes(server.Registry) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
ctx := context.Background()
client := svctest.NewResourceServiceBuilder().
WithV2Tenancy(useV2Tenancy).
WithRegisterFns(demo.RegisterTypes).
Run(t)
// Create a resource with a finalizer // Create a resource with a finalizer
res := rtest.Resource(demo.TypeV1Artist, "manwithnoname"). res := rtest.Resource(demo.TypeV1Artist, "manwithnoname").
WithTenancy(resource.DefaultClusteredTenancy()). WithTenancy(resource.DefaultClusteredTenancy()).
WithData(t, &pbdemo.Artist{Name: "Man With No Name"}). WithData(t, &pbdemo.Artist{Name: "Man With No Name"}).
WithMeta(resource.FinalizerKey, "finalizer1"). WithMeta(resource.FinalizerKey, "finalizer1").
Write(t, client) Write(t, client)
// Delete it // Delete it
_, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id}) _, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Verify resource has been marked for deletion // Verify resource has been marked for deletion
rsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id}) rsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
require.True(t, resource.IsMarkedForDeletion(rsp.Resource)) require.True(t, resource.IsMarkedForDeletion(rsp.Resource))
// Delete again - should be no-op // Delete again - should be no-op
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id}) _, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Verify no-op by checking version still the same // Verify no-op by checking version still the same
rsp2, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id}) rsp2, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
rtest.RequireVersionUnchanged(t, rsp2.Resource, rsp.Resource.Version) rtest.RequireVersionUnchanged(t, rsp2.Resource, rsp.Resource.Version)
})
}
} }
func TestDelete_ImmediatelyDeletedAfterFinalizersRemoved(t *testing.T) { func TestDelete_ImmediatelyDeletedAfterFinalizersRemoved(t *testing.T) {
server, client, ctx := testDeps(t) for _, useV2Tenancy := range []bool{false, true} {
demo.RegisterTypes(server.Registry) t.Run(fmt.Sprintf("v2tenancy %v", useV2Tenancy), func(t *testing.T) {
ctx := context.Background()
client := svctest.NewResourceServiceBuilder().
WithV2Tenancy(useV2Tenancy).
WithRegisterFns(demo.RegisterTypes).
Run(t)
// Create a resource with a finalizer // Create a resource with a finalizer
res := rtest.Resource(demo.TypeV1Artist, "manwithnoname"). res := rtest.Resource(demo.TypeV1Artist, "manwithnoname").
WithTenancy(resource.DefaultClusteredTenancy()). WithTenancy(resource.DefaultClusteredTenancy()).
WithData(t, &pbdemo.Artist{Name: "Man With No Name"}). WithData(t, &pbdemo.Artist{Name: "Man With No Name"}).
WithMeta(resource.FinalizerKey, "finalizer1"). WithMeta(resource.FinalizerKey, "finalizer1").
Write(t, client) Write(t, client)
// Delete should mark it for deletion // Delete should mark it for deletion
_, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id}) _, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Remove the finalizer // Remove the finalizer
rsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id}) rsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
resource.RemoveFinalizer(rsp.Resource, "finalizer1") resource.RemoveFinalizer(rsp.Resource, "finalizer1")
_, err = client.Write(ctx, &pbresource.WriteRequest{Resource: rsp.Resource}) _, err = client.Write(ctx, &pbresource.WriteRequest{Resource: rsp.Resource})
require.NoError(t, err) require.NoError(t, err)
// Delete should be immediate // Delete should be immediate
_, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: rsp.Resource.Id}) _, err = client.Delete(ctx, &pbresource.DeleteRequest{Id: rsp.Resource.Id})
require.NoError(t, err) require.NoError(t, err)
// Verify deleted // Verify deleted
_, err = client.Read(ctx, &pbresource.ReadRequest{Id: rsp.Resource.Id}) _, err = client.Read(ctx, &pbresource.ReadRequest{Id: rsp.Resource.Id})
require.Error(t, err) require.Error(t, err)
require.Equal(t, codes.NotFound.String(), status.Code(err).String()) require.Equal(t, codes.NotFound.String(), status.Code(err).String())
})
}
} }
func testDeps(t *testing.T) (*Server, pbresource.ResourceServiceClient, context.Context) { func TestDelete_BlockDeleteDefaultNamespace(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().WithV2Tenancy(true).Run(t)
client := testClient(t, server)
return server, client, context.Background() id := &pbresource.ID{
Name: resource.DefaultNamespaceName,
Type: pbtenancy.NamespaceType,
Tenancy: &pbresource.Tenancy{Partition: resource.DefaultPartitionName},
}
_, err := client.Delete(context.Background(), &pbresource.DeleteRequest{Id: id})
require.Error(t, err)
require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String())
require.ErrorContains(t, err, "cannot delete default namespace")
} }
type deleteTestCase struct { type deleteTestCase struct {

View File

@ -1,7 +1,7 @@
// // Copyright (c) HashiCorp, Inc. // // Copyright (c) HashiCorp, Inc.
// // SPDX-License-Identifier: BUSL-1.1 // // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -17,6 +17,8 @@ import (
"google.golang.org/protobuf/proto" "google.golang.org/protobuf/proto"
"github.com/hashicorp/consul/acl" "github.com/hashicorp/consul/acl"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/internal/resource/resourcetest" "github.com/hashicorp/consul/internal/resource/resourcetest"
@ -25,10 +27,12 @@ import (
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestListByOwner_InputValidation(t *testing.T) { func TestListByOwner_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
type testCase struct { type testCase struct {
modFn func(artistId, recordlabelId, executiveId *pbresource.ID) *pbresource.ID modFn func(artistId, recordlabelId, executiveId *pbresource.ID) *pbresource.ID
@ -152,8 +156,7 @@ func TestListByOwner_InputValidation(t *testing.T) {
} }
func TestListByOwner_TypeNotRegistered(t *testing.T) { func TestListByOwner_TypeNotRegistered(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().Run(t)
client := testClient(t, server)
_, err := client.ListByOwner(context.Background(), &pbresource.ListByOwnerRequest{ _, err := client.ListByOwner(context.Background(), &pbresource.ListByOwnerRequest{
Owner: &pbresource.ID{ Owner: &pbresource.ID{
@ -169,9 +172,9 @@ func TestListByOwner_TypeNotRegistered(t *testing.T) {
} }
func TestListByOwner_Empty(t *testing.T) { func TestListByOwner_Empty(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -185,9 +188,9 @@ func TestListByOwner_Empty(t *testing.T) {
} }
func TestListByOwner_Many(t *testing.T) { func TestListByOwner_Many(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -245,9 +248,9 @@ func TestListByOwner_OwnerTenancyDoesNotExist(t *testing.T) {
} }
for desc, tc := range tenancyCases { for desc, tc := range tenancyCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
recordLabel := resourcetest.Resource(demo.TypeV1RecordLabel, "looney-tunes"). recordLabel := resourcetest.Resource(demo.TypeV1RecordLabel, "looney-tunes").
WithTenancy(resource.DefaultPartitionedTenancy()). WithTenancy(resource.DefaultPartitionedTenancy()).
@ -271,9 +274,9 @@ func TestListByOwner_OwnerTenancyDoesNotExist(t *testing.T) {
func TestListByOwner_Tenancy_Defaults_And_Normalization(t *testing.T) { func TestListByOwner_Tenancy_Defaults_And_Normalization(t *testing.T) {
for tenancyDesc, modFn := range tenancyCases() { for tenancyDesc, modFn := range tenancyCases() {
t.Run(tenancyDesc, func(t *testing.T) { t.Run(tenancyDesc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
// Create partition scoped recordLabel. // Create partition scoped recordLabel.
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
@ -343,9 +346,9 @@ func TestListByOwner_ACL_PerTypeAllowed(t *testing.T) {
// roundtrip a ListByOwner which attempts to return a single resource // roundtrip a ListByOwner which attempts to return a single resource
func roundTripListByOwner(t *testing.T, authz acl.Authorizer) (*pbresource.Resource, *pbresource.ListByOwnerResponse, error) { func roundTripListByOwner(t *testing.T, authz acl.Authorizer) (*pbresource.Resource, *pbresource.ListByOwnerResponse, error) {
server := testServer(t) builder := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes)
demo.RegisterTypes(server.Registry) client := builder.Run(t)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -361,10 +364,11 @@ func roundTripListByOwner(t *testing.T, authz acl.Authorizer) (*pbresource.Resou
album = rsp2.Resource album = rsp2.Resource
require.NoError(t, err) require.NoError(t, err)
mockACLResolver := &MockACLResolver{} // Mock has to be put in place after the above writes so writes will succeed.
mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(authz, nil) Return(authz, nil)
server.ACLResolver = mockACLResolver builder.ServiceImpl().ACLResolver = mockACLResolver
rsp3, err := client.ListByOwner(testContext(t), &pbresource.ListByOwnerRequest{Owner: artist.Id}) rsp3, err := client.ListByOwner(testContext(t), &pbresource.ListByOwnerRequest{Owner: artist.Id})
return album, rsp3, err return album, rsp3, err

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -10,25 +10,29 @@ import (
"strings" "strings"
"testing" "testing"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
"github.com/hashicorp/consul/acl" "github.com/hashicorp/consul/acl"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/agent/grpc-external/testutils" "github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/internal/storage" "github.com/hashicorp/consul/internal/storage"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestList_InputValidation(t *testing.T) { func TestList_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
type testCase struct { type testCase struct {
modReqFn func(req *pbresource.ListRequest) modReqFn func(req *pbresource.ListRequest)
@ -93,8 +97,7 @@ func TestList_InputValidation(t *testing.T) {
} }
func TestList_TypeNotFound(t *testing.T) { func TestList_TypeNotFound(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().Run(t)
client := testClient(t, server)
_, err := client.List(context.Background(), &pbresource.ListRequest{ _, err := client.List(context.Background(), &pbresource.ListRequest{
Type: demo.TypeV2Artist, Type: demo.TypeV2Artist,
@ -109,9 +112,9 @@ func TestList_TypeNotFound(t *testing.T) {
func TestList_Empty(t *testing.T) { func TestList_Empty(t *testing.T) {
for desc, tc := range listTestCases() { for desc, tc := range listTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
rsp, err := client.List(tc.ctx, &pbresource.ListRequest{ rsp, err := client.List(tc.ctx, &pbresource.ListRequest{
Type: demo.TypeV1Artist, Type: demo.TypeV1Artist,
@ -127,9 +130,9 @@ func TestList_Empty(t *testing.T) {
func TestList_Many(t *testing.T) { func TestList_Many(t *testing.T) {
for desc, tc := range listTestCases() { for desc, tc := range listTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
resources := make([]*pbresource.Resource, 10) resources := make([]*pbresource.Resource, 10)
for i := 0; i < len(resources); i++ { for i := 0; i < len(resources); i++ {
@ -159,9 +162,9 @@ func TestList_Many(t *testing.T) {
func TestList_NamePrefix(t *testing.T) { func TestList_NamePrefix(t *testing.T) {
for desc, tc := range listTestCases() { for desc, tc := range listTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
expectedResources := []*pbresource.Resource{} expectedResources := []*pbresource.Resource{}
@ -201,9 +204,9 @@ func TestList_Tenancy_Defaults_And_Normalization(t *testing.T) {
ctx := context.Background() ctx := context.Background()
for desc, tc := range wildcardTenancyCases() { for desc, tc := range wildcardTenancyCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
// Write partition scoped record label // Write partition scoped record label
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
@ -236,14 +239,14 @@ func TestList_Tenancy_Defaults_And_Normalization(t *testing.T) {
func TestList_GroupVersionMismatch(t *testing.T) { func TestList_GroupVersionMismatch(t *testing.T) {
for desc, tc := range listTestCases() { for desc, tc := range listTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
_, err = server.Backend.WriteCAS(tc.ctx, artist) _, err = client.Write(tc.ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
rsp, err := client.List(tc.ctx, &pbresource.ListRequest{ rsp, err := client.List(tc.ctx, &pbresource.ListRequest{
@ -261,7 +264,7 @@ func TestList_VerifyReadConsistencyArg(t *testing.T) {
// Uses a mockBackend instead of the inmem Backend to verify the ReadConsistency argument is set correctly. // Uses a mockBackend instead of the inmem Backend to verify the ReadConsistency argument is set correctly.
for desc, tc := range listTestCases() { for desc, tc := range listTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
mockBackend := NewMockBackend(t) mockBackend := svc.NewMockBackend(t)
server := testServer(t) server := testServer(t)
server.Backend = mockBackend server.Backend = mockBackend
demo.RegisterTypes(server.Registry) demo.RegisterTypes(server.Registry)
@ -322,25 +325,24 @@ func TestList_ACL_ListAllowed_ReadAllowed(t *testing.T) {
prototest.AssertDeepEqual(t, artist, rsp.Resources[0]) prototest.AssertDeepEqual(t, artist, rsp.Resources[0])
} }
// roundtrip a List which attempts to return a single resource
func roundTripList(t *testing.T, authz acl.Authorizer) (*pbresource.Resource, *pbresource.ListResponse, error) { func roundTripList(t *testing.T, authz acl.Authorizer) (*pbresource.Resource, *pbresource.ListResponse, error) {
server := testServer(t)
client := testClient(t, server)
ctx := testContext(t) ctx := testContext(t)
builder := svctest.NewResourceServiceBuilder().WithRegisterFns(demo.RegisterTypes)
mockACLResolver := &MockACLResolver{} client := builder.Run(t)
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(authz, nil)
server.ACLResolver = mockACLResolver
demo.RegisterTypes(server.Registry)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
artist, err = server.Backend.WriteCAS(ctx, artist) rsp1, err := client.Write(ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
rsp, err := client.List( // Put ACLResolver in place after above writes so writes not subject to ACLs
mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(authz, nil)
builder.ServiceImpl().Config.ACLResolver = mockACLResolver
rsp2, err := client.List(
ctx, ctx,
&pbresource.ListRequest{ &pbresource.ListRequest{
Type: artist.Id.Type, Type: artist.Id.Type,
@ -348,8 +350,7 @@ func roundTripList(t *testing.T, authz acl.Authorizer) (*pbresource.Resource, *p
NamePrefix: "", NamePrefix: "",
}, },
) )
return rsp1.Resource, rsp2, err
return artist, rsp, err
} }
type listTestCase struct { type listTestCase struct {

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -19,21 +19,23 @@ import (
"github.com/hashicorp/consul/acl" "github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/agent/grpc-external/testutils" "github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/internal/storage" "github.com/hashicorp/consul/internal/storage"
"github.com/hashicorp/consul/internal/tenancy"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
"github.com/hashicorp/consul/sdk/testutil" "github.com/hashicorp/consul/sdk/testutil"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestRead_InputValidation(t *testing.T) { func TestRead_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
tenancy.RegisterTypes(server.Registry) Run(t)
demo.RegisterTypes(server.Registry)
type testCase struct { type testCase struct {
modFn func(artistId, recordlabelId, executiveId *pbresource.ID) *pbresource.ID modFn func(artistId, recordlabelId, executiveId *pbresource.ID) *pbresource.ID
@ -148,7 +150,7 @@ func TestRead_InputValidation(t *testing.T) {
} }
func TestRead_TypeNotFound(t *testing.T) { func TestRead_TypeNotFound(t *testing.T) {
server := NewServer(Config{Registry: resource.NewRegistry()}) server := svc.NewServer(svc.Config{Registry: resource.NewRegistry()})
client := testClient(t, server) client := testClient(t, server)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
@ -202,18 +204,19 @@ func TestRead_ResourceNotFound(t *testing.T) {
} }
for tenancyDesc, tenancyCase := range tenancyCases { for tenancyDesc, tenancyCase := range tenancyCases {
t.Run(tenancyDesc, func(t *testing.T) { t.Run(tenancyDesc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithV2Tenancy(true).
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err) require.NoError(t, err)
recordLabel, err = server.Backend.WriteCAS(tc.ctx, recordLabel) _, err = client.Write(context.Background(), &pbresource.WriteRequest{Resource: recordLabel})
require.NoError(t, err) require.NoError(t, err)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
artist, err = server.Backend.WriteCAS(tc.ctx, artist) _, err = client.Write(context.Background(), &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
// Each tenancy test case picks which resource to use based on the resource type's scope. // Each tenancy test case picks which resource to use based on the resource type's scope.
@ -230,15 +233,14 @@ func TestRead_ResourceNotFound(t *testing.T) {
func TestRead_GroupVersionMismatch(t *testing.T) { func TestRead_GroupVersionMismatch(t *testing.T) {
for desc, tc := range readTestCases() { for desc, tc := range readTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
client := testClient(t, server)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
_, err = server.Backend.WriteCAS(tc.ctx, artist) _, err = client.Write(tc.ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
id := clone(artist.Id) id := clone(artist.Id)
@ -257,18 +259,20 @@ func TestRead_Success(t *testing.T) {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
for tenancyDesc, modFn := range tenancyCases() { for tenancyDesc, modFn := range tenancyCases() {
t.Run(tenancyDesc, func(t *testing.T) { t.Run(tenancyDesc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
client := testClient(t, server) Run(t)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err) require.NoError(t, err)
recordLabel, err = server.Backend.WriteCAS(tc.ctx, recordLabel) rsp1, err := client.Write(tc.ctx, &pbresource.WriteRequest{Resource: recordLabel})
recordLabel = rsp1.Resource
require.NoError(t, err) require.NoError(t, err)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
artist, err = server.Backend.WriteCAS(tc.ctx, artist) rsp2, err := client.Write(tc.ctx, &pbresource.WriteRequest{Resource: artist})
artist = rsp2.Resource
require.NoError(t, err) require.NoError(t, err)
// Each tenancy test case picks which resource to use based on the resource type's scope. // Each tenancy test case picks which resource to use based on the resource type's scope.
@ -295,7 +299,7 @@ func TestRead_VerifyReadConsistencyArg(t *testing.T) {
for desc, tc := range readTestCases() { for desc, tc := range readTestCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) server := testServer(t)
mockBackend := NewMockBackend(t) mockBackend := svc.NewMockBackend(t)
server.Backend = mockBackend server.Backend = mockBackend
demo.RegisterTypes(server.Registry) demo.RegisterTypes(server.Registry)
@ -364,15 +368,14 @@ func TestRead_ACLs(t *testing.T) {
for desc, tc := range testcases { for desc, tc := range testcases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t)
client := testClient(t, server)
dr := &dummyACLResolver{ dr := &dummyACLResolver{
result: testutils.ACLsDisabled(t), result: testutils.ACLsDisabled(t),
} }
server.ACLResolver = dr
demo.RegisterTypes(server.Registry) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
WithACLResolver(dr).
Run(t)
dr.SetResult(tc.authz) dr.SetResult(tc.authz)
testutil.RunStep(t, "does not exist", func(t *testing.T) { testutil.RunStep(t, "does not exist", func(t *testing.T) {
@ -410,7 +413,7 @@ type dummyACLResolver struct {
result resolver.Result result resolver.Result
} }
var _ ACLResolver = (*dummyACLResolver)(nil) var _ svc.ACLResolver = (*dummyACLResolver)(nil)
func (r *dummyACLResolver) SetResult(result resolver.Result) { func (r *dummyACLResolver) SetResult(result resolver.Result) {
r.lock.Lock() r.lock.Lock()

View File

@ -3,7 +3,7 @@
//go:build !consulent //go:build !consulent
package resource package resource_test
import "github.com/hashicorp/consul/acl" import "github.com/hashicorp/consul/acl"

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -11,12 +11,14 @@ import (
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"google.golang.org/grpc" "google.golang.org/grpc"
"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/types/known/anypb" "google.golang.org/protobuf/types/known/anypb"
"github.com/hashicorp/go-uuid" "github.com/hashicorp/go-uuid"
"github.com/hashicorp/consul/acl" "github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
"github.com/hashicorp/consul/agent/grpc-external/testutils" "github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
@ -51,7 +53,8 @@ func AuthorizerFrom(t *testing.T, policyStrs ...string) resolver.Result {
} }
} }
func testServer(t *testing.T) *Server { // Deprecated: use NewResourceServiceBuilder instead
func testServer(t *testing.T) *svc.Server {
t.Helper() t.Helper()
backend, err := inmem.NewBackend() backend, err := inmem.NewBackend()
@ -59,7 +62,7 @@ func testServer(t *testing.T) *Server {
go backend.Run(testContext(t)) go backend.Run(testContext(t))
// Mock the ACL Resolver to "allow all" for testing. // Mock the ACL Resolver to "allow all" for testing.
mockACLResolver := &MockACLResolver{} mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(testutils.ACLsDisabled(t), nil). Return(testutils.ACLsDisabled(t), nil).
Run(func(args mock.Arguments) { Run(func(args mock.Arguments) {
@ -76,7 +79,7 @@ func testServer(t *testing.T) *Server {
}) })
// Mock the tenancy bridge since we can't use the real thing. // Mock the tenancy bridge since we can't use the real thing.
mockTenancyBridge := &MockTenancyBridge{} mockTenancyBridge := &svc.MockTenancyBridge{}
mockTenancyBridge.On("PartitionExists", resource.DefaultPartitionName).Return(true, nil) mockTenancyBridge.On("PartitionExists", resource.DefaultPartitionName).Return(true, nil)
mockTenancyBridge.On("NamespaceExists", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(true, nil) mockTenancyBridge.On("NamespaceExists", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(true, nil)
mockTenancyBridge.On("PartitionExists", mock.Anything).Return(false, nil) mockTenancyBridge.On("PartitionExists", mock.Anything).Return(false, nil)
@ -84,7 +87,7 @@ func testServer(t *testing.T) *Server {
mockTenancyBridge.On("IsPartitionMarkedForDeletion", resource.DefaultPartitionName).Return(false, nil) mockTenancyBridge.On("IsPartitionMarkedForDeletion", resource.DefaultPartitionName).Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(false, nil) mockTenancyBridge.On("IsNamespaceMarkedForDeletion", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(false, nil)
return NewServer(Config{ return svc.NewServer(svc.Config{
Logger: testutil.Logger(t), Logger: testutil.Logger(t),
Registry: resource.NewRegistry(), Registry: resource.NewRegistry(),
Backend: backend, Backend: backend,
@ -93,7 +96,8 @@ func testServer(t *testing.T) *Server {
}) })
} }
func testClient(t *testing.T, server *Server) pbresource.ResourceServiceClient { // Deprecated: use NewResourceServiceBuilder instead
func testClient(t *testing.T, server *svc.Server) pbresource.ResourceServiceClient {
t.Helper() t.Helper()
addr := testutils.RunTestServer(t, server) addr := testutils.RunTestServer(t, server)
@ -253,3 +257,5 @@ func tenancyCases() map[string]func(artistId, recordlabelId *pbresource.ID) *pbr
} }
return tenancyCases return tenancyCases
} }
func clone[T proto.Message](v T) T { return proto.Clone(v).(T) }

View File

@ -0,0 +1,201 @@
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package testing
import (
"context"
"testing"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure"
"github.com/hashicorp/consul/acl"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
internal "github.com/hashicorp/consul/agent/grpc-internal"
"github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/storage/inmem"
"github.com/hashicorp/consul/internal/tenancy"
"github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/sdk/testutil"
)
type builder struct {
registry resource.Registry
registerFns []func(resource.Registry)
useV2Tenancy bool
tenancies []*pbresource.Tenancy
aclResolver svc.ACLResolver
serviceImpl *svc.Server
}
// NewResourceServiceBuilder is the preferred way to configure and run
// an isolated in-process instance of the resource service for unit
// testing. The final call to `Run()` returns a client you can use for
// making requests.
func NewResourceServiceBuilder() *builder {
b := &builder{
useV2Tenancy: false,
registry: resource.NewRegistry(),
// Regardless of whether using mock of v2tenancy, always make sure
// the builtin tenancy exists.
tenancies: []*pbresource.Tenancy{resource.DefaultNamespacedTenancy()},
}
return b
}
// WithV2Tenancy configures which tenancy bridge is used.
//
// true => real v2 default partition and namespace via v2 tenancy bridge
// false => mock default partition and namespace since v1 tenancy bridge can't be used (not spinning up an entire server here)
func (b *builder) WithV2Tenancy(useV2Tenancy bool) *builder {
b.useV2Tenancy = useV2Tenancy
return b
}
// Registry provides access to the constructed registry post-Run() when
// needed by other test dependencies.
func (b *builder) Registry() resource.Registry {
return b.registry
}
// ServiceImpl provides access to the actual server side implemenation of the resource service. This should never be used
// used/accessed without good reason. The current justifying use case is to monkeypatch the ACL resolver post-creation
// to allow unfettered writes which some ACL related tests require to put test data in place.
func (b *builder) ServiceImpl() *svc.Server {
return b.serviceImpl
}
func (b *builder) WithRegisterFns(registerFns ...func(resource.Registry)) *builder {
for _, registerFn := range registerFns {
b.registerFns = append(b.registerFns, registerFn)
}
return b
}
func (b *builder) WithACLResolver(aclResolver svc.ACLResolver) *builder {
b.aclResolver = aclResolver
return b
}
// WithTenancies adds additional partitions and namespaces if default/default
// is not sufficient.
func (b *builder) WithTenancies(tenancies ...*pbresource.Tenancy) *builder {
for _, tenancy := range tenancies {
b.tenancies = append(b.tenancies, tenancy)
}
return b
}
// Run starts the resource service and returns a client.
func (b *builder) Run(t *testing.T) pbresource.ResourceServiceClient {
// backend cannot be customized
backend, err := inmem.NewBackend()
require.NoError(t, err)
// start the backend and add teardown hook
ctx, cancel := context.WithCancel(context.Background())
t.Cleanup(cancel)
go backend.Run(ctx)
// Automatically add tenancy types if v2 tenancy enabled
if b.useV2Tenancy {
b.registerFns = append(b.registerFns, tenancy.RegisterTypes)
}
for _, registerFn := range b.registerFns {
registerFn(b.registry)
}
var tenancyBridge resource.TenancyBridge
if !b.useV2Tenancy {
// use mock tenancy bridge. default/default has already been added out of the box
mockTenancyBridge := &svc.MockTenancyBridge{}
for _, tenancy := range b.tenancies {
mockTenancyBridge.On("PartitionExists", tenancy.Partition).Return(true, nil)
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, tenancy.Namespace).Return(true, nil)
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenancy.Partition).Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, tenancy.Namespace).Return(false, nil)
}
tenancyBridge = mockTenancyBridge
} else {
// use v2 tenancy bridge. population comes later after client injected.
tenancyBridge = tenancy.NewV2TenancyBridge()
}
if b.aclResolver == nil {
// When not provided (regardless of V1 tenancy or V2 tenancy), configure an ACL resolver
// that has ACLs disabled and fills in "default" for the partition and namespace when
// not provided. This is similar to user initiated requests.
//
// Controllers under test should be providing full tenancy since they will run with the DANGER_NO_AUTH.
mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(testutils.ACLsDisabled(t), nil).
Run(func(args mock.Arguments) {
// Caller expecting passed in tokenEntMeta and authorizerContext to be filled in.
tokenEntMeta := args.Get(1).(*acl.EnterpriseMeta)
if tokenEntMeta != nil {
FillEntMeta(tokenEntMeta)
}
authzContext := args.Get(2).(*acl.AuthorizerContext)
if authzContext != nil {
FillAuthorizerContext(authzContext)
}
})
b.aclResolver = mockACLResolver
}
config := svc.Config{
Logger: testutil.Logger(t),
Registry: b.registry,
Backend: backend,
ACLResolver: b.aclResolver,
TenancyBridge: tenancyBridge,
UseV2Tenancy: b.useV2Tenancy,
}
server := grpc.NewServer()
b.serviceImpl = svc.NewServer(config)
b.serviceImpl.Register(server)
pipe := internal.NewPipeListener()
go server.Serve(pipe)
t.Cleanup(server.Stop)
conn, err := grpc.Dial("",
grpc.WithTransportCredentials(insecure.NewCredentials()),
grpc.WithContextDialer(pipe.DialContext),
grpc.WithBlock(),
)
require.NoError(t, err)
t.Cleanup(func() { _ = conn.Close() })
client := pbresource.NewResourceServiceClient(conn)
// HACK ALERT: The client needs to be injected into the V2TenancyBridge
// after it has been created due the the circular dependency. This will
// go away when the tenancy bridge is removed and V1 is no more, however
// long that takes.
switch config.TenancyBridge.(type) {
case *tenancy.V2TenancyBridge:
config.TenancyBridge.(*tenancy.V2TenancyBridge).WithClient(client)
// Default partition namespace can finally be created
require.NoError(t, initTenancy(ctx, backend))
for _, tenancy := range b.tenancies {
if tenancy.Partition == resource.DefaultPartitionName && tenancy.Namespace == resource.DefaultNamespaceName {
continue
}
t.Fatalf("TODO: implement creation of passed in v2 tenancy: %v", tenancy)
}
}
return client
}

View File

@ -4,27 +4,15 @@
package testing package testing
import ( import (
"context"
"testing" "testing"
"github.com/hashicorp/go-uuid"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials/insecure" "github.com/hashicorp/go-uuid"
"github.com/hashicorp/consul/acl" "github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
internal "github.com/hashicorp/consul/agent/grpc-internal"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/internal/resource"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
"github.com/hashicorp/consul/internal/storage/inmem"
"github.com/hashicorp/consul/internal/tenancy"
"github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/sdk/testutil"
) )
func randomACLIdentity(t *testing.T) structs.ACLIdentity { func randomACLIdentity(t *testing.T) structs.ACLIdentity {
@ -50,129 +38,3 @@ func AuthorizerFrom(t *testing.T, policyStrs ...string) resolver.Result {
ACLIdentity: randomACLIdentity(t), ACLIdentity: randomACLIdentity(t),
} }
} }
// RunResourceService runs a Resource Service for the duration of the test and
// returns a client to interact with it. ACLs will be disabled and only the
// default partition and namespace are available.
func RunResourceService(t *testing.T, registerFns ...func(resource.Registry)) pbresource.ResourceServiceClient {
return RunResourceServiceWithConfig(t, svc.Config{}, registerFns...)
}
// RunResourceServiceWithTenancies runs a Resource Service with tenancies returned from TestTenancies.
func RunResourceServiceWithTenancies(t *testing.T, registerFns ...func(resource.Registry)) pbresource.ResourceServiceClient {
mockTenancyBridge := &svc.MockTenancyBridge{}
for _, tenant := range rtest.TestTenancies() {
mockTenancyBridge.On("PartitionExists", tenant.Partition).Return(true, nil)
mockTenancyBridge.On("NamespaceExists", tenant.Partition, tenant.Namespace).Return(true, nil)
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenant.Partition).Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenant.Partition, tenant.Namespace).Return(false, nil)
}
cfg := &svc.Config{
TenancyBridge: mockTenancyBridge,
}
return RunResourceServiceWithConfig(t, *cfg, registerFns...)
}
// RunResourceServiceWithConfig runs a ResourceService with caller injectable config to ease mocking dependencies.
// Any nil config field is replaced with a reasonable default with the following behavior:
//
// config.Backend - cannot be configured and must be nil
// config.Registry - empty registry
// config.TenancyBridge - mock provided with only the default partition and namespace
// config.ACLResolver - mock provided with ACLs disabled. Fills entMeta and authzContext with default partition and namespace
func RunResourceServiceWithConfig(t *testing.T, config svc.Config, registerFns ...func(resource.Registry)) pbresource.ResourceServiceClient {
t.Helper()
if config.Backend != nil {
panic("backend can not be configured")
}
backend, err := inmem.NewBackend()
require.NoError(t, err)
ctx, cancel := context.WithCancel(context.Background())
t.Cleanup(cancel)
go backend.Run(ctx)
config.Backend = backend
if config.Registry == nil {
config.Registry = resource.NewRegistry()
}
for _, fn := range registerFns {
fn(config.Registry)
}
server := grpc.NewServer()
if config.TenancyBridge == nil {
mockTenancyBridge := &svc.MockTenancyBridge{}
mockTenancyBridge.On("PartitionExists", resource.DefaultPartitionName).Return(true, nil)
mockTenancyBridge.On("PartitionExists", "foo").Return(true, nil)
mockTenancyBridge.On("NamespaceExists", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(true, nil)
mockTenancyBridge.On("PartitionExists", "foo").Return(true, nil)
mockTenancyBridge.On("IsPartitionMarkedForDeletion", resource.DefaultPartitionName).Return(false, nil)
mockTenancyBridge.On("IsPartitionMarkedForDeletion", "foo").Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", resource.DefaultPartitionName, resource.DefaultNamespaceName).Return(false, nil)
config.TenancyBridge = mockTenancyBridge
} else {
switch config.TenancyBridge.(type) {
case *tenancy.V2TenancyBridge:
err = initTenancy(ctx, backend)
require.NoError(t, err)
}
}
if config.ACLResolver == nil {
// Provide a resolver which will default partition and namespace when not provided. This is similar to user
// initiated requests.
//
// Controllers under test should be providing full tenancy since they will run with the DANGER_NO_AUTH.
mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(testutils.ACLsDisabled(t), nil).
Run(func(args mock.Arguments) {
// Caller expecting passed in tokenEntMeta and authorizerContext to be filled in.
tokenEntMeta := args.Get(1).(*acl.EnterpriseMeta)
if tokenEntMeta != nil {
FillEntMeta(tokenEntMeta)
}
authzContext := args.Get(2).(*acl.AuthorizerContext)
if authzContext != nil {
FillAuthorizerContext(authzContext)
}
})
config.ACLResolver = mockACLResolver
}
if config.Logger == nil {
config.Logger = testutil.Logger(t)
}
svc.NewServer(config).Register(server)
pipe := internal.NewPipeListener()
go server.Serve(pipe)
t.Cleanup(server.Stop)
conn, err := grpc.Dial("",
grpc.WithTransportCredentials(insecure.NewCredentials()),
grpc.WithContextDialer(pipe.DialContext),
grpc.WithBlock(),
)
require.NoError(t, err)
t.Cleanup(func() { _ = conn.Close() })
client := pbresource.NewResourceServiceClient(conn)
if config.TenancyBridge != nil {
switch config.TenancyBridge.(type) {
case *tenancy.V2TenancyBridge:
config.TenancyBridge.(*tenancy.V2TenancyBridge).WithClient(client)
}
}
return client
}

View File

@ -31,8 +31,6 @@ func FillAuthorizerContext(authzContext *acl.AuthorizerContext) {
// initTenancy create the base tenancy objects (default/default) // initTenancy create the base tenancy objects (default/default)
func initTenancy(ctx context.Context, b *inmem.Backend) error { func initTenancy(ctx context.Context, b *inmem.Backend) error {
//TODO(dhiaayachi): This is now called for testing purpose but at some point we need to add something similar
// when bootstrapping a server, probably in the tenancy controllers.
nsData, err := anypb.New(&pbtenancy.Namespace{Description: "default namespace in default partition"}) nsData, err := anypb.New(&pbtenancy.Namespace{Description: "default namespace in default partition"})
if err != nil { if err != nil {
return err return err
@ -61,5 +59,4 @@ func initTenancy(ctx context.Context, b *inmem.Backend) error {
} }
} }
return nil return nil
} }

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -11,24 +11,28 @@ import (
"testing" "testing"
"time" "time"
"github.com/hashicorp/consul/acl"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"google.golang.org/grpc/codes" "google.golang.org/grpc/codes"
"google.golang.org/grpc/status" "google.golang.org/grpc/status"
"google.golang.org/protobuf/proto" "google.golang.org/protobuf/proto"
"github.com/hashicorp/consul/acl"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/agent/grpc-external/testutils"
"github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestWatchList_InputValidation(t *testing.T) { func TestWatchList_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
type testCase struct { type testCase struct {
modFn func(*pbresource.WatchListRequest) modFn func(*pbresource.WatchListRequest)
@ -108,8 +112,7 @@ func TestWatchList_InputValidation(t *testing.T) {
func TestWatchList_TypeNotFound(t *testing.T) { func TestWatchList_TypeNotFound(t *testing.T) {
t.Parallel() t.Parallel()
server := testServer(t) client := svctest.NewResourceServiceBuilder().Run(t)
client := testClient(t, server)
stream, err := client.WatchList(context.Background(), &pbresource.WatchListRequest{ stream, err := client.WatchList(context.Background(), &pbresource.WatchListRequest{
Type: demo.TypeV2Artist, Type: demo.TypeV2Artist,
@ -171,9 +174,9 @@ func TestWatchList_Tenancy_Defaults_And_Normalization(t *testing.T) {
for desc, tc := range wildcardTenancyCases() { for desc, tc := range wildcardTenancyCases() {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
ctx := context.Background() ctx := context.Background()
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
// Create a watch. // Create a watch.
stream, err := client.WatchList(ctx, &pbresource.WatchListRequest{ stream, err := client.WatchList(ctx, &pbresource.WatchListRequest{
@ -191,17 +194,17 @@ func TestWatchList_Tenancy_Defaults_And_Normalization(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
// Create and verify upsert event received. // Create and verify upsert event received.
recordLabel, err = server.Backend.WriteCAS(ctx, recordLabel) rlRsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: recordLabel})
require.NoError(t, err) require.NoError(t, err)
artist, err = server.Backend.WriteCAS(ctx, artist) artistRsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: artist})
require.NoError(t, err) require.NoError(t, err)
var expected *pbresource.Resource var expected *pbresource.Resource
switch { switch {
case proto.Equal(tc.typ, demo.TypeV1RecordLabel): case proto.Equal(tc.typ, demo.TypeV1RecordLabel):
expected = recordLabel expected = rlRsp.Resource
case proto.Equal(tc.typ, demo.TypeV2Artist): case proto.Equal(tc.typ, demo.TypeV2Artist):
expected = artist expected = artistRsp.Resource
default: default:
require.Fail(t, "unsupported type", tc.typ) require.Fail(t, "unsupported type", tc.typ)
} }
@ -210,7 +213,6 @@ func TestWatchList_Tenancy_Defaults_And_Normalization(t *testing.T) {
require.Equal(t, pbresource.WatchEvent_OPERATION_UPSERT, rsp.Operation) require.Equal(t, pbresource.WatchEvent_OPERATION_UPSERT, rsp.Operation)
prototest.AssertDeepEqual(t, expected, rsp.Resource) prototest.AssertDeepEqual(t, expected, rsp.Resource)
}) })
} }
} }
@ -304,7 +306,7 @@ func roundTripACL(t *testing.T, authz acl.Authorizer) (<-chan resourceOrError, *
server := testServer(t) server := testServer(t)
client := testClient(t, server) client := testClient(t, server)
mockACLResolver := &MockACLResolver{} mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(authz, nil) Return(authz, nil)
server.ACLResolver = mockACLResolver server.ACLResolver = mockACLResolver
@ -395,10 +397,11 @@ type resourceOrError struct {
func TestWatchList_NoTenancy(t *testing.T) { func TestWatchList_NoTenancy(t *testing.T) {
t.Parallel() t.Parallel()
ctx := context.Background() ctx := context.Background()
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
// Create a watch. // Create a watch.
stream, err := client.WatchList(ctx, &pbresource.WatchListRequest{ stream, err := client.WatchList(ctx, &pbresource.WatchListRequest{
@ -411,11 +414,11 @@ func TestWatchList_NoTenancy(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
// Create and verify upsert event received. // Create and verify upsert event received.
recordLabel, err = server.Backend.WriteCAS(ctx, recordLabel) rsp1, err := client.Write(ctx, &pbresource.WriteRequest{Resource: recordLabel})
require.NoError(t, err) require.NoError(t, err)
rsp := mustGetResource(t, rspCh) rsp2 := mustGetResource(t, rspCh)
require.Equal(t, pbresource.WatchEvent_OPERATION_UPSERT, rsp.Operation) require.Equal(t, pbresource.WatchEvent_OPERATION_UPSERT, rsp2.Operation)
prototest.AssertDeepEqual(t, recordLabel, rsp.Resource) prototest.AssertDeepEqual(t, rsp1.Resource, rsp2.Resource)
} }

View File

@ -357,8 +357,9 @@ func ensureDataUnchanged(input *pbresource.Resource, existing *pbresource.Resour
return nil return nil
} }
// ensureFinalizerRemoved ensures at least one finalizer was removed. // EnsureFinalizerRemoved ensures at least one finalizer was removed.
func ensureFinalizerRemoved(input *pbresource.Resource, existing *pbresource.Resource) error { // TODO: only public for test to access
func EnsureFinalizerRemoved(input *pbresource.Resource, existing *pbresource.Resource) error {
inputFinalizers := resource.GetFinalizers(input) inputFinalizers := resource.GetFinalizers(input)
existingFinalizers := resource.GetFinalizers(existing) existingFinalizers := resource.GetFinalizers(existing)
if !inputFinalizers.IsProperSubset(existingFinalizers) { if !inputFinalizers.IsProperSubset(existingFinalizers) {
@ -419,13 +420,13 @@ func vetIfDeleteRelated(input, existing *pbresource.Resource, tenancyMarkedForDe
if err := ensureDataUnchanged(input, existing); err != nil { if err := ensureDataUnchanged(input, existing); err != nil {
return err return err
} }
if err := ensureFinalizerRemoved(input, existing); err != nil { if err := EnsureFinalizerRemoved(input, existing); err != nil {
return err return err
} }
} }
// Classify writes that just remove finalizer as deleteRelated regardless of deletion state. // Classify writes that just remove finalizer as deleteRelated regardless of deletion state.
if err := ensureFinalizerRemoved(input, existing); err == nil { if err := EnsureFinalizerRemoved(input, existing); err == nil {
if err := ensureDataUnchanged(input, existing); err == nil { if err := ensureDataUnchanged(input, existing); err == nil {
deleteRelated = deleteRelated || true deleteRelated = deleteRelated || true
} }

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"fmt" "fmt"
@ -16,11 +16,15 @@ import (
"google.golang.org/protobuf/types/known/timestamppb" "google.golang.org/protobuf/types/known/timestamppb"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestWriteStatus_ACL(t *testing.T) { func TestWriteStatus_ACL(t *testing.T) {
type testCase struct { type testCase struct {
authz resolver.Result authz resolver.Result
@ -44,9 +48,8 @@ func TestWriteStatus_ACL(t *testing.T) {
for desc, tc := range testcases { for desc, tc := range testcases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) builder := svctest.NewResourceServiceBuilder().WithRegisterFns(demo.RegisterTypes)
client := testClient(t, server) client := builder.Run(t)
demo.RegisterTypes(server.Registry)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -56,10 +59,10 @@ func TestWriteStatus_ACL(t *testing.T) {
artist = rsp.Resource artist = rsp.Resource
// Defer mocking out authz since above write is necessary to set up the test resource. // Defer mocking out authz since above write is necessary to set up the test resource.
mockACLResolver := &MockACLResolver{} mockACLResolver := &svc.MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(tc.authz, nil) Return(tc.authz, nil)
server.ACLResolver = mockACLResolver builder.ServiceImpl().Config.ACLResolver = mockACLResolver
// exercise ACL // exercise ACL
_, err = client.WriteStatus(testContext(t), validWriteStatusRequest(t, artist)) _, err = client.WriteStatus(testContext(t), validWriteStatusRequest(t, artist))
@ -69,9 +72,9 @@ func TestWriteStatus_ACL(t *testing.T) {
} }
func TestWriteStatus_InputValidation(t *testing.T) { func TestWriteStatus_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
testCases := map[string]struct { testCases := map[string]struct {
typ *pbresource.Type typ *pbresource.Type
@ -259,9 +262,9 @@ func TestWriteStatus_Success(t *testing.T) {
"Non CAS": func(req *pbresource.WriteStatusRequest) { req.Version = "" }, "Non CAS": func(req *pbresource.WriteStatusRequest) { req.Version = "" },
} { } {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -331,9 +334,9 @@ func TestWriteStatus_Tenancy_Defaults(t *testing.T) {
}, },
} { } {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
// Pick resource based on scope of type in testcase. // Pick resource based on scope of type in testcase.
var res *pbresource.Resource var res *pbresource.Resource
@ -395,9 +398,10 @@ func TestWriteStatus_Tenancy_NotFound(t *testing.T) {
}, },
} { } {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithV2Tenancy(true).
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
Run(t)
// Pick resource based on scope of type in testcase. // Pick resource based on scope of type in testcase.
var res *pbresource.Resource var res *pbresource.Resource
@ -428,10 +432,9 @@ func TestWriteStatus_Tenancy_NotFound(t *testing.T) {
} }
func TestWriteStatus_CASFailure(t *testing.T) { func TestWriteStatus_CASFailure(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -449,8 +452,7 @@ func TestWriteStatus_CASFailure(t *testing.T) {
} }
func TestWriteStatus_TypeNotFound(t *testing.T) { func TestWriteStatus_TypeNotFound(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().Run(t)
client := testClient(t, server)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -464,9 +466,9 @@ func TestWriteStatus_TypeNotFound(t *testing.T) {
} }
func TestWriteStatus_ResourceNotFound(t *testing.T) { func TestWriteStatus_ResourceNotFound(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -479,9 +481,9 @@ func TestWriteStatus_ResourceNotFound(t *testing.T) {
} }
func TestWriteStatus_WrongUid(t *testing.T) { func TestWriteStatus_WrongUid(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)

View File

@ -1,7 +1,7 @@
// Copyright (c) HashiCorp, Inc. // Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1 // SPDX-License-Identifier: BUSL-1.1
package resource package resource_test
import ( import (
"context" "context"
@ -18,6 +18,8 @@ import (
"google.golang.org/protobuf/types/known/anypb" "google.golang.org/protobuf/types/known/anypb"
"github.com/hashicorp/consul/acl/resolver" "github.com/hashicorp/consul/acl/resolver"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest" rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
@ -29,10 +31,12 @@ import (
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
) )
// TODO: Update all tests to use true/false table test for v2tenancy
func TestWrite_InputValidation(t *testing.T) { func TestWrite_InputValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
type testCase struct { type testCase struct {
modFn func(artist, recordLabel *pbresource.Resource) *pbresource.Resource modFn func(artist, recordLabel *pbresource.Resource) *pbresource.Resource
@ -154,9 +158,9 @@ func TestWrite_InputValidation(t *testing.T) {
} }
func TestWrite_OwnerValidation(t *testing.T) { func TestWrite_OwnerValidation(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
type testCase struct { type testCase struct {
modReqFn func(req *pbresource.WriteRequest) modReqFn func(req *pbresource.WriteRequest)
@ -230,8 +234,7 @@ func TestWrite_OwnerValidation(t *testing.T) {
} }
func TestWrite_TypeNotFound(t *testing.T) { func TestWrite_TypeNotFound(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().Run(t)
client := testClient(t, server)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -265,14 +268,14 @@ func TestWrite_ACLs(t *testing.T) {
for desc, tc := range testcases { for desc, tc := range testcases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) mockACLResolver := &svc.MockACLResolver{}
client := testClient(t, server)
mockACLResolver := &MockACLResolver{}
mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything). mockACLResolver.On("ResolveTokenAndDefaultMeta", mock.Anything, mock.Anything, mock.Anything).
Return(tc.authz, nil) Return(tc.authz, nil)
server.ACLResolver = mockACLResolver
demo.RegisterTypes(server.Registry) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
WithACLResolver(mockACLResolver).
Run(t)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -285,9 +288,9 @@ func TestWrite_ACLs(t *testing.T) {
} }
func TestWrite_Mutate(t *testing.T) { func TestWrite_Mutate(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -389,9 +392,9 @@ func TestWrite_Create_Success(t *testing.T) {
} }
for desc, tc := range testCases { for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
demo.RegisterTypes(server.Registry) Run(t)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err) require.NoError(t, err)
@ -442,9 +445,10 @@ func TestWrite_Create_Tenancy_NotFound(t *testing.T) {
} }
for desc, tc := range testCases { for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithV2Tenancy(true).
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
Run(t)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err) require.NoError(t, err)
@ -461,9 +465,10 @@ func TestWrite_Create_Tenancy_NotFound(t *testing.T) {
} }
func TestWrite_Create_With_DeletionTimestamp_Fails(t *testing.T) { func TestWrite_Create_With_DeletionTimestamp_Fails(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithV2Tenancy(true).
demo.RegisterTypes(server.Registry) WithRegisterFns(demo.RegisterTypes).
Run(t)
res := rtest.Resource(demo.TypeV1Artist, "blur"). res := rtest.Resource(demo.TypeV1Artist, "blur").
WithTenancy(resource.DefaultNamespacedTenancy()). WithTenancy(resource.DefaultNamespacedTenancy()).
@ -480,18 +485,18 @@ func TestWrite_Create_With_DeletionTimestamp_Fails(t *testing.T) {
func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) { func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) {
// Verify resource write fails when its partition or namespace is marked for deletion. // Verify resource write fails when its partition or namespace is marked for deletion.
testCases := map[string]struct { testCases := map[string]struct {
modFn func(artist, recordLabel *pbresource.Resource, mockTenancyBridge *MockTenancyBridge) *pbresource.Resource modFn func(artist, recordLabel *pbresource.Resource, mockTenancyBridge *svc.MockTenancyBridge) *pbresource.Resource
errContains string errContains string
}{ }{
"namespaced resources partition marked for deletion": { "namespaced resources partition marked for deletion": {
modFn: func(artist, _ *pbresource.Resource, mockTenancyBridge *MockTenancyBridge) *pbresource.Resource { modFn: func(artist, _ *pbresource.Resource, mockTenancyBridge *svc.MockTenancyBridge) *pbresource.Resource {
mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(true, nil) mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(true, nil)
return artist return artist
}, },
errContains: "tenancy marked for deletion", errContains: "tenancy marked for deletion",
}, },
"namespaced resources namespace marked for deletion": { "namespaced resources namespace marked for deletion": {
modFn: func(artist, _ *pbresource.Resource, mockTenancyBridge *MockTenancyBridge) *pbresource.Resource { modFn: func(artist, _ *pbresource.Resource, mockTenancyBridge *svc.MockTenancyBridge) *pbresource.Resource {
mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(false, nil) mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", "ap1", "ns1").Return(true, nil) mockTenancyBridge.On("IsNamespaceMarkedForDeletion", "ap1", "ns1").Return(true, nil)
return artist return artist
@ -499,7 +504,7 @@ func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) {
errContains: "tenancy marked for deletion", errContains: "tenancy marked for deletion",
}, },
"partitioned resources partition marked for deletion": { "partitioned resources partition marked for deletion": {
modFn: func(_, recordLabel *pbresource.Resource, mockTenancyBridge *MockTenancyBridge) *pbresource.Resource { modFn: func(_, recordLabel *pbresource.Resource, mockTenancyBridge *svc.MockTenancyBridge) *pbresource.Resource {
mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(true, nil) mockTenancyBridge.On("IsPartitionMarkedForDeletion", "ap1").Return(true, nil)
return recordLabel return recordLabel
}, },
@ -511,6 +516,7 @@ func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) {
server := testServer(t) server := testServer(t)
client := testClient(t, server) client := testClient(t, server)
demo.RegisterTypes(server.Registry) demo.RegisterTypes(server.Registry)
recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes") recordLabel, err := demo.GenerateV1RecordLabel("looney-tunes")
require.NoError(t, err) require.NoError(t, err)
recordLabel.Id.Tenancy.Partition = "ap1" recordLabel.Id.Tenancy.Partition = "ap1"
@ -520,7 +526,7 @@ func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) {
artist.Id.Tenancy.Partition = "ap1" artist.Id.Tenancy.Partition = "ap1"
artist.Id.Tenancy.Namespace = "ns1" artist.Id.Tenancy.Namespace = "ns1"
mockTenancyBridge := &MockTenancyBridge{} mockTenancyBridge := &svc.MockTenancyBridge{}
mockTenancyBridge.On("PartitionExists", "ap1").Return(true, nil) mockTenancyBridge.On("PartitionExists", "ap1").Return(true, nil)
mockTenancyBridge.On("NamespaceExists", "ap1", "ns1").Return(true, nil) mockTenancyBridge.On("NamespaceExists", "ap1", "ns1").Return(true, nil)
server.TenancyBridge = mockTenancyBridge server.TenancyBridge = mockTenancyBridge
@ -534,10 +540,9 @@ func TestWrite_Create_With_TenancyMarkedForDeletion_Fails(t *testing.T) {
} }
func TestWrite_CASUpdate_Success(t *testing.T) { func TestWrite_CASUpdate_Success(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -556,10 +561,9 @@ func TestWrite_CASUpdate_Success(t *testing.T) {
} }
func TestWrite_ResourceCreation_StatusProvided(t *testing.T) { func TestWrite_ResourceCreation_StatusProvided(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -575,10 +579,9 @@ func TestWrite_ResourceCreation_StatusProvided(t *testing.T) {
} }
func TestWrite_CASUpdate_Failure(t *testing.T) { func TestWrite_CASUpdate_Failure(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -596,10 +599,9 @@ func TestWrite_CASUpdate_Failure(t *testing.T) {
} }
func TestWrite_Update_WrongUid(t *testing.T) { func TestWrite_Update_WrongUid(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -617,10 +619,9 @@ func TestWrite_Update_WrongUid(t *testing.T) {
} }
func TestWrite_Update_StatusModified(t *testing.T) { func TestWrite_Update_StatusModified(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -647,10 +648,9 @@ func TestWrite_Update_StatusModified(t *testing.T) {
} }
func TestWrite_Update_NilStatus(t *testing.T) { func TestWrite_Update_NilStatus(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -671,10 +671,9 @@ func TestWrite_Update_NilStatus(t *testing.T) {
} }
func TestWrite_Update_NoUid(t *testing.T) { func TestWrite_Update_NoUid(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -690,10 +689,9 @@ func TestWrite_Update_NoUid(t *testing.T) {
} }
func TestWrite_Update_GroupVersion(t *testing.T) { func TestWrite_Update_GroupVersion(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -720,10 +718,9 @@ func TestWrite_Update_GroupVersion(t *testing.T) {
} }
func TestWrite_NonCASUpdate_Success(t *testing.T) { func TestWrite_NonCASUpdate_Success(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -743,7 +740,6 @@ func TestWrite_NonCASUpdate_Success(t *testing.T) {
func TestWrite_NonCASUpdate_Retry(t *testing.T) { func TestWrite_NonCASUpdate_Retry(t *testing.T) {
server := testServer(t) server := testServer(t)
client := testClient(t, server) client := testClient(t, server)
demo.RegisterTypes(server.Registry) demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
@ -788,10 +784,9 @@ func TestWrite_NonCASUpdate_Retry(t *testing.T) {
} }
func TestWrite_NoData(t *testing.T) { func TestWrite_NoData(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
res, err := demo.GenerateV1Concept("jazz") res, err := demo.GenerateV1Concept("jazz")
require.NoError(t, err) require.NoError(t, err)
@ -806,10 +801,9 @@ func TestWrite_Owner_Immutable(t *testing.T) {
// Use of proto.Equal(..) in implementation covers all permutations // Use of proto.Equal(..) in implementation covers all permutations
// (nil -> non-nil, non-nil -> nil, owner1 -> owner2) so only the first one // (nil -> non-nil, non-nil -> nil, owner1 -> owner2) so only the first one
// is tested. // is tested.
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
require.NoError(t, err) require.NoError(t, err)
@ -834,10 +828,9 @@ func TestWrite_Owner_Immutable(t *testing.T) {
} }
func TestWrite_Owner_Uid(t *testing.T) { func TestWrite_Owner_Uid(t *testing.T) {
server := testServer(t) client := svctest.NewResourceServiceBuilder().
client := testClient(t, server) WithRegisterFns(demo.RegisterTypes).
Run(t)
demo.RegisterTypes(server.Registry)
t.Run("uid given", func(t *testing.T) { t.Run("uid given", func(t *testing.T) {
artist, err := demo.GenerateV2Artist() artist, err := demo.GenerateV2Artist()
@ -991,7 +984,7 @@ func TestEnsureFinalizerRemoved(t *testing.T) {
tc.mod(input, existing) tc.mod(input, existing)
err := ensureFinalizerRemoved(input, existing) err := svc.EnsureFinalizerRemoved(input, existing)
if tc.errContains != "" { if tc.errContains != "" {
require.Error(t, err) require.Error(t, err)
require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String()) require.Equal(t, codes.InvalidArgument.String(), status.Code(err).String())
@ -1054,8 +1047,10 @@ func TestWrite_ResourceFrozenAfterMarkedForDeletion(t *testing.T) {
for desc, tc := range testCases { for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server, client, ctx := testDeps(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithV2Tenancy(true).
WithRegisterFns(demo.RegisterTypes).
Run(t)
// Create a resource with finalizers // Create a resource with finalizers
res := rtest.Resource(demo.TypeV1Artist, "joydivision"). res := rtest.Resource(demo.TypeV1Artist, "joydivision").
@ -1065,11 +1060,11 @@ func TestWrite_ResourceFrozenAfterMarkedForDeletion(t *testing.T) {
Write(t, client) Write(t, client)
// Mark for deletion - resource should now be frozen // Mark for deletion - resource should now be frozen
_, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id}) _, err := client.Delete(context.Background(), &pbresource.DeleteRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Verify marked for deletion // Verify marked for deletion
rsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id}) rsp, err := client.Read(context.Background(), &pbresource.ReadRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
require.True(t, resource.IsMarkedForDeletion(rsp.Resource)) require.True(t, resource.IsMarkedForDeletion(rsp.Resource))
@ -1077,7 +1072,7 @@ func TestWrite_ResourceFrozenAfterMarkedForDeletion(t *testing.T) {
tc.modFn(rsp.Resource) tc.modFn(rsp.Resource)
// Verify write results // Verify write results
_, err = client.Write(ctx, &pbresource.WriteRequest{Resource: rsp.Resource}) _, err = client.Write(context.Background(), &pbresource.WriteRequest{Resource: rsp.Resource})
if tc.errContains == "" { if tc.errContains == "" {
require.NoError(t, err) require.NoError(t, err)
} else { } else {
@ -1120,8 +1115,10 @@ func TestWrite_NonCASWritePreservesFinalizers(t *testing.T) {
for desc, tc := range testCases { for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server, client, ctx := testDeps(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithV2Tenancy(true).
WithRegisterFns(demo.RegisterTypes).
Run(t)
// Create the resource based on tc.existingMetadata // Create the resource based on tc.existingMetadata
builder := rtest.Resource(demo.TypeV1Artist, "joydivision"). builder := rtest.Resource(demo.TypeV1Artist, "joydivision").
@ -1148,7 +1145,7 @@ func TestWrite_NonCASWritePreservesFinalizers(t *testing.T) {
userRes := builder.Build() userRes := builder.Build()
// Perform the user write // Perform the user write
rsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: userRes}) rsp, err := client.Write(context.Background(), &pbresource.WriteRequest{Resource: userRes})
require.NoError(t, err) require.NoError(t, err)
// Verify write result preserved metadata based on testcase.expecteMetadata // Verify write result preserved metadata based on testcase.expecteMetadata
@ -1184,8 +1181,10 @@ func TestWrite_NonCASWritePreservesDeletionTimestamp(t *testing.T) {
for desc, tc := range testCases { for desc, tc := range testCases {
t.Run(desc, func(t *testing.T) { t.Run(desc, func(t *testing.T) {
server, client, ctx := testDeps(t) client := svctest.NewResourceServiceBuilder().
demo.RegisterTypes(server.Registry) WithV2Tenancy(true).
WithRegisterFns(demo.RegisterTypes).
Run(t)
// Create the resource based on tc.existingMetadata // Create the resource based on tc.existingMetadata
builder := rtest.Resource(demo.TypeV1Artist, "joydivision"). builder := rtest.Resource(demo.TypeV1Artist, "joydivision").
@ -1200,11 +1199,11 @@ func TestWrite_NonCASWritePreservesDeletionTimestamp(t *testing.T) {
res := builder.Write(t, client) res := builder.Write(t, client)
// Mark for deletion // Mark for deletion
_, err := client.Delete(ctx, &pbresource.DeleteRequest{Id: res.Id}) _, err := client.Delete(context.Background(), &pbresource.DeleteRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Re-read the deleted res for future comparison of deletionTimestamp // Re-read the deleted res for future comparison of deletionTimestamp
delRsp, err := client.Read(ctx, &pbresource.ReadRequest{Id: res.Id}) delRsp, err := client.Read(context.Background(), &pbresource.ReadRequest{Id: res.Id})
require.NoError(t, err) require.NoError(t, err)
// Build resource for user write based on tc.inputMetadata // Build resource for user write based on tc.inputMetadata
@ -1220,7 +1219,7 @@ func TestWrite_NonCASWritePreservesDeletionTimestamp(t *testing.T) {
userRes := builder.Build() userRes := builder.Build()
// Perform the non-CAS user write // Perform the non-CAS user write
rsp, err := client.Write(ctx, &pbresource.WriteRequest{Resource: userRes}) rsp, err := client.Write(context.Background(), &pbresource.WriteRequest{Resource: userRes})
require.NoError(t, err) require.NoError(t, err)
// Verify write result preserved metadata based on testcase.expecteMetadata // Verify write result preserved metadata based on testcase.expecteMetadata

View File

@ -41,7 +41,11 @@ func (suite *controllerSuite) SetupTest() {
suite.isEnterprise = versiontest.IsEnterprise() suite.isEnterprise = versiontest.IsEnterprise()
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
client := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.client = rtest.NewClient(client) suite.client = rtest.NewClient(client)
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: suite.client, Client: suite.client,

View File

@ -26,7 +26,9 @@ func runInMemResourceServiceAndControllers(t *testing.T, deps controllers.Depend
ctx := testutil.TestContext(t) ctx := testutil.TestContext(t)
// Create the in-mem resource service // Create the in-mem resource service
client := svctest.RunResourceService(t, catalog.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(catalog.RegisterTypes).
Run(t)
// Setup/Run the controller manager // Setup/Run the controller manager
mgr := controller.NewManager(client, testutil.Logger(t)) mgr := controller.NewManager(client, testutil.Logger(t))

View File

@ -449,7 +449,10 @@ type controllerSuite struct {
func (suite *controllerSuite) SetupTest() { func (suite *controllerSuite) SetupTest() {
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
client := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: client, Client: client,
Logger: testutil.Logger(suite.T()), Logger: testutil.Logger(suite.T()),

View File

@ -49,9 +49,11 @@ type reconciliationDataSuite struct {
func (suite *reconciliationDataSuite) SetupTest() { func (suite *reconciliationDataSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
suite.tenancies = rtest.TestTenancies() suite.tenancies = rtest.TestTenancies()
resourceClient := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.client = resourcetest.NewClient(resourceClient) suite.client = resourcetest.NewClient(resourceClient)
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: suite.client, Client: suite.client,

View File

@ -36,17 +36,19 @@ type controllerSuite struct {
} }
func (suite *controllerSuite) SetupTest() { func (suite *controllerSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.tenancies = rtest.TestTenancies()
client := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, types.RegisterDNSPolicy) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, types.RegisterDNSPolicy).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: client, Client: client,
Logger: testutil.Logger(suite.T()), Logger: testutil.Logger(suite.T()),
} }
suite.client = rtest.NewClient(client) suite.client = rtest.NewClient(client)
suite.failoverMapper = failovermapper.New() suite.failoverMapper = failovermapper.New()
suite.ctx = testutil.TestContext(suite.T())
suite.tenancies = rtest.TestTenancies()
} }
func (suite *controllerSuite) TestController() { func (suite *controllerSuite) TestController() {

View File

@ -14,7 +14,6 @@ import (
"google.golang.org/grpc/codes" "google.golang.org/grpc/codes"
"google.golang.org/grpc/status" "google.golang.org/grpc/status"
mockres "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/catalog/internal/types" "github.com/hashicorp/consul/internal/catalog/internal/types"
"github.com/hashicorp/consul/internal/controller" "github.com/hashicorp/consul/internal/controller"
@ -82,18 +81,12 @@ func (suite *nodeHealthControllerTestSuite) writeNode(name string, tenancy *pbre
} }
func (suite *nodeHealthControllerTestSuite) SetupTest() { func (suite *nodeHealthControllerTestSuite) SetupTest() {
mockTenancyBridge := &mockres.MockTenancyBridge{}
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
for _, tenancy := range suite.tenancies { client := svctest.NewResourceServiceBuilder().
mockTenancyBridge.On("PartitionExists", tenancy.Partition).Return(true, nil) WithRegisterFns(types.Register, types.RegisterDNSPolicy).
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, tenancy.Namespace).Return(true, nil) WithTenancies(suite.tenancies...).
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenancy.Partition).Return(false, nil) Run(suite.T())
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, tenancy.Namespace).Return(false, nil)
}
cfg := mockres.Config{
TenancyBridge: mockTenancyBridge,
}
client := svctest.RunResourceServiceWithConfig(suite.T(), cfg, types.Register, types.RegisterDNSPolicy)
suite.resourceClient = resourcetest.NewClient(client) suite.resourceClient = resourcetest.NewClient(client)
suite.runtime = controller.Runtime{Client: suite.resourceClient, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: suite.resourceClient, Logger: testutil.Logger(suite.T())}
suite.isEnterprise = versiontest.IsEnterprise() suite.isEnterprise = versiontest.IsEnterprise()

View File

@ -15,7 +15,6 @@ import (
"google.golang.org/grpc/status" "google.golang.org/grpc/status"
"google.golang.org/protobuf/testing/protocmp" "google.golang.org/protobuf/testing/protocmp"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/catalog/internal/controllers/nodehealth" "github.com/hashicorp/consul/internal/catalog/internal/controllers/nodehealth"
"github.com/hashicorp/consul/internal/catalog/internal/mappers/nodemapper" "github.com/hashicorp/consul/internal/catalog/internal/mappers/nodemapper"
@ -93,15 +92,10 @@ type controllerSuite struct {
func (suite *controllerSuite) SetupTest() { func (suite *controllerSuite) SetupTest() {
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
mockTenancyBridge := &svc.MockTenancyBridge{} suite.client = svctest.NewResourceServiceBuilder().
for _, tenancy := range suite.tenancies { WithRegisterFns(types.Register).
mockTenancyBridge.On("PartitionExists", tenancy.Partition).Return(true, nil) WithTenancies(suite.tenancies...).
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenancy.Partition).Return(false, nil) Run(suite.T())
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, tenancy.Namespace).Return(true, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, tenancy.Namespace).Return(false, nil)
}
suite.client = svctest.RunResourceServiceWithConfig(suite.T(), svc.Config{TenancyBridge: mockTenancyBridge}, types.Register)
suite.runtime = controller.Runtime{Client: suite.client, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: suite.client, Logger: testutil.Logger(suite.T())}
suite.isEnterprise = versiontest.IsEnterprise() suite.isEnterprise = versiontest.IsEnterprise()
} }

View File

@ -23,7 +23,9 @@ func TestController_API(t *testing.T) {
t.Parallel() t.Parallel()
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
concertsChan := make(chan controller.Event) concertsChan := make(chan controller.Event)
defer close(concertsChan) defer close(concertsChan)
@ -164,7 +166,9 @@ func TestController_Placement(t *testing.T) {
t.Run("singleton", func(t *testing.T) { t.Run("singleton", func(t *testing.T) {
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
ctrl := controller. ctrl := controller.
ForType(demo.TypeV2Artist). ForType(demo.TypeV2Artist).
@ -197,7 +201,9 @@ func TestController_Placement(t *testing.T) {
t.Run("each server", func(t *testing.T) { t.Run("each server", func(t *testing.T) {
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
ctrl := controller. ctrl := controller.
ForType(demo.TypeV2Artist). ForType(demo.TypeV2Artist).
@ -233,7 +239,10 @@ func TestController_String(t *testing.T) {
} }
func TestController_NoReconciler(t *testing.T) { func TestController_NoReconciler(t *testing.T) {
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
mgr := controller.NewManager(client, testutil.Logger(t)) mgr := controller.NewManager(client, testutil.Logger(t))
ctrl := controller.ForType(demo.TypeV2Artist) ctrl := controller.ForType(demo.TypeV2Artist)
@ -248,7 +257,9 @@ func TestController_Watch(t *testing.T) {
t.Run("partitioned scoped resources", func(t *testing.T) { t.Run("partitioned scoped resources", func(t *testing.T) {
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
ctrl := controller. ctrl := controller.
ForType(demo.TypeV1RecordLabel). ForType(demo.TypeV1RecordLabel).
@ -274,7 +285,9 @@ func TestController_Watch(t *testing.T) {
t.Run("cluster scoped resources", func(t *testing.T) { t.Run("cluster scoped resources", func(t *testing.T) {
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
ctrl := controller. ctrl := controller.
ForType(demo.TypeV1Executive). ForType(demo.TypeV1Executive).
@ -299,7 +312,9 @@ func TestController_Watch(t *testing.T) {
t.Run("namespace scoped resources", func(t *testing.T) { t.Run("namespace scoped resources", func(t *testing.T) {
rec := newTestReconciler() rec := newTestReconciler()
client := svctest.RunResourceService(t, demo.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
ctrl := controller. ctrl := controller.
ForType(demo.TypeV2Artist). ForType(demo.TypeV2Artist).

View File

@ -192,7 +192,10 @@ func TestFindDuplicates(t *testing.T) {
func (suite *controllerTestSuite) SetupTest() { func (suite *controllerTestSuite) SetupTest() {
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
resourceClient := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, catalog.RegisterTypes) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())}
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
suite.client = resourcetest.NewClient(resourceClient) suite.client = resourcetest.NewClient(resourceClient)

View File

@ -53,7 +53,10 @@ type controllerTestSuite struct {
func (suite *controllerTestSuite) SetupTest() { func (suite *controllerTestSuite) SetupTest() {
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
resourceClient := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, catalog.RegisterTypes) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.client = resourcetest.NewClient(resourceClient) suite.client = resourcetest.NewClient(resourceClient)
suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())}
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())

View File

@ -94,7 +94,9 @@ func TestSortProxyConfigurations(t *testing.T) {
for name, c := range cases { for name, c := range cases {
t.Run(name, func(t *testing.T) { t.Run(name, func(t *testing.T) {
resourceClient := svctest.RunResourceService(t, types.Register) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register).
Run(t)
var decProxyCfgs []*types.DecodedProxyConfiguration var decProxyCfgs []*types.DecodedProxyConfiguration
for i, ws := range c.selectors { for i, ws := range c.selectors {

View File

@ -37,7 +37,11 @@ type controllerSuite struct {
func (suite *controllerSuite) SetupTest() { func (suite *controllerSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
suite.tenancies = rtest.TestTenancies() suite.tenancies = rtest.TestTenancies()
client := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, catalog.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: client, Client: client,
Logger: testutil.Logger(suite.T()), Logger: testutil.Logger(suite.T()),

View File

@ -7,11 +7,10 @@ import (
"testing" "testing"
"time" "time"
"github.com/hashicorp/go-hclog"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"google.golang.org/protobuf/types/known/durationpb" "google.golang.org/protobuf/types/known/durationpb"
pbmesh "github.com/hashicorp/consul/proto-public/pbmesh/v2beta1" "github.com/hashicorp/go-hclog"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/catalog" "github.com/hashicorp/consul/internal/catalog"
@ -21,6 +20,7 @@ import (
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest" rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
pbcatalog "github.com/hashicorp/consul/proto-public/pbcatalog/v2beta1" pbcatalog "github.com/hashicorp/consul/proto-public/pbcatalog/v2beta1"
pbmesh "github.com/hashicorp/consul/proto-public/pbmesh/v2beta1"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
"github.com/hashicorp/consul/sdk/testutil" "github.com/hashicorp/consul/sdk/testutil"
@ -28,7 +28,9 @@ import (
func TestLoadResourcesForComputedRoutes(t *testing.T) { func TestLoadResourcesForComputedRoutes(t *testing.T) {
ctx := testutil.TestContext(t) ctx := testutil.TestContext(t)
rclient := svctest.RunResourceService(t, types.Register, catalog.RegisterTypes) rclient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
Run(t)
rt := controller.Runtime{ rt := controller.Runtime{
Client: rclient, Client: rclient,
Logger: testutil.Logger(t), Logger: testutil.Logger(t),

View File

@ -5,7 +5,6 @@ package cache
import ( import (
"context" "context"
"github.com/hashicorp/consul/internal/mesh/internal/controllers/routes/routestest"
"testing" "testing"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
@ -13,6 +12,7 @@ import (
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/catalog" "github.com/hashicorp/consul/internal/catalog"
"github.com/hashicorp/consul/internal/controller" "github.com/hashicorp/consul/internal/controller"
"github.com/hashicorp/consul/internal/mesh/internal/controllers/routes/routestest"
"github.com/hashicorp/consul/internal/mesh/internal/types" "github.com/hashicorp/consul/internal/mesh/internal/types"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/resourcetest" "github.com/hashicorp/consul/internal/resource/resourcetest"
@ -82,7 +82,10 @@ func TestIdentities(t *testing.T) {
func TestMapComputedTrafficPermissions(t *testing.T) { func TestMapComputedTrafficPermissions(t *testing.T) {
resourcetest.RunWithTenancies(func(tenancy *pbresource.Tenancy) { resourcetest.RunWithTenancies(func(tenancy *pbresource.Tenancy) {
client := svctest.RunResourceService(t, types.Register, catalog.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
Run(t)
ctp := resourcetest.Resource(pbauth.ComputedTrafficPermissionsType, "workload-identity-1"). ctp := resourcetest.Resource(pbauth.ComputedTrafficPermissionsType, "workload-identity-1").
WithTenancy(tenancy). WithTenancy(tenancy).
WithData(t, &pbauth.ComputedTrafficPermissions{}). WithData(t, &pbauth.ComputedTrafficPermissions{}).
@ -137,7 +140,9 @@ func TestUnified_AllMappingsToProxyStateTemplate(t *testing.T) {
resourcetest.RunWithTenancies(func(tenancy *pbresource.Tenancy) { resourcetest.RunWithTenancies(func(tenancy *pbresource.Tenancy) {
var ( var (
cache = New() cache = New()
client = svctest.RunResourceService(t, types.Register, catalog.RegisterTypes) client = svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
Run(t)
) )
anyServiceData := &pbcatalog.Service{ anyServiceData := &pbcatalog.Service{

View File

@ -12,7 +12,6 @@ import (
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"github.com/stretchr/testify/suite" "github.com/stretchr/testify/suite"
mockres "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/envoyextensions/xdscommon" "github.com/hashicorp/consul/envoyextensions/xdscommon"
"github.com/hashicorp/consul/internal/auth" "github.com/hashicorp/consul/internal/auth"
@ -85,17 +84,11 @@ type apiData struct {
func (suite *controllerTestSuite) SetupTest() { func (suite *controllerTestSuite) SetupTest() {
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
mockTenancyBridge := &mockres.MockTenancyBridge{} resourceClient := svctest.NewResourceServiceBuilder().
for _, tenancy := range suite.tenancies { WithRegisterFns(types.Register, catalog.RegisterTypes, auth.RegisterTypes).
mockTenancyBridge.On("PartitionExists", tenancy.Partition).Return(true, nil) WithTenancies(suite.tenancies...).
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, tenancy.Namespace).Return(true, nil) Run(suite.T())
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenancy.Partition).Return(false, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, tenancy.Namespace).Return(false, nil)
}
cfg := mockres.Config{
TenancyBridge: mockTenancyBridge,
}
resourceClient := svctest.RunResourceServiceWithConfig(suite.T(), cfg, types.Register, catalog.RegisterTypes, auth.RegisterTypes)
suite.client = resourcetest.NewClient(resourceClient) suite.client = resourcetest.NewClient(resourceClient)
suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())}
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())

View File

@ -53,8 +53,10 @@ type dataFetcherSuite struct {
func (suite *dataFetcherSuite) SetupTest() { func (suite *dataFetcherSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
suite.tenancies = resourcetest.TestTenancies() suite.tenancies = resourcetest.TestTenancies()
suite.tenancies = resourcetest.TestTenancies() suite.client = svctest.NewResourceServiceBuilder().
suite.client = svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, catalog.RegisterTypes) WithRegisterFns(types.Register, catalog.RegisterTypes).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.resourceClient = resourcetest.NewClient(suite.client) suite.resourceClient = resourcetest.NewClient(suite.client)
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: suite.client, Client: suite.client,

View File

@ -70,7 +70,12 @@ type xdsControllerTestSuite struct {
func (suite *xdsControllerTestSuite) SetupTest() { func (suite *xdsControllerTestSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
resourceClient := svctest.RunResourceServiceWithTenancies(suite.T(), types.Register, catalog.RegisterTypes) suite.tenancies = resourcetest.TestTenancies()
resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
WithTenancies(suite.tenancies...).
Run(suite.T())
suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())} suite.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(suite.T())}
suite.client = resourcetest.NewClient(resourceClient) suite.client = resourcetest.NewClient(resourceClient)
suite.fetcher = mockFetcher suite.fetcher = mockFetcher
@ -100,8 +105,6 @@ func (suite *xdsControllerTestSuite) SetupTest() {
leafCertEvents: suite.leafCertEvents, leafCertEvents: suite.leafCertEvents,
datacenter: "dc1", datacenter: "dc1",
} }
suite.tenancies = resourcetest.TestTenancies()
} }
func mockFetcher() (*pbproxystate.TrustBundle, error) { func mockFetcher() (*pbproxystate.TrustBundle, error) {

View File

@ -21,7 +21,9 @@ import (
) )
func TestMapSelector(t *testing.T) { func TestMapSelector(t *testing.T) {
client := svctest.RunResourceService(t, types.Register, catalog.RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
Run(t)
// Create some workloads. // Create some workloads.
// For this test, we don't care about the workload data, so we will re-use // For this test, we don't care about the workload data, so we will re-use

View File

@ -22,7 +22,10 @@ import (
) )
func TestMapToComputedType(t *testing.T) { func TestMapToComputedType(t *testing.T) {
resourceClient := svctest.RunResourceService(t, types.Register, catalog.RegisterTypes) resourceClient := svctest.NewResourceServiceBuilder().
WithRegisterFns(types.Register, catalog.RegisterTypes).
Run(t)
mapper := New[*pbmesh.ProxyConfiguration](pbmesh.ComputedProxyConfigurationType) mapper := New[*pbmesh.ProxyConfiguration](pbmesh.ComputedProxyConfigurationType)
workloadData := &pbcatalog.Workload{ workloadData := &pbcatalog.Workload{

View File

@ -11,9 +11,8 @@ import (
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"github.com/stretchr/testify/suite" "github.com/stretchr/testify/suite"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
cat "github.com/hashicorp/consul/internal/catalog" "github.com/hashicorp/consul/internal/catalog"
"github.com/hashicorp/consul/internal/controller" "github.com/hashicorp/consul/internal/controller"
"github.com/hashicorp/consul/internal/multicluster/internal/types" "github.com/hashicorp/consul/internal/multicluster/internal/types"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
@ -39,20 +38,12 @@ type controllerSuite struct {
func (suite *controllerSuite) SetupTest() { func (suite *controllerSuite) SetupTest() {
suite.ctx = testutil.TestContext(suite.T()) suite.ctx = testutil.TestContext(suite.T())
suite.tenancies = rtest.TestTenancies() suite.tenancies = rtest.TestTenancies()
mockTenancyBridge := &svc.MockTenancyBridge{} client := svctest.NewResourceServiceBuilder().
for _, tenancy := range suite.tenancies { WithRegisterFns(types.Register, catalog.RegisterTypes).
mockTenancyBridge.On("PartitionExists", tenancy.Partition).Return(true, nil) WithTenancies(suite.tenancies...).
mockTenancyBridge.On("IsPartitionMarkedForDeletion", tenancy.Partition).Return(false, nil) WithTenancies(rtest.Tenancy("default.app"), rtest.Tenancy("foo.app")).
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, tenancy.Namespace).Return(true, nil) Run(suite.T())
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, tenancy.Namespace).Return(false, nil)
mockTenancyBridge.On("NamespaceExists", tenancy.Partition, "app").Return(true, nil)
mockTenancyBridge.On("IsNamespaceMarkedForDeletion", tenancy.Partition, "app").Return(false, nil)
}
config := svc.Config{
TenancyBridge: mockTenancyBridge,
}
client := svctest.RunResourceServiceWithConfig(suite.T(), config, types.Register, cat.RegisterTypes)
suite.client = rtest.NewClient(client) suite.client = rtest.NewClient(client)
suite.rt = controller.Runtime{ suite.rt = controller.Runtime{
Client: suite.client, Client: suite.client,

View File

@ -22,7 +22,7 @@ import (
func TestGetDecodedResource(t *testing.T) { func TestGetDecodedResource(t *testing.T) {
var ( var (
baseClient = svctest.RunResourceService(t, demo.RegisterTypes) baseClient = svctest.NewResourceServiceBuilder().WithRegisterFns(demo.RegisterTypes).Run(t)
client = rtest.NewClient(baseClient) client = rtest.NewClient(baseClient)
ctx = testutil.TestContext(t) ctx = testutil.TestContext(t)
) )

View File

@ -16,7 +16,9 @@ import (
) )
func TestArtistReconciler(t *testing.T) { func TestArtistReconciler(t *testing.T) {
client := svctest.RunResourceService(t, RegisterTypes) client := svctest.NewResourceServiceBuilder().
WithRegisterFns(RegisterTypes).
Run(t)
// Seed the database with an artist. // Seed the database with an artist.
res, err := GenerateV2Artist() res, err := GenerateV2Artist()

View File

@ -12,17 +12,17 @@ import (
"strings" "strings"
"testing" "testing"
"github.com/hashicorp/go-hclog"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
resourceSvc "github.com/hashicorp/consul/agent/grpc-external/services/resource" "github.com/hashicorp/go-hclog"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
pbdemov1 "github.com/hashicorp/consul/proto/private/pbdemo/v1"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
pbdemov1 "github.com/hashicorp/consul/proto/private/pbdemo/v1"
pbdemov2 "github.com/hashicorp/consul/proto/private/pbdemo/v2" pbdemov2 "github.com/hashicorp/consul/proto/private/pbdemo/v2"
"github.com/hashicorp/consul/sdk/testutil" "github.com/hashicorp/consul/sdk/testutil"
) )
@ -44,7 +44,11 @@ func TestResourceHandler_InputValidation(t *testing.T) {
expectedResponseCode int expectedResponseCode int
responseBodyContains string responseBodyContains string
} }
client := svctest.RunResourceService(t, demo.RegisterTypes)
client := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
Run(t)
resourceHandler := resourceHandler{ resourceHandler := resourceHandler{
resource.Registration{ resource.Registration{
Type: demo.TypeV2Artist, Type: demo.TypeV2Artist,
@ -125,17 +129,17 @@ func TestResourceHandler_InputValidation(t *testing.T) {
} }
func TestResourceWriteHandler(t *testing.T) { func TestResourceWriteHandler(t *testing.T) {
aclResolver := &resourceSvc.MockACLResolver{} aclResolver := &svc.MockACLResolver{}
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV1ReadPolicy, demo.ArtistV2ReadPolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV1ReadPolicy, demo.ArtistV2ReadPolicy), nil)
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV1WritePolicy, demo.ArtistV2WritePolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV1WritePolicy, demo.ArtistV2WritePolicy), nil)
client := svctest.RunResourceServiceWithConfig(t, resourceSvc.Config{ACLResolver: aclResolver}, demo.RegisterTypes) builder := svctest.NewResourceServiceBuilder().
WithACLResolver(aclResolver).
r := resource.NewRegistry() WithRegisterFns(demo.RegisterTypes)
demo.RegisterTypes(r) client := builder.Run(t)
handler := NewHandler(client, r, parseToken, hclog.NewNullLogger()) handler := NewHandler(client, builder.Registry(), parseToken, hclog.NewNullLogger())
t.Run("should be blocked if the token is not authorized", func(t *testing.T) { t.Run("should be blocked if the token is not authorized", func(t *testing.T) {
rsp := httptest.NewRecorder() rsp := httptest.NewRecorder()
@ -157,6 +161,7 @@ func TestResourceWriteHandler(t *testing.T) {
require.Equal(t, http.StatusForbidden, rsp.Result().StatusCode) require.Equal(t, http.StatusForbidden, rsp.Result().StatusCode)
}) })
var readRsp *pbresource.ReadResponse var readRsp *pbresource.ReadResponse
t.Run("should write to the resource backend", func(t *testing.T) { t.Run("should write to the resource backend", func(t *testing.T) {
rsp := httptest.NewRecorder() rsp := httptest.NewRecorder()
@ -352,7 +357,7 @@ func deleteResource(t *testing.T, artistHandler http.Handler, resourceUri *Resou
} }
func TestResourceReadHandler(t *testing.T) { func TestResourceReadHandler(t *testing.T) {
aclResolver := &resourceSvc.MockACLResolver{} aclResolver := &svc.MockACLResolver{}
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV1ReadPolicy, demo.ArtistV2ReadPolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV1ReadPolicy, demo.ArtistV2ReadPolicy), nil)
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything).
@ -360,11 +365,11 @@ func TestResourceReadHandler(t *testing.T) {
aclResolver.On("ResolveTokenAndDefaultMeta", fakeToken, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", fakeToken, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, ""), nil) Return(svctest.AuthorizerFrom(t, ""), nil)
client := svctest.RunResourceServiceWithConfig(t, resourceSvc.Config{ACLResolver: aclResolver}, demo.RegisterTypes) builder := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
r := resource.NewRegistry() WithACLResolver(aclResolver)
demo.RegisterTypes(r) client := builder.Run(t)
handler := NewHandler(client, r, parseToken, hclog.NewNullLogger()) handler := NewHandler(client, builder.Registry(), parseToken, hclog.NewNullLogger())
createdResource := createResource(t, handler, nil) createdResource := createResource(t, handler, nil)
@ -407,18 +412,17 @@ func TestResourceReadHandler(t *testing.T) {
} }
func TestResourceDeleteHandler(t *testing.T) { func TestResourceDeleteHandler(t *testing.T) {
aclResolver := &resourceSvc.MockACLResolver{} aclResolver := &svc.MockACLResolver{}
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistReadPolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV2ReadPolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV2ReadPolicy), nil)
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV2WritePolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV2WritePolicy), nil)
client := svctest.RunResourceServiceWithConfig(t, resourceSvc.Config{ACLResolver: aclResolver}, demo.RegisterTypes) builder := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
r := resource.NewRegistry() WithACLResolver(aclResolver)
demo.RegisterTypes(r) client := builder.Run(t)
handler := NewHandler(client, builder.Registry(), parseToken, hclog.NewNullLogger())
handler := NewHandler(client, r, parseToken, hclog.NewNullLogger())
t.Run("should surface PermissionDenied error from resource service", func(t *testing.T) { t.Run("should surface PermissionDenied error from resource service", func(t *testing.T) {
createResource(t, handler, nil) createResource(t, handler, nil)
@ -484,18 +488,17 @@ func TestResourceDeleteHandler(t *testing.T) {
} }
func TestResourceListHandler(t *testing.T) { func TestResourceListHandler(t *testing.T) {
aclResolver := &resourceSvc.MockACLResolver{} aclResolver := &svc.MockACLResolver{}
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistListPolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistListPolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV2ListPolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV2ListPolicy), nil)
aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything). aclResolver.On("ResolveTokenAndDefaultMeta", testACLTokenArtistWritePolicy, mock.Anything, mock.Anything).
Return(svctest.AuthorizerFrom(t, demo.ArtistV2WritePolicy), nil) Return(svctest.AuthorizerFrom(t, demo.ArtistV2WritePolicy), nil)
client := svctest.RunResourceServiceWithConfig(t, resourceSvc.Config{ACLResolver: aclResolver}, demo.RegisterTypes) builder := svctest.NewResourceServiceBuilder().
WithRegisterFns(demo.RegisterTypes).
r := resource.NewRegistry() WithACLResolver(aclResolver)
demo.RegisterTypes(r) client := builder.Run(t)
handler := NewHandler(client, builder.Registry(), parseToken, hclog.NewNullLogger())
handler := NewHandler(client, r, parseToken, hclog.NewNullLogger())
t.Run("should return MethodNotAllowed", func(t *testing.T) { t.Run("should return MethodNotAllowed", func(t *testing.T) {
rsp := httptest.NewRecorder() rsp := httptest.NewRecorder()

View File

@ -4,7 +4,6 @@
package reaper package reaper
import ( import (
"github.com/hashicorp/consul/internal/resource/resourcetest"
"testing" "testing"
"time" "time"
@ -16,13 +15,14 @@ import (
"github.com/hashicorp/consul/internal/controller" "github.com/hashicorp/consul/internal/controller"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
"github.com/hashicorp/consul/internal/resource/resourcetest"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
"github.com/hashicorp/consul/sdk/testutil" "github.com/hashicorp/consul/sdk/testutil"
) )
func TestReconcile_ResourceWithNoChildren(t *testing.T) { func TestReconcile_ResourceWithNoChildren(t *testing.T) {
client := svctest.RunResourceServiceWithTenancies(t, demo.RegisterTypes) client := setupResourceService(t)
runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) { runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) {
// Seed the database with an artist. // Seed the database with an artist.
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
@ -79,8 +79,7 @@ func TestReconcile_ResourceWithNoChildren(t *testing.T) {
} }
func TestReconcile_ResourceWithChildren(t *testing.T) { func TestReconcile_ResourceWithChildren(t *testing.T) {
client := svctest.RunResourceServiceWithTenancies(t, demo.RegisterTypes) client := setupResourceService(t)
runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) { runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) {
// Seed the database with an artist // Seed the database with an artist
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
@ -162,8 +161,7 @@ func TestReconcile_ResourceWithChildren(t *testing.T) {
} }
func TestReconcile_RequeueWithDelayWhenSecondPassDelayNotElapsed(t *testing.T) { func TestReconcile_RequeueWithDelayWhenSecondPassDelayNotElapsed(t *testing.T) {
client := svctest.RunResourceServiceWithTenancies(t, demo.RegisterTypes) client := setupResourceService(t)
runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) { runReaperTestCaseWithTenancies(func(tenancy *pbresource.Tenancy) {
// Seed the database with an artist. // Seed the database with an artist.
res, err := demo.GenerateV2Artist() res, err := demo.GenerateV2Artist()
@ -216,3 +214,10 @@ func runReaperTestCaseWithTenancies(testCase func(tenancy *pbresource.Tenancy))
testCase(tenancy) testCase(tenancy)
} }
} }
func setupResourceService(t *testing.T) pbresource.ResourceServiceClient {
return svctest.NewResourceServiceBuilder().
WithTenancies(rtest.TestTenancies()...).
WithRegisterFns(demo.RegisterTypes).
Run(t)
}

View File

@ -10,13 +10,11 @@ import (
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"github.com/stretchr/testify/suite" "github.com/stretchr/testify/suite"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/controller" "github.com/hashicorp/consul/internal/controller"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
"github.com/hashicorp/consul/internal/resource/demo" "github.com/hashicorp/consul/internal/resource/demo"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest" rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
"github.com/hashicorp/consul/internal/tenancy"
"github.com/hashicorp/consul/internal/tenancy/internal/controllers/common" "github.com/hashicorp/consul/internal/tenancy/internal/controllers/common"
"github.com/hashicorp/consul/internal/tenancy/internal/controllers/namespace" "github.com/hashicorp/consul/internal/tenancy/internal/controllers/namespace"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
@ -31,24 +29,21 @@ import (
type nsTestSuite struct { type nsTestSuite struct {
suite.Suite suite.Suite
client *rtest.Client client *rtest.Client
runtime controller.Runtime runtime controller.Runtime
ctx context.Context ctx context.Context
registry resource.Registry
} }
func (ts *nsTestSuite) SetupTest() { func (ts *nsTestSuite) SetupTest() {
ts.registry = resource.NewRegistry() builder := svctest.NewResourceServiceBuilder().
tenancyBridge := tenancy.NewV2TenancyBridge() WithV2Tenancy(true).
config := svc.Config{TenancyBridge: tenancyBridge, Registry: ts.registry, UseV2Tenancy: true} WithRegisterFns(demo.RegisterTypes)
resourceClient := svctest.RunResourceServiceWithConfig(ts.T(), config, tenancy.RegisterTypes, demo.RegisterTypes) ts.client = rtest.NewClient(builder.Run(ts.T()))
tenancyBridge.WithClient(resourceClient) ts.runtime = controller.Runtime{Client: ts.client, Logger: testutil.Logger(ts.T())}
ts.client = rtest.NewClient(resourceClient)
ts.runtime = controller.Runtime{Client: resourceClient, Logger: testutil.Logger(ts.T())}
ts.ctx = testutil.TestContext(ts.T()) ts.ctx = testutil.TestContext(ts.T())
mgr := controller.NewManager(ts.client, testutil.Logger(ts.T())) mgr := controller.NewManager(ts.client, testutil.Logger(ts.T()))
mgr.Register(namespace.Controller(ts.registry)) mgr.Register(namespace.Controller(builder.Registry()))
mgr.SetRaftLeader(true) mgr.SetRaftLeader(true)
ctx, cancel := context.WithCancel(context.Background()) ctx, cancel := context.WithCancel(context.Background())
ts.T().Cleanup(cancel) ts.T().Cleanup(cancel)

View File

@ -11,21 +11,16 @@ import (
"google.golang.org/grpc/codes" "google.golang.org/grpc/codes"
"google.golang.org/grpc/status" "google.golang.org/grpc/status"
svc "github.com/hashicorp/consul/agent/grpc-external/services/resource"
svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing" svctest "github.com/hashicorp/consul/agent/grpc-external/services/resource/testing"
"github.com/hashicorp/consul/internal/resource" "github.com/hashicorp/consul/internal/resource"
rtest "github.com/hashicorp/consul/internal/resource/resourcetest" rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
"github.com/hashicorp/consul/internal/tenancy"
"github.com/hashicorp/consul/proto-public/pbresource" "github.com/hashicorp/consul/proto-public/pbresource"
pbtenancy "github.com/hashicorp/consul/proto-public/pbtenancy/v2beta1" pbtenancy "github.com/hashicorp/consul/proto-public/pbtenancy/v2beta1"
"github.com/hashicorp/consul/proto/private/prototest" "github.com/hashicorp/consul/proto/private/prototest"
) )
func TestWriteNamespace_Success(t *testing.T) { func TestWriteNamespace_Success(t *testing.T) {
v2TenancyBridge := tenancy.NewV2TenancyBridge() cl := rtest.NewClient(svctest.NewResourceServiceBuilder().WithV2Tenancy(true).Run(t))
config := svc.Config{TenancyBridge: v2TenancyBridge, UseV2Tenancy: true}
client := svctest.RunResourceServiceWithConfig(t, config, tenancy.RegisterTypes)
cl := rtest.NewClient(client)
res := rtest.Resource(pbtenancy.NamespaceType, "ns1"). res := rtest.Resource(pbtenancy.NamespaceType, "ns1").
WithTenancy(resource.DefaultPartitionedTenancy()). WithTenancy(resource.DefaultPartitionedTenancy()).
@ -41,10 +36,7 @@ func TestWriteNamespace_Success(t *testing.T) {
} }
func TestReadNamespace_Success(t *testing.T) { func TestReadNamespace_Success(t *testing.T) {
v2TenancyBridge := tenancy.NewV2TenancyBridge() cl := rtest.NewClient(svctest.NewResourceServiceBuilder().WithV2Tenancy(true).Run(t))
config := svc.Config{TenancyBridge: v2TenancyBridge, UseV2Tenancy: true}
client := svctest.RunResourceServiceWithConfig(t, config, tenancy.RegisterTypes)
cl := rtest.NewClient(client)
res := rtest.Resource(pbtenancy.NamespaceType, "ns1"). res := rtest.Resource(pbtenancy.NamespaceType, "ns1").
WithData(t, validNamespace()). WithData(t, validNamespace()).
@ -74,10 +66,7 @@ func TestReadNamespace_Success(t *testing.T) {
} }
func TestDeleteNamespace_Success(t *testing.T) { func TestDeleteNamespace_Success(t *testing.T) {
v2TenancyBridge := tenancy.NewV2TenancyBridge() cl := rtest.NewClient(svctest.NewResourceServiceBuilder().WithV2Tenancy(true).Run(t))
config := svc.Config{TenancyBridge: v2TenancyBridge, UseV2Tenancy: true}
client := svctest.RunResourceServiceWithConfig(t, config, tenancy.RegisterTypes)
cl := rtest.NewClient(client)
res := rtest.Resource(pbtenancy.NamespaceType, "ns1"). res := rtest.Resource(pbtenancy.NamespaceType, "ns1").
WithData(t, validNamespace()).Write(t, cl) WithData(t, validNamespace()).Write(t, cl)
@ -96,10 +85,7 @@ func TestDeleteNamespace_Success(t *testing.T) {
} }
func TestListNamespace_Success(t *testing.T) { func TestListNamespace_Success(t *testing.T) {
v2TenancyBridge := tenancy.NewV2TenancyBridge() cl := rtest.NewClient(svctest.NewResourceServiceBuilder().WithV2Tenancy(true).Run(t))
config := svc.Config{TenancyBridge: v2TenancyBridge, UseV2Tenancy: true}
client := svctest.RunResourceServiceWithConfig(t, config, tenancy.RegisterTypes)
cl := rtest.NewClient(client)
res := rtest.Resource(pbtenancy.NamespaceType, "ns1"). res := rtest.Resource(pbtenancy.NamespaceType, "ns1").
WithData(t, validNamespace()).Write(t, cl) WithData(t, validNamespace()).Write(t, cl)