You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
4.7 KiB
202 lines
4.7 KiB
10 years ago
|
package donut
|
||
10 years ago
|
|
||
|
import (
|
||
|
"regexp"
|
||
10 years ago
|
"strings"
|
||
10 years ago
|
"time"
|
||
|
"unicode/utf8"
|
||
|
)
|
||
|
|
||
10 years ago
|
// BucketACL - bucket level access control
|
||
|
type BucketACL string
|
||
|
|
||
|
// different types of ACL's currently supported for buckets
|
||
|
const (
|
||
|
BucketPrivate = BucketACL("private")
|
||
|
BucketPublicRead = BucketACL("public-read")
|
||
|
BucketPublicReadWrite = BucketACL("public-read-write")
|
||
|
)
|
||
|
|
||
|
func (b BucketACL) String() string {
|
||
|
return string(b)
|
||
|
}
|
||
|
|
||
|
// IsPrivate - is acl Private
|
||
|
func (b BucketACL) IsPrivate() bool {
|
||
|
return b == BucketACL("private")
|
||
|
}
|
||
|
|
||
|
// IsPublicRead - is acl PublicRead
|
||
|
func (b BucketACL) IsPublicRead() bool {
|
||
|
return b == BucketACL("public-read")
|
||
|
}
|
||
|
|
||
|
// IsPublicReadWrite - is acl PublicReadWrite
|
||
|
func (b BucketACL) IsPublicReadWrite() bool {
|
||
|
return b == BucketACL("public-read-write")
|
||
|
}
|
||
|
|
||
10 years ago
|
// FilterMode type
|
||
|
type FilterMode int
|
||
|
|
||
|
// FilterMode list
|
||
|
const (
|
||
|
DelimiterPrefixMode FilterMode = iota
|
||
|
DelimiterMode
|
||
|
PrefixMode
|
||
|
DefaultMode
|
||
|
)
|
||
|
|
||
10 years ago
|
// PartMetadata - various types of individual part resources
|
||
|
type PartMetadata struct {
|
||
|
PartNumber int
|
||
|
LastModified time.Time
|
||
|
ETag string
|
||
|
Size int64
|
||
|
}
|
||
|
|
||
|
// ObjectResourcesMetadata - various types of object resources
|
||
|
type ObjectResourcesMetadata struct {
|
||
10 years ago
|
Bucket string
|
||
|
EncodingType string
|
||
|
Key string
|
||
|
UploadID string
|
||
10 years ago
|
StorageClass string
|
||
|
PartNumberMarker int
|
||
|
NextPartNumberMarker int
|
||
|
MaxParts int
|
||
|
IsTruncated bool
|
||
|
|
||
|
Part []*PartMetadata
|
||
|
}
|
||
|
|
||
10 years ago
|
// UploadMetadata container capturing metadata on in progress multipart upload in a given bucket
|
||
|
type UploadMetadata struct {
|
||
|
Key string
|
||
|
UploadID string
|
||
|
StorageClass string
|
||
|
Initiated time.Time
|
||
|
}
|
||
|
|
||
|
// BucketMultipartResourcesMetadata - various types of bucket resources for inprogress multipart uploads
|
||
|
type BucketMultipartResourcesMetadata struct {
|
||
|
KeyMarker string
|
||
|
UploadIDMarker string
|
||
|
NextKeyMarker string
|
||
|
NextUploadIDMarker string
|
||
|
EncodingType string
|
||
|
MaxUploads int
|
||
|
IsTruncated bool
|
||
|
Upload []*UploadMetadata
|
||
|
Prefix string
|
||
|
Delimiter string
|
||
|
CommonPrefixes []string
|
||
|
}
|
||
|
|
||
10 years ago
|
// BucketResourcesMetadata - various types of bucket resources
|
||
|
type BucketResourcesMetadata struct {
|
||
|
Prefix string
|
||
|
Marker string
|
||
10 years ago
|
NextMarker string
|
||
10 years ago
|
Maxkeys int
|
||
10 years ago
|
EncodingType string
|
||
10 years ago
|
Delimiter string
|
||
|
IsTruncated bool
|
||
|
CommonPrefixes []string
|
||
|
Mode FilterMode
|
||
|
}
|
||
|
|
||
|
// GetMode - Populate filter mode
|
||
|
func GetMode(resources BucketResourcesMetadata) FilterMode {
|
||
|
var f FilterMode
|
||
|
switch true {
|
||
|
case resources.Delimiter != "" && resources.Prefix != "":
|
||
|
f = DelimiterPrefixMode
|
||
|
case resources.Delimiter != "" && resources.Prefix == "":
|
||
|
f = DelimiterMode
|
||
|
case resources.Delimiter == "" && resources.Prefix != "":
|
||
|
f = PrefixMode
|
||
|
case resources.Delimiter == "" && resources.Prefix == "":
|
||
|
f = DefaultMode
|
||
|
}
|
||
|
|
||
|
return f
|
||
|
}
|
||
|
|
||
10 years ago
|
// IsValidBucketACL - is provided acl string supported
|
||
|
func IsValidBucketACL(acl string) bool {
|
||
|
switch acl {
|
||
|
case "private":
|
||
|
fallthrough
|
||
|
case "public-read":
|
||
|
fallthrough
|
||
|
case "public-read-write":
|
||
|
return true
|
||
|
case "":
|
||
|
// by default its "private"
|
||
|
return true
|
||
|
default:
|
||
|
return false
|
||
|
}
|
||
|
}
|
||
|
|
||
10 years ago
|
// IsDelimiterPrefixSet Delimiter and Prefix set
|
||
|
func (b BucketResourcesMetadata) IsDelimiterPrefixSet() bool {
|
||
|
return b.Mode == DelimiterPrefixMode
|
||
|
}
|
||
|
|
||
|
// IsDelimiterSet Delimiter set
|
||
|
func (b BucketResourcesMetadata) IsDelimiterSet() bool {
|
||
|
return b.Mode == DelimiterMode
|
||
|
}
|
||
|
|
||
|
// IsPrefixSet Prefix set
|
||
|
func (b BucketResourcesMetadata) IsPrefixSet() bool {
|
||
|
return b.Mode == PrefixMode
|
||
|
}
|
||
|
|
||
|
// IsDefault No query values
|
||
|
func (b BucketResourcesMetadata) IsDefault() bool {
|
||
|
return b.Mode == DefaultMode
|
||
|
}
|
||
|
|
||
|
// IsValidBucket - verify bucket name in accordance with
|
||
|
// - http://docs.aws.amazon.com/AmazonS3/latest/dev/UsingBucket.html
|
||
|
func IsValidBucket(bucket string) bool {
|
||
|
if len(bucket) < 3 || len(bucket) > 63 {
|
||
|
return false
|
||
|
}
|
||
|
if bucket[0] == '.' || bucket[len(bucket)-1] == '.' {
|
||
|
return false
|
||
|
}
|
||
|
if match, _ := regexp.MatchString("\\.\\.", bucket); match == true {
|
||
|
return false
|
||
|
}
|
||
|
// We don't support buckets with '.' in them
|
||
|
match, _ := regexp.MatchString("^[a-zA-Z][a-zA-Z0-9\\-]+[a-zA-Z0-9]$", bucket)
|
||
|
return match
|
||
|
}
|
||
|
|
||
10 years ago
|
// IsValidObjectName - verify object name in accordance with
|
||
10 years ago
|
// - http://docs.aws.amazon.com/AmazonS3/latest/dev/UsingMetadata.html
|
||
10 years ago
|
func IsValidObjectName(object string) bool {
|
||
10 years ago
|
if strings.TrimSpace(object) == "" {
|
||
10 years ago
|
return false
|
||
10 years ago
|
}
|
||
10 years ago
|
if len(object) > 1024 || len(object) == 0 {
|
||
|
return false
|
||
|
}
|
||
|
if !utf8.ValidString(object) {
|
||
|
return false
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
10 years ago
|
|
||
|
// IsValidPrefix - verify prefix name is correct, an empty prefix is valid
|
||
|
func IsValidPrefix(prefix string) bool {
|
||
|
if strings.TrimSpace(prefix) == "" {
|
||
|
return true
|
||
|
}
|
||
|
return IsValidObjectName(prefix)
|
||
|
}
|