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.
187 lines
5.0 KiB
187 lines
5.0 KiB
7 years ago
|
/*
|
||
|
* Minio Cloud Storage, (C) 2018 Minio, Inc.
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
package condition
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
// Key - conditional key which is used to fetch values for any condition.
|
||
|
// Refer https://docs.aws.amazon.com/IAM/latest/UserGuide/list_s3.html
|
||
|
// for more information about available condition keys.
|
||
|
type Key string
|
||
|
|
||
|
const (
|
||
|
// S3XAmzCopySource - key representing x-amz-copy-source HTTP header applicable to PutObject API only.
|
||
|
S3XAmzCopySource Key = "s3:x-amz-copy-source"
|
||
|
|
||
|
// S3XAmzServerSideEncryption - key representing x-amz-server-side-encryption HTTP header applicable
|
||
|
// to PutObject API only.
|
||
|
S3XAmzServerSideEncryption = "s3:x-amz-server-side-encryption"
|
||
|
|
||
|
// S3XAmzServerSideEncryptionAwsKMSKeyID - key representing x-amz-server-side-encryption-aws-kms-key-id
|
||
|
// HTTP header applicable to PutObject API only.
|
||
|
S3XAmzServerSideEncryptionAwsKMSKeyID = "s3:x-amz-server-side-encryption-aws-kms-key-id"
|
||
|
|
||
|
// S3XAmzServerSideEncryptionCustomerAlgorithm - key representing
|
||
|
// x-amz-server-side-encryption-customer-algorithm HTTP header applicable to PutObject API only.
|
||
|
S3XAmzServerSideEncryptionCustomerAlgorithm = "s3:x-amz-server-side-encryption-customer-algorithm"
|
||
|
|
||
|
// S3XAmzMetadataDirective - key representing x-amz-metadata-directive HTTP header applicable to
|
||
|
// PutObject API only.
|
||
|
S3XAmzMetadataDirective = "s3:x-amz-metadata-directive"
|
||
|
|
||
|
// S3XAmzStorageClass - key representing x-amz-storage-class HTTP header applicable to PutObject API
|
||
|
// only.
|
||
|
S3XAmzStorageClass = "s3:x-amz-storage-class"
|
||
|
|
||
|
// S3LocationConstraint - key representing LocationConstraint XML tag of CreateBucket API only.
|
||
|
S3LocationConstraint = "s3:LocationConstraint"
|
||
|
|
||
|
// S3Prefix - key representing prefix query parameter of ListBucket API only.
|
||
|
S3Prefix = "s3:prefix"
|
||
|
|
||
|
// S3Delimiter - key representing delimiter query parameter of ListBucket API only.
|
||
|
S3Delimiter = "s3:delimiter"
|
||
|
|
||
|
// S3MaxKeys - key representing max-keys query parameter of ListBucket API only.
|
||
|
S3MaxKeys = "s3:max-keys"
|
||
|
|
||
|
// AWSReferer - key representing Referer header of any API.
|
||
|
AWSReferer = "aws:Referer"
|
||
|
|
||
|
// AWSSourceIP - key representing client's IP address (not intermittent proxies) of any API.
|
||
|
AWSSourceIP = "aws:SourceIp"
|
||
|
)
|
||
|
|
||
|
// IsValid - checks if key is valid or not.
|
||
|
func (key Key) IsValid() bool {
|
||
|
switch key {
|
||
|
case S3XAmzCopySource, S3XAmzServerSideEncryption, S3XAmzServerSideEncryptionAwsKMSKeyID:
|
||
|
fallthrough
|
||
|
case S3XAmzMetadataDirective, S3XAmzStorageClass, S3LocationConstraint, S3Prefix:
|
||
|
fallthrough
|
||
|
case S3Delimiter, S3MaxKeys, AWSReferer, AWSSourceIP:
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
// MarshalJSON - encodes Key to JSON data.
|
||
|
func (key Key) MarshalJSON() ([]byte, error) {
|
||
|
if !key.IsValid() {
|
||
|
return nil, fmt.Errorf("unknown key %v", key)
|
||
|
}
|
||
|
|
||
|
return json.Marshal(string(key))
|
||
|
}
|
||
|
|
||
|
// Name - returns key name which is stripped value of prefixes "aws:" and "s3:"
|
||
|
func (key Key) Name() string {
|
||
|
keyString := string(key)
|
||
|
|
||
|
if strings.HasPrefix(keyString, "aws:") {
|
||
|
return strings.TrimPrefix(keyString, "aws:")
|
||
|
}
|
||
|
|
||
|
return strings.TrimPrefix(keyString, "s3:")
|
||
|
}
|
||
|
|
||
|
// UnmarshalJSON - decodes JSON data to Key.
|
||
|
func (key *Key) UnmarshalJSON(data []byte) error {
|
||
|
var s string
|
||
|
if err := json.Unmarshal(data, &s); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
parsedKey, err := parseKey(s)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
*key = parsedKey
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func parseKey(s string) (Key, error) {
|
||
|
key := Key(s)
|
||
|
|
||
|
if key.IsValid() {
|
||
|
return key, nil
|
||
|
}
|
||
|
|
||
|
return key, fmt.Errorf("invalid condition key '%v'", s)
|
||
|
}
|
||
|
|
||
|
// KeySet - set representation of slice of keys.
|
||
|
type KeySet map[Key]struct{}
|
||
|
|
||
|
// Add - add a key to key set.
|
||
|
func (set KeySet) Add(key Key) {
|
||
|
set[key] = struct{}{}
|
||
|
}
|
||
|
|
||
|
// Difference - returns a key set contains difference of two keys.
|
||
|
// Example:
|
||
|
// keySet1 := ["one", "two", "three"]
|
||
|
// keySet2 := ["two", "four", "three"]
|
||
|
// keySet1.Difference(keySet2) == ["one"]
|
||
|
func (set KeySet) Difference(sset KeySet) KeySet {
|
||
|
nset := make(KeySet)
|
||
|
|
||
|
for k := range set {
|
||
|
if _, ok := sset[k]; !ok {
|
||
|
nset.Add(k)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nset
|
||
|
}
|
||
|
|
||
|
// IsEmpty - returns whether key set is empty or not.
|
||
|
func (set KeySet) IsEmpty() bool {
|
||
|
return len(set) == 0
|
||
|
}
|
||
|
|
||
|
func (set KeySet) String() string {
|
||
|
return fmt.Sprintf("%v", set.ToSlice())
|
||
|
}
|
||
|
|
||
|
// ToSlice - returns slice of keys.
|
||
|
func (set KeySet) ToSlice() []Key {
|
||
|
keys := []Key{}
|
||
|
|
||
|
for key := range set {
|
||
|
keys = append(keys, key)
|
||
|
}
|
||
|
|
||
|
return keys
|
||
|
}
|
||
|
|
||
|
// NewKeySet - returns new KeySet contains given keys.
|
||
|
func NewKeySet(keys ...Key) KeySet {
|
||
|
set := make(KeySet)
|
||
|
for _, key := range keys {
|
||
|
set.Add(key)
|
||
|
}
|
||
|
|
||
|
return set
|
||
|
}
|