115 lines
3.7 KiB
Go
115 lines
3.7 KiB
Go
package meta
|
|
|
|
import (
|
|
"context"
|
|
"math/rand"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/internal/testutil"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-node/pkg/local_object_storage/shard/mode"
|
|
apistatus "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/client/status"
|
|
cidtest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/container/id/test"
|
|
objectSDK "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object"
|
|
oid "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id"
|
|
oidtest "git.frostfs.info/TrueCloudLab/frostfs-sdk-go/object/id/test"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestInhumeECObject(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
db := New(
|
|
WithPath(filepath.Join(t.TempDir(), "metabase")),
|
|
WithPermissions(0o600),
|
|
WithEpochState(epochState{uint64(12)}),
|
|
)
|
|
|
|
require.NoError(t, db.Open(context.Background(), mode.ReadWrite))
|
|
require.NoError(t, db.Init(context.Background()))
|
|
defer func() { require.NoError(t, db.Close(context.Background())) }()
|
|
|
|
cnr := cidtest.ID()
|
|
ecChunk := oidtest.ID()
|
|
ecChunk2 := oidtest.ID()
|
|
ecParent := oidtest.ID()
|
|
tombstoneID := oidtest.ID()
|
|
|
|
chunkObj := testutil.GenerateObjectWithCID(cnr)
|
|
chunkObj.SetID(ecChunk)
|
|
chunkObj.SetPayload([]byte{0, 1, 2, 3, 4})
|
|
chunkObj.SetPayloadSize(uint64(5))
|
|
chunkObj.SetECHeader(objectSDK.NewECHeader(objectSDK.ECParentInfo{ID: ecParent}, 0, 3, []byte{}, 0))
|
|
|
|
chunkObj2 := testutil.GenerateObjectWithCID(cnr)
|
|
chunkObj2.SetID(ecChunk2)
|
|
chunkObj2.SetPayload([]byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9})
|
|
chunkObj2.SetPayloadSize(uint64(10))
|
|
chunkObj2.SetECHeader(objectSDK.NewECHeader(objectSDK.ECParentInfo{ID: ecParent}, 1, 3, []byte{}, 0))
|
|
|
|
// put object with EC
|
|
|
|
var prm PutPrm
|
|
prm.SetObject(chunkObj)
|
|
prm.SetStorageID([]byte("0/0"))
|
|
_, err := db.Put(context.Background(), prm)
|
|
require.NoError(t, err)
|
|
|
|
prm.SetObject(chunkObj2)
|
|
_, err = db.Put(context.Background(), prm)
|
|
require.NoError(t, err)
|
|
|
|
var ecChunkAddress oid.Address
|
|
ecChunkAddress.SetContainer(cnr)
|
|
ecChunkAddress.SetObject(ecChunk)
|
|
|
|
var ecParentAddress oid.Address
|
|
ecParentAddress.SetContainer(cnr)
|
|
ecParentAddress.SetObject(ecParent)
|
|
|
|
var chunkObjectAddress oid.Address
|
|
chunkObjectAddress.SetContainer(cnr)
|
|
chunkObjectAddress.SetObject(ecChunk)
|
|
|
|
var getPrm GetPrm
|
|
|
|
getPrm.SetAddress(ecChunkAddress)
|
|
_, err = db.Get(context.Background(), getPrm)
|
|
require.NoError(t, err)
|
|
|
|
var ecInfoError *objectSDK.ECInfoError
|
|
getPrm.SetAddress(ecParentAddress)
|
|
_, err = db.Get(context.Background(), getPrm)
|
|
require.ErrorAs(t, err, &ecInfoError)
|
|
require.True(t, len(ecInfoError.ECInfo().Chunks) == 2 &&
|
|
ecInfoError.ECInfo().Chunks[0].Index == 0 &&
|
|
ecInfoError.ECInfo().Chunks[0].Total == 3)
|
|
|
|
// inhume Chunk
|
|
var inhumePrm InhumePrm
|
|
var tombAddress oid.Address
|
|
inhumePrm.SetAddresses(chunkObjectAddress)
|
|
res, err := db.Inhume(context.Background(), inhumePrm)
|
|
require.NoError(t, err)
|
|
require.True(t, len(res.deletionDetails) == 1)
|
|
require.True(t, res.deletionDetails[0].Size == 5)
|
|
|
|
// inhume EC parent (like Delete does)
|
|
tombAddress.SetContainer(cnr)
|
|
tombAddress.SetObject(tombstoneID)
|
|
inhumePrm.SetAddresses(ecParentAddress)
|
|
inhumePrm.SetTombstoneAddress(tombAddress, rand.Uint64())
|
|
res, err = db.Inhume(context.Background(), inhumePrm)
|
|
require.NoError(t, err)
|
|
// Previously deleted chunk shouldn't be in the details, because it is marked as garbage
|
|
require.True(t, len(res.deletionDetails) == 1)
|
|
require.True(t, res.deletionDetails[0].Size == 10)
|
|
|
|
getPrm.SetAddress(ecParentAddress)
|
|
_, err = db.Get(context.Background(), getPrm)
|
|
require.ErrorAs(t, err, new(*apistatus.ObjectAlreadyRemoved))
|
|
|
|
getPrm.SetAddress(ecChunkAddress)
|
|
_, err = db.Get(context.Background(), getPrm)
|
|
require.ErrorAs(t, err, new(*apistatus.ObjectAlreadyRemoved))
|
|
}
|