From 5c72a34fa82e6f7e434a925b396d807ba1b5c72e Mon Sep 17 00:00:00 2001 From: Harshavardhana Date: Wed, 4 Nov 2020 07:56:58 -0800 Subject: [PATCH] fix: honor delimiter as per AWS S3 spec (#10823) --- cmd/metacache-entries.go | 29 +++++++++++++++++++++ cmd/object-api-listobjects_test.go | 42 +++++++++++++++++++++++++++++- 2 files changed, 70 insertions(+), 1 deletion(-) diff --git a/cmd/metacache-entries.go b/cmd/metacache-entries.go index 88bbafc5d..2481e8716 100644 --- a/cmd/metacache-entries.go +++ b/cmd/metacache-entries.go @@ -301,6 +301,20 @@ func (m *metaCacheEntriesSorted) fileInfoVersions(bucket, prefix, delimiter, aft prevPrefix := "" for _, entry := range m.o { if entry.isObject() { + if delimiter != "" { + idx := strings.Index(strings.TrimPrefix(entry.name, prefix), delimiter) + if idx >= 0 { + idx = len(prefix) + idx + len(delimiter) + currPrefix := entry.name[:idx] + if currPrefix == prevPrefix { + continue + } + prevPrefix = currPrefix + commonPrefixes = append(commonPrefixes, currPrefix) + continue + } + } + fiv, err := entry.fileInfoVersions(bucket) if afterV != "" { // Forward first entry to specified version @@ -314,6 +328,7 @@ func (m *metaCacheEntriesSorted) fileInfoVersions(bucket, prefix, delimiter, aft } continue } + if entry.isDir() { if delimiter == "" { continue @@ -343,6 +358,20 @@ func (m *metaCacheEntriesSorted) fileInfos(bucket, prefix, delimiter string) (ob prevPrefix := "" for _, entry := range m.o { if entry.isObject() { + if delimiter != "" { + idx := strings.Index(strings.TrimPrefix(entry.name, prefix), delimiter) + if idx >= 0 { + idx = len(prefix) + idx + len(delimiter) + currPrefix := entry.name[:idx] + if currPrefix == prevPrefix { + continue + } + prevPrefix = currPrefix + commonPrefixes = append(commonPrefixes, currPrefix) + continue + } + } + fi, err := entry.fileInfo(bucket) if err == nil { objects = append(objects, fi.ToObjectInfo(bucket, entry.name)) diff --git a/cmd/object-api-listobjects_test.go b/cmd/object-api-listobjects_test.go index 3fdedfe9c..bb202d3c1 100644 --- a/cmd/object-api-listobjects_test.go +++ b/cmd/object-api-listobjects_test.go @@ -47,6 +47,8 @@ func testListObjects(obj ObjectLayer, instanceType string, t1 TestErrHandler) { "empty-bucket", // Listing the case where the marker > last object. "test-bucket-single-object", + // Listing uncommon delimiter. + "test-bucket-delimiter", } for _, bucket := range testBuckets { err := obj.MakeBucketWithLocation(context.Background(), bucket, BucketOptions{}) @@ -75,6 +77,8 @@ func testListObjects(obj ObjectLayer, instanceType string, t1 TestErrHandler) { {testBuckets[1], "obj2", "obj2", nil}, {testBuckets[1], "temporary/0/", "", nil}, {testBuckets[3], "A/B", "contentstring", nil}, + {testBuckets[4], "file1/receipt.json", "content", nil}, + {testBuckets[4], "file1/guidSplunk-aaaa/file", "content", nil}, } for _, object := range testObjects { md5Bytes := md5.Sum([]byte(object.content)) @@ -455,6 +459,14 @@ func testListObjects(obj ObjectLayer, instanceType string, t1 TestErrHandler) { IsTruncated: false, Objects: []ObjectInfo{}, }, + // ListObjectsResult-35 list with custom uncommon delimiter + { + IsTruncated: false, + Objects: []ObjectInfo{ + {Name: "file1/receipt.json"}, + }, + Prefixes: []string{"file1/guidSplunk"}, + }, } testCases := []struct { @@ -573,6 +585,10 @@ func testListObjects(obj ObjectLayer, instanceType string, t1 TestErrHandler) { {"test-bucket-single-object", "", "A/C", "", 1000, resultCases[34], nil, true}, // Test listing an object with a trailing slash and a slash delimiter (66) {"test-bucket-list-object", "Asia-maps.png/", "", "/", 1000, resultCases[34], nil, true}, + // Test listing an object with uncommon delimiter + {testBuckets[4], "", "", "guidSplunk", 1000, resultCases[35], nil, true}, + // Test listing an object with uncommon delimiter and matching prefix + {testBuckets[4], "file1/", "", "guidSplunk", 1000, resultCases[35], nil, true}, } for i, testCase := range testCases { @@ -696,10 +712,16 @@ func testListObjectVersions(obj ObjectLayer, instanceType string, t1 TestErrHand "empty-bucket", // Listing the case where the marker > last object. "test-bucket-single-object", + // Listing uncommon delimiter. + "test-bucket-delimiter", } for _, bucket := range testBuckets { - err := obj.MakeBucketWithLocation(context.Background(), bucket, BucketOptions{}) + err := obj.MakeBucketWithLocation(context.Background(), bucket, BucketOptions{VersioningEnabled: true}) if err != nil { + if _, ok := err.(NotImplemented); ok { + // Skip test for FS mode. + continue + } t.Fatalf("%s : %s", instanceType, err.Error()) } } @@ -724,12 +746,18 @@ func testListObjectVersions(obj ObjectLayer, instanceType string, t1 TestErrHand {testBuckets[1], "obj2", "obj2", nil}, {testBuckets[1], "temporary/0/", "", nil}, {testBuckets[3], "A/B", "contentstring", nil}, + {testBuckets[4], "file1/receipt.json", "content", nil}, + {testBuckets[4], "file1/guidSplunk-aaaa/file", "content", nil}, } for _, object := range testObjects { md5Bytes := md5.Sum([]byte(object.content)) _, err = obj.PutObject(context.Background(), object.parentBucket, object.name, mustGetPutObjReader(t, bytes.NewBufferString(object.content), int64(len(object.content)), hex.EncodeToString(md5Bytes[:]), ""), ObjectOptions{UserDefined: object.meta}) if err != nil { + if _, ok := err.(BucketNotFound); ok { + // Skip test failure for FS mode. + continue + } t.Fatalf("%s : %s", instanceType, err.Error()) } @@ -1104,6 +1132,14 @@ func testListObjectVersions(obj ObjectLayer, instanceType string, t1 TestErrHand IsTruncated: false, Objects: []ObjectInfo{}, }, + // ListObjectsResult-35 list with custom uncommon delimiter + { + IsTruncated: false, + Objects: []ObjectInfo{ + {Name: "file1/receipt.json"}, + }, + Prefixes: []string{"file1/guidSplunk"}, + }, } testCases := []struct { @@ -1222,6 +1258,10 @@ func testListObjectVersions(obj ObjectLayer, instanceType string, t1 TestErrHand {"test-bucket-single-object", "", "A/C", "", 1000, resultCases[34], nil, true}, // Test listing an object with a trailing slash and a slash delimiter (64) {"test-bucket-list-object", "Asia-maps.png/", "", "/", 1000, resultCases[34], nil, true}, + // Test listing an object with uncommon delimiter + {testBuckets[4], "", "", "guidSplunk", 1000, resultCases[35], nil, true}, + // Test listing an object with uncommon delimiter and matching prefix + {testBuckets[4], "file1/", "", "guidSplunk", 1000, resultCases[35], nil, true}, } for i, testCase := range testCases {