lifecycle: Support tags with special characters (#14906)

Object tags can have special characters such as whitespace. However
the current code doesn't properly consider those characters while
evaluating the lifecycle document.

ObjectInfo.UserTags contains an url encoded form of object tags
(e.g. key+1=val)

This commit fixes the issue by using the tags package to parse object tags.
This commit is contained in:
Anis Elleuch 2022-05-14 18:25:55 +01:00 committed by GitHub
parent f28a8eca91
commit b0e2c2da78
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 44 additions and 17 deletions

View File

@ -20,6 +20,9 @@ package lifecycle
import ( import (
"encoding/xml" "encoding/xml"
"io" "io"
"log"
"github.com/minio/minio-go/v7/pkg/tags"
) )
var errInvalidFilter = Errorf("Filter must have exactly one of Prefix, Tag, or And specified") var errInvalidFilter = Errorf("Filter must have exactly one of Prefix, Tag, or And specified")
@ -36,8 +39,9 @@ type Filter struct {
Tag Tag Tag Tag
tagSet bool tagSet bool
// Caching tags, only once // Caching tags, only once
cachedTags []string cachedTags map[string]string
} }
// MarshalXML - produces the xml representation of the Filter struct // MarshalXML - produces the xml representation of the Filter struct
@ -150,27 +154,42 @@ func (f Filter) Validate() error {
// TestTags tests if the object tags satisfy the Filter tags requirement, // TestTags tests if the object tags satisfy the Filter tags requirement,
// it returns true if there is no tags in the underlying Filter. // it returns true if there is no tags in the underlying Filter.
func (f Filter) TestTags(tags []string) bool { func (f Filter) TestTags(userTags string) bool {
if f.cachedTags == nil { if f.cachedTags == nil {
tags := make([]string, 0) cache := make(map[string]string)
for _, t := range append(f.And.Tags, f.Tag) { for _, t := range append(f.And.Tags, f.Tag) {
if !t.IsEmpty() { if !t.IsEmpty() {
tags = append(tags, t.String()) cache[t.Key] = t.Value
} }
} }
f.cachedTags = tags f.cachedTags = cache
} }
for _, ct := range f.cachedTags {
foundTag := false // This filter does not have any tags, always return true
for _, t := range tags { if len(f.cachedTags) == 0 {
if ct == t { return true
foundTag = true }
break
} parsedTags, err := tags.ParseObjectTags(userTags)
} if err != nil {
if !foundTag { log.Printf("Unexpected object tag found: `%s`\n", userTags)
return false return false
}
tagsMap := parsedTags.ToMap()
// This filter has tags configured but this object
// does not have any tag, skip this object
if len(tagsMap) == 0 {
return false
}
// Both filter and object have tags, find a match,
// skip this object otherwise
for k, cv := range f.cachedTags {
v, ok := tagsMap[k]
if ok && v == cv {
return true
} }
} }
return true return false
} }

View File

@ -259,7 +259,7 @@ func (lc Lifecycle) FilterActionableRules(obj ObjectOpts) []Rule {
continue continue
} }
if rule.Filter.TestTags(strings.Split(obj.UserTags, "&")) { if rule.Filter.TestTags(obj.UserTags) {
rules = append(rules, rule) rules = append(rules, rule)
} }
if !rule.Transition.IsNull() { if !rule.Transition.IsNull() {

View File

@ -325,6 +325,14 @@ func TestComputeActions(t *testing.T) {
objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago objectModTime: time.Now().UTC().Add(-24 * time.Hour), // Created 1 day ago
expectedAction: DeleteAction, expectedAction: DeleteAction,
}, },
// Should remove (Tags with encoded chars)
{
inputConfig: `<LifecycleConfiguration><Rule><Filter><And><Tag><Key>factory</Key><Value>true</Value></Tag><Tag><Key>store forever</Key><Value>false</Value></Tag></And></Filter><Status>Enabled</Status><Expiration><Date>` + time.Now().Truncate(24*time.Hour).UTC().Add(-24*time.Hour).Format(time.RFC3339) + `</Date></Expiration></Rule></LifecycleConfiguration>`,
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) // Should not remove (Tags don't match)
{ {