2020-11-23 13:30:56 +00:00
|
|
|
package meta
|
|
|
|
|
|
|
|
import (
|
2021-02-16 15:56:59 +00:00
|
|
|
"bytes"
|
2022-09-08 11:54:21 +00:00
|
|
|
"crypto/sha256"
|
2022-05-31 17:00:41 +00:00
|
|
|
"fmt"
|
2020-11-23 13:30:56 +00:00
|
|
|
|
2021-11-10 07:08:33 +00:00
|
|
|
cid "github.com/nspcc-dev/neofs-sdk-go/container/id"
|
|
|
|
"github.com/nspcc-dev/neofs-sdk-go/object"
|
2022-05-31 17:00:41 +00:00
|
|
|
oid "github.com/nspcc-dev/neofs-sdk-go/object/id"
|
2021-09-13 08:23:58 +00:00
|
|
|
"go.etcd.io/bbolt"
|
2020-11-23 13:30:56 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2021-04-08 14:19:31 +00:00
|
|
|
// graveyardBucketName stores rows with the objects that have been
|
|
|
|
// covered with Tombstone objects. That objects should not be returned
|
|
|
|
// from the node and should not be accepted by the node from other
|
|
|
|
// nodes.
|
2022-09-08 11:54:21 +00:00
|
|
|
graveyardBucketName = []byte{graveyardPrefix}
|
2021-04-08 14:19:31 +00:00
|
|
|
// garbageBucketName stores rows with the objects that should be physically
|
|
|
|
// deleted by the node (Garbage Collector routine).
|
2022-09-08 11:54:21 +00:00
|
|
|
garbageBucketName = []byte{garbagePrefix}
|
|
|
|
toMoveItBucketName = []byte{toMoveItPrefix}
|
|
|
|
containerVolumeBucketName = []byte{containerVolumePrefix}
|
2020-11-23 13:30:56 +00:00
|
|
|
|
|
|
|
zeroValue = []byte{0xFF}
|
2022-09-08 11:54:21 +00:00
|
|
|
)
|
2020-11-23 13:30:56 +00:00
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
// Prefix bytes for database keys. All ids and addresses are encoded in binary
|
|
|
|
// unless specified otherwise.
|
|
|
|
const (
|
|
|
|
// graveyardPrefix is used for the graveyard bucket.
|
|
|
|
// Key: object address
|
|
|
|
// Value: tombstone address
|
|
|
|
graveyardPrefix = iota
|
|
|
|
// garbagePrefix is used for the garbage bucket.
|
|
|
|
// Key: object address
|
|
|
|
// Value: dummy value
|
|
|
|
garbagePrefix
|
|
|
|
// toMoveItPrefix is used for bucket containing IDs of objects that are candidates for moving
|
|
|
|
// to another shard.
|
|
|
|
toMoveItPrefix
|
|
|
|
// containerVolumePrefix is used for storing container size estimations.
|
|
|
|
// Key: container ID
|
|
|
|
// Value: container size in bytes as little-endian uint64
|
|
|
|
containerVolumePrefix
|
|
|
|
// lockedPrefix is used for storing locked objects information.
|
|
|
|
// Key: container ID
|
|
|
|
// Value: bucket mapping objects locked to the list of corresponding LOCK objects.
|
|
|
|
lockedPrefix
|
|
|
|
// shardInfoPrefix is used for storing shard ID. All keys are custom and are not connected to the container.
|
|
|
|
shardInfoPrefix
|
|
|
|
|
|
|
|
//======================
|
|
|
|
// Unique index buckets.
|
|
|
|
//======================
|
|
|
|
|
|
|
|
// primaryPrefix is used for prefixing buckets containing objects of REGULAR type.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: marshalled object
|
|
|
|
primaryPrefix
|
|
|
|
// lockersPrefix is used for prefixing buckets containing objects of LOCK type.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: marshalled object
|
|
|
|
lockersPrefix
|
|
|
|
// storageGroupPrefix is used for prefixing buckets containing objects of STORAGEGROUP type.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: marshaled object
|
|
|
|
storageGroupPrefix
|
|
|
|
// tombstonePrefix is used for prefixing buckets containing objects of TOMBSTONE type.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: marshaled object
|
|
|
|
tombstonePrefix
|
|
|
|
// smallPrefix is used for prefixing buckets mapping objects to the blobovniczas they are stored in.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: blobovnicza ID
|
|
|
|
smallPrefix
|
|
|
|
// rootPrefix is used for prefixing buckets mapping parent object to the split info.
|
|
|
|
// Key: object ID
|
|
|
|
// Value: split info
|
|
|
|
rootPrefix
|
|
|
|
|
|
|
|
//====================
|
|
|
|
// FKBT index buckets.
|
|
|
|
//====================
|
|
|
|
|
|
|
|
// ownerPrefix is used for prefixing FKBT index buckets mapping owner to object IDs.
|
|
|
|
// Key: owner ID
|
|
|
|
// Value: bucket containing object IDs as keys
|
|
|
|
ownerPrefix
|
|
|
|
// userAttributePrefix is used for prefixing FKBT index buckets containing objects.
|
|
|
|
// Key: attribute value
|
|
|
|
// Value: bucket containing object IDs as keys
|
|
|
|
userAttributePrefix
|
|
|
|
|
|
|
|
//====================
|
|
|
|
// List index buckets.
|
|
|
|
//====================
|
|
|
|
|
|
|
|
// payloadHashPrefix is used for prefixing List index buckets mapping payload hash to a list of object IDs.
|
|
|
|
// Key: payload hash
|
|
|
|
// Value: list of object IDs
|
|
|
|
payloadHashPrefix
|
|
|
|
// parentPrefix is used for prefixing List index buckets mapping parent ID to a list of children IDs.
|
|
|
|
// Key: parent ID
|
|
|
|
// Value: list of object IDs
|
|
|
|
parentPrefix
|
|
|
|
// splitPrefix is used for prefixing List index buckets mapping split ID to a list of object IDs.
|
|
|
|
// Key: split ID
|
|
|
|
// Value: list of object IDs
|
|
|
|
splitPrefix
|
|
|
|
)
|
2020-12-03 09:31:51 +00:00
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
const (
|
|
|
|
cidSize = sha256.Size
|
|
|
|
bucketKeySize = 1 + cidSize
|
|
|
|
objectKeySize = sha256.Size
|
|
|
|
addressKeySize = cidSize + objectKeySize
|
2020-11-23 13:30:56 +00:00
|
|
|
)
|
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
var splitInfoError *object.SplitInfoError // for errors.As comparisons
|
|
|
|
|
|
|
|
func bucketName(cnr cid.ID, prefix byte, key []byte) []byte {
|
|
|
|
key[0] = prefix
|
|
|
|
cnr.Encode(key[1:])
|
|
|
|
return key[:bucketKeySize]
|
|
|
|
}
|
|
|
|
|
2020-11-23 13:30:56 +00:00
|
|
|
// primaryBucketName returns <CID>.
|
2022-09-08 11:54:21 +00:00
|
|
|
func primaryBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, primaryPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// tombstoneBucketName returns <CID>_TS.
|
2022-09-08 11:54:21 +00:00
|
|
|
func tombstoneBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, tombstonePrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// storageGroupBucketName returns <CID>_SG.
|
2022-09-08 11:54:21 +00:00
|
|
|
func storageGroupBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, storageGroupPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// smallBucketName returns <CID>_small.
|
2022-09-08 11:54:21 +00:00
|
|
|
func smallBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, smallPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// attributeBucketName returns <CID>_attr_<attributeKey>.
|
2022-09-08 11:54:21 +00:00
|
|
|
func attributeBucketName(cnr cid.ID, attributeKey string, key []byte) []byte {
|
|
|
|
key[0] = userAttributePrefix
|
|
|
|
cnr.Encode(key[1:])
|
|
|
|
return append(key[:bucketKeySize], attributeKey...)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
2021-02-16 15:56:59 +00:00
|
|
|
// returns <CID> from attributeBucketName result, nil otherwise.
|
|
|
|
func cidFromAttributeBucket(val []byte, attributeKey string) []byte {
|
2022-09-08 11:54:21 +00:00
|
|
|
if len(val) < bucketKeySize || val[0] != userAttributePrefix || !bytes.Equal(val[bucketKeySize:], []byte(attributeKey)) {
|
2021-02-16 15:56:59 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
return val[1:bucketKeySize]
|
2021-02-16 15:56:59 +00:00
|
|
|
}
|
|
|
|
|
2020-11-23 13:30:56 +00:00
|
|
|
// payloadHashBucketName returns <CID>_payloadhash.
|
2022-09-08 11:54:21 +00:00
|
|
|
func payloadHashBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, payloadHashPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// rootBucketName returns <CID>_root.
|
2022-09-08 11:54:21 +00:00
|
|
|
func rootBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, rootPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ownerBucketName returns <CID>_ownerid.
|
2022-09-08 11:54:21 +00:00
|
|
|
func ownerBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, ownerPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
2020-11-30 08:46:18 +00:00
|
|
|
// parentBucketName returns <CID>_parent.
|
2022-09-08 11:54:21 +00:00
|
|
|
func parentBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, parentPrefix, key)
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
2020-11-30 08:46:18 +00:00
|
|
|
// splitBucketName returns <CID>_splitid.
|
2022-09-08 11:54:21 +00:00
|
|
|
func splitBucketName(cnr cid.ID, key []byte) []byte {
|
|
|
|
return bucketName(cnr, splitPrefix, key)
|
2020-11-30 08:46:18 +00:00
|
|
|
}
|
|
|
|
|
2020-11-23 13:30:56 +00:00
|
|
|
// addressKey returns key for K-V tables when key is a whole address.
|
2022-09-08 11:54:21 +00:00
|
|
|
func addressKey(addr oid.Address, key []byte) []byte {
|
|
|
|
addr.Container().Encode(key)
|
|
|
|
addr.Object().Encode(key[cidSize:])
|
|
|
|
return key[:addressKeySize]
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
|
|
|
|
2021-02-16 07:57:43 +00:00
|
|
|
// parses object address formed by addressKey.
|
2022-05-31 17:00:41 +00:00
|
|
|
func decodeAddressFromKey(dst *oid.Address, k []byte) error {
|
2022-09-08 11:54:21 +00:00
|
|
|
if len(k) != addressKeySize {
|
|
|
|
return fmt.Errorf("invalid length")
|
|
|
|
}
|
|
|
|
|
|
|
|
var cnr cid.ID
|
|
|
|
if err := cnr.Decode(k[:cidSize]); err != nil {
|
|
|
|
return err
|
2022-05-31 17:00:41 +00:00
|
|
|
}
|
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
var obj oid.ID
|
|
|
|
if err := obj.Decode(k[cidSize:]); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
dst.SetObject(obj)
|
|
|
|
dst.SetContainer(cnr)
|
2022-05-31 17:00:41 +00:00
|
|
|
return nil
|
2021-02-16 07:57:43 +00:00
|
|
|
}
|
|
|
|
|
2020-11-23 13:30:56 +00:00
|
|
|
// objectKey returns key for K-V tables when key is an object id.
|
2022-09-08 11:54:21 +00:00
|
|
|
func objectKey(obj oid.ID, key []byte) []byte {
|
|
|
|
obj.Encode(key)
|
|
|
|
return key[:objectKeySize]
|
2020-11-23 13:30:56 +00:00
|
|
|
}
|
2021-09-13 08:23:58 +00:00
|
|
|
|
|
|
|
// removes all bucket elements.
|
|
|
|
func resetBucket(b *bbolt.Bucket) error {
|
|
|
|
return b.ForEach(func(k, v []byte) error {
|
|
|
|
if v != nil {
|
|
|
|
return b.Delete(k)
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.DeleteBucket(k)
|
|
|
|
})
|
|
|
|
}
|
2022-02-15 11:35:13 +00:00
|
|
|
|
|
|
|
// if meets irregular object container in objs - returns its type, otherwise returns object.TypeRegular.
|
|
|
|
//
|
|
|
|
// firstIrregularObjectType(tx, cnr, obj) usage allows getting object type.
|
|
|
|
func firstIrregularObjectType(tx *bbolt.Tx, idCnr cid.ID, objs ...[]byte) object.Type {
|
|
|
|
if len(objs) == 0 {
|
|
|
|
panic("empty object list in firstIrregularObjectType")
|
|
|
|
}
|
|
|
|
|
2022-09-08 11:54:21 +00:00
|
|
|
var keys [3][1 + cidSize]byte
|
|
|
|
|
2022-02-15 11:35:13 +00:00
|
|
|
irregularTypeBuckets := [...]struct {
|
|
|
|
typ object.Type
|
|
|
|
name []byte
|
|
|
|
}{
|
2022-09-08 11:54:21 +00:00
|
|
|
{object.TypeTombstone, tombstoneBucketName(idCnr, keys[0][:])},
|
|
|
|
{object.TypeStorageGroup, storageGroupBucketName(idCnr, keys[1][:])},
|
|
|
|
{object.TypeLock, bucketNameLockers(idCnr, keys[2][:])},
|
2022-02-15 11:35:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for i := range objs {
|
|
|
|
for j := range irregularTypeBuckets {
|
|
|
|
if inBucket(tx, irregularTypeBuckets[j].name, objs[i]) {
|
|
|
|
return irregularTypeBuckets[j].typ
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return object.TypeRegular
|
|
|
|
}
|
2022-05-26 17:38:32 +00:00
|
|
|
|
|
|
|
// return true if provided object is of LOCK type.
|
|
|
|
func isLockObject(tx *bbolt.Tx, idCnr cid.ID, obj oid.ID) bool {
|
2022-09-08 11:54:21 +00:00
|
|
|
return inBucket(tx,
|
|
|
|
bucketNameLockers(idCnr, make([]byte, bucketKeySize)),
|
|
|
|
objectKey(obj, make([]byte, objectKeySize)))
|
2022-05-26 17:38:32 +00:00
|
|
|
}
|