swift backend: limit http concurrency in Save(), Stat(), Test(), Remove(),
List(). move comment regarding problematic List() backend api (it's s3's ListObjects that has a problem, NOT swift's ObjectsWalk). As per discussion in PR #1399.
This commit is contained in:
parent
8515d093e0
commit
0268d0e7d6
2 changed files with 20 additions and 8 deletions
|
@ -433,6 +433,9 @@ func (be *Backend) List(ctx context.Context, t restic.FileType) <-chan string {
|
||||||
prefix += "/"
|
prefix += "/"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NB: unfortunately we can't protect this with be.sem.GetToken() here.
|
||||||
|
// Doing so would enable a deadlock situation (gh-1399), as ListObjects()
|
||||||
|
// starts its own goroutine and returns results via a channel.
|
||||||
listresp := be.client.ListObjects(be.cfg.Bucket, prefix, true, ctx.Done())
|
listresp := be.client.ListObjects(be.cfg.Bucket, prefix, true, ctx.Done())
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
|
|
|
@ -163,6 +163,9 @@ func (be *beSwift) Save(ctx context.Context, h restic.Handle, rd io.Reader) (err
|
||||||
|
|
||||||
debug.Log("Save %v at %v", h, objName)
|
debug.Log("Save %v at %v", h, objName)
|
||||||
|
|
||||||
|
be.sem.GetToken()
|
||||||
|
defer be.sem.ReleaseToken()
|
||||||
|
|
||||||
// Check key does not already exist
|
// Check key does not already exist
|
||||||
switch _, _, err = be.conn.Object(be.container, objName); err {
|
switch _, _, err = be.conn.Object(be.container, objName); err {
|
||||||
case nil:
|
case nil:
|
||||||
|
@ -176,11 +179,6 @@ func (be *beSwift) Save(ctx context.Context, h restic.Handle, rd io.Reader) (err
|
||||||
return errors.Wrap(err, "conn.Object")
|
return errors.Wrap(err, "conn.Object")
|
||||||
}
|
}
|
||||||
|
|
||||||
be.sem.GetToken()
|
|
||||||
defer func() {
|
|
||||||
be.sem.ReleaseToken()
|
|
||||||
}()
|
|
||||||
|
|
||||||
encoding := "binary/octet-stream"
|
encoding := "binary/octet-stream"
|
||||||
|
|
||||||
debug.Log("PutObject(%v, %v, %v)", be.container, objName, encoding)
|
debug.Log("PutObject(%v, %v, %v)", be.container, objName, encoding)
|
||||||
|
@ -196,6 +194,9 @@ func (be *beSwift) Stat(ctx context.Context, h restic.Handle) (bi restic.FileInf
|
||||||
|
|
||||||
objName := be.Filename(h)
|
objName := be.Filename(h)
|
||||||
|
|
||||||
|
be.sem.GetToken()
|
||||||
|
defer be.sem.ReleaseToken()
|
||||||
|
|
||||||
obj, _, err := be.conn.Object(be.container, objName)
|
obj, _, err := be.conn.Object(be.container, objName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
debug.Log("Object() err %v", err)
|
debug.Log("Object() err %v", err)
|
||||||
|
@ -208,6 +209,10 @@ func (be *beSwift) Stat(ctx context.Context, h restic.Handle) (bi restic.FileInf
|
||||||
// Test returns true if a blob of the given type and name exists in the backend.
|
// Test returns true if a blob of the given type and name exists in the backend.
|
||||||
func (be *beSwift) Test(ctx context.Context, h restic.Handle) (bool, error) {
|
func (be *beSwift) Test(ctx context.Context, h restic.Handle) (bool, error) {
|
||||||
objName := be.Filename(h)
|
objName := be.Filename(h)
|
||||||
|
|
||||||
|
be.sem.GetToken()
|
||||||
|
defer be.sem.ReleaseToken()
|
||||||
|
|
||||||
switch _, _, err := be.conn.Object(be.container, objName); err {
|
switch _, _, err := be.conn.Object(be.container, objName); err {
|
||||||
case nil:
|
case nil:
|
||||||
return true, nil
|
return true, nil
|
||||||
|
@ -223,6 +228,10 @@ func (be *beSwift) Test(ctx context.Context, h restic.Handle) (bool, error) {
|
||||||
// Remove removes the blob with the given name and type.
|
// Remove removes the blob with the given name and type.
|
||||||
func (be *beSwift) Remove(ctx context.Context, h restic.Handle) error {
|
func (be *beSwift) Remove(ctx context.Context, h restic.Handle) error {
|
||||||
objName := be.Filename(h)
|
objName := be.Filename(h)
|
||||||
|
|
||||||
|
be.sem.GetToken()
|
||||||
|
defer be.sem.ReleaseToken()
|
||||||
|
|
||||||
err := be.conn.ObjectDelete(be.container, objName)
|
err := be.conn.ObjectDelete(be.container, objName)
|
||||||
debug.Log("Remove(%v) -> err %v", h, err)
|
debug.Log("Remove(%v) -> err %v", h, err)
|
||||||
return errors.Wrap(err, "conn.ObjectDelete")
|
return errors.Wrap(err, "conn.ObjectDelete")
|
||||||
|
@ -240,12 +249,12 @@ func (be *beSwift) List(ctx context.Context, t restic.FileType) <-chan string {
|
||||||
go func() {
|
go func() {
|
||||||
defer close(ch)
|
defer close(ch)
|
||||||
|
|
||||||
// NB: unfortunately we can't protect this with be.sem.GetToken() here.
|
|
||||||
// Doing so would enable a deadlock situation (PR: gh-1399), as ObjectsWalk()
|
|
||||||
// starts its own goroutine and returns results via a channel.
|
|
||||||
err := be.conn.ObjectsWalk(be.container, &swift.ObjectsOpts{Prefix: prefix},
|
err := be.conn.ObjectsWalk(be.container, &swift.ObjectsOpts{Prefix: prefix},
|
||||||
func(opts *swift.ObjectsOpts) (interface{}, error) {
|
func(opts *swift.ObjectsOpts) (interface{}, error) {
|
||||||
|
be.sem.GetToken()
|
||||||
newObjects, err := be.conn.ObjectNames(be.container, opts)
|
newObjects, err := be.conn.ObjectNames(be.container, opts)
|
||||||
|
be.sem.ReleaseToken()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "conn.ObjectNames")
|
return nil, errors.Wrap(err, "conn.ObjectNames")
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue