555 lines
14 KiB
Go
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)
|
|
}
|
|
})
|
|
}
|
|
}
|