[#131] pool: Refactor retry сalls
Signed-off-by: Leonard Lyubich <leonard@nspcc.ru>
This commit is contained in:
parent
d5ce5d63b4
commit
b508f3b71e
1 changed files with 132 additions and 68 deletions
176
pool/pool.go
176
pool/pool.go
|
@ -259,8 +259,9 @@ type innerPool struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
DefaultSessionTokenExpirationDuration = 100 // in blocks
|
defaultSessionTokenExpirationDuration = 100 // in blocks
|
||||||
DefaultSessionTokenThreshold = 5 * time.Second
|
|
||||||
|
defaultSessionTokenThreshold = 5 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
func newPool(ctx context.Context, options *BuilderOptions) (Pool, error) {
|
func newPool(ctx context.Context, options *BuilderOptions) (Pool, error) {
|
||||||
|
@ -270,11 +271,11 @@ func newPool(ctx context.Context, options *BuilderOptions) (Pool, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if options.SessionExpirationDuration == 0 {
|
if options.SessionExpirationDuration == 0 {
|
||||||
options.SessionExpirationDuration = DefaultSessionTokenExpirationDuration
|
options.SessionExpirationDuration = defaultSessionTokenExpirationDuration
|
||||||
}
|
}
|
||||||
|
|
||||||
if options.SessionTokenThreshold <= 0 {
|
if options.SessionTokenThreshold <= 0 {
|
||||||
options.SessionTokenThreshold = DefaultSessionTokenThreshold
|
options.SessionTokenThreshold = defaultSessionTokenThreshold
|
||||||
}
|
}
|
||||||
|
|
||||||
ownerID := owner.NewIDFromPublicKey(&options.Key.PublicKey)
|
ownerID := owner.NewIDFromPublicKey(&options.Key.PublicKey)
|
||||||
|
@ -588,8 +589,8 @@ func (p *pool) removeSessionTokenAfterThreshold(cfg *callConfig) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
type callContext struct {
|
type callContext struct {
|
||||||
// context for RPC
|
// base context for RPC
|
||||||
ctxBase context.Context
|
context.Context
|
||||||
|
|
||||||
client Client
|
client Client
|
||||||
|
|
||||||
|
@ -601,51 +602,96 @@ type callContext struct {
|
||||||
|
|
||||||
// flag to open default session if session token is missing
|
// flag to open default session if session token is missing
|
||||||
sessionDefault bool
|
sessionDefault bool
|
||||||
sessionToken *session.Token
|
sessionTarget func(session.Token)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *pool) prepareCallContext(ctx *callContext, cfg *callConfig) error {
|
func (p *pool) initCallContext(ctx *callContext, cfg *callConfig) error {
|
||||||
cp, err := p.connection()
|
cp, err := p.connection()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.endpoint = cp.address
|
|
||||||
ctx.client = cp.client
|
|
||||||
|
|
||||||
ctx.key = cfg.key
|
ctx.key = cfg.key
|
||||||
if ctx.key == nil {
|
if ctx.key == nil {
|
||||||
|
// use pool key if caller didn't specify its own
|
||||||
ctx.key = p.key
|
ctx.key = p.key
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.sessionDefault = cfg.useDefaultSession
|
ctx.endpoint = cp.address
|
||||||
ctx.sessionToken = cfg.stoken
|
ctx.client = cp.client
|
||||||
|
|
||||||
if ctx.sessionToken == nil && ctx.sessionDefault {
|
if ctx.sessionTarget == nil && cfg.stoken != nil {
|
||||||
|
ctx.sessionTarget(*cfg.stoken)
|
||||||
|
}
|
||||||
|
|
||||||
|
// note that we don't override session provided by the caller
|
||||||
|
ctx.sessionDefault = cfg.stoken == nil && cfg.useDefaultSession
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
type callContextWithRetry struct {
|
||||||
|
callContext
|
||||||
|
|
||||||
|
noRetry bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *pool) initCallContextWithRetry(ctx *callContextWithRetry, cfg *callConfig) error {
|
||||||
|
err := p.initCallContext(&ctx.callContext, cfg)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// don't retry if session was specified by the caller
|
||||||
|
ctx.noRetry = cfg.stoken != nil
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *pool) openDefaultSession(ctx *callContext) error {
|
||||||
cacheKey := formCacheKey(ctx.endpoint, ctx.key)
|
cacheKey := formCacheKey(ctx.endpoint, ctx.key)
|
||||||
|
|
||||||
ctx.sessionToken = p.cache.Get(cacheKey)
|
tok := p.cache.Get(cacheKey)
|
||||||
if ctx.sessionToken == nil {
|
if tok != nil {
|
||||||
|
// use cached token
|
||||||
|
ctx.sessionTarget(*tok)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// open new session
|
||||||
var cliPrm client.CreateSessionPrm
|
var cliPrm client.CreateSessionPrm
|
||||||
|
|
||||||
cliPrm.SetExp(math.MaxUint32)
|
cliPrm.SetExp(math.MaxUint32)
|
||||||
|
|
||||||
cliRes, err := ctx.client.CreateSession(ctx.ctxBase, cliPrm)
|
cliRes, err := ctx.client.CreateSession(ctx, cliPrm)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("default session: %w", err)
|
return fmt.Errorf("session API client: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.sessionToken = sessionTokenForOwner(owner.NewIDFromPublicKey(&ctx.key.PublicKey), cliRes)
|
tok = sessionTokenForOwner(owner.NewIDFromPublicKey(&ctx.key.PublicKey), cliRes)
|
||||||
|
|
||||||
_ = p.cache.Put(cacheKey, ctx.sessionToken)
|
// sign the token
|
||||||
}
|
err = tok.Sign(ctx.key)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("sign token of the opened session: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctx.sessionToken != nil && ctx.sessionToken.Signature() == nil {
|
// cache the opened session
|
||||||
err = ctx.sessionToken.Sign(ctx.key)
|
p.cache.Put(cacheKey, tok)
|
||||||
|
|
||||||
|
ctx.sessionTarget(*tok)
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return err
|
func (p *pool) handleAttemptError(ctx *callContextWithRetry, err error) bool {
|
||||||
|
isTokenErr := p.checkSessionTokenErr(err, ctx.endpoint)
|
||||||
|
// note that checkSessionTokenErr must be called
|
||||||
|
res := isTokenErr && !ctx.noRetry
|
||||||
|
|
||||||
|
ctx.noRetry = true
|
||||||
|
|
||||||
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *pool) PutObject(ctx context.Context, hdr object.Object, payload io.Reader, opts ...CallOption) (*oid.ID, error) {
|
func (p *pool) PutObject(ctx context.Context, hdr object.Object, payload io.Reader, opts ...CallOption) (*oid.ID, error) {
|
||||||
|
@ -668,11 +714,11 @@ func (p *pool) PutObject(ctx context.Context, hdr object.Object, payload io.Read
|
||||||
|
|
||||||
var ctxCall callContext
|
var ctxCall callContext
|
||||||
|
|
||||||
ctxCall.ctxBase = ctx
|
ctxCall.Context = ctx
|
||||||
|
|
||||||
err = p.prepareCallContext(&ctxCall, cfg)
|
err = p.initCallContext(&ctxCall, cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("init call context")
|
||||||
}
|
}
|
||||||
|
|
||||||
var prm client.PrmObjectPutInit
|
var prm client.PrmObjectPutInit
|
||||||
|
@ -682,11 +728,15 @@ func (p *pool) PutObject(ctx context.Context, hdr object.Object, payload io.Read
|
||||||
return nil, fmt.Errorf("init writing on API client: %w", err)
|
return nil, fmt.Errorf("init writing on API client: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
wObj.UseKey(*ctxCall.key)
|
if ctxCall.sessionDefault {
|
||||||
|
ctxCall.sessionTarget = wObj.WithinSession
|
||||||
if ctxCall.sessionToken != nil {
|
err = p.openDefaultSession(&ctxCall)
|
||||||
wObj.WithinSession(*ctxCall.sessionToken)
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("open default session: %w", err)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wObj.UseKey(*ctxCall.key)
|
||||||
|
|
||||||
if cfg.btoken != nil {
|
if cfg.btoken != nil {
|
||||||
wObj.WithBearerToken(*cfg.btoken)
|
wObj.WithBearerToken(*cfg.btoken)
|
||||||
|
@ -786,20 +836,42 @@ type ResGetObject struct {
|
||||||
Payload io.ReadCloser
|
Payload io.ReadCloser
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *pool) callWithRetry(ctx *callContextWithRetry, f func() error) error {
|
||||||
|
var err error
|
||||||
|
|
||||||
|
if ctx.sessionDefault {
|
||||||
|
err = p.openDefaultSession(&ctx.callContext)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("open default session: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = f()
|
||||||
|
|
||||||
|
if p.checkSessionTokenErr(err, ctx.endpoint) && !ctx.noRetry {
|
||||||
|
// don't retry anymore
|
||||||
|
ctx.noRetry = true
|
||||||
|
return p.callWithRetry(ctx, f)
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
func (p *pool) GetObject(ctx context.Context, addr address.Address, opts ...CallOption) (*ResGetObject, error) {
|
func (p *pool) GetObject(ctx context.Context, addr address.Address, opts ...CallOption) (*ResGetObject, error) {
|
||||||
cfg := cfgFromOpts(append(opts, useDefaultSession())...)
|
cfg := cfgFromOpts(append(opts, useDefaultSession())...)
|
||||||
|
|
||||||
var ctxCall callContext
|
var prm client.PrmObjectGet
|
||||||
|
|
||||||
ctxCall.ctxBase = ctx
|
var cc callContextWithRetry
|
||||||
|
|
||||||
err := p.prepareCallContext(&ctxCall, cfg)
|
cc.Context = ctx
|
||||||
|
cc.sessionTarget = prm.WithinSession
|
||||||
|
|
||||||
|
err := p.initCallContextWithRetry(&cc, cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var prm client.PrmObjectGet
|
|
||||||
|
|
||||||
if cnr := addr.ContainerID(); cnr != nil {
|
if cnr := addr.ContainerID(); cnr != nil {
|
||||||
prm.FromContainer(*cnr)
|
prm.FromContainer(*cnr)
|
||||||
}
|
}
|
||||||
|
@ -808,34 +880,26 @@ func (p *pool) GetObject(ctx context.Context, addr address.Address, opts ...Call
|
||||||
prm.ByID(*obj)
|
prm.ByID(*obj)
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctxCall.sessionToken != nil {
|
|
||||||
prm.WithinSession(*ctxCall.sessionToken)
|
|
||||||
}
|
|
||||||
|
|
||||||
if cfg.btoken != nil {
|
|
||||||
prm.WithBearerToken(*cfg.btoken)
|
|
||||||
}
|
|
||||||
|
|
||||||
rObj, err := ctxCall.client.ObjectGetInit(ctx, prm)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
rObj.UseKey(*ctxCall.key)
|
|
||||||
|
|
||||||
var res ResGetObject
|
var res ResGetObject
|
||||||
|
|
||||||
|
err = p.callWithRetry(&cc, func() error {
|
||||||
|
rObj, err := cc.client.ObjectGetInit(ctx, prm)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("init object reading on client: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
rObj.UseKey(*cc.key)
|
||||||
|
|
||||||
if !rObj.ReadHeader(&res.Header) {
|
if !rObj.ReadHeader(&res.Header) {
|
||||||
_, err = rObj.Close()
|
_, err = rObj.Close()
|
||||||
if p.checkSessionTokenErr(err, ctxCall.endpoint) && !cfg.isRetry {
|
return fmt.Errorf("read header: %w", err)
|
||||||
return p.GetObject(ctx, addr, append(opts, retry())...)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, fmt.Errorf("read header: %w", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
res.Payload = (*objectReadCloser)(rObj)
|
res.Payload = (*objectReadCloser)(rObj)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
return &res, nil
|
return &res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue