-
Notifications
You must be signed in to change notification settings - Fork 116
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix: Support retry on milvus errors and retry on ErrInconsistentReque…
…ry (#675) 1. Support retry on milvus errors. 2. Retry on ErrInconsistentRequery for search and hybrid search operations. issue: milvus-io/milvus#29656 related milvus pr: milvus-io/milvus#31343 --------- Signed-off-by: bigsheeper <yihao.dai@zilliz.com>
- Loading branch information
1 parent
d927718
commit bede290
Showing
5 changed files
with
369 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,141 @@ | ||
// 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" | ||
|
||
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb" | ||
) | ||
|
||
type config struct { | ||
attempts uint | ||
sleep time.Duration | ||
maxSleepTime time.Duration | ||
merrCodes []int32 | ||
} | ||
|
||
func newDefaultConfig() *config { | ||
return &config{ | ||
attempts: uint(10), | ||
sleep: 200 * time.Millisecond, | ||
maxSleepTime: 3 * time.Second, | ||
} | ||
} | ||
|
||
// Option is used to config the retry function. | ||
type Option func(*config) | ||
|
||
// Attempts is used to config the max retry times. | ||
func Attempts(attempts uint) Option { | ||
return func(c *config) { | ||
c.attempts = attempts | ||
} | ||
} | ||
|
||
// Sleep is used to config the initial interval time of each execution. | ||
func Sleep(sleep time.Duration) Option { | ||
return func(c *config) { | ||
c.sleep = sleep | ||
// ensure max retry interval is always larger than retry interval | ||
if c.sleep*2 > c.maxSleepTime { | ||
c.maxSleepTime = 2 * c.sleep | ||
} | ||
} | ||
} | ||
|
||
// MaxSleepTime is used to config the max interval time of each execution. | ||
func MaxSleepTime(maxSleepTime time.Duration) Option { | ||
return func(c *config) { | ||
// ensure max retry interval is always larger than retry interval | ||
if c.sleep*2 > maxSleepTime { | ||
c.maxSleepTime = 2 * c.sleep | ||
} else { | ||
c.maxSleepTime = maxSleepTime | ||
} | ||
} | ||
} | ||
|
||
func OnMerrCodes(codes ...int32) Option { | ||
return func(c *config) { | ||
c.merrCodes = append(c.merrCodes, codes...) | ||
} | ||
} | ||
|
||
func contains(codes []int32, target int32) bool { | ||
for _, c := range codes { | ||
if c == target { | ||
return true | ||
} | ||
} | ||
return false | ||
} | ||
|
||
func RetryOnMilvusErrors(ctx context.Context, fn func() (interface{}, error), opts ...Option) (interface{}, error) { | ||
if ctx.Err() != nil { | ||
return nil, ctx.Err() | ||
} | ||
|
||
c := newDefaultConfig() | ||
for _, opt := range opts { | ||
opt(c) | ||
} | ||
|
||
if len(c.merrCodes) == 0 { | ||
return fn() | ||
} | ||
|
||
var lastResp interface{} | ||
for i := uint(0); i < c.attempts; i++ { | ||
resp, err := fn() | ||
if err != nil { | ||
return resp, err | ||
} | ||
var code int32 | ||
switch r := resp.(type) { | ||
case interface{ GetStatus() *commonpb.Status }: | ||
code = r.GetStatus().GetCode() | ||
case interface{ GetCode() int32 }: | ||
code = r.GetCode() | ||
default: | ||
return resp, nil | ||
} | ||
if code == 0 || !contains(c.merrCodes, code) { | ||
return resp, nil | ||
} | ||
|
||
deadline, ok := ctx.Deadline() | ||
if ok && time.Until(deadline) < c.sleep { | ||
return resp, context.DeadlineExceeded | ||
} | ||
|
||
lastResp = resp | ||
|
||
select { | ||
case <-time.After(c.sleep): | ||
case <-ctx.Done(): | ||
return lastResp, ctx.Err() | ||
} | ||
|
||
c.sleep *= 2 | ||
if c.sleep > c.maxSleepTime { | ||
c.sleep = c.maxSleepTime | ||
} | ||
} | ||
return lastResp, nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,180 @@ | ||
// 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" | ||
"errors" | ||
"testing" | ||
"time" | ||
|
||
"github.com/stretchr/testify/assert" | ||
|
||
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb" | ||
) | ||
|
||
func TestRetryOnMilvusErrors(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
n := 0 | ||
testFn := func() (interface{}, error) { | ||
if n < 3 { | ||
n++ | ||
return &commonpb.Status{ | ||
Code: 0, | ||
}, nil | ||
} | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100)) | ||
assert.Equal(t, int32(0), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.NoError(t, err) | ||
t.Log(resp) | ||
} | ||
|
||
func TestOnNoCode(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
n := 0 | ||
testFn := func() (interface{}, error) { | ||
if n < 3 { | ||
n++ | ||
return &commonpb.Status{ | ||
Code: 0, | ||
}, nil | ||
} | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn) | ||
assert.Equal(t, int32(0), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.NoError(t, err) | ||
t.Log(resp) | ||
} | ||
|
||
func TestReturnErr(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
testFn := func() (interface{}, error) { | ||
return nil, errors.New("mock err") | ||
} | ||
|
||
_, err := RetryOnMilvusErrors(ctx, testFn) | ||
assert.Error(t, err) | ||
t.Log(err) | ||
} | ||
|
||
func TestAttempts(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
testFn := func() (interface{}, error) { | ||
t.Log("executed") | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100), Attempts(1)) | ||
assert.Equal(t, int32(100), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.NoError(t, err) | ||
t.Log(resp) | ||
} | ||
|
||
func TestMaxSleepTime(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
testFn := func() (interface{}, error) { | ||
t.Log("executed") | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100), Attempts(3), MaxSleepTime(200*time.Millisecond)) | ||
assert.Equal(t, int32(100), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.NoError(t, err) | ||
t.Log(resp) | ||
} | ||
|
||
func TestSleep(t *testing.T) { | ||
ctx := context.Background() | ||
|
||
testFn := func() (interface{}, error) { | ||
t.Log("executed") | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100), Attempts(3), Sleep(500*time.Millisecond)) | ||
assert.Equal(t, int32(100), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.NoError(t, err) | ||
t.Log(resp) | ||
} | ||
|
||
func TestContextDeadline(t *testing.T) { | ||
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second) | ||
defer cancel() | ||
|
||
testFn := func() (interface{}, error) { | ||
t.Log("executed") | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100)) | ||
assert.Equal(t, int32(100), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.Error(t, err) | ||
assert.ErrorIs(t, err, context.DeadlineExceeded) | ||
t.Log(resp) | ||
} | ||
|
||
func TestContextCancel(t *testing.T) { | ||
ctx, cancel := context.WithCancel(context.Background()) | ||
|
||
testFn := func() (interface{}, error) { | ||
t.Log("executed") | ||
return &commonpb.Status{ | ||
Reason: "mock err", | ||
Code: 100, | ||
}, nil | ||
} | ||
|
||
go func() { | ||
time.Sleep(100 * time.Millisecond) | ||
cancel() | ||
}() | ||
|
||
resp, err := RetryOnMilvusErrors(ctx, testFn, OnMerrCodes(100)) | ||
assert.Equal(t, int32(100), resp.(interface{ GetCode() int32 }).GetCode()) | ||
assert.Error(t, err) | ||
assert.ErrorIs(t, err, context.Canceled) | ||
t.Log(resp) | ||
t.Log(err) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.