400 lines
11 KiB
Go
400 lines
11 KiB
Go
|
/*
|
||
|
* MinIO Cloud Storage, (C) 2020 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 cmd
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"context"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"sync"
|
||
|
|
||
|
"github.com/minio/minio-go/v6/pkg/tags"
|
||
|
bucketsse "github.com/minio/minio/pkg/bucket/encryption"
|
||
|
"github.com/minio/minio/pkg/bucket/lifecycle"
|
||
|
objectlock "github.com/minio/minio/pkg/bucket/object/lock"
|
||
|
"github.com/minio/minio/pkg/bucket/policy"
|
||
|
"github.com/minio/minio/pkg/event"
|
||
|
"github.com/minio/minio/pkg/madmin"
|
||
|
"github.com/minio/minio/pkg/sync/errgroup"
|
||
|
)
|
||
|
|
||
|
// BucketMetadataSys captures all bucket metadata for a given cluster.
|
||
|
type BucketMetadataSys struct {
|
||
|
sync.RWMutex
|
||
|
metadataMap map[string]BucketMetadata
|
||
|
}
|
||
|
|
||
|
// Remove bucket metadata from memory.
|
||
|
func (sys *BucketMetadataSys) Remove(bucket string) {
|
||
|
if globalIsGateway {
|
||
|
return
|
||
|
}
|
||
|
sys.Lock()
|
||
|
delete(sys.metadataMap, bucket)
|
||
|
sys.Unlock()
|
||
|
}
|
||
|
|
||
|
// Set - sets a new metadata in-memory.
|
||
|
// Only a shallow copy is saved and fields with references
|
||
|
// cannot be modified without causing a race condition,
|
||
|
// so they should be replaced atomically and not appended to, etc.
|
||
|
// Data is not persisted to disk.
|
||
|
func (sys *BucketMetadataSys) Set(bucket string, meta BucketMetadata) {
|
||
|
if globalIsGateway {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if bucket != minioMetaBucket {
|
||
|
sys.Lock()
|
||
|
sys.metadataMap[bucket] = meta
|
||
|
sys.Unlock()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Update update bucket metadata for the specified config file.
|
||
|
// The configData data should not be modified after being sent here.
|
||
|
func (sys *BucketMetadataSys) Update(bucket string, configFile string, configData []byte) error {
|
||
|
objAPI := newObjectLayerWithoutSafeModeFn()
|
||
|
if objAPI == nil {
|
||
|
return errServerNotInitialized
|
||
|
}
|
||
|
|
||
|
if globalIsGateway {
|
||
|
// This code is needed only for gateway implementations.
|
||
|
switch configFile {
|
||
|
case bucketSSEConfig:
|
||
|
if globalGatewayName == "nas" {
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
meta.EncryptionConfigXML = configData
|
||
|
return meta.Save(GlobalContext, objAPI)
|
||
|
}
|
||
|
case bucketLifecycleConfig:
|
||
|
if globalGatewayName == "nas" {
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
meta.LifecycleConfigXML = configData
|
||
|
return meta.Save(GlobalContext, objAPI)
|
||
|
}
|
||
|
case bucketTaggingConfig:
|
||
|
if globalGatewayName == "nas" {
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
meta.TaggingConfigXML = configData
|
||
|
return meta.Save(GlobalContext, objAPI)
|
||
|
}
|
||
|
case bucketNotificationConfig:
|
||
|
if globalGatewayName == "nas" {
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
meta.NotificationConfigXML = configData
|
||
|
return meta.Save(GlobalContext, objAPI)
|
||
|
}
|
||
|
case bucketPolicyConfig:
|
||
|
if configData == nil {
|
||
|
return objAPI.DeleteBucketPolicy(GlobalContext, bucket)
|
||
|
}
|
||
|
config, err := policy.ParseConfig(bytes.NewReader(configData), bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
return objAPI.SetBucketPolicy(GlobalContext, bucket, config)
|
||
|
}
|
||
|
return NotImplemented{}
|
||
|
}
|
||
|
|
||
|
if bucket == minioMetaBucket {
|
||
|
return errInvalidArgument
|
||
|
}
|
||
|
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
switch configFile {
|
||
|
case bucketPolicyConfig:
|
||
|
meta.PolicyConfigJSON = configData
|
||
|
case bucketNotificationConfig:
|
||
|
meta.NotificationConfigXML = configData
|
||
|
case bucketLifecycleConfig:
|
||
|
meta.LifecycleConfigXML = configData
|
||
|
case bucketSSEConfig:
|
||
|
meta.EncryptionConfigXML = configData
|
||
|
case bucketTaggingConfig:
|
||
|
meta.TaggingConfigXML = configData
|
||
|
case objectLockConfig:
|
||
|
meta.ObjectLockConfigXML = configData
|
||
|
case bucketQuotaConfigFile:
|
||
|
meta.QuotaConfigJSON = configData
|
||
|
default:
|
||
|
return fmt.Errorf("Unknown bucket %s metadata update requested %s", bucket, configFile)
|
||
|
}
|
||
|
|
||
|
if err := meta.Save(GlobalContext, objAPI); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
sys.Set(bucket, meta)
|
||
|
globalNotificationSys.LoadBucketMetadata(GlobalContext, bucket)
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Get metadata for a bucket.
|
||
|
// If no metadata exists errConfigNotFound is returned and a new metadata is returned.
|
||
|
// Only a shallow copy is returned, so referenced data should not be modified,
|
||
|
// but can be replaced atomically.
|
||
|
func (sys *BucketMetadataSys) Get(bucket string) (BucketMetadata, error) {
|
||
|
if globalIsGateway || bucket == minioMetaBucket {
|
||
|
return newBucketMetadata(bucket), errConfigNotFound
|
||
|
}
|
||
|
|
||
|
sys.RLock()
|
||
|
defer sys.RUnlock()
|
||
|
|
||
|
meta, ok := sys.metadataMap[bucket]
|
||
|
if !ok {
|
||
|
return newBucketMetadata(bucket), errConfigNotFound
|
||
|
}
|
||
|
|
||
|
return meta, nil
|
||
|
}
|
||
|
|
||
|
// GetTaggingConfig returns configured tagging config
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetTaggingConfig(bucket string) (*tags.Tags, error) {
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, errConfigNotFound) {
|
||
|
return nil, BucketTaggingNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return nil, err
|
||
|
}
|
||
|
if meta.taggingConfig == nil {
|
||
|
return nil, BucketTaggingNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return meta.taggingConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetObjectLockConfig returns configured object lock config
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetObjectLockConfig(bucket string) (*objectlock.Config, error) {
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, errConfigNotFound) {
|
||
|
return nil, BucketObjectLockConfigNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return nil, err
|
||
|
}
|
||
|
if meta.objectLockConfig == nil {
|
||
|
return nil, BucketObjectLockConfigNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return meta.objectLockConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetLifecycleConfig returns configured lifecycle config
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetLifecycleConfig(bucket string) (*lifecycle.Lifecycle, error) {
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, errConfigNotFound) {
|
||
|
return nil, BucketLifecycleNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return nil, err
|
||
|
}
|
||
|
if meta.lifecycleConfig == nil {
|
||
|
return nil, BucketLifecycleNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return meta.lifecycleConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetNotificationConfig returns configured notification config
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetNotificationConfig(bucket string) (*event.Config, error) {
|
||
|
if globalIsGateway && globalGatewayName == "nas" {
|
||
|
// Only needed in case of NAS gateway.
|
||
|
objAPI := newObjectLayerWithoutSafeModeFn()
|
||
|
if objAPI == nil {
|
||
|
return nil, errServerNotInitialized
|
||
|
}
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return meta.notificationConfig, nil
|
||
|
}
|
||
|
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return meta.notificationConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetSSEConfig returns configured SSE config
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetSSEConfig(bucket string) (*bucketsse.BucketSSEConfig, error) {
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, errConfigNotFound) {
|
||
|
return nil, BucketSSEConfigNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return nil, err
|
||
|
}
|
||
|
if meta.sseConfig == nil {
|
||
|
return nil, BucketSSEConfigNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return meta.sseConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetPolicyConfig returns configured bucket policy
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetPolicyConfig(bucket string) (*policy.Policy, error) {
|
||
|
if globalIsGateway {
|
||
|
objAPI := newObjectLayerWithoutSafeModeFn()
|
||
|
if objAPI == nil {
|
||
|
return nil, errServerNotInitialized
|
||
|
}
|
||
|
return objAPI.GetBucketPolicy(GlobalContext, bucket)
|
||
|
}
|
||
|
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
if errors.Is(err, errConfigNotFound) {
|
||
|
return nil, BucketPolicyNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return nil, err
|
||
|
}
|
||
|
if meta.policyConfig == nil {
|
||
|
return nil, BucketPolicyNotFound{Bucket: bucket}
|
||
|
}
|
||
|
return meta.policyConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetQuotaConfig returns configured bucket quota
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetQuotaConfig(bucket string) (*madmin.BucketQuota, error) {
|
||
|
meta, err := sys.GetConfig(bucket)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return meta.quotaConfig, nil
|
||
|
}
|
||
|
|
||
|
// GetConfig returns the current bucket metadata
|
||
|
// The returned object may not be modified.
|
||
|
func (sys *BucketMetadataSys) GetConfig(bucket string) (BucketMetadata, error) {
|
||
|
objAPI := newObjectLayerWithoutSafeModeFn()
|
||
|
if objAPI == nil {
|
||
|
return newBucketMetadata(bucket), errServerNotInitialized
|
||
|
}
|
||
|
|
||
|
if globalIsGateway {
|
||
|
return newBucketMetadata(bucket), NotImplemented{}
|
||
|
}
|
||
|
|
||
|
if bucket == minioMetaBucket {
|
||
|
return newBucketMetadata(bucket), errInvalidArgument
|
||
|
}
|
||
|
|
||
|
sys.RLock()
|
||
|
meta, ok := sys.metadataMap[bucket]
|
||
|
sys.RUnlock()
|
||
|
if ok {
|
||
|
return meta, nil
|
||
|
}
|
||
|
meta, err := loadBucketMetadata(GlobalContext, objAPI, bucket)
|
||
|
if err != nil {
|
||
|
return meta, err
|
||
|
}
|
||
|
sys.Lock()
|
||
|
sys.metadataMap[bucket] = meta
|
||
|
sys.Unlock()
|
||
|
return meta, nil
|
||
|
}
|
||
|
|
||
|
// Init - initializes bucket metadata system for all buckets.
|
||
|
func (sys *BucketMetadataSys) Init(ctx context.Context, buckets []BucketInfo, objAPI ObjectLayer) error {
|
||
|
if objAPI == nil {
|
||
|
return errServerNotInitialized
|
||
|
}
|
||
|
|
||
|
// In gateway mode, we don't need to load the policies
|
||
|
// from the backend.
|
||
|
if globalIsGateway {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Load PolicySys once during boot.
|
||
|
return sys.load(ctx, buckets, objAPI)
|
||
|
}
|
||
|
|
||
|
// concurrently load bucket metadata to speed up loading bucket metadata.
|
||
|
func (sys *BucketMetadataSys) concurrentLoad(ctx context.Context, buckets []BucketInfo, objAPI ObjectLayer) error {
|
||
|
g := errgroup.WithNErrs(len(buckets))
|
||
|
for index := range buckets {
|
||
|
index := index
|
||
|
g.Go(func() error {
|
||
|
meta, err := loadBucketMetadata(ctx, objAPI, buckets[index].Name)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
sys.Lock()
|
||
|
sys.metadataMap[buckets[index].Name] = meta
|
||
|
sys.Unlock()
|
||
|
return nil
|
||
|
}, index)
|
||
|
}
|
||
|
for _, err := range g.Wait() {
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Loads bucket metadata for all buckets into BucketMetadataSys.
|
||
|
func (sys *BucketMetadataSys) load(ctx context.Context, buckets []BucketInfo, objAPI ObjectLayer) error {
|
||
|
count := 100 // load 100 bucket metadata at a time.
|
||
|
for {
|
||
|
if len(buckets) < count {
|
||
|
return sys.concurrentLoad(ctx, buckets, objAPI)
|
||
|
}
|
||
|
if err := sys.concurrentLoad(ctx, buckets[:count], objAPI); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
buckets = buckets[count:]
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// NewBucketMetadataSys - creates new policy system.
|
||
|
func NewBucketMetadataSys() *BucketMetadataSys {
|
||
|
return &BucketMetadataSys{
|
||
|
metadataMap: make(map[string]BucketMetadata),
|
||
|
}
|
||
|
}
|