forked from TrueCloudLab/frostfs-node
337049b2ce
Signed-off-by: Pavel Karpy <p.karpy@yadro.com>
266 lines
6.6 KiB
Go
266 lines
6.6 KiB
Go
package meta_test
|
|
|
|
import (
|
|
"testing"
|
|
|
|
objectcore "github.com/TrueCloudLab/frostfs-node/pkg/core/object"
|
|
meta "github.com/TrueCloudLab/frostfs-node/pkg/local_object_storage/metabase"
|
|
apistatus "github.com/TrueCloudLab/frostfs-sdk-go/client/status"
|
|
cidtest "github.com/TrueCloudLab/frostfs-sdk-go/container/id/test"
|
|
"github.com/TrueCloudLab/frostfs-sdk-go/object"
|
|
oid "github.com/TrueCloudLab/frostfs-sdk-go/object/id"
|
|
oidtest "github.com/TrueCloudLab/frostfs-sdk-go/object/id/test"
|
|
objecttest "github.com/TrueCloudLab/frostfs-sdk-go/object/test"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestDB_Lock(t *testing.T) {
|
|
cnr := cidtest.ID()
|
|
db := newDB(t)
|
|
|
|
t.Run("empty locked list", func(t *testing.T) {
|
|
require.Panics(t, func() { _ = db.Lock(cnr, oid.ID{}, nil) })
|
|
require.Panics(t, func() { _ = db.Lock(cnr, oid.ID{}, []oid.ID{}) })
|
|
})
|
|
|
|
t.Run("(ir)regular", func(t *testing.T) {
|
|
for _, typ := range [...]object.Type{
|
|
object.TypeTombstone,
|
|
object.TypeStorageGroup,
|
|
object.TypeLock,
|
|
object.TypeRegular,
|
|
} {
|
|
obj := objecttest.Object()
|
|
obj.SetType(typ)
|
|
obj.SetContainerID(cnr)
|
|
|
|
// save irregular object
|
|
err := metaPut(db, obj, nil)
|
|
require.NoError(t, err, typ)
|
|
|
|
var e apistatus.LockNonRegularObject
|
|
|
|
id, _ := obj.ID()
|
|
|
|
// try to lock it
|
|
err = db.Lock(cnr, oidtest.ID(), []oid.ID{id})
|
|
if typ == object.TypeRegular {
|
|
require.NoError(t, err, typ)
|
|
} else {
|
|
require.ErrorAs(t, err, &e, typ)
|
|
}
|
|
}
|
|
})
|
|
|
|
t.Run("removing lock object", func(t *testing.T) {
|
|
objs, lockObj := putAndLockObj(t, db, 1)
|
|
|
|
objAddr := objectcore.AddressOf(objs[0])
|
|
lockAddr := objectcore.AddressOf(lockObj)
|
|
|
|
var inhumePrm meta.InhumePrm
|
|
inhumePrm.SetGCMark()
|
|
|
|
// check locking relation
|
|
|
|
inhumePrm.SetAddresses(objAddr)
|
|
_, err := db.Inhume(inhumePrm)
|
|
require.ErrorAs(t, err, new(apistatus.ObjectLocked))
|
|
|
|
inhumePrm.SetTombstoneAddress(oidtest.Address())
|
|
_, err = db.Inhume(inhumePrm)
|
|
require.ErrorAs(t, err, new(apistatus.ObjectLocked))
|
|
|
|
// try to remove lock object
|
|
inhumePrm.SetAddresses(lockAddr)
|
|
_, err = db.Inhume(inhumePrm)
|
|
require.Error(t, err)
|
|
|
|
// check that locking relation has not been
|
|
// dropped
|
|
|
|
inhumePrm.SetAddresses(objAddr)
|
|
_, err = db.Inhume(inhumePrm)
|
|
require.ErrorAs(t, err, new(apistatus.ObjectLocked))
|
|
|
|
inhumePrm.SetTombstoneAddress(oidtest.Address())
|
|
_, err = db.Inhume(inhumePrm)
|
|
require.ErrorAs(t, err, new(apistatus.ObjectLocked))
|
|
})
|
|
|
|
t.Run("lock-unlock scenario", func(t *testing.T) {
|
|
objs, lockObj := putAndLockObj(t, db, 1)
|
|
|
|
objAddr := objectcore.AddressOf(objs[0])
|
|
lockAddr := objectcore.AddressOf(lockObj)
|
|
|
|
// try to inhume locked object using tombstone
|
|
err := metaInhume(db, objAddr, lockAddr)
|
|
require.ErrorAs(t, err, new(apistatus.ObjectLocked))
|
|
|
|
// free locked object
|
|
var inhumePrm meta.InhumePrm
|
|
inhumePrm.SetAddresses(lockAddr)
|
|
inhumePrm.SetForceGCMark()
|
|
inhumePrm.SetLockObjectHandling()
|
|
|
|
res, err := db.Inhume(inhumePrm)
|
|
require.NoError(t, err)
|
|
require.Len(t, res.DeletedLockObjects(), 1)
|
|
require.Equal(t, objectcore.AddressOf(lockObj), res.DeletedLockObjects()[0])
|
|
|
|
err = db.FreeLockedBy([]oid.Address{lockAddr})
|
|
require.NoError(t, err)
|
|
|
|
inhumePrm.SetAddresses(objAddr)
|
|
inhumePrm.SetGCMark()
|
|
|
|
// now we can inhume the object
|
|
_, err = db.Inhume(inhumePrm)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("force removing lock objects", func(t *testing.T) {
|
|
const objsNum = 3
|
|
|
|
// put and lock `objsNum` objects
|
|
objs, lockObj := putAndLockObj(t, db, objsNum)
|
|
|
|
// force remove objects
|
|
|
|
var inhumePrm meta.InhumePrm
|
|
inhumePrm.SetForceGCMark()
|
|
inhumePrm.SetAddresses(objectcore.AddressOf(lockObj))
|
|
inhumePrm.SetLockObjectHandling()
|
|
|
|
res, err := db.Inhume(inhumePrm)
|
|
require.NoError(t, err)
|
|
require.Len(t, res.DeletedLockObjects(), 1)
|
|
require.Equal(t, objectcore.AddressOf(lockObj), res.DeletedLockObjects()[0])
|
|
|
|
// unlock just objects that were locked by
|
|
// just removed locker
|
|
err = db.FreeLockedBy([]oid.Address{res.DeletedLockObjects()[0]})
|
|
require.NoError(t, err)
|
|
|
|
// removing objects after unlock
|
|
|
|
inhumePrm.SetGCMark()
|
|
|
|
for i := 0; i < objsNum; i++ {
|
|
inhumePrm.SetAddresses(objectcore.AddressOf(objs[i]))
|
|
|
|
res, err = db.Inhume(inhumePrm)
|
|
require.NoError(t, err)
|
|
require.Len(t, res.DeletedLockObjects(), 0)
|
|
}
|
|
})
|
|
|
|
t.Run("skipping lock object handling", func(t *testing.T) {
|
|
_, lockObj := putAndLockObj(t, db, 1)
|
|
|
|
var inhumePrm meta.InhumePrm
|
|
inhumePrm.SetForceGCMark()
|
|
inhumePrm.SetAddresses(objectcore.AddressOf(lockObj))
|
|
|
|
res, err := db.Inhume(inhumePrm)
|
|
require.NoError(t, err)
|
|
require.Len(t, res.DeletedLockObjects(), 0)
|
|
})
|
|
}
|
|
|
|
func TestDB_Lock_Expired(t *testing.T) {
|
|
es := &epochState{e: 123}
|
|
|
|
db := newDB(t, meta.WithEpochState(es))
|
|
|
|
// put an object
|
|
addr := putWithExpiration(t, db, object.TypeRegular, 124)
|
|
|
|
// expire the obj
|
|
es.e = 125
|
|
_, err := metaGet(db, addr, false)
|
|
require.ErrorIs(t, err, meta.ErrObjectIsExpired)
|
|
|
|
// lock the obj
|
|
require.NoError(t, db.Lock(addr.Container(), oidtest.ID(), []oid.ID{addr.Object()}))
|
|
|
|
// object is expired but locked, thus, must be available
|
|
_, err = metaGet(db, addr, false)
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func TestDB_IsLocked(t *testing.T) {
|
|
db := newDB(t)
|
|
|
|
// existing and locked objs
|
|
|
|
objs, _ := putAndLockObj(t, db, 5)
|
|
var prm meta.IsLockedPrm
|
|
|
|
for _, obj := range objs {
|
|
prm.SetAddress(objectcore.AddressOf(obj))
|
|
|
|
res, err := db.IsLocked(prm)
|
|
require.NoError(t, err)
|
|
|
|
require.True(t, res.Locked())
|
|
}
|
|
|
|
// some rand obj
|
|
|
|
prm.SetAddress(oidtest.Address())
|
|
|
|
res, err := db.IsLocked(prm)
|
|
require.NoError(t, err)
|
|
|
|
require.False(t, res.Locked())
|
|
|
|
// existing but not locked obj
|
|
|
|
obj := objecttest.Object()
|
|
|
|
var putPrm meta.PutPrm
|
|
putPrm.SetObject(obj)
|
|
|
|
_, err = db.Put(putPrm)
|
|
require.NoError(t, err)
|
|
|
|
prm.SetAddress(objectcore.AddressOf(obj))
|
|
|
|
res, err = db.IsLocked(prm)
|
|
require.NoError(t, err)
|
|
|
|
require.False(t, res.Locked())
|
|
}
|
|
|
|
// putAndLockObj puts object, returns it and its locker.
|
|
func putAndLockObj(t *testing.T, db *meta.DB, numOfLockedObjs int) ([]*object.Object, *object.Object) {
|
|
cnr := cidtest.ID()
|
|
|
|
lockedObjs := make([]*object.Object, 0, numOfLockedObjs)
|
|
lockedObjIDs := make([]oid.ID, 0, numOfLockedObjs)
|
|
|
|
for i := 0; i < numOfLockedObjs; i++ {
|
|
obj := generateObjectWithCID(t, cnr)
|
|
err := putBig(db, obj)
|
|
require.NoError(t, err)
|
|
|
|
id, _ := obj.ID()
|
|
|
|
lockedObjs = append(lockedObjs, obj)
|
|
lockedObjIDs = append(lockedObjIDs, id)
|
|
}
|
|
|
|
lockObj := generateObjectWithCID(t, cnr)
|
|
lockID, _ := lockObj.ID()
|
|
lockObj.SetType(object.TypeLock)
|
|
|
|
err := putBig(db, lockObj)
|
|
require.NoError(t, err)
|
|
|
|
err = db.Lock(cnr, lockID, lockedObjIDs)
|
|
require.NoError(t, err)
|
|
|
|
return lockedObjs, lockObj
|
|
}
|