// Copyright (c) 2015-2021 MinIO, Inc. // // This file is part of MinIO Object Storage stack // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . package lifecycle import ( "bytes" "encoding/xml" "fmt" "net/http" "net/http/httptest" "strconv" "strings" "testing" "time" "github.com/dustin/go-humanize" "github.com/google/uuid" xhttp "github.com/minio/minio/internal/http" ) func TestParseAndValidateLifecycleConfig(t *testing.T) { testCases := []struct { inputConfig string expectedParsingErr error expectedValidationErr error }{ { // Valid lifecycle config inputConfig: ` testRule1 prefix Enabled 3 testRule2 another-prefix Enabled 3 `, expectedParsingErr: nil, expectedValidationErr: nil, }, { // Valid lifecycle config inputConfig: ` key1val1key2val2 3 `, expectedParsingErr: errDuplicatedXMLTag, expectedValidationErr: nil, }, { // lifecycle config with no rules inputConfig: ` `, expectedParsingErr: nil, expectedValidationErr: errLifecycleNoRule, }, { // lifecycle config with rules having overlapping prefix inputConfig: `rule1Enabled/a/b3rule2Enabled/a/b/ckey1val13 `, expectedParsingErr: nil, expectedValidationErr: nil, }, { // lifecycle config with rules having duplicate ID inputConfig: `duplicateIDEnabled/a/b3duplicateIDEnabled/x/zkey1val14`, expectedParsingErr: nil, expectedValidationErr: errLifecycleDuplicateID, }, // Missing in { inputConfig: `sample-rule-2/a/b/cEnabled1`, expectedParsingErr: nil, expectedValidationErr: errXMLNotWellFormed, }, // Lifecycle with the deprecated Prefix tag { inputConfig: `ruleEnabled1`, expectedParsingErr: nil, expectedValidationErr: nil, }, // Lifecycle with empty Filter tag { inputConfig: `ruleEnabled1`, expectedParsingErr: nil, expectedValidationErr: nil, }, // Lifecycle with zero Transition Days { inputConfig: `ruleEnabled0S3TIER-1`, expectedParsingErr: nil, expectedValidationErr: nil, }, // Lifecycle with max noncurrent versions { inputConfig: `ruleEnabled5`, expectedParsingErr: nil, expectedValidationErr: nil, }, // Lifecycle with delmarker expiration { inputConfig: `ruleEnabled5`, expectedParsingErr: nil, expectedValidationErr: nil, }, // Lifecycle with empty delmarker expiration { inputConfig: `ruleEnabled`, expectedParsingErr: errInvalidDaysDelMarkerExpiration, expectedValidationErr: nil, }, } for i, tc := range testCases { t.Run(fmt.Sprintf("Test %d", i+1), func(t *testing.T) { lc, err := ParseLifecycleConfig(bytes.NewReader([]byte(tc.inputConfig))) if err != tc.expectedParsingErr { t.Fatalf("%d: Expected %v during parsing but got %v", i+1, tc.expectedParsingErr, err) } if tc.expectedParsingErr != nil { // We already expect a parsing error, // no need to continue this test. return } err = lc.Validate() if err != tc.expectedValidationErr { t.Fatalf("%d: Expected %v during validation but got %v", i+1, tc.expectedValidationErr, err) } }) } } // TestMarshalLifecycleConfig checks if lifecycleconfig xml // marshaling/unmarshaling can handle output from each other func TestMarshalLifecycleConfig(t *testing.T) { // Time at midnight UTC midnightTS := ExpirationDate{time.Date(2019, time.April, 20, 0, 0, 0, 0, time.UTC)} lc := Lifecycle{ Rules: []Rule{ { Status: "Enabled", Filter: Filter{Prefix: Prefix{string: "prefix-1", set: true}}, Expiration: Expiration{Days: ExpirationDays(3)}, }, { Status: "Enabled", Filter: Filter{Prefix: Prefix{string: "prefix-1", set: true}}, Expiration: Expiration{Date: midnightTS}, }, { Status: "Enabled", Filter: Filter{Prefix: Prefix{string: "prefix-1", set: true}}, Expiration: Expiration{Date: midnightTS}, NoncurrentVersionTransition: NoncurrentVersionTransition{NoncurrentDays: TransitionDays(2), StorageClass: "TEST"}, }, }, } b, err := xml.MarshalIndent(&lc, "", "\t") if err != nil { t.Fatal(err) } var lc1 Lifecycle err = xml.Unmarshal(b, &lc1) if err != nil { t.Fatal(err) } ruleSet := make(map[string]struct{}) for _, rule := range lc.Rules { ruleBytes, err := xml.Marshal(rule) if err != nil { t.Fatal(err) } ruleSet[string(ruleBytes)] = struct{}{} } for _, rule := range lc1.Rules { ruleBytes, err := xml.Marshal(rule) if err != nil { t.Fatal(err) } if _, ok := ruleSet[string(ruleBytes)]; !ok { t.Fatalf("Expected %v to be equal to %v, %v missing", lc, lc1, rule) } } } func TestExpectedExpiryTime(t *testing.T) { testCases := []struct { modTime time.Time days ExpirationDays expected time.Time }{ { time.Date(2020, time.March, 15, 10, 10, 10, 0, time.UTC), 4, time.Date(2020, time.March, 20, 0, 0, 0, 0, time.UTC), }, { time.Date(2020, time.March, 15, 0, 0, 0, 0, time.UTC), 1, time.Date(2020, time.March, 17, 0, 0, 0, 0, time.UTC), }, } for i, tc := range testCases { t.Run(fmt.Sprintf("Test %d", i+1), func(t *testing.T) { got := ExpectedExpiryTime(tc.modTime, int(tc.days)) if !got.Equal(tc.expected) { t.Fatalf("Expected %v to be equal to %v", got, tc.expected) } }) } } func TestEval(t *testing.T) { testCases := []struct { inputConfig string objectName string objectTags string objectModTime time.Time isDelMarker bool hasManyVersions bool expectedAction Action isNoncurrent bool objectSuccessorModTime time.Time versionID string }{ // Empty object name (unexpected case) should always return NoneAction { inputConfig: `prefixEnabled5`, expectedAction: NoneAction, }, // Disabled should always return NoneAction { inputConfig: `foodir/Disabled5`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago expectedAction: NoneAction, }, // No modTime, should be none-action { inputConfig: `foodir/Enabled5`, objectName: "foodir/fooobject", expectedAction: NoneAction, }, // Prefix not matched { inputConfig: `foodir/Enabled5`, objectName: "foxdir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago expectedAction: NoneAction, }, // Test rule with empty prefix e.g. for whole bucket { inputConfig: `Enabled5`, objectName: "foxdir/fooobject/foo.txt", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago expectedAction: DeleteAction, }, // Too early to remove (test Days) { inputConfig: `foodir/Enabled5`, objectName: "foxdir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago expectedAction: NoneAction, }, // Should remove (test Days) { inputConfig: `foodir/Enabled5`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-6 * 24 * time.Hour), // Created 6 days ago expectedAction: DeleteAction, }, // Too early to remove (test Date) { inputConfig: `foodir/Enabled` + time.Now().UTC().Truncate(24*time.Hour).Add(24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: NoneAction, }, // Should remove (test Days) { inputConfig: `foodir/Enabled` + time.Now().UTC().Truncate(24*time.Hour).Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove (Tags match) { inputConfig: `foodir/tag1value1Enabled` + time.Now().UTC().Truncate(24*time.Hour).Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectTags: "tag1=value1&tag2=value2", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove (Multiple Rules, Tags match) { inputConfig: `foodir/tag1value1tag2value2Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + `abc/tag2valueEnabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectTags: "tag1=value1&tag2=value2", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove (Tags match) { inputConfig: `foodir/tag1value1tag2value2Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectTags: "tag1=value1&tag2=value2", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove (Tags match with inverted order) { inputConfig: `factorytruestoreforeverfalseEnabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "fooobject", objectTags: "storeforever=false&factory=true", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove (Tags with encoded chars) { inputConfig: `factorytruestore foreverfalseEnabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "fooobject", objectTags: "store+forever=false&factory=true", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should not remove (Tags don't match) { inputConfig: `foodir/tagvalue1Enabled` + time.Now().UTC().Truncate(24*time.Hour).Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectTags: "tag1=value1", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: NoneAction, }, // Should not remove (Tags match, but prefix doesn't match) { inputConfig: `foodir/tag1value1Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foxdir/fooobject", objectTags: "tag1=value1", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: NoneAction, }, // Should remove - empty prefix, tags match, date expiration kicked in { inputConfig: `tag1value1Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foxdir/fooobject", objectTags: "tag1=value1", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should remove - empty prefix, tags match, object is expired based on specified Days { inputConfig: `tag1value1Enabled1`, objectName: "foxdir/fooobject", objectTags: "tag1=value1", objectModTime: time.Now().UTC().Add(-48 * time.Hour), // Created 2 day ago expectedAction: DeleteAction, }, // Should remove, the second rule has expiration kicked in { inputConfig: `Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(24*time.Hour).Format(time.RFC3339) + `foxdir/Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foxdir/fooobject", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should accept BucketLifecycleConfiguration root tag { inputConfig: `foodir/Enabled` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + ``, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago expectedAction: DeleteAction, }, // Should delete expired delete marker right away { inputConfig: `trueEnabled`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-1 * time.Hour), // Created one hour ago isDelMarker: true, expectedAction: DeleteVersionAction, }, // Should not expire a delete marker; ExpiredObjectDeleteAllVersions applies only when current version is not a DEL marker. { inputConfig: `1trueEnabled`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago isDelMarker: true, hasManyVersions: true, expectedAction: NoneAction, }, // Should delete all versions of this object since the latest version has past the expiry days criteria { inputConfig: `1trueEnabled`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago hasManyVersions: true, expectedAction: DeleteAllVersionsAction, }, // TransitionAction applies since object doesn't meet the age criteria for DeleteAllVersions { inputConfig: `30true10WARM-1Enabled`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-11 * 24 * time.Hour), // Created 11 days ago hasManyVersions: true, expectedAction: TransitionAction, }, // Should not delete expired marker if its time has not come yet { inputConfig: `Enabled1`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-12 * time.Hour), // Created 12 hours ago isDelMarker: true, expectedAction: NoneAction, }, // Should delete expired marker since its time has come { inputConfig: `Enabled1`, objectName: "foodir/fooobject", objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago isDelMarker: true, expectedAction: DeleteVersionAction, }, // Should transition immediately when Transition days is zero { inputConfig: `Enabled0S3TIER-1`, objectName: "foodir/fooobject", objectModTime: time.Now().Add(-1 * time.Nanosecond).UTC(), // Created now expectedAction: TransitionAction, }, // Should transition immediately when NoncurrentVersion Transition days is zero { inputConfig: `Enabled0S3TIER-1`, objectName: "foodir/fooobject", objectModTime: time.Now().Add(-1 * time.Nanosecond).UTC(), // Created now expectedAction: TransitionVersionAction, isNoncurrent: true, objectSuccessorModTime: time.Now().Add(-1 * time.Nanosecond).UTC(), versionID: uuid.New().String(), }, // Lifecycle rules with NewerNoncurrentVersions specified must return NoneAction. { inputConfig: `foodir/Enabled5`, objectName: "foodir/fooobject", versionID: uuid.NewString(), objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago expectedAction: NoneAction, }, // Disabled rules with NewerNoncurrentVersions shouldn't affect outcome. { inputConfig: `foodir/Enabled5foodir/Disabled5`, objectName: "foodir/fooobject", versionID: uuid.NewString(), objectModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago objectSuccessorModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // Created 10 days ago isNoncurrent: true, expectedAction: DeleteVersionAction, }, { inputConfig: ` Rule 1 Enabled 365 Rule 2 logs/ Enabled STANDARD_IA 30 `, objectName: "logs/obj-1", objectModTime: time.Now().UTC().Add(-31 * 24 * time.Hour), expectedAction: TransitionAction, }, { inputConfig: ` Rule 1 logs/ Enabled 365 Rule 2 logs/ Enabled STANDARD_IA 365 `, objectName: "logs/obj-1", objectModTime: time.Now().UTC().Add(-366 * 24 * time.Hour), expectedAction: DeleteAction, }, { inputConfig: ` Rule 1 tag1 value1 Enabled GLACIER 365 Rule 2 tag2 value2 Enabled 14 `, objectName: "obj-1", objectTags: "tag1=value1&tag2=value2", objectModTime: time.Now().UTC().Add(-15 * 24 * time.Hour), expectedAction: DeleteAction, }, { inputConfig: ` Rule 1 Enabled WARM-1 30 60 `, objectName: "obj-1", objectModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), expectedAction: DeleteAction, }, { inputConfig: ` Rule 2 Enabled 60 WARM-1 30 `, objectName: "obj-1", isNoncurrent: true, objectModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), objectSuccessorModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), expectedAction: DeleteVersionAction, }, { // DelMarkerExpiration is preferred since object age is past both transition and expiration days. inputConfig: ` DelMarkerExpiration with Transition Enabled 60 WARM-1 30 `, objectName: "obj-1", objectModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), isDelMarker: true, expectedAction: DelMarkerDeleteAllVersionsAction, }, { // NoneAction since object doesn't qualify for DelMarkerExpiration yet. // Note: TransitionAction doesn't apply to DEL marker inputConfig: ` DelMarkerExpiration with Transition Enabled 60 WARM-1 30 `, objectName: "obj-1", objectModTime: time.Now().UTC().Add(-50 * 24 * time.Hour), isDelMarker: true, expectedAction: NoneAction, }, { inputConfig: ` DelMarkerExpiration with non DEL-marker object Enabled 60 `, objectName: "obj-1", objectModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), expectedAction: NoneAction, }, { inputConfig: ` DelMarkerExpiration with noncurrent DEL-marker Enabled 60 `, objectName: "obj-1", objectModTime: time.Now().UTC().Add(-90 * 24 * time.Hour), objectSuccessorModTime: time.Now().UTC().Add(-60 * 24 * time.Hour), isDelMarker: true, isNoncurrent: true, expectedAction: NoneAction, }, } for _, tc := range testCases { tc := tc t.Run("", func(t *testing.T) { lc, err := ParseLifecycleConfig(bytes.NewReader([]byte(tc.inputConfig))) if err != nil { t.Fatalf("Got unexpected error: %v", err) } opts := ObjectOpts{ Name: tc.objectName, UserTags: tc.objectTags, ModTime: tc.objectModTime, DeleteMarker: tc.isDelMarker, IsLatest: !tc.isNoncurrent, SuccessorModTime: tc.objectSuccessorModTime, VersionID: tc.versionID, } opts.NumVersions = 1 if tc.hasManyVersions { opts.NumVersions = 2 // at least one noncurrent version } if res := lc.Eval(opts); res.Action != tc.expectedAction { t.Fatalf("Expected action: `%v`, got: `%v`", tc.expectedAction, res.Action) } }) } } func TestHasActiveRules(t *testing.T) { testCases := []struct { inputConfig string prefix string want bool }{ { inputConfig: `foodir/Enabled5`, prefix: "foodir/foobject", want: true, }, { // empty prefix inputConfig: `Enabled5`, prefix: "foodir/foobject/foo.txt", want: true, }, { inputConfig: `foodir/Enabled5`, prefix: "zdir/foobject", want: false, }, { inputConfig: `foodir/zdir/Enabled5`, prefix: "foodir/", want: true, }, { inputConfig: `Disabled5`, prefix: "foodir/", want: false, }, { inputConfig: `foodir/Enabled2999-01-01T00:00:00.000Z`, prefix: "foodir/foobject", want: false, }, { inputConfig: `EnabledS3TIER-1`, prefix: "foodir/foobject/foo.txt", want: true, }, { inputConfig: `EnabledS3TIER-1`, prefix: "foodir/foobject/foo.txt", want: true, }, { inputConfig: `Enabledtrue`, prefix: "", want: true, }, { inputConfig: `Enabled42true`, prefix: "", want: true, }, } for i, tc := range testCases { tc := tc t.Run(fmt.Sprintf("Test_%d", i+1), func(t *testing.T) { lc, err := ParseLifecycleConfig(bytes.NewReader([]byte(tc.inputConfig))) if err != nil { t.Fatalf("Got unexpected error: %v", err) } // To ensure input lifecycle configurations are valid if err := lc.Validate(); err != nil { t.Fatalf("Invalid test case: %d %v", i+1, err) } if got := lc.HasActiveRules(tc.prefix); got != tc.want { t.Fatalf("Expected result: `%v`, got: `%v`", tc.want, got) } }) } } func TestSetPredictionHeaders(t *testing.T) { lc := Lifecycle{ Rules: []Rule{ { ID: "rule-1", Status: "Enabled", Expiration: Expiration{ Days: ExpirationDays(3), set: true, }, }, { ID: "rule-2", Status: "Enabled", Transition: Transition{ Days: TransitionDays(3), StorageClass: "TIER-1", set: true, }, }, { ID: "rule-3", Status: "Enabled", NoncurrentVersionTransition: NoncurrentVersionTransition{ NoncurrentDays: TransitionDays(5), StorageClass: "TIER-2", set: true, }, }, }, } // current version obj1 := ObjectOpts{ Name: "obj1", IsLatest: true, } // non-current version obj2 := ObjectOpts{ Name: "obj2", } tests := []struct { obj ObjectOpts expRuleID int transRuleID int }{ { obj: obj1, expRuleID: 0, transRuleID: 1, }, { obj: obj2, expRuleID: 0, transRuleID: 2, }, } for i, tc := range tests { w := httptest.NewRecorder() lc.SetPredictionHeaders(w, tc.obj) if expHdrs, ok := w.Header()[xhttp.AmzExpiration]; ok && !strings.Contains(expHdrs[0], lc.Rules[tc.expRuleID].ID) { t.Fatalf("Test %d: Expected %s header", i+1, xhttp.AmzExpiration) } if transHdrs, ok := w.Header()[xhttp.MinIOTransition]; ok { if !strings.Contains(transHdrs[0], lc.Rules[tc.transRuleID].ID) { t.Fatalf("Test %d: Expected %s header", i+1, xhttp.MinIOTransition) } if tc.obj.IsLatest { if expectedDue, _ := lc.Rules[tc.transRuleID].Transition.NextDue(tc.obj); !strings.Contains(transHdrs[0], expectedDue.Format(http.TimeFormat)) { t.Fatalf("Test %d: Expected transition time %s", i+1, expectedDue) } } else { if expectedDue, _ := lc.Rules[tc.transRuleID].NoncurrentVersionTransition.NextDue(tc.obj); !strings.Contains(transHdrs[0], expectedDue.Format(http.TimeFormat)) { t.Fatalf("Test %d: Expected transition time %s", i+1, expectedDue) } } } } } func TestTransitionTier(t *testing.T) { lc := Lifecycle{ Rules: []Rule{ { ID: "rule-1", Status: "Enabled", Transition: Transition{ Days: TransitionDays(3), StorageClass: "TIER-1", }, }, { ID: "rule-2", Status: "Enabled", NoncurrentVersionTransition: NoncurrentVersionTransition{ NoncurrentDays: TransitionDays(3), StorageClass: "TIER-2", }, }, }, } now := time.Now().UTC() obj1 := ObjectOpts{ Name: "obj1", IsLatest: true, ModTime: now, } obj2 := ObjectOpts{ Name: "obj2", ModTime: now, } // Go back seven days in the past now = now.Add(7 * 24 * time.Hour) evt := lc.eval(obj1, now) if evt.Action != TransitionAction { t.Fatalf("Expected action: %s but got %s", TransitionAction, evt.Action) } if evt.StorageClass != "TIER-1" { t.Fatalf("Expected TIER-1 but got %s", evt.StorageClass) } evt = lc.eval(obj2, now) if evt.Action != TransitionVersionAction { t.Fatalf("Expected action: %s but got %s", TransitionVersionAction, evt.Action) } if evt.StorageClass != "TIER-2" { t.Fatalf("Expected TIER-2 but got %s", evt.StorageClass) } } func TestTransitionTierWithPrefixAndTags(t *testing.T) { lc := Lifecycle{ Rules: []Rule{ { ID: "rule-1", Status: "Enabled", Filter: Filter{ Prefix: Prefix{ set: true, string: "abcd/", }, }, Transition: Transition{ Days: TransitionDays(3), StorageClass: "TIER-1", }, }, { ID: "rule-2", Status: "Enabled", Filter: Filter{ tagSet: true, Tag: Tag{ Key: "priority", Value: "low", }, }, Transition: Transition{ Days: TransitionDays(3), StorageClass: "TIER-2", }, }, }, } now := time.Now().UTC() obj1 := ObjectOpts{ Name: "obj1", IsLatest: true, ModTime: now, } obj2 := ObjectOpts{ Name: "abcd/obj2", IsLatest: true, ModTime: now, } obj3 := ObjectOpts{ Name: "obj3", IsLatest: true, ModTime: now, UserTags: "priority=low", } // Go back seven days in the past now = now.Add(7 * 24 * time.Hour) // Eval object 1 evt := lc.eval(obj1, now) if evt.Action != NoneAction { t.Fatalf("Expected action: %s but got %s", NoneAction, evt.Action) } // Eval object 2 evt = lc.eval(obj2, now) if evt.Action != TransitionAction { t.Fatalf("Expected action: %s but got %s", TransitionAction, evt.Action) } if evt.StorageClass != "TIER-1" { t.Fatalf("Expected TIER-1 but got %s", evt.StorageClass) } // Eval object 3 evt = lc.eval(obj3, now) if evt.Action != TransitionAction { t.Fatalf("Expected action: %s but got %s", TransitionAction, evt.Action) } if evt.StorageClass != "TIER-2" { t.Fatalf("Expected TIER-2 but got %s", evt.StorageClass) } } func TestNoncurrentVersionsLimit(t *testing.T) { // test that the lowest max noncurrent versions limit is returned among // matching rules var rules []Rule for i := 1; i <= 10; i++ { rules = append(rules, Rule{ ID: strconv.Itoa(i), Status: "Enabled", NoncurrentVersionExpiration: NoncurrentVersionExpiration{ NewerNoncurrentVersions: i, NoncurrentDays: ExpirationDays(i), }, }) } lc := Lifecycle{ Rules: rules, } if event := lc.NoncurrentVersionsExpirationLimit(ObjectOpts{Name: "obj"}); event.RuleID != "1" || event.NoncurrentDays != 1 || event.NewerNoncurrentVersions != 1 { t.Fatalf("Expected (ruleID, days, lim) to be (\"1\", 1, 1) but got (%s, %d, %d)", event.RuleID, event.NoncurrentDays, event.NewerNoncurrentVersions) } } func TestMaxNoncurrentBackwardCompat(t *testing.T) { testCases := []struct { xml string expected NoncurrentVersionExpiration }{ { xml: `13`, expected: NoncurrentVersionExpiration{ XMLName: xml.Name{ Local: "NoncurrentVersionExpiration", }, NoncurrentDays: 1, NewerNoncurrentVersions: 3, set: true, }, }, { xml: `24`, expected: NoncurrentVersionExpiration{ XMLName: xml.Name{ Local: "NoncurrentVersionExpiration", }, NoncurrentDays: 2, NewerNoncurrentVersions: 4, set: true, }, }, } for i, tc := range testCases { var got NoncurrentVersionExpiration dec := xml.NewDecoder(strings.NewReader(tc.xml)) if err := dec.Decode(&got); err != nil || got != tc.expected { if err != nil { t.Fatalf("%d: Failed to unmarshal xml %v", i+1, err) } t.Fatalf("%d: Expected %v but got %v", i+1, tc.expected, got) } } } func TestParseLifecycleConfigWithID(t *testing.T) { r := bytes.NewReader([]byte(` rule-1 prefix Enabled 3 another-prefix Enabled 3 `)) lc, err := ParseLifecycleConfigWithID(r) if err != nil { t.Fatalf("Expected parsing to succeed but failed with %v", err) } for _, rule := range lc.Rules { if rule.ID == "" { t.Fatalf("Expected all rules to have a unique id assigned %#v", rule) } } } func TestFilterAndSetPredictionHeaders(t *testing.T) { lc := Lifecycle{ Rules: []Rule{ { ID: "rule-1", Status: "Enabled", Filter: Filter{ set: true, Prefix: Prefix{ string: "folder1/folder1/exp_dt=2022-", set: true, }, }, Expiration: Expiration{ Days: 1, set: true, }, }, }, } tests := []struct { opts ObjectOpts lc Lifecycle want int }{ { opts: ObjectOpts{ Name: "folder1/folder1/exp_dt=2022-08-01/obj-1", ModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), VersionID: "", IsLatest: true, NumVersions: 1, }, want: 1, lc: lc, }, { opts: ObjectOpts{ Name: "folder1/folder1/exp_dt=9999-01-01/obj-1", ModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), VersionID: "", IsLatest: true, NumVersions: 1, }, want: 0, lc: lc, }, } for i, tc := range tests { t.Run(fmt.Sprintf("test-%d", i+1), func(t *testing.T) { if got := tc.lc.FilterRules(tc.opts); len(got) != tc.want { t.Fatalf("Expected %d rules to match but got %d", tc.want, len(got)) } w := httptest.NewRecorder() tc.lc.SetPredictionHeaders(w, tc.opts) expHdr, ok := w.Header()[xhttp.AmzExpiration] switch { case ok && tc.want == 0: t.Fatalf("Expected no rule to match but found x-amz-expiration header set: %v", expHdr) case !ok && tc.want > 0: t.Fatal("Expected x-amz-expiration header to be set but not found") } }) } } func TestFilterRules(t *testing.T) { rules := []Rule{ { ID: "rule-1", Status: "Enabled", Filter: Filter{ set: true, Tag: Tag{ Key: "key1", Value: "val1", }, }, Expiration: Expiration{ set: true, Days: 1, }, }, { ID: "rule-with-sz-lt", Status: "Enabled", Filter: Filter{ set: true, ObjectSizeLessThan: 100 * humanize.MiByte, }, Expiration: Expiration{ set: true, Days: 1, }, }, { ID: "rule-with-sz-gt", Status: "Enabled", Filter: Filter{ set: true, ObjectSizeGreaterThan: 1 * humanize.MiByte, }, Expiration: Expiration{ set: true, Days: 1, }, }, { ID: "rule-with-sz-lt-and-tag", Status: "Enabled", Filter: Filter{ set: true, And: And{ ObjectSizeLessThan: 100 * humanize.MiByte, Tags: []Tag{ { Key: "key1", Value: "val1", }, }, }, }, Expiration: Expiration{ set: true, Days: 1, }, }, { ID: "rule-with-sz-gt-and-tag", Status: "Enabled", Filter: Filter{ set: true, And: And{ ObjectSizeGreaterThan: 1 * humanize.MiByte, Tags: []Tag{ { Key: "key1", Value: "val1", }, }, }, }, Expiration: Expiration{ set: true, Days: 1, }, }, { ID: "rule-with-sz-lt-and-gt", Status: "Enabled", Filter: Filter{ set: true, And: And{ ObjectSizeGreaterThan: 101 * humanize.MiByte, ObjectSizeLessThan: 200 * humanize.MiByte, }, }, Expiration: Expiration{ set: true, Days: 1, }, }, } tests := []struct { lc Lifecycle opts ObjectOpts hasRules bool }{ { // Delete marker shouldn't match filter without tags lc: Lifecycle{ Rules: []Rule{ rules[0], }, }, opts: ObjectOpts{ DeleteMarker: true, IsLatest: true, Name: "obj-1", }, hasRules: false, }, { // PUT version with no matching tags lc: Lifecycle{ Rules: []Rule{ rules[0], }, }, opts: ObjectOpts{ IsLatest: true, Name: "obj-1", Size: 1 * humanize.MiByte, }, hasRules: false, }, { // PUT version with matching tags lc: Lifecycle{ Rules: []Rule{ rules[0], }, }, opts: ObjectOpts{ IsLatest: true, UserTags: "key1=val1", Name: "obj-1", Size: 2 * humanize.MiByte, }, hasRules: true, }, { // PUT version with size based filters lc: Lifecycle{ Rules: []Rule{ rules[1], rules[2], rules[3], rules[4], rules[5], }, }, opts: ObjectOpts{ IsLatest: true, UserTags: "key1=val1", Name: "obj-1", Size: 1*humanize.MiByte - 1, }, hasRules: true, }, { // PUT version with size based filters lc: Lifecycle{ Rules: []Rule{ rules[1], rules[2], rules[3], rules[4], rules[5], }, }, opts: ObjectOpts{ IsLatest: true, Name: "obj-1", Size: 1*humanize.MiByte + 1, }, hasRules: true, }, { // DEL version with size based filters lc: Lifecycle{ Rules: []Rule{ rules[1], rules[2], rules[3], rules[4], rules[5], }, }, opts: ObjectOpts{ DeleteMarker: true, IsLatest: true, Name: "obj-1", }, hasRules: true, }, } for i, tc := range tests { t.Run(fmt.Sprintf("test-%d", i+1), func(t *testing.T) { if err := tc.lc.Validate(); err != nil { t.Fatalf("Lifecycle validation failed - %v", err) } rules := tc.lc.FilterRules(tc.opts) if tc.hasRules && len(rules) == 0 { t.Fatalf("%d: Expected at least one rule to match but none matched", i+1) } if !tc.hasRules && len(rules) > 0 { t.Fatalf("%d: Expected no rules to match but got matches %v", i+1, rules) } }) } } // TestDeleteAllVersions tests ordering among events, especially ones which // expire all versions like ExpiredObjectDeleteAllVersions and // DelMarkerExpiration func TestDeleteAllVersions(t *testing.T) { // ExpiredObjectDeleteAllVersions lc := Lifecycle{ Rules: []Rule{ { ID: "ExpiredObjectDeleteAllVersions-20", Status: "Enabled", Expiration: Expiration{ set: true, DeleteAll: Boolean{val: true, set: true}, Days: 20, }, }, { ID: "Transition-10", Status: "Enabled", Transition: Transition{ set: true, StorageClass: "WARM-1", Days: 10, }, }, }, } opts := ObjectOpts{ Name: "foo.txt", ModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // created 10 days ago Size: 0, VersionID: uuid.New().String(), IsLatest: true, NumVersions: 4, } event := lc.eval(opts, time.Time{}) if event.Action != TransitionAction { t.Fatalf("Expected %v action but got %v", TransitionAction, event.Action) } // The earlier upcoming lifecycle event must be picked, i.e rule with id "Transition-10" if exp := ExpectedExpiryTime(opts.ModTime, 10); exp != event.Due { t.Fatalf("Expected due %v but got %v, ruleID=%v", exp, event.Due, event.RuleID) } // DelMarkerExpiration lc = Lifecycle{ Rules: []Rule{ { ID: "delmarker-exp-20", Status: "Enabled", DelMarkerExpiration: DelMarkerExpiration{ Days: 20, }, }, { ID: "delmarker-exp-10", Status: "Enabled", DelMarkerExpiration: DelMarkerExpiration{ Days: 10, }, }, }, } opts = ObjectOpts{ Name: "foo.txt", ModTime: time.Now().UTC().Add(-10 * 24 * time.Hour), // created 10 days ago Size: 0, VersionID: uuid.New().String(), IsLatest: true, DeleteMarker: true, NumVersions: 4, } event = lc.eval(opts, time.Time{}) if event.Action != DelMarkerDeleteAllVersionsAction { t.Fatalf("Expected %v action but got %v", DelMarkerDeleteAllVersionsAction, event.Action) } // The earlier upcoming lifecycle event must be picked, i.e rule with id "delmarker-exp-10" if exp := ExpectedExpiryTime(opts.ModTime, 10); exp != event.Due { t.Fatalf("Expected due %v but got %v, ruleID=%v", exp, event.Due, event.RuleID) } }