2021-12-16 02:07:10 +00:00
|
|
|
// Licensed to the LF AI & Data foundation under one
|
|
|
|
// or more contributor license agreements. See the NOTICE file
|
|
|
|
// distributed with this work for additional information
|
|
|
|
// regarding copyright ownership. The ASF licenses this file
|
|
|
|
// to you under the Apache License, Version 2.0 (the
|
|
|
|
// "License"); you may not use this file except in compliance
|
2021-04-19 03:12:56 +00:00
|
|
|
// with the License. You may obtain a copy of the License at
|
|
|
|
//
|
2021-12-16 02:07:10 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2021-04-19 03:12:56 +00:00
|
|
|
//
|
2021-12-16 02:07:10 +00:00
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2021-06-18 13:30:08 +00:00
|
|
|
package rootcoord
|
2021-01-21 02:01:29 +00:00
|
|
|
|
|
|
|
import (
|
2022-08-04 03:04:34 +00:00
|
|
|
"encoding/json"
|
2021-10-21 06:04:36 +00:00
|
|
|
"errors"
|
2021-01-21 02:01:29 +00:00
|
|
|
"fmt"
|
|
|
|
"math/rand"
|
2022-08-04 03:04:34 +00:00
|
|
|
"strings"
|
2021-01-21 02:01:29 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2022-09-05 05:29:11 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/util/funcutil"
|
|
|
|
|
2022-09-16 08:56:49 +00:00
|
|
|
"github.com/milvus-io/milvus/api/commonpb"
|
|
|
|
"github.com/milvus-io/milvus/api/milvuspb"
|
2022-07-22 02:20:29 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/common"
|
2021-04-22 06:45:57 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/kv"
|
2022-08-20 02:24:51 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/metastore/kv/rootcoord"
|
2022-07-22 02:20:29 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
2022-08-11 04:12:38 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/util"
|
2021-04-22 06:45:57 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
2021-01-21 02:01:29 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2022-08-11 04:12:38 +00:00
|
|
|
"go.uber.org/zap"
|
2021-01-21 02:01:29 +00:00
|
|
|
)
|
|
|
|
|
2021-04-09 08:10:12 +00:00
|
|
|
type mockTestKV struct {
|
2022-08-04 03:04:34 +00:00
|
|
|
kv.SnapShotKV
|
2021-04-09 08:10:12 +00:00
|
|
|
|
2021-05-18 06:18:02 +00:00
|
|
|
loadWithPrefix func(key string, ts typeutil.Timestamp) ([]string, []string, error)
|
2021-08-18 06:36:10 +00:00
|
|
|
save func(key, value string, ts typeutil.Timestamp) error
|
2021-10-13 07:54:33 +00:00
|
|
|
multiSave func(kvs map[string]string, ts typeutil.Timestamp) error
|
|
|
|
multiSaveAndRemoveWithPrefix func(saves map[string]string, removals []string, ts typeutil.Timestamp) error
|
2021-04-09 08:10:12 +00:00
|
|
|
}
|
|
|
|
|
2021-05-18 06:18:02 +00:00
|
|
|
func (m *mockTestKV) LoadWithPrefix(key string, ts typeutil.Timestamp) ([]string, []string, error) {
|
|
|
|
return m.loadWithPrefix(key, ts)
|
|
|
|
}
|
|
|
|
func (m *mockTestKV) Load(key string, ts typeutil.Timestamp) (string, error) {
|
|
|
|
return "", nil
|
2021-04-09 08:10:12 +00:00
|
|
|
}
|
|
|
|
|
2021-08-18 06:36:10 +00:00
|
|
|
func (m *mockTestKV) Save(key, value string, ts typeutil.Timestamp) error {
|
|
|
|
return m.save(key, value, ts)
|
2021-04-09 08:10:12 +00:00
|
|
|
}
|
|
|
|
|
2021-10-13 07:54:33 +00:00
|
|
|
func (m *mockTestKV) MultiSave(kvs map[string]string, ts typeutil.Timestamp) error {
|
|
|
|
return m.multiSave(kvs, ts)
|
2021-04-09 08:10:12 +00:00
|
|
|
}
|
|
|
|
|
2021-10-13 07:54:33 +00:00
|
|
|
func (m *mockTestKV) MultiSaveAndRemoveWithPrefix(saves map[string]string, removals []string, ts typeutil.Timestamp) error {
|
|
|
|
return m.multiSaveAndRemoveWithPrefix(saves, removals, ts)
|
2021-04-12 07:03:23 +00:00
|
|
|
}
|
2021-04-09 08:10:12 +00:00
|
|
|
|
2021-10-21 06:04:36 +00:00
|
|
|
type mockTestTxnKV struct {
|
|
|
|
kv.TxnKV
|
|
|
|
loadWithPrefix func(key string) ([]string, []string, error)
|
|
|
|
save func(key, value string) error
|
|
|
|
multiSave func(kvs map[string]string) error
|
|
|
|
multiSaveAndRemoveWithPrefix func(saves map[string]string, removals []string) error
|
2022-04-11 11:49:34 +00:00
|
|
|
remove func(key string) error
|
2022-06-17 10:08:12 +00:00
|
|
|
multiRemove func(keys []string) error
|
2022-08-04 03:04:34 +00:00
|
|
|
load func(key string) (string, error)
|
2022-08-26 11:22:56 +00:00
|
|
|
removeWithPrefix func(key string) error
|
2021-10-21 06:04:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockTestTxnKV) LoadWithPrefix(key string) ([]string, []string, error) {
|
|
|
|
return m.loadWithPrefix(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockTestTxnKV) Save(key, value string) error {
|
|
|
|
return m.save(key, value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockTestTxnKV) MultiSave(kvs map[string]string) error {
|
|
|
|
return m.multiSave(kvs)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockTestTxnKV) MultiSaveAndRemoveWithPrefix(saves map[string]string, removals []string) error {
|
|
|
|
return m.multiSaveAndRemoveWithPrefix(saves, removals)
|
|
|
|
}
|
|
|
|
|
2022-04-11 11:49:34 +00:00
|
|
|
func (m *mockTestTxnKV) Remove(key string) error {
|
|
|
|
return m.remove(key)
|
|
|
|
}
|
|
|
|
|
2022-06-17 10:08:12 +00:00
|
|
|
func (m *mockTestTxnKV) MultiRemove(keys []string) error {
|
|
|
|
return m.multiRemove(keys)
|
|
|
|
}
|
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
func (m *mockTestTxnKV) Load(key string) (string, error) {
|
|
|
|
return m.load(key)
|
|
|
|
}
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
func (m *mockTestTxnKV) RemoveWithPrefix(key string) error {
|
|
|
|
return m.removeWithPrefix(key)
|
|
|
|
}
|
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
func generateMetaTable(t *testing.T) (*MetaTable, *mockTestKV, *mockTestTxnKV, func()) {
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
|
|
|
Params.Init()
|
|
|
|
|
|
|
|
mockSnapshotKV := &mockTestKV{
|
2022-07-22 02:20:29 +00:00
|
|
|
loadWithPrefix: func(key string, ts typeutil.Timestamp) ([]string, []string, error) {
|
|
|
|
return nil, nil, nil
|
|
|
|
},
|
|
|
|
}
|
2021-10-21 06:04:36 +00:00
|
|
|
mockTxnKV := &mockTestTxnKV{
|
2022-09-05 05:29:11 +00:00
|
|
|
loadWithPrefix: func(key string) ([]string, []string, error) { return nil, nil, nil },
|
|
|
|
save: func(key, value string) error { return nil },
|
|
|
|
multiSave: func(kvs map[string]string) error { return nil },
|
|
|
|
multiSaveAndRemoveWithPrefix: func(kvs map[string]string, removal []string) error { return nil },
|
|
|
|
remove: func(key string) error { return nil },
|
2021-10-21 06:04:36 +00:00
|
|
|
}
|
2022-07-22 02:20:29 +00:00
|
|
|
|
2022-09-05 05:29:11 +00:00
|
|
|
mockMt := &MetaTable{catalog: &rootcoord.Catalog{Txn: mockTxnKV, Snapshot: mockSnapshotKV}}
|
|
|
|
return mockMt, mockSnapshotKV, mockTxnKV, func() {}
|
2021-01-21 02:01:29 +00:00
|
|
|
}
|
2021-05-18 09:12:17 +00:00
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
func TestRbacCreateRole(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: ""})
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return nil
|
|
|
|
}
|
2022-08-23 02:26:53 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", common.NewKeyNotExistError(key)
|
|
|
|
}
|
2022-08-24 02:02:52 +00:00
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, nil
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role1"})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
2022-08-23 02:26:53 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", fmt.Errorf("load error")
|
|
|
|
}
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role1"})
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role1"})
|
|
|
|
assert.Equal(t, true, common.IsIgnorableError(err))
|
|
|
|
|
2022-08-24 02:02:52 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", common.NewKeyNotExistError(key)
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return fmt.Errorf("save error")
|
|
|
|
}
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role2"})
|
|
|
|
assert.NotNil(t, err)
|
2022-08-24 02:02:52 +00:00
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("loadWithPrefix error")
|
|
|
|
}
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role2"})
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
Params.ProxyCfg.MaxRoleNum = 2
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/a", key + "/b"}, []string{}, nil
|
|
|
|
}
|
|
|
|
err = mt.CreateRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role2"})
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
Params.ProxyCfg.MaxRoleNum = 10
|
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacDropRole(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
err = mt.DropRole(util.DefaultTenant, "role1")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return fmt.Errorf("delete error")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = mt.DropRole(util.DefaultTenant, "role2")
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacOperateRole(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: " "}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType_AddUserToRole)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: " "}, milvuspb.OperateUserRoleType_AddUserToRole)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType(100))
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-23 02:26:53 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", common.NewKeyNotExistError(key)
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType_AddUserToRole)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return fmt.Errorf("save error")
|
|
|
|
}
|
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType_AddUserToRole)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return nil
|
|
|
|
}
|
2022-08-23 02:26:53 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", nil
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType_RemoveUserFromRole)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return fmt.Errorf("remove error")
|
|
|
|
}
|
|
|
|
err = mt.OperateUserRole(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, &milvuspb.RoleEntity{Name: "role"}, milvuspb.OperateUserRoleType_RemoveUserFromRole)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacSelectRole(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
_, err = mt.SelectRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: ""}, false)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix error")
|
|
|
|
}
|
|
|
|
_, err = mt.SelectRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role"}, true)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/key1", key + "/key2", key + "/a/err"}, []string{"value1", "value2", "values3"}, nil
|
|
|
|
}
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
results, _ := mt.SelectRole(util.DefaultTenant, nil, false)
|
|
|
|
assert.Equal(t, 2, len(results))
|
|
|
|
results, _ = mt.SelectRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role"}, false)
|
|
|
|
assert.Equal(t, 1, len(results))
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix error")
|
|
|
|
}
|
|
|
|
_, err = mt.SelectRole(util.DefaultTenant, nil, false)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", fmt.Errorf("load error")
|
|
|
|
}
|
|
|
|
_, err = mt.SelectRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: "role"}, false)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
roleName := "role1"
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/user1/" + roleName, key + "/user2/role2", key + "/user3/" + roleName, key + "/err"}, []string{"value1", "value2", "values3", "value4"}, nil
|
|
|
|
}
|
|
|
|
results, err = mt.SelectRole(util.DefaultTenant, &milvuspb.RoleEntity{Name: roleName}, true)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 1, len(results))
|
|
|
|
assert.Equal(t, 2, len(results[0].Users))
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
2022-08-20 02:24:51 +00:00
|
|
|
if key == rootcoord.RoleMappingPrefix {
|
2022-08-04 03:04:34 +00:00
|
|
|
return []string{key + "/user1/role2", key + "/user2/role2", key + "/user1/role1", key + "/user2/role1"}, []string{"value1", "value2", "values3", "value4"}, nil
|
2022-08-20 02:24:51 +00:00
|
|
|
} else if key == rootcoord.RolePrefix {
|
2022-08-04 03:04:34 +00:00
|
|
|
return []string{key + "/role1", key + "/role2", key + "/role3"}, []string{"value1", "value2", "values3"}, nil
|
|
|
|
} else {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
results, err = mt.SelectRole(util.DefaultTenant, nil, true)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 3, len(results))
|
|
|
|
for _, result := range results {
|
|
|
|
if result.Role.Name == "role1" {
|
|
|
|
assert.Equal(t, 2, len(result.Users))
|
|
|
|
} else if result.Role.Name == "role2" {
|
|
|
|
assert.Equal(t, 2, len(result.Users))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacSelectUser(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
_, err = mt.SelectUser(util.DefaultTenant, &milvuspb.UserEntity{Name: ""}, false)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
credentialInfo := internalpb.CredentialInfo{
|
|
|
|
EncryptedPassword: "password",
|
|
|
|
}
|
|
|
|
credentialInfoByte, _ := json.Marshal(credentialInfo)
|
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return string(credentialInfoByte), nil
|
|
|
|
}
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/key1", key + "/key2"}, []string{string(credentialInfoByte), string(credentialInfoByte)}, nil
|
|
|
|
}
|
|
|
|
results, err := mt.SelectUser(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, false)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 1, len(results))
|
|
|
|
results, err = mt.SelectUser(util.DefaultTenant, nil, false)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 2, len(results))
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/key1", key + "/key2", key + "/a/err"}, []string{"value1", "value2", "values3"}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return string(credentialInfoByte), nil
|
|
|
|
}
|
|
|
|
results, err = mt.SelectUser(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, true)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 1, len(results))
|
|
|
|
assert.Equal(t, 2, len(results[0].Roles))
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
logger.Debug("simfg", zap.String("key", key))
|
2022-08-20 02:24:51 +00:00
|
|
|
if strings.Contains(key, rootcoord.RoleMappingPrefix) {
|
2022-08-04 03:04:34 +00:00
|
|
|
if strings.Contains(key, "user1") {
|
|
|
|
return []string{key + "/role2", key + "/role1", key + "/role3"}, []string{"value1", "value4", "value2"}, nil
|
|
|
|
} else if strings.Contains(key, "user2") {
|
|
|
|
return []string{key + "/role2"}, []string{"value1"}, nil
|
|
|
|
}
|
|
|
|
return []string{}, []string{}, nil
|
2022-08-20 02:24:51 +00:00
|
|
|
} else if key == rootcoord.CredentialPrefix {
|
2022-08-04 03:04:34 +00:00
|
|
|
return []string{key + "/user1", key + "/user2", key + "/user3"}, []string{string(credentialInfoByte), string(credentialInfoByte), string(credentialInfoByte)}, nil
|
|
|
|
} else {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
results, err = mt.SelectUser(util.DefaultTenant, nil, true)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 3, len(results))
|
|
|
|
for _, result := range results {
|
|
|
|
if result.User.Name == "user1" {
|
|
|
|
assert.Equal(t, 3, len(result.Roles))
|
|
|
|
} else if result.User.Name == "user2" {
|
|
|
|
assert.Equal(t, 1, len(result.Roles))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, nil
|
|
|
|
}
|
|
|
|
_, err = mt.SelectUser(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, true)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix error")
|
|
|
|
}
|
|
|
|
_, err = mt.SelectUser(util.DefaultTenant, &milvuspb.UserEntity{Name: "user"}, true)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
_, err = mt.SelectUser(util.DefaultTenant, nil, true)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacOperatePrivilege(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
entity := &milvuspb.GrantEntity{}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
entity.ObjectName = "col1"
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
entity.Object = &milvuspb.ObjectEntity{Name: commonpb.ObjectType_Collection.String()}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
entity.Role = &milvuspb.RoleEntity{Name: "admin"}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
entity.Grantor = &milvuspb.GrantorEntity{}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
entity.Grantor.Privilege = &milvuspb.PrivilegeEntity{Name: commonpb.ObjectPrivilege_PrivilegeLoad.String()}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, 100)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
return "fail", fmt.Errorf("load error")
|
|
|
|
}
|
|
|
|
entity.Grantor.User = &milvuspb.UserEntity{Name: "user2"}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Revoke)
|
|
|
|
assert.NotNil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
return "", common.NewKeyNotExistError(key)
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Revoke)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.NotNil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.True(t, common.IsIgnorableError(err))
|
2022-08-04 03:04:34 +00:00
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
return "", common.NewKeyNotExistError(key)
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
granteeKey := funcutil.HandleTenantForEtcdKey(rootcoord.GranteePrefix, util.DefaultTenant, fmt.Sprintf("%s/%s/%s", entity.Role.Name, entity.Object.Name, entity.ObjectName))
|
|
|
|
granteeID := "123456"
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == granteeKey {
|
|
|
|
return granteeID, nil
|
|
|
|
}
|
|
|
|
return "", errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == granteeKey {
|
|
|
|
return granteeID, nil
|
|
|
|
}
|
|
|
|
return "", common.NewKeyNotExistError(key)
|
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Revoke)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
assert.True(t, common.IsIgnorableError(err))
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return errors.New("test error")
|
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
|
|
|
mockTxnKV.save = func(key, value string) error {
|
|
|
|
return nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
|
|
|
assert.Nil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
|
2022-08-23 02:26:53 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == granteeKey {
|
|
|
|
return granteeID, nil
|
|
|
|
}
|
|
|
|
return "", nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Grant)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.NotNil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.True(t, common.IsIgnorableError(err))
|
|
|
|
|
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
|
|
|
if key == granteeKey {
|
|
|
|
return granteeID, nil
|
|
|
|
}
|
|
|
|
return "", nil
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Revoke)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
mockTxnKV.remove = func(key string) error {
|
|
|
|
return errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
err = mt.OperatePrivilege(util.DefaultTenant, entity, milvuspb.OperatePrivilegeType_Revoke)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.NotNil(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacSelectGrant(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
entity := &milvuspb.GrantEntity{}
|
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
entity.Role = &milvuspb.RoleEntity{Name: ""}
|
2022-08-04 03:04:34 +00:00
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
entity.Role = &milvuspb.RoleEntity{Name: "admin"}
|
|
|
|
entity.ObjectName = "Collection"
|
|
|
|
entity.Object = &milvuspb.ObjectEntity{Name: "col"}
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
return "", errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
granteeID := "123456"
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.load = func(key string) (string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
return granteeID, nil
|
|
|
|
}
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return nil, nil, errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/PrivilegeInsert", key + "/*", key + "/a/b"}, []string{"root", "root", "root"}, nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
grantEntities, err := mt.SelectGrant(util.DefaultTenant, entity)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.Nil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.Equal(t, 2, len(grantEntities))
|
2022-08-04 03:04:34 +00:00
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
entity.Role = &milvuspb.RoleEntity{Name: "role1"}
|
|
|
|
entity.ObjectName = ""
|
|
|
|
entity.Object = nil
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return nil, nil, errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
granteeID1 := "123456"
|
|
|
|
granteeID2 := "147258"
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
if key == funcutil.HandleTenantForEtcdKey(rootcoord.GranteePrefix, util.DefaultTenant, entity.Role.Name) {
|
|
|
|
return []string{key + "/Collection/col1", key + "/Collection/col2", key + "/Collection/col1/x"}, []string{granteeID1, granteeID1, granteeID2}, nil
|
|
|
|
}
|
|
|
|
return nil, nil, errors.New("test error")
|
|
|
|
}
|
|
|
|
_, err = mt.SelectGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NotNil(t, err)
|
2022-08-04 03:04:34 +00:00
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == funcutil.HandleTenantForEtcdKey(rootcoord.GranteePrefix, util.DefaultTenant, entity.Role.Name) {
|
|
|
|
return []string{key + "/Collection/col1", key + "/Collection/col2", key + "/Collection/col1/x"}, []string{granteeID1, granteeID1, granteeID2}, nil
|
|
|
|
}
|
|
|
|
return []string{key + "/PrivilegeInsert", key + "/*", key + "/a/b"}, []string{"root", "root", "root"}, nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
grantEntities, err = mt.SelectGrant(util.DefaultTenant, entity)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.Nil(t, err)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.Equal(t, 4, len(grantEntities))
|
|
|
|
}
|
2022-08-04 03:04:34 +00:00
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
func TestRbacDropGrant(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
var (
|
|
|
|
roleName = "foo"
|
|
|
|
entity *milvuspb.RoleEntity
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
err = mt.DropGrant(util.DefaultTenant, nil)
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
|
|
|
entity = &milvuspb.RoleEntity{Name: ""}
|
|
|
|
err = mt.DropGrant(util.DefaultTenant, entity)
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
|
|
|
entity.Name = roleName
|
|
|
|
mockTxnKV.removeWithPrefix = func(key string) error {
|
|
|
|
return nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
err = mt.DropGrant(util.DefaultTenant, entity)
|
|
|
|
assert.NoError(t, err)
|
2022-08-04 03:04:34 +00:00
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
mockTxnKV.removeWithPrefix = func(key string) error {
|
|
|
|
return errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
err = mt.DropGrant(util.DefaultTenant, entity)
|
|
|
|
assert.Error(t, err)
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacListPolicy(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix err")
|
|
|
|
}
|
|
|
|
policies, err := mt.ListPolicy(util.DefaultTenant)
|
2022-08-26 11:22:56 +00:00
|
|
|
assert.Error(t, err)
|
2022-08-04 03:04:34 +00:00
|
|
|
assert.Equal(t, 0, len(policies))
|
|
|
|
|
2022-08-26 11:22:56 +00:00
|
|
|
granteeID1 := "123456"
|
|
|
|
granteeID2 := "147258"
|
2022-08-04 03:04:34 +00:00
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == funcutil.HandleTenantForEtcdKey(rootcoord.GranteePrefix, util.DefaultTenant, "") {
|
|
|
|
return []string{key + "/alice/collection/col1", key + "/tom/collection/col2", key + "/tom/collection/a/col2"}, []string{granteeID1, granteeID2, granteeID2}, nil
|
|
|
|
}
|
|
|
|
return []string{}, []string{}, errors.New("test error")
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
_, err = mt.ListPolicy(util.DefaultTenant)
|
|
|
|
assert.Error(t, err)
|
2022-08-04 03:04:34 +00:00
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
2022-08-26 11:22:56 +00:00
|
|
|
if key == funcutil.HandleTenantForEtcdKey(rootcoord.GranteePrefix, util.DefaultTenant, "") {
|
|
|
|
return []string{key + "/alice/collection/col1", key + "/tom/collection/col2", key + "/tom/collection/a/col2"}, []string{granteeID1, granteeID2, granteeID2}, nil
|
|
|
|
}
|
|
|
|
return []string{key + "/PrivilegeInsert", key + "/*", key + "/a/b"}, []string{"root", "root", "root"}, nil
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
2022-08-26 11:22:56 +00:00
|
|
|
policies, err = mt.ListPolicy(util.DefaultTenant)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, 4, len(policies))
|
2022-08-04 03:04:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRbacListUserRole(t *testing.T) {
|
|
|
|
mt, _, mockTxnKV, closeCli := generateMetaTable(t)
|
|
|
|
defer closeCli()
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{}, []string{}, fmt.Errorf("load with prefix err")
|
|
|
|
}
|
|
|
|
userRoles, err := mt.ListUserRole(util.DefaultTenant)
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
assert.Equal(t, 0, len(userRoles))
|
|
|
|
|
|
|
|
mockTxnKV.loadWithPrefix = func(key string) ([]string, []string, error) {
|
|
|
|
return []string{key + "/user1/role2", key + "/user2/role2", key + "/user1/role1", key + "/user2/role1", key + "/user2/role1/a"}, []string{"value1", "value2", "values3", "value4", "value5"}, nil
|
|
|
|
}
|
|
|
|
userRoles, err = mt.ListUserRole(util.DefaultTenant)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, 4, len(userRoles))
|
|
|
|
}
|