milvus/pkg/util/funcutil/policy.go
sthuang 70605cf5b3
enhance: Support custom privilege group for RBAC (#37087)
issue: #37031

---------

Signed-off-by: shaoting-huang <shaoting.huang@zilliz.com>
2024-11-09 08:44:28 +08:00

150 lines
4.9 KiB
Go

package funcutil
import (
"fmt"
"strings"
"github.com/cockroachdb/errors"
"github.com/samber/lo"
"go.uber.org/zap"
"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/reflect/protoreflect"
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
"github.com/milvus-io/milvus-proto/go-api/v2/milvuspb"
"github.com/milvus-io/milvus/pkg/log"
"github.com/milvus-io/milvus/pkg/util"
)
func GetVersion(m interface{}) (string, error) {
pbMsg, ok := m.(proto.Message)
if !ok {
err := fmt.Errorf("MessageDescriptorProto result is nil")
log.RatedInfo(60, "GetVersion failed", zap.Error(err))
return "", err
}
if !proto.HasExtension(pbMsg.ProtoReflect().Descriptor().Options(), milvuspb.E_MilvusExtObj) {
err := errors.New("Extension not found")
log.Error("GetExtension fail", zap.Error(err))
return "", err
}
extObj := proto.GetExtension(pbMsg.ProtoReflect().Descriptor().Options(), milvuspb.E_MilvusExtObj)
version := extObj.(*milvuspb.MilvusExt).Version
log.Debug("GetVersion success", zap.String("version", version))
return version, nil
}
func GetPrivilegeExtObj(m interface{}) (commonpb.PrivilegeExt, error) {
pbMsg, ok := m.(proto.Message)
if !ok {
err := fmt.Errorf("MessageDescriptorProto result is nil")
log.RatedInfo(60, "GetPrivilegeExtObj failed", zap.Error(err))
return commonpb.PrivilegeExt{}, err
}
if !proto.HasExtension(pbMsg.ProtoReflect().Descriptor().Options(), commonpb.E_PrivilegeExtObj) {
err := errors.New("Extension not found")
log.RatedWarn(60, "GetPrivilegeExtObj failed", zap.Error(err))
return commonpb.PrivilegeExt{}, err
}
extObj := proto.GetExtension(pbMsg.ProtoReflect().Descriptor().Options(), commonpb.E_PrivilegeExtObj)
privilegeExt := extObj.(*commonpb.PrivilegeExt)
log.RatedDebug(60, "GetPrivilegeExtObj success", zap.String("resource_type", privilegeExt.ObjectType.String()), zap.String("resource_privilege", privilegeExt.ObjectPrivilege.String()))
return commonpb.PrivilegeExt{
ObjectType: privilegeExt.ObjectType,
ObjectPrivilege: privilegeExt.ObjectPrivilege,
ObjectNameIndex: privilegeExt.ObjectNameIndex,
ObjectNameIndexs: privilegeExt.ObjectNameIndexs,
}, nil
}
// GetObjectName get object name from the grpc message according to the field index. The field is a string.
func GetObjectName(m interface{}, index int32) string {
if index <= 0 {
return util.AnyWord
}
pbMsg, ok := m.(proto.Message)
if !ok {
err := fmt.Errorf("MessageDescriptorProto result is nil")
log.RatedInfo(60, "GetObjectName fail", zap.Error(err))
return util.AnyWord
}
msgDesc := pbMsg.ProtoReflect().Descriptor()
value := pbMsg.ProtoReflect().Get(msgDesc.Fields().ByNumber(protoreflect.FieldNumber(index)))
user, ok := value.Interface().(protoreflect.Message)
if ok {
userDesc := user.Descriptor()
value = user.Get(userDesc.Fields().ByNumber(protoreflect.FieldNumber(1)))
if value.String() == "" {
return util.AnyWord
}
}
return value.String()
}
// GetObjectNames get object names from the grpc message according to the field index. The field is an array.
func GetObjectNames(m interface{}, index int32) []string {
if index <= 0 {
return []string{}
}
pbMsg, ok := m.(proto.Message)
if !ok {
err := fmt.Errorf("MessageDescriptorProto result is nil")
log.RatedInfo(60, "GetObjectNames fail", zap.Error(err))
return []string{}
}
msgDesc := pbMsg.ProtoReflect().Descriptor()
value := pbMsg.ProtoReflect().Get(msgDesc.Fields().ByNumber(protoreflect.FieldNumber(index)))
names, ok := value.Interface().(protoreflect.List)
if !ok {
return []string{}
}
res := make([]string, names.Len())
for i := 0; i < names.Len(); i++ {
res[i] = names.Get(i).String()
}
return res
}
func PolicyForPrivilege(roleName string, objectType string, objectName string, privilege string, dbName string) string {
return fmt.Sprintf(`{"PType":"p","V0":"%s","V1":"%s","V2":"%s"}`, roleName, PolicyForResource(dbName, objectType, objectName), privilege)
}
func PolicyForPrivileges(grants []*milvuspb.GrantEntity) string {
return strings.Join(lo.Map(grants, func(r *milvuspb.GrantEntity, _ int) string {
return PolicyForPrivilege(r.Role.Name, r.Object.Name, r.ObjectName, r.Grantor.Privilege.Name, r.DbName)
}), "|")
}
func PrivilegesForPolicy(policy string) []string {
return strings.Split(policy, "|")
}
func PolicyForResource(dbName string, objectType string, objectName string) string {
return fmt.Sprintf("%s-%s", objectType, CombineObjectName(dbName, objectName))
}
func CombineObjectName(dbName string, objectName string) string {
if dbName == "" {
dbName = util.DefaultDBName
}
return fmt.Sprintf("%s.%s", dbName, objectName)
}
func SplitObjectName(objectName string) (string, string) {
if !strings.Contains(objectName, ".") {
return util.DefaultDBName, objectName
}
names := strings.Split(objectName, ".")
return names[0], names[1]
}
func PolicyCheckerWithRole(policy, roleName string) bool {
return strings.Contains(policy, fmt.Sprintf(`"V0":"%s"`, roleName))
}