2024-05-17 02:39:37 +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
|
|
|
|
// 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 client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
2024-05-20 12:47:38 +00:00
|
|
|
grpc_retry "github.com/grpc-ecosystem/go-grpc-middleware/retry"
|
2024-05-17 02:39:37 +00:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/milvuspb"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
authorizationHeader = `authorization`
|
|
|
|
|
|
|
|
identifierHeader = `identifier`
|
|
|
|
|
|
|
|
databaseHeader = `dbname`
|
|
|
|
)
|
|
|
|
|
|
|
|
func (c *Client) MetadataUnaryInterceptor() grpc.UnaryClientInterceptor {
|
|
|
|
return func(ctx context.Context, method string, req, reply any, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
|
|
|
|
ctx = c.metadata(ctx)
|
|
|
|
ctx = c.state(ctx)
|
|
|
|
|
|
|
|
return invoker(ctx, method, req, reply, cc, opts...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) metadata(ctx context.Context) context.Context {
|
|
|
|
for k, v := range c.config.metadataHeaders {
|
|
|
|
ctx = metadata.AppendToOutgoingContext(ctx, k, v)
|
|
|
|
}
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) state(ctx context.Context) context.Context {
|
|
|
|
c.stateMut.RLock()
|
|
|
|
defer c.stateMut.RUnlock()
|
|
|
|
|
|
|
|
if c.currentDB != "" {
|
|
|
|
ctx = metadata.AppendToOutgoingContext(ctx, databaseHeader, c.currentDB)
|
|
|
|
}
|
|
|
|
if c.identifier != "" {
|
|
|
|
ctx = metadata.AppendToOutgoingContext(ctx, identifierHeader, c.identifier)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
// ref: https://github.com/grpc-ecosystem/go-grpc-middleware
|
|
|
|
|
|
|
|
type ctxKey int
|
|
|
|
|
|
|
|
const (
|
|
|
|
RetryOnRateLimit ctxKey = iota
|
|
|
|
)
|
|
|
|
|
|
|
|
// RetryOnRateLimitInterceptor returns a new retrying unary client interceptor.
|
|
|
|
func RetryOnRateLimitInterceptor(maxRetry uint, maxBackoff time.Duration, backoffFunc grpc_retry.BackoffFuncContext) grpc.UnaryClientInterceptor {
|
|
|
|
return func(parentCtx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
|
|
|
|
if maxRetry == 0 {
|
|
|
|
return invoker(parentCtx, method, req, reply, cc, opts...)
|
|
|
|
}
|
|
|
|
var lastErr error
|
|
|
|
for attempt := uint(0); attempt < maxRetry; attempt++ {
|
|
|
|
_, err := waitRetryBackoff(parentCtx, attempt, maxBackoff, backoffFunc)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
lastErr = invoker(parentCtx, method, req, reply, cc, opts...)
|
|
|
|
rspStatus := getResultStatus(reply)
|
|
|
|
if retryOnRateLimit(parentCtx) && rspStatus.GetErrorCode() == commonpb.ErrorCode_RateLimit {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return lastErr
|
|
|
|
}
|
|
|
|
return lastErr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func retryOnRateLimit(ctx context.Context) bool {
|
|
|
|
retry, ok := ctx.Value(RetryOnRateLimit).(bool)
|
|
|
|
if !ok {
|
|
|
|
return true // default true
|
|
|
|
}
|
|
|
|
return retry
|
|
|
|
}
|
|
|
|
|
|
|
|
// getResultStatus returns status of response.
|
|
|
|
func getResultStatus(reply interface{}) *commonpb.Status {
|
|
|
|
switch r := reply.(type) {
|
|
|
|
case *commonpb.Status:
|
|
|
|
return r
|
|
|
|
case *milvuspb.MutationResult:
|
|
|
|
return r.GetStatus()
|
|
|
|
case *milvuspb.BoolResponse:
|
|
|
|
return r.GetStatus()
|
|
|
|
case *milvuspb.SearchResults:
|
|
|
|
return r.GetStatus()
|
|
|
|
case *milvuspb.QueryResults:
|
|
|
|
return r.GetStatus()
|
|
|
|
case *milvuspb.FlushResponse:
|
|
|
|
return r.GetStatus()
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func contextErrToGrpcErr(err error) error {
|
|
|
|
switch err {
|
|
|
|
case context.DeadlineExceeded:
|
|
|
|
return status.Error(codes.DeadlineExceeded, err.Error())
|
|
|
|
case context.Canceled:
|
|
|
|
return status.Error(codes.Canceled, err.Error())
|
|
|
|
default:
|
|
|
|
return status.Error(codes.Unknown, err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func waitRetryBackoff(parentCtx context.Context, attempt uint, maxBackoff time.Duration, backoffFunc grpc_retry.BackoffFuncContext) (time.Duration, error) {
|
|
|
|
var waitTime time.Duration
|
|
|
|
if attempt > 0 {
|
|
|
|
waitTime = backoffFunc(parentCtx, attempt)
|
|
|
|
}
|
|
|
|
if waitTime > 0 {
|
|
|
|
if waitTime > maxBackoff {
|
|
|
|
waitTime = maxBackoff
|
|
|
|
}
|
|
|
|
timer := time.NewTimer(waitTime)
|
|
|
|
select {
|
|
|
|
case <-parentCtx.Done():
|
|
|
|
timer.Stop()
|
|
|
|
return waitTime, contextErrToGrpcErr(parentCtx.Err())
|
|
|
|
case <-timer.C:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return waitTime, nil
|
|
|
|
}
|