feature/165-speed_up_listing #294
2 changed files with 61 additions and 26 deletions
|
@ -15,7 +15,6 @@ import (
|
||||||
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data"
|
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/data"
|
||||||
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer"
|
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer"
|
||||||
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption"
|
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/api/layer/encryption"
|
||||||
"git.frostfs.info/TrueCloudLab/frostfs-s3-gw/pkg/service/tree"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
"go.uber.org/zap/zaptest"
|
"go.uber.org/zap/zaptest"
|
||||||
)
|
)
|
||||||
|
@ -74,37 +73,68 @@ func TestListObjectsWithOldTreeNodes(t *testing.T) {
|
||||||
|
|
||||||
srcEnc, err := encryption.NewParams([]byte("1234567890qwertyuiopasdfghjklzxc"))
|
srcEnc, err := encryption.NewParams([]byte("1234567890qwertyuiopasdfghjklzxc"))
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
objInfo := createTestObject(hc, bktInfo, objName, *srcEnc)
|
|
||||||
_ = objInfo
|
|
||||||
|
|
||||||
prm := &tree.GetNodesParams{
|
n := 10
|
||||||
BktInfo: bktInfo,
|
objInfos := make([]*data.ObjectInfo, n)
|
||||||
TreeID: "version",
|
for i := 0; i < n; i++ {
|
||||||
Path: []string{objName},
|
objInfos[i] = createTestObject(hc, bktInfo, objName+strconv.Itoa(i), *srcEnc)
|
||||||
LatestOnly: true,
|
|
||||||
AllAttrs: true,
|
|
||||||
}
|
}
|
||||||
nodes, err := hc.treeMock.GetNodes(hc.Context(), prm)
|
sort.Slice(objInfos, func(i, j int) bool { return objInfos[i].Name < objInfos[j].Name })
|
||||||
require.NoError(t, err)
|
|
||||||
require.Len(t, nodes, 1)
|
makeAllTreeObjectsOld(hc, bktInfo)
|
||||||
node := nodes[0]
|
|
||||||
meta := make(map[string]string, len(node.GetMeta()))
|
listV1 := listObjectsV1(hc, bktName, "", "", "", -1)
|
||||||
for _, m := range node.GetMeta() {
|
checkListOldNodes(hc, listV1.Contents, objInfos)
|
||||||
if m.GetKey() != "Created" && m.GetKey() != "Owner" {
|
|
||||||
meta[m.GetKey()] = string(m.GetValue())
|
listV2 := listObjectsV2(hc, bktName, "", "", "", "", -1)
|
||||||
|
checkListOldNodes(hc, listV2.Contents, objInfos)
|
||||||
|
|
||||||
|
listVers := listObjectsVersions(hc, bktName, "", "", "", "", -1)
|
||||||
|
checkListVersionsOldNodes(hc, listVers.Version, objInfos)
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeAllTreeObjectsOld(hc *handlerContext, bktInfo *data.BucketInfo) {
|
||||||
|
nodes, err := hc.treeMock.GetSubTree(hc.Context(), bktInfo, "version", 0, 0)
|
||||||
|
require.NoError(hc.t, err)
|
||||||
|
|
||||||
|
for _, node := range nodes {
|
||||||
|
if node.GetNodeID() == 0 {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
meta := make(map[string]string, len(node.GetMeta()))
|
||||||
|
for _, m := range node.GetMeta() {
|
||||||
|
if m.GetKey() != "Created" && m.GetKey() != "Owner" {
|
||||||
|
meta[m.GetKey()] = string(m.GetValue())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = hc.treeMock.MoveNode(hc.Context(), bktInfo, "version", node.GetNodeID(), node.GetParentID(), meta)
|
||||||
|
require.NoError(hc.t, err)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
err = hc.treeMock.MoveNode(hc.Context(), bktInfo, "version", node.GetNodeID(), node.GetParentID(), meta)
|
func checkListOldNodes(hc *handlerContext, list []Object, objInfos []*data.ObjectInfo) {
|
||||||
require.NoError(t, err)
|
require.Len(hc.t, list, len(objInfos))
|
||||||
|
for i := range list {
|
||||||
|
require.Equal(hc.t, objInfos[i].Name, list[i].Key)
|
||||||
|
realSize, err := layer.GetObjectSize(objInfos[i])
|
||||||
|
require.NoError(hc.t, err)
|
||||||
|
require.Equal(hc.t, objInfos[i].Owner.EncodeToString(), list[i].Owner.ID)
|
||||||
|
require.Equal(hc.t, objInfos[i].Created.UTC().Format(time.RFC3339), list[i].LastModified)
|
||||||
|
require.Equal(hc.t, realSize, list[i].Size)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
list := listObjectsV1(hc, bktName, "", "", "", -1)
|
func checkListVersionsOldNodes(hc *handlerContext, list []ObjectVersionResponse, objInfos []*data.ObjectInfo) {
|
||||||
require.Len(t, list.Contents, 1)
|
require.Len(hc.t, list, len(objInfos))
|
||||||
realSize, err := layer.GetObjectSize(objInfo)
|
for i := range list {
|
||||||
require.NoError(t, err)
|
require.Equal(hc.t, objInfos[i].Name, list[i].Key)
|
||||||
require.Equal(t, objInfo.Owner.EncodeToString(), list.Contents[0].Owner.ID)
|
realSize, err := layer.GetObjectSize(objInfos[i])
|
||||||
require.Equal(t, objInfo.Created.UTC().Format(time.RFC3339), list.Contents[0].LastModified)
|
require.NoError(hc.t, err)
|
||||||
require.Equal(t, realSize, list.Contents[0].Size)
|
require.Equal(hc.t, objInfos[i].Owner.EncodeToString(), list[i].Owner.ID)
|
||||||
|
require.Equal(hc.t, objInfos[i].Created.UTC().Format(time.RFC3339), list[i].LastModified)
|
||||||
|
require.Equal(hc.t, realSize, list[i].Size)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestListObjectsContextCanceled(t *testing.T) {
|
func TestListObjectsContextCanceled(t *testing.T) {
|
||||||
|
@ -657,6 +687,7 @@ func listObjectsV2(hc *handlerContext, bktName, prefix, delimiter, startAfter, c
|
||||||
|
|
||||||
func listObjectsV2Ext(hc *handlerContext, bktName, prefix, delimiter, startAfter, continuationToken, encodingType string, maxKeys int) *ListObjectsV2Response {
|
func listObjectsV2Ext(hc *handlerContext, bktName, prefix, delimiter, startAfter, continuationToken, encodingType string, maxKeys int) *ListObjectsV2Response {
|
||||||
query := prepareCommonListObjectsQuery(prefix, delimiter, maxKeys)
|
query := prepareCommonListObjectsQuery(prefix, delimiter, maxKeys)
|
||||||
|
query.Add("fetch-owner", "true")
|
||||||
if len(startAfter) != 0 {
|
if len(startAfter) != 0 {
|
||||||
query.Add("start-after", startAfter)
|
query.Add("start-after", startAfter)
|
||||||
}
|
}
|
||||||
|
|
|
@ -214,6 +214,8 @@ func (n *layer) getLatestObjectsVersions(ctx context.Context, p commonLatestVers
|
||||||
return nil, nil, fmt.Errorf("failed to get next object from stream: %w", err)
|
return nil, nil, fmt.Errorf("failed to get next object from stream: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sort.Slice(objects, func(i, j int) bool { return objects[i].NodeVersion.FilePath < objects[j].NodeVersion.FilePath })
|
||||||
|
|
||||||
if len(objects) > p.MaxKeys {
|
if len(objects) > p.MaxKeys {
|
||||||
next = objects[p.MaxKeys]
|
next = objects[p.MaxKeys]
|
||||||
n.putListLatestVersionsSession(ctx, p, session, objects)
|
n.putListLatestVersionsSession(ctx, p, session, objects)
|
||||||
|
@ -241,6 +243,8 @@ func (n *layer) getAllObjectsVersions(ctx context.Context, p commonVersionsListi
|
||||||
|
|
||||||
allObjects := handleGeneratedVersions(objOutCh, p, session)
|
allObjects := handleGeneratedVersions(objOutCh, p, session)
|
||||||
|
|
||||||
|
sort.SliceStable(allObjects, func(i, j int) bool { return allObjects[i].NodeVersion.FilePath < allObjects[j].NodeVersion.FilePath })
|
||||||
|
|
||||||
if err = <-errorCh; err != nil {
|
if err = <-errorCh; err != nil {
|
||||||
return nil, false, fmt.Errorf("failed to get next object from stream: %w", err)
|
return nil, false, fmt.Errorf("failed to get next object from stream: %w", err)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue