mirror of
https://github.com/minio/minio.git
synced 2025-01-03 19:13:22 -05:00
7926401cbd
i.e., this rule element doesn't apply to DEL markers. This is a breaking change to how ExpiredObejctDeleteAllVersions functions today. This is necessary to avoid the following highly probable footgun scenario in the future. Scenario: The user uses tags-based filtering to select an object's time to live(TTL). The application sometimes deletes objects, too, making its latest version a DEL marker. The previous implementation skipped tag-based filters if the newest version was DEL marker, voiding the tag-based TTL. The user is surprised to find objects that have expired sooner than expected. * Add DelMarkerExpiration action This ILM action removes all versions of an object if its the latest version is a DEL marker. ```xml <DelMarkerObjectExpiration> <Days> 10 </Days> </DelMarkerObjectExpiration> ``` 1. Applies only to objects whose, • The latest version is a DEL marker. • satisfies the number of days criteria 2. Deletes all versions of this object 3. Associated rule can't have tag-based filtering Includes, - New bucket event type for deletion due to DelMarkerExpiration
149 lines
4.9 KiB
Go
149 lines
4.9 KiB
Go
// 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 <http://www.gnu.org/licenses/>.
|
|
|
|
package lifecycle
|
|
|
|
import (
|
|
"encoding/xml"
|
|
"fmt"
|
|
"testing"
|
|
)
|
|
|
|
// TestInvalidRules checks if Rule xml with invalid elements returns
|
|
// appropriate errors on validation
|
|
func TestInvalidRules(t *testing.T) {
|
|
invalidTestCases := []struct {
|
|
inputXML string
|
|
expectedErr error
|
|
}{
|
|
{ // Rule with ID longer than 255 characters
|
|
inputXML: ` <Rule>
|
|
<ID> babababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababababab </ID>
|
|
</Rule>`,
|
|
expectedErr: errInvalidRuleID,
|
|
},
|
|
{ // Rule with empty ID
|
|
inputXML: `<Rule>
|
|
<ID></ID>
|
|
<Filter><Prefix></Prefix></Filter>
|
|
<Expiration>
|
|
<Days>365</Days>
|
|
</Expiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: nil,
|
|
},
|
|
{ // Rule with empty status
|
|
inputXML: ` <Rule>
|
|
<ID>rule with empty status</ID>
|
|
<Status></Status>
|
|
</Rule>`,
|
|
expectedErr: errEmptyRuleStatus,
|
|
},
|
|
{ // Rule with invalid status
|
|
inputXML: ` <Rule>
|
|
<ID>rule with invalid status</ID>
|
|
<Status>OK</Status>
|
|
</Rule>`,
|
|
expectedErr: errInvalidRuleStatus,
|
|
},
|
|
{ // Rule with negative values for ObjectSizeLessThan
|
|
inputXML: `<Rule>
|
|
<ID>negative-obj-size-less-than</ID>
|
|
<Filter><ObjectSizeLessThan>-1</ObjectSizeLessThan></Filter>
|
|
<Expiration>
|
|
<Days>365</Days>
|
|
</Expiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errXMLNotWellFormed,
|
|
},
|
|
{ // Rule with negative values for And>ObjectSizeLessThan
|
|
inputXML: `<Rule>
|
|
<ID>negative-and-obj-size-less-than</ID>
|
|
<Filter><And><ObjectSizeLessThan>-1</ObjectSizeLessThan></And></Filter>
|
|
<Expiration>
|
|
<Days>365</Days>
|
|
</Expiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errXMLNotWellFormed,
|
|
},
|
|
{ // Rule with negative values for ObjectSizeGreaterThan
|
|
inputXML: `<Rule>
|
|
<ID>negative-obj-size-greater-than</ID>
|
|
<Filter><ObjectSizeGreaterThan>-1</ObjectSizeGreaterThan></Filter>
|
|
<Expiration>
|
|
<Days>365</Days>
|
|
</Expiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errXMLNotWellFormed,
|
|
},
|
|
{ // Rule with negative values for And>ObjectSizeGreaterThan
|
|
inputXML: `<Rule>
|
|
<ID>negative-and-obj-size-greater-than</ID>
|
|
<Filter><And><ObjectSizeGreaterThan>-1</ObjectSizeGreaterThan></And></Filter>
|
|
<Expiration>
|
|
<Days>365</Days>
|
|
</Expiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errXMLNotWellFormed,
|
|
},
|
|
{
|
|
inputXML: `<Rule>
|
|
<ID>Rule with a tag and DelMarkerExpiration</ID>
|
|
<Filter><Tag><Key>k1</Key><Value>v1</Value></Tag></Filter>
|
|
<DelMarkerExpiration>
|
|
<Days>365</Days>
|
|
</DelMarkerExpiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errInvalidRuleDelMarkerExpiration,
|
|
},
|
|
{
|
|
inputXML: `<Rule>
|
|
<ID>Rule with multiple tags and DelMarkerExpiration</ID>
|
|
<Filter><And>
|
|
<Tag><Key>k1</Key><Value>v1</Value></Tag>
|
|
<Tag><Key>k2</Key><Value>v2</Value></Tag>
|
|
</And></Filter>
|
|
<DelMarkerExpiration>
|
|
<Days>365</Days>
|
|
</DelMarkerExpiration>
|
|
<Status>Enabled</Status>
|
|
</Rule>`,
|
|
expectedErr: errInvalidRuleDelMarkerExpiration,
|
|
},
|
|
}
|
|
|
|
for i, tc := range invalidTestCases {
|
|
t.Run(fmt.Sprintf("Test %d", i+1), func(t *testing.T) {
|
|
var rule Rule
|
|
err := xml.Unmarshal([]byte(tc.inputXML), &rule)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if err := rule.Validate(); err != tc.expectedErr {
|
|
t.Fatalf("%d: Expected %v but got %v", i+1, tc.expectedErr, err)
|
|
}
|
|
})
|
|
}
|
|
}
|