``` goos: linux goarch: amd64 pkg: git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/metabase cpu: 11th Gen Intel(R) Core(TM) i5-1135G7 @ 2.40GHz │ master │ expired │ │ sec/op │ sec/op vs base │ Select/string_equal-8 4.007m ± 10% 3.529m ± 11% -11.94% (p=0.000 n=10) Select/string_not_equal-8 3.834m ± 12% 3.440m ± 7% -10.29% (p=0.029 n=10) Select/common_prefix-8 3.470m ± 9% 3.240m ± 6% ~ (p=0.105 n=10) Select/unknown-8 3.156m ± 3% 3.198m ± 6% ~ (p=0.631 n=10) geomean 3.602m 3.349m -7.03% │ master │ expired │ │ B/op │ B/op vs base │ Select/string_equal-8 1.907Mi ± 0% 1.885Mi ± 0% -1.18% (p=0.000 n=10) Select/string_not_equal-8 1.907Mi ± 0% 1.885Mi ± 0% -1.18% (p=0.000 n=10) Select/common_prefix-8 1.907Mi ± 0% 1.885Mi ± 0% -1.18% (p=0.000 n=10) Select/unknown-8 1.900Mi ± 0% 1.877Mi ± 0% -1.18% (p=0.000 n=10) geomean 1.905Mi 1.883Mi -1.18% │ master │ expired │ │ allocs/op │ allocs/op vs base │ Select/string_equal-8 47.03k ± 0% 46.04k ± 0% -2.12% (p=0.000 n=10) Select/string_not_equal-8 47.03k ± 0% 46.04k ± 0% -2.12% (p=0.000 n=10) Select/common_prefix-8 47.03k ± 0% 46.04k ± 0% -2.12% (p=0.000 n=10) Select/unknown-8 46.04k ± 0% 45.05k ± 0% -2.16% (p=0.000 n=10) geomean 46.78k 45.79k -2.13% ``` Change-Id: I9c7a5e1f5c8b9eb3f25a563fd74c6ad2a9d1b92e Signed-off-by: Evgenii Stratonikov <e.stratonikov@yadro.com>
113 lines
3 KiB
Go
113 lines
3 KiB
Go
package meta
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"errors"
|
|
"strconv"
|
|
"time"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/internal/metaerr"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-observability/tracing"
|
|
cid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id"
|
|
oid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id"
|
|
"go.etcd.io/bbolt"
|
|
"go.opentelemetry.io/otel/attribute"
|
|
"go.opentelemetry.io/otel/trace"
|
|
)
|
|
|
|
var errInvalidEpochValueLength = errors.New("could not parse expiration epoch: invalid data length")
|
|
|
|
// FilterExpired return expired items from addresses.
|
|
// Address considered expired if metabase does contain information about expiration and
|
|
// expiration epoch is less than epoch.
|
|
func (db *DB) FilterExpired(ctx context.Context, epoch uint64, addresses []oid.Address) ([]oid.Address, error) {
|
|
var (
|
|
startedAt = time.Now()
|
|
success = true
|
|
)
|
|
defer func() {
|
|
db.metrics.AddMethodDuration("FilterExpired", time.Since(startedAt), success)
|
|
}()
|
|
|
|
_, span := tracing.StartSpanFromContext(ctx, "metabase.FilterExpired",
|
|
trace.WithAttributes(
|
|
attribute.String("epoch", strconv.FormatUint(epoch, 10)),
|
|
attribute.Int("addr_count", len(addresses)),
|
|
))
|
|
defer span.End()
|
|
|
|
db.modeMtx.RLock()
|
|
defer db.modeMtx.RUnlock()
|
|
|
|
if db.mode.NoMetabase() {
|
|
return nil, ErrDegradedMode
|
|
}
|
|
|
|
result := make([]oid.Address, 0, len(addresses))
|
|
containerIDToObjectIDs := make(map[cid.ID][]oid.ID)
|
|
for _, addr := range addresses {
|
|
containerIDToObjectIDs[addr.Container()] = append(containerIDToObjectIDs[addr.Container()], addr.Object())
|
|
}
|
|
|
|
err := db.boltDB.View(func(tx *bbolt.Tx) error {
|
|
for containerID, objectIDs := range containerIDToObjectIDs {
|
|
select {
|
|
case <-ctx.Done():
|
|
return ErrInterruptIterator
|
|
default:
|
|
}
|
|
|
|
expired, err := selectExpiredObjects(tx, epoch, containerID, objectIDs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
result = append(result, expired...)
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, metaerr.Wrap(err)
|
|
}
|
|
success = true
|
|
return result, nil
|
|
}
|
|
|
|
func isExpired(tx *bbolt.Tx, addr oid.Address, currEpoch uint64) (bool, error) {
|
|
return isExpiredWithCache(nil, tx, addr, currEpoch)
|
|
}
|
|
|
|
func isExpiredWithCache(bc *bucketCache, tx *bbolt.Tx, addr oid.Address, currEpoch uint64) (bool, error) {
|
|
b := getExpiredBucket(bc, tx, addr.Container())
|
|
if b == nil {
|
|
return false, nil
|
|
}
|
|
key := make([]byte, objectKeySize)
|
|
addr.Object().Encode(key)
|
|
val := b.Get(key)
|
|
if len(val) == 0 {
|
|
return false, nil
|
|
}
|
|
if len(val) != epochSize {
|
|
return false, errInvalidEpochValueLength
|
|
}
|
|
expEpoch := binary.LittleEndian.Uint64(val)
|
|
return expEpoch < currEpoch, nil
|
|
}
|
|
|
|
func selectExpiredObjects(tx *bbolt.Tx, epoch uint64, containerID cid.ID, objectIDs []oid.ID) ([]oid.Address, error) {
|
|
result := make([]oid.Address, 0)
|
|
var addr oid.Address
|
|
addr.SetContainer(containerID)
|
|
for _, objID := range objectIDs {
|
|
addr.SetObject(objID)
|
|
expired, err := isExpired(tx, addr, epoch)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if expired {
|
|
result = append(result, addr)
|
|
}
|
|
}
|
|
return result, nil
|
|
}
|