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-12-12 15:02:37 +01:00
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-01-11 08:43:18 -08:00
|
|
|
"errors"
|
2021-07-13 18:42:11 +01:00
|
|
|
"strings"
|
2023-09-18 22:13:03 -07:00
|
|
|
"time"
|
2019-12-12 15:02:37 +01:00
|
|
|
|
2020-01-08 03:31:43 -08:00
|
|
|
jsoniter "github.com/json-iterator/go"
|
2024-02-23 13:28:14 -08:00
|
|
|
"github.com/minio/minio/internal/cachevalue"
|
2021-06-01 14:59:40 -07:00
|
|
|
"github.com/minio/minio/internal/logger"
|
2019-12-12 15:02:37 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-04-28 01:16:57 -07:00
|
|
|
dataUsageRoot = SlashSeparator
|
|
|
|
dataUsageBucket = minioMetaBucket + SlashSeparator + bucketMetaPrefix
|
|
|
|
|
2022-01-03 10:22:58 -08:00
|
|
|
dataUsageObjName = ".usage.json"
|
|
|
|
dataUsageObjNamePath = bucketMetaPrefix + SlashSeparator + dataUsageObjName
|
|
|
|
dataUsageBloomName = ".bloomcycle.bin"
|
|
|
|
dataUsageBloomNamePath = bucketMetaPrefix + SlashSeparator + dataUsageBloomName
|
|
|
|
|
2022-04-07 16:10:40 +01:00
|
|
|
backgroundHealInfoPath = bucketMetaPrefix + SlashSeparator + ".background-heal.json"
|
|
|
|
|
2020-06-12 10:28:21 -07:00
|
|
|
dataUsageCacheName = ".usage-cache.bin"
|
2019-12-12 15:02:37 +01:00
|
|
|
)
|
|
|
|
|
2024-01-17 07:48:54 +08:00
|
|
|
// storeDataUsageInBackend will store all objects sent on the dui channel until closed.
|
2021-09-18 16:31:35 -04:00
|
|
|
func storeDataUsageInBackend(ctx context.Context, objAPI ObjectLayer, dui <-chan DataUsageInfo) {
|
2023-09-14 11:53:52 -07:00
|
|
|
attempts := 1
|
2020-12-07 13:47:48 -08:00
|
|
|
for dataUsageInfo := range dui {
|
2022-01-02 09:15:06 -08:00
|
|
|
json := jsoniter.ConfigCompatibleWithStandardLibrary
|
2020-03-19 09:47:47 -07:00
|
|
|
dataUsageJSON, err := json.Marshal(dataUsageInfo)
|
2020-03-19 00:19:29 +01:00
|
|
|
if err != nil {
|
|
|
|
logger.LogIf(ctx, err)
|
|
|
|
continue
|
|
|
|
}
|
2023-09-14 11:53:52 -07:00
|
|
|
if attempts > 10 {
|
|
|
|
saveConfig(ctx, objAPI, dataUsageObjNamePath+".bkp", dataUsageJSON) // Save a backup every 10th update.
|
|
|
|
attempts = 1
|
|
|
|
}
|
2022-01-03 10:22:58 -08:00
|
|
|
if err = saveConfig(ctx, objAPI, dataUsageObjNamePath, dataUsageJSON); err != nil {
|
2023-12-12 16:11:17 -08:00
|
|
|
logger.LogOnceIf(ctx, err, dataUsageObjNamePath)
|
2020-04-28 01:16:57 -07:00
|
|
|
}
|
2023-09-14 11:53:52 -07:00
|
|
|
attempts++
|
2019-12-12 15:02:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-23 13:28:14 -08:00
|
|
|
var prefixUsageCache = cachevalue.New[map[string]uint64]()
|
2023-09-18 22:13:03 -07:00
|
|
|
|
2021-07-13 18:42:11 +01:00
|
|
|
// loadPrefixUsageFromBackend returns prefix usages found in passed buckets
|
2022-08-26 12:52:29 -07:00
|
|
|
//
|
|
|
|
// e.g.: /testbucket/prefix => 355601334
|
2021-07-13 18:42:11 +01:00
|
|
|
func loadPrefixUsageFromBackend(ctx context.Context, objAPI ObjectLayer, bucket string) (map[string]uint64, error) {
|
|
|
|
z, ok := objAPI.(*erasureServerPools)
|
|
|
|
if !ok {
|
2021-07-17 01:17:35 -07:00
|
|
|
// Prefix usage is empty
|
|
|
|
return map[string]uint64{}, nil
|
2021-07-13 18:42:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
cache := dataUsageCache{}
|
|
|
|
|
2023-09-18 22:13:03 -07:00
|
|
|
prefixUsageCache.Once.Do(func() {
|
|
|
|
prefixUsageCache.TTL = 30 * time.Second
|
2021-07-13 18:42:11 +01:00
|
|
|
|
2023-09-18 22:13:03 -07:00
|
|
|
// No need to fail upon Update() error, fallback to old value.
|
2024-02-26 10:49:19 -08:00
|
|
|
prefixUsageCache.ReturnLastGood = true
|
|
|
|
prefixUsageCache.NoWait = true
|
2024-02-23 09:21:38 -08:00
|
|
|
prefixUsageCache.Update = func() (map[string]uint64, error) {
|
2023-09-18 22:13:03 -07:00
|
|
|
m := make(map[string]uint64)
|
|
|
|
for _, pool := range z.serverPools {
|
|
|
|
for _, er := range pool.sets {
|
|
|
|
// Load bucket usage prefixes
|
|
|
|
ctx, done := context.WithTimeout(context.Background(), 2*time.Second)
|
|
|
|
ok := cache.load(ctx, er, bucket+slashSeparator+dataUsageCacheName) == nil
|
|
|
|
done()
|
|
|
|
if ok {
|
|
|
|
root := cache.find(bucket)
|
|
|
|
if root == nil {
|
|
|
|
// We dont have usage information for this bucket in this
|
|
|
|
// set, go to the next set
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for id, usageInfo := range cache.flattenChildrens(*root) {
|
|
|
|
prefix := decodeDirObject(strings.TrimPrefix(id, bucket+slashSeparator))
|
|
|
|
// decodeDirObject to avoid any __XLDIR__ objects
|
|
|
|
m[prefix] += uint64(usageInfo.Size)
|
|
|
|
}
|
|
|
|
}
|
2021-07-13 18:42:11 +01:00
|
|
|
}
|
|
|
|
}
|
2023-09-18 22:13:03 -07:00
|
|
|
return m, nil
|
2021-07-13 18:42:11 +01:00
|
|
|
}
|
2023-09-18 22:13:03 -07:00
|
|
|
})
|
|
|
|
|
2024-02-23 09:21:38 -08:00
|
|
|
return prefixUsageCache.Get()
|
2021-07-13 18:42:11 +01:00
|
|
|
}
|
|
|
|
|
2021-09-18 16:31:35 -04:00
|
|
|
func loadDataUsageFromBackend(ctx context.Context, objAPI ObjectLayer) (DataUsageInfo, error) {
|
2022-01-03 10:22:58 -08:00
|
|
|
buf, err := readConfig(ctx, objAPI, dataUsageObjNamePath)
|
2019-12-12 15:02:37 +01:00
|
|
|
if err != nil {
|
2023-09-14 11:53:52 -07:00
|
|
|
buf, err = readConfig(ctx, objAPI, dataUsageObjNamePath+".bkp")
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, errConfigNotFound) {
|
|
|
|
return DataUsageInfo{}, nil
|
|
|
|
}
|
|
|
|
return DataUsageInfo{}, toObjectErr(err, minioMetaBucket, dataUsageObjNamePath)
|
2020-01-21 14:07:49 -08:00
|
|
|
}
|
2019-12-12 15:02:37 +01:00
|
|
|
}
|
|
|
|
|
2021-09-18 16:31:35 -04:00
|
|
|
var dataUsageInfo DataUsageInfo
|
2022-01-02 09:15:06 -08:00
|
|
|
json := jsoniter.ConfigCompatibleWithStandardLibrary
|
2022-01-03 10:22:58 -08:00
|
|
|
if err = json.Unmarshal(buf, &dataUsageInfo); err != nil {
|
2021-09-18 16:31:35 -04:00
|
|
|
return DataUsageInfo{}, err
|
2019-12-12 15:02:37 +01:00
|
|
|
}
|
2020-05-27 06:45:43 -07:00
|
|
|
// For forward compatibility reasons, we need to add this code.
|
|
|
|
if len(dataUsageInfo.BucketsUsage) == 0 {
|
2021-09-18 16:31:35 -04:00
|
|
|
dataUsageInfo.BucketsUsage = make(map[string]BucketUsageInfo, len(dataUsageInfo.BucketSizes))
|
2020-05-27 06:45:43 -07:00
|
|
|
for bucket, size := range dataUsageInfo.BucketSizes {
|
2021-09-18 16:31:35 -04:00
|
|
|
dataUsageInfo.BucketsUsage[bucket] = BucketUsageInfo{Size: size}
|
2020-05-27 06:45:43 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// For backward compatibility reasons, we need to add this code.
|
|
|
|
if len(dataUsageInfo.BucketSizes) == 0 {
|
|
|
|
dataUsageInfo.BucketSizes = make(map[string]uint64, len(dataUsageInfo.BucketsUsage))
|
|
|
|
for bucket, bui := range dataUsageInfo.BucketsUsage {
|
|
|
|
dataUsageInfo.BucketSizes[bucket] = bui.Size
|
|
|
|
}
|
|
|
|
}
|
2021-09-18 16:31:35 -04:00
|
|
|
// For forward compatibility reasons, we need to add this code.
|
|
|
|
for bucket, bui := range dataUsageInfo.BucketsUsage {
|
|
|
|
if bui.ReplicatedSizeV1 > 0 || bui.ReplicationFailedCountV1 > 0 ||
|
|
|
|
bui.ReplicationFailedSizeV1 > 0 || bui.ReplicationPendingCountV1 > 0 {
|
2021-11-19 14:46:14 -08:00
|
|
|
cfg, _ := getReplicationConfig(GlobalContext, bucket)
|
|
|
|
if cfg != nil && cfg.RoleArn != "" {
|
2024-01-30 00:54:20 +08:00
|
|
|
if dataUsageInfo.ReplicationInfo == nil {
|
|
|
|
dataUsageInfo.ReplicationInfo = make(map[string]BucketTargetUsageInfo)
|
|
|
|
}
|
2021-11-19 14:46:14 -08:00
|
|
|
dataUsageInfo.ReplicationInfo[cfg.RoleArn] = BucketTargetUsageInfo{
|
|
|
|
ReplicationFailedSize: bui.ReplicationFailedSizeV1,
|
|
|
|
ReplicationFailedCount: bui.ReplicationFailedCountV1,
|
|
|
|
ReplicatedSize: bui.ReplicatedSizeV1,
|
|
|
|
ReplicationPendingCount: bui.ReplicationPendingCountV1,
|
|
|
|
ReplicationPendingSize: bui.ReplicationPendingSizeV1,
|
|
|
|
}
|
2021-09-18 16:31:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-12 15:02:37 +01:00
|
|
|
return dataUsageInfo, nil
|
|
|
|
}
|