congqixia 7b8ecdaad5
enhance: Add accesslog field for template value length info (#44723)
Related to #36672

Add accesslog field displaying value length for search/query request may
help developers debug related issues

---------

Signed-off-by: Congqi Xia <congqi.xia@zilliz.com>
2025-10-11 18:23:57 +08:00

153 lines
4.1 KiB
Go

// 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
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// 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.
package info
import (
"context"
"encoding/json"
"fmt"
"strings"
"github.com/cockroachdb/errors"
"go.uber.org/atomic"
"google.golang.org/grpc/metadata"
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
"github.com/milvus-io/milvus-proto/go-api/v2/schemapb"
"github.com/milvus-io/milvus/pkg/v2/util"
"github.com/milvus-io/milvus/pkg/v2/util/crypto"
"github.com/milvus-io/milvus/pkg/v2/util/funcutil"
)
var ClusterPrefix atomic.String
func getCurUserFromContext(ctx context.Context) (string, error) {
md, ok := metadata.FromIncomingContext(ctx)
if !ok {
return "", errors.New("fail to get md from the context")
}
authorization, ok := md[strings.ToLower(util.HeaderAuthorize)]
if !ok || len(authorization) < 1 {
return "", fmt.Errorf("fail to get authorization from the md, authorize:[%s]", util.HeaderAuthorize)
}
token := authorization[0]
rawToken, err := crypto.Base64Decode(token)
if err != nil {
return "", fmt.Errorf("fail to decode the token, token: %s", token)
}
secrets := strings.SplitN(rawToken, util.CredentialSeperator, 2)
if len(secrets) < 2 {
return "", fmt.Errorf("fail to get user info from the raw token, raw token: %s", rawToken)
}
username := secrets[0]
return username, nil
}
func getSdkVersionByUserAgent(ctx context.Context) string {
md, ok := metadata.FromIncomingContext(ctx)
if !ok {
return Unknown
}
UserAgent, ok := md[util.HeaderUserAgent]
if !ok {
return Unknown
}
SdkType, ok := getSdkTypeByUserAgent(UserAgent)
if !ok {
return Unknown
}
return SdkType + "-" + Unknown
}
func getSdkTypeByUserAgent(userAgents []string) (string, bool) {
if len(userAgents) == 0 {
return "", false
}
userAgent := userAgents[0]
switch {
case strings.HasPrefix(userAgent, "grpc-node-js"):
return "nodejs", true
case strings.HasPrefix(userAgent, "grpc-python"):
return "Python", true
case strings.HasPrefix(userAgent, "grpc-go"):
return "Golang", true
case strings.HasPrefix(userAgent, "grpc-java"):
return "Java", true
default:
return "", false
}
}
func getAnnsFieldFromKvs(kvs []*commonpb.KeyValuePair) string {
field, err := funcutil.GetAttrByKeyFromRepeatedKV("anns_field", kvs)
if err != nil {
return "default"
}
return field
}
func getLengthFromTemplateValue(tv *schemapb.TemplateValue) int {
arrayValues := tv.GetArrayVal()
// other single value data type
if arrayValues == nil {
return 1
}
switch arrayValues.GetData().(type) {
case *schemapb.TemplateArrayValue_BoolData:
return len(arrayValues.GetBoolData().GetData())
case *schemapb.TemplateArrayValue_LongData:
return len(arrayValues.GetLongData().GetData())
case *schemapb.TemplateArrayValue_DoubleData:
return len(arrayValues.GetDoubleData().GetData())
case *schemapb.TemplateArrayValue_StringData:
return len(arrayValues.GetStringData().GetData())
case *schemapb.TemplateArrayValue_JsonData:
return len(arrayValues.GetJsonData().GetData())
default:
// undefined
return -1
}
}
func listToString(strs []string) string {
result := "["
for i, str := range strs {
if i != 0 {
result += ", "
}
result += "\"" + str + "\""
}
return result + "]"
}
func kvsToString(kvs []*commonpb.KeyValuePair) string {
m := make(map[string]string)
for _, kv := range kvs {
m[kv.GetKey()] = kv.GetValue()
}
v, err := json.Marshal(m)
if err != nil {
return Unknown
}
return string(v)
}