rclone/vendor/storj.io/uplink/buckets.go
Caleb Case 6cd8d3c4a0 backend/tardigrade: Upgrade to uplink v1.0.6
This fixes an important bug with listing that affects users with more
than 500 objects in a listing operation.
2020-05-29 18:01:20 +01:00

129 lines
2.6 KiB
Go

// Copyright (C) 2020 Storj Labs, Inc.
// See LICENSE for copying information.
package uplink
import (
"context"
"storj.io/common/storj"
)
// ListBucketsOptions defines bucket listing options.
type ListBucketsOptions struct {
// Cursor sets the starting position of the iterator. The first item listed will be the one after the cursor.
Cursor string
}
// ListBuckets returns an iterator over the buckets.
func (project *Project) ListBuckets(ctx context.Context, options *ListBucketsOptions) *BucketIterator {
defer mon.Func().RestartTrace(&ctx)(nil)
opts := storj.BucketListOptions{
Direction: storj.After,
}
if options != nil {
opts.Cursor = options.Cursor
}
buckets := BucketIterator{
ctx: ctx,
project: project,
options: opts,
}
return &buckets
}
// BucketIterator is an iterator over a collection of buckets.
type BucketIterator struct {
ctx context.Context
project *Project
options storj.BucketListOptions
list *storj.BucketList
position int
completed bool
err error
}
// Next prepares next Bucket for reading.
// It returns false if the end of the iteration is reached and there are no more buckets, or if there is an error.
func (buckets *BucketIterator) Next() bool {
if buckets.err != nil {
buckets.completed = true
return false
}
if buckets.list == nil {
more := buckets.loadNext()
buckets.completed = !more
return more
}
if buckets.position >= len(buckets.list.Items)-1 {
if !buckets.list.More {
buckets.completed = true
return false
}
more := buckets.loadNext()
buckets.completed = !more
return more
}
buckets.position++
return true
}
func (buckets *BucketIterator) loadNext() bool {
list, err := buckets.project.db.ListBuckets(buckets.ctx, buckets.options)
if err != nil {
buckets.err = convertKnownErrors(err, "", "")
return false
}
buckets.list = &list
if list.More {
buckets.options = buckets.options.NextPage(list)
}
buckets.position = 0
return len(list.Items) > 0
}
// Err returns error, if one happened during iteration.
func (buckets *BucketIterator) Err() error {
return packageError.Wrap(buckets.err)
}
// Item returns the current bucket in the iterator.
func (buckets *BucketIterator) Item() *Bucket {
item := buckets.item()
if item == nil {
return nil
}
return &Bucket{
Name: item.Name,
Created: item.Created,
}
}
func (buckets *BucketIterator) item() *storj.Bucket {
if buckets.completed {
return nil
}
if buckets.err != nil {
return nil
}
if buckets.list == nil {
return nil
}
if len(buckets.list.Items) == 0 {
return nil
}
return &buckets.list.Items[buckets.position]
}