influxdb/testing/user_resource_mapping_servi...

555 lines
14 KiB
Go

package testing
import (
"bytes"
"context"
"fmt"
"sort"
"testing"
"github.com/google/go-cmp/cmp"
platform "github.com/influxdata/influxdb/v2"
platform2 "github.com/influxdata/influxdb/v2/kit/platform"
)
var mappingCmpOptions = cmp.Options{
cmp.Comparer(func(x, y []byte) bool {
return bytes.Equal(x, y)
}),
cmp.Transformer("Sort", func(in []*platform.UserResourceMapping) []*platform.UserResourceMapping {
out := append([]*platform.UserResourceMapping(nil), in...) // Copy input to avoid mutating it
sort.Slice(out, func(i, j int) bool {
return out[i].ResourceID.String() > out[j].ResourceID.String()
})
return out
}),
}
// UserResourceFields includes prepopulated data for mapping tests
type UserResourceFields struct {
Organizations []*platform.Organization
Users []*platform.User
Buckets []*platform.Bucket
UserResourceMappings []*platform.UserResourceMapping
}
type userResourceMappingServiceF func(
init func(UserResourceFields, *testing.T) (platform.UserResourceMappingService, func()),
t *testing.T,
)
// UserResourceMappingService tests all the service functions.
func UserResourceMappingService(
init func(UserResourceFields, *testing.T) (platform.UserResourceMappingService, func()),
t *testing.T,
) {
tests := []struct {
name string
fn userResourceMappingServiceF
}{
{
name: "CreateUserResourceMapping",
fn: CreateUserResourceMapping,
},
{
name: "FindUserResourceMappings",
fn: FindUserResourceMappings,
},
{
name: "DeleteUserResourceMapping",
fn: DeleteUserResourceMapping,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
tt := tt
t.Parallel()
tt.fn(init, t)
})
}
}
// baseUserResourceFields creates base fields to create URMs.
// Users for URMs must exist in order not to fail on creation.
func baseUserResourceFields() UserResourceFields {
return UserResourceFields{
Users: []*platform.User{
{
Name: "user1",
ID: MustIDBase16(userOneID),
},
{
Name: "user2",
ID: MustIDBase16(userTwoID),
},
},
}
}
func CreateUserResourceMapping(
init func(UserResourceFields, *testing.T) (platform.UserResourceMappingService, func()),
t *testing.T,
) {
type args struct {
mapping *platform.UserResourceMapping
}
type wants struct {
err error
mappings []*platform.UserResourceMapping
}
tests := []struct {
name string
fields UserResourceFields
args args
wants wants
}{
{
name: "basic create user resource mapping",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
mapping: &platform.UserResourceMapping{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
},
},
{
name: "duplicate mappings are not allowed",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
mapping: &platform.UserResourceMapping{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
//lint:ignore ST1005 Error is capitalized in the tested code.
err: fmt.Errorf("Unexpected error when assigning user to a resource: mapping for user %s already exists", userOneID),
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
s, done := init(tt.fields, t)
defer done()
ctx := context.Background()
err := s.CreateUserResourceMapping(ctx, tt.args.mapping)
if (err != nil) != (tt.wants.err != nil) {
t.Fatalf("expected error '%v' got '%v'", tt.wants.err, err)
}
if err != nil && tt.wants.err != nil {
if err.Error() != tt.wants.err.Error() {
t.Fatalf("expected error messages to match '%v' got '%v'", tt.wants.err, err.Error())
}
}
defer s.DeleteUserResourceMapping(ctx, tt.args.mapping.ResourceID, tt.args.mapping.UserID)
mappings, _, err := s.FindUserResourceMappings(ctx, platform.UserResourceMappingFilter{})
if err != nil {
t.Fatalf("failed to retrieve mappings: %v", err)
}
if diff := cmp.Diff(mappings, tt.wants.mappings, mappingCmpOptions...); diff != "" {
t.Errorf("mappings are different -got/+want\ndiff %s", diff)
}
})
}
}
func DeleteUserResourceMapping(
init func(UserResourceFields, *testing.T) (platform.UserResourceMappingService, func()),
t *testing.T,
) {
type args struct {
resourceID platform2.ID
userID platform2.ID
}
type wants struct {
err error
mappings []*platform.UserResourceMapping
}
tests := []struct {
name string
fields UserResourceFields
args args
wants wants
}{
{
name: "basic delete user resource mapping",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
resourceID: idOne,
userID: MustIDBase16(userOneID),
},
wants: wants{
mappings: []*platform.UserResourceMapping{},
},
},
{
name: "deleting a non-existent user",
fields: UserResourceFields{
UserResourceMappings: []*platform.UserResourceMapping{},
},
args: args{
resourceID: idOne,
userID: MustIDBase16(userOneID),
},
wants: wants{
mappings: []*platform.UserResourceMapping{},
err: fmt.Errorf("user to resource mapping not found"),
},
},
{
name: "delete user resource mapping for org",
fields: UserResourceFields{
Organizations: []*platform.Organization{
{
ID: idOne,
Name: "organization1",
},
},
Users: []*platform.User{
{
ID: MustIDBase16(userOneID),
Name: "user1",
},
},
Buckets: []*platform.Bucket{
{
ID: idOne,
Name: "bucket1",
OrgID: idOne,
},
},
UserResourceMappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
ResourceType: platform.OrgsResourceType,
MappingType: platform.UserMappingType,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
},
},
},
args: args{
resourceID: idOne,
userID: MustIDBase16(userOneID),
},
wants: wants{
mappings: []*platform.UserResourceMapping{},
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
s, done := init(tt.fields, t)
defer done()
ctx := context.Background()
err := s.DeleteUserResourceMapping(ctx, tt.args.resourceID, tt.args.userID)
if (err != nil) != (tt.wants.err != nil) {
t.Fatalf("expected error '%v' got '%v'", tt.wants.err, err)
}
if err != nil && tt.wants.err != nil {
if err.Error() != tt.wants.err.Error() {
t.Fatalf("expected error messages to match '%v' got '%v'", tt.wants.err, err.Error())
}
}
mappings, _, err := s.FindUserResourceMappings(ctx, platform.UserResourceMappingFilter{})
if err != nil {
t.Fatalf("failed to retrieve mappings: %v", err)
}
if diff := cmp.Diff(mappings, tt.wants.mappings, mappingCmpOptions...); diff != "" {
t.Errorf("mappings are different -got/+want\ndiff %s", diff)
}
})
}
}
func FindUserResourceMappings(
init func(UserResourceFields, *testing.T) (platform.UserResourceMappingService, func()),
t *testing.T,
) {
type args struct {
filter platform.UserResourceMappingFilter
}
type wants struct {
err error
mappings []*platform.UserResourceMapping
}
tests := []struct {
name string
fields UserResourceFields
args args
wants wants
}{
{
name: "basic find mappings",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
filter: platform.UserResourceMappingFilter{},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
},
},
{
name: "find mappings filtered by user",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
filter: platform.UserResourceMappingFilter{
UserID: MustIDBase16(userOneID),
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
},
},
{
name: "find mappings filtered by resource",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
filter: platform.UserResourceMappingFilter{
ResourceID: idOne,
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
},
},
{
name: "find mappings filtered by user type",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idOne,
UserID: MustIDBase16(userOneID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Owner,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
filter: platform.UserResourceMappingFilter{
UserType: platform.Owner,
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Owner,
ResourceType: platform.BucketsResourceType,
},
},
},
},
{
name: "find mappings filtered by resource type",
fields: func() UserResourceFields {
f := baseUserResourceFields()
f.UserResourceMappings = []*platform.UserResourceMapping{
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
}
return f
}(),
args: args{
filter: platform.UserResourceMappingFilter{
ResourceType: platform.BucketsResourceType,
},
},
wants: wants{
mappings: []*platform.UserResourceMapping{
{
ResourceID: idTwo,
UserID: MustIDBase16(userTwoID),
UserType: platform.Member,
ResourceType: platform.BucketsResourceType,
},
},
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
s, done := init(tt.fields, t)
defer done()
ctx := context.Background()
mappings, _, err := s.FindUserResourceMappings(ctx, tt.args.filter)
if (err != nil) != (tt.wants.err != nil) {
t.Fatalf("expected error '%v' got '%v'", tt.wants.err, err)
}
if err != nil && tt.wants.err != nil {
if err.Error() != tt.wants.err.Error() {
t.Fatalf("expected error messages to match '%v' got '%v'", tt.wants.err, err.Error())
}
}
if diff := cmp.Diff(mappings, tt.wants.mappings, mappingCmpOptions...); diff != "" {
t.Errorf("mappings are different -got/+want\ndiff %s", diff)
}
})
}
}