2021-04-18 12:41:13 -07:00
|
|
|
// 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/>.
|
2019-07-19 13:20:33 -07:00
|
|
|
|
|
|
|
package lifecycle
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/xml"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-06-12 20:04:01 -07:00
|
|
|
errLifecycleInvalidDate = Errorf("Date must be provided in ISO 8601 format")
|
|
|
|
errLifecycleInvalidDays = Errorf("Days must be positive integer when used with Expiration")
|
2020-08-25 04:56:50 +08:00
|
|
|
errLifecycleInvalidExpiration = Errorf("Exactly one of Days (positive integer) or Date (positive ISO 8601 format) should be present inside Expiration.")
|
2020-06-12 20:04:01 -07:00
|
|
|
errLifecycleInvalidDeleteMarker = Errorf("Delete marker cannot be specified with Days or Date in a Lifecycle Expiration Policy")
|
|
|
|
errLifecycleDateNotMidnight = Errorf("'Date' must be at midnight GMT")
|
2019-07-19 13:20:33 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// ExpirationDays is a type alias to unmarshal Days in Expiration
|
|
|
|
type ExpirationDays int
|
|
|
|
|
|
|
|
// UnmarshalXML parses number of days from Expiration and validates if
|
|
|
|
// greater than zero
|
|
|
|
func (eDays *ExpirationDays) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
|
|
|
|
var numDays int
|
|
|
|
err := d.DecodeElement(&numDays, &startElement)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if numDays <= 0 {
|
|
|
|
return errLifecycleInvalidDays
|
|
|
|
}
|
|
|
|
*eDays = ExpirationDays(numDays)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalXML encodes number of days to expire if it is non-zero and
|
|
|
|
// encodes empty string otherwise
|
2020-07-23 16:01:25 +01:00
|
|
|
func (eDays ExpirationDays) MarshalXML(e *xml.Encoder, startElement xml.StartElement) error {
|
|
|
|
if eDays == 0 {
|
2019-07-19 13:20:33 -07:00
|
|
|
return nil
|
|
|
|
}
|
2020-07-23 16:01:25 +01:00
|
|
|
return e.EncodeElement(int(eDays), startElement)
|
2019-07-19 13:20:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// ExpirationDate is a embedded type containing time.Time to unmarshal
|
|
|
|
// Date in Expiration
|
|
|
|
type ExpirationDate struct {
|
|
|
|
time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalXML parses date from Expiration and validates date format
|
|
|
|
func (eDate *ExpirationDate) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
|
|
|
|
var dateStr string
|
|
|
|
err := d.DecodeElement(&dateStr, &startElement)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// While AWS documentation mentions that the date specified
|
|
|
|
// must be present in ISO 8601 format, in reality they allow
|
|
|
|
// users to provide RFC 3339 compliant dates.
|
|
|
|
expDate, err := time.Parse(time.RFC3339, dateStr)
|
|
|
|
if err != nil {
|
|
|
|
return errLifecycleInvalidDate
|
|
|
|
}
|
|
|
|
// Allow only date timestamp specifying midnight GMT
|
|
|
|
hr, min, sec := expDate.Clock()
|
|
|
|
nsec := expDate.Nanosecond()
|
|
|
|
loc := expDate.Location()
|
|
|
|
if !(hr == 0 && min == 0 && sec == 0 && nsec == 0 && loc.String() == time.UTC.String()) {
|
|
|
|
return errLifecycleDateNotMidnight
|
|
|
|
}
|
|
|
|
|
|
|
|
*eDate = ExpirationDate{expDate}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalXML encodes expiration date if it is non-zero and encodes
|
|
|
|
// empty string otherwise
|
2020-07-23 16:01:25 +01:00
|
|
|
func (eDate ExpirationDate) MarshalXML(e *xml.Encoder, startElement xml.StartElement) error {
|
|
|
|
if eDate.Time.IsZero() {
|
2019-07-19 13:20:33 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return e.EncodeElement(eDate.Format(time.RFC3339), startElement)
|
|
|
|
}
|
|
|
|
|
2020-06-12 20:04:01 -07:00
|
|
|
// ExpireDeleteMarker represents value of ExpiredObjectDeleteMarker field in Expiration XML element.
|
2020-08-25 20:38:59 +01:00
|
|
|
type ExpireDeleteMarker struct {
|
|
|
|
val bool
|
|
|
|
set bool
|
|
|
|
}
|
2020-06-12 20:04:01 -07:00
|
|
|
|
2019-07-19 13:20:33 -07:00
|
|
|
// Expiration - expiration actions for a rule in lifecycle configuration.
|
|
|
|
type Expiration struct {
|
2020-06-12 20:04:01 -07:00
|
|
|
XMLName xml.Name `xml:"Expiration"`
|
|
|
|
Days ExpirationDays `xml:"Days,omitempty"`
|
|
|
|
Date ExpirationDate `xml:"Date,omitempty"`
|
2020-08-25 20:38:59 +01:00
|
|
|
DeleteMarker ExpireDeleteMarker `xml:"ExpiredObjectDeleteMarker"`
|
|
|
|
|
|
|
|
set bool
|
2020-06-12 20:04:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalXML encodes delete marker boolean into an XML form.
|
2020-07-05 17:08:42 +01:00
|
|
|
func (b ExpireDeleteMarker) MarshalXML(e *xml.Encoder, startElement xml.StartElement) error {
|
2020-08-25 20:38:59 +01:00
|
|
|
if !b.set {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return e.EncodeElement(b.val, startElement)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalXML decodes delete marker boolean from the XML form.
|
|
|
|
func (b *ExpireDeleteMarker) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
|
|
|
|
var exp bool
|
|
|
|
err := d.DecodeElement(&exp, &startElement)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.val = exp
|
|
|
|
b.set = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalXML encodes expiration field into an XML form.
|
|
|
|
func (e Expiration) MarshalXML(enc *xml.Encoder, startElement xml.StartElement) error {
|
|
|
|
if !e.set {
|
2020-06-12 20:04:01 -07:00
|
|
|
return nil
|
|
|
|
}
|
2020-08-25 20:38:59 +01:00
|
|
|
type expirationWrapper Expiration
|
|
|
|
return enc.EncodeElement(expirationWrapper(e), startElement)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalXML decodes expiration field from the XML form.
|
|
|
|
func (e *Expiration) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
|
|
|
|
type expirationWrapper Expiration
|
|
|
|
var exp expirationWrapper
|
|
|
|
err := d.DecodeElement(&exp, &startElement)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*e = Expiration(exp)
|
|
|
|
e.set = true
|
|
|
|
return nil
|
2019-07-19 13:20:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Validate - validates the "Expiration" element
|
|
|
|
func (e Expiration) Validate() error {
|
2020-08-25 20:38:59 +01:00
|
|
|
if !e.set {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-06-12 20:04:01 -07:00
|
|
|
// DeleteMarker cannot be specified if date or dates are specified.
|
2020-08-25 20:38:59 +01:00
|
|
|
if (!e.IsDaysNull() || !e.IsDateNull()) && e.DeleteMarker.set {
|
2020-06-12 20:04:01 -07:00
|
|
|
return errLifecycleInvalidDeleteMarker
|
|
|
|
}
|
|
|
|
|
2020-08-25 20:38:59 +01:00
|
|
|
if !e.DeleteMarker.set && e.IsDaysNull() && e.IsDateNull() {
|
|
|
|
return errXMLNotWellFormed
|
2019-07-19 13:20:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Both expiration days and date are specified
|
2019-08-09 18:02:41 +01:00
|
|
|
if !e.IsDaysNull() && !e.IsDateNull() {
|
2019-07-19 13:20:33 -07:00
|
|
|
return errLifecycleInvalidExpiration
|
|
|
|
}
|
2020-06-12 20:04:01 -07:00
|
|
|
|
2019-07-19 13:20:33 -07:00
|
|
|
return nil
|
|
|
|
}
|
2019-08-09 18:02:41 +01:00
|
|
|
|
|
|
|
// IsDaysNull returns true if days field is null
|
|
|
|
func (e Expiration) IsDaysNull() bool {
|
|
|
|
return e.Days == ExpirationDays(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsDateNull returns true if date field is null
|
|
|
|
func (e Expiration) IsDateNull() bool {
|
2020-06-12 10:28:21 -07:00
|
|
|
return e.Date.Time.IsZero()
|
2019-08-09 18:02:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsNull returns true if both date and days fields are null
|
|
|
|
func (e Expiration) IsNull() bool {
|
|
|
|
return e.IsDaysNull() && e.IsDateNull()
|
|
|
|
}
|