package writecache import ( "errors" "fmt" "github.com/nspcc-dev/neofs-node/pkg/local_object_storage/blobstor/fstree" addressSDK "github.com/nspcc-dev/neofs-sdk-go/object/address" "go.etcd.io/bbolt" ) // ErrNoDefaultBucket is returned by IterateDB when default bucket for objects is missing. var ErrNoDefaultBucket = errors.New("no default bucket") // IterationPrm contains iteration parameters. type IterationPrm struct { handler func([]byte) error ignoreErrors bool } // WithHandler sets a callback to be executed on every object. func (p *IterationPrm) WithHandler(f func([]byte) error) *IterationPrm { p.handler = f return p } // WithIgnoreErrors sets a flag indicating that errors should be ignored. func (p *IterationPrm) WithIgnoreErrors(ignore bool) *IterationPrm { p.ignoreErrors = ignore return p } // Iterate iterates over all objects present in write cache. // This is very difficult to do correctly unless write-cache is put in read-only mode. // Thus we silently fail if shard is not in read-only mode to avoid reporting misleading results. func (c *cache) Iterate(prm *IterationPrm) error { c.modeMtx.RLock() defer c.modeMtx.RUnlock() if c.mode != ModeReadOnly { return nil } err := c.db.View(func(tx *bbolt.Tx) error { b := tx.Bucket(defaultBucket) return b.ForEach(func(k, data []byte) error { if _, ok := c.flushed.Peek(string(k)); ok { return nil } return prm.handler(data) }) }) if err != nil { return err } return c.fsTree.Iterate(new(fstree.IterationPrm).WithHandler(func(addr *addressSDK.Address, data []byte) error { if _, ok := c.flushed.Peek(addr.String()); ok { return nil } return prm.handler(data) }).WithIgnoreErrors(prm.ignoreErrors)) } // IterateDB iterates over all objects stored in bbolt.DB instance and passes them to f until error return. // It is assumed that db is an underlying database of some WriteCache instance. // // Returns ErrNoDefaultBucket if there is no default bucket in db. // // DB must not be nil and should be opened. func IterateDB(db *bbolt.DB, f func(*addressSDK.Address) error) error { return db.View(func(tx *bbolt.Tx) error { b := tx.Bucket(defaultBucket) if b == nil { return ErrNoDefaultBucket } var addr *addressSDK.Address return b.ForEach(func(k, v []byte) error { if addr == nil { addr = addressSDK.NewAddress() } err := addr.Parse(string(k)) if err != nil { return fmt.Errorf("could not parse object address: %w", err) } return f(addr) }) }) }