2021-04-19 10:30:42 -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/>.
|
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
|
2023-06-19 17:53:08 -07:00
|
|
|
"github.com/minio/madmin-go/v3"
|
2022-12-23 02:35:54 +08:00
|
|
|
xhttp "github.com/minio/minio/internal/http"
|
2021-04-19 10:30:42 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// WarmBackendGetOpts is used to express byte ranges within an object. The zero
|
|
|
|
// value represents the entire byte range of an object.
|
|
|
|
type WarmBackendGetOpts struct {
|
|
|
|
startOffset int64
|
|
|
|
length int64
|
|
|
|
}
|
|
|
|
|
|
|
|
// WarmBackend provides interface to be implemented by remote tier backends
|
|
|
|
type WarmBackend interface {
|
2021-06-03 14:26:51 -07:00
|
|
|
Put(ctx context.Context, object string, r io.Reader, length int64) (remoteVersionID, error)
|
|
|
|
Get(ctx context.Context, object string, rv remoteVersionID, opts WarmBackendGetOpts) (io.ReadCloser, error)
|
|
|
|
Remove(ctx context.Context, object string, rv remoteVersionID) error
|
2021-04-19 10:30:42 -07:00
|
|
|
InUse(ctx context.Context) (bool, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
const probeObject = "probeobject"
|
|
|
|
|
|
|
|
// checkWarmBackend checks if tier config credentials have sufficient privileges
|
|
|
|
// to perform all operations defined in the WarmBackend interface.
|
|
|
|
func checkWarmBackend(ctx context.Context, w WarmBackend) error {
|
|
|
|
var empty bytes.Reader
|
2023-09-28 22:33:24 -07:00
|
|
|
remoteVersionID, err := w.Put(ctx, probeObject, &empty, 0)
|
2021-04-19 10:30:42 -07:00
|
|
|
if err != nil {
|
2023-03-06 17:56:10 +01:00
|
|
|
if _, ok := err.(BackendDown); ok {
|
2021-11-10 22:33:17 -08:00
|
|
|
return err
|
|
|
|
}
|
2021-04-19 10:30:42 -07:00
|
|
|
return tierPermErr{
|
|
|
|
Op: tierPut,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-26 00:14:56 -07:00
|
|
|
r, err := w.Get(ctx, probeObject, "", WarmBackendGetOpts{})
|
2022-12-23 02:35:54 +08:00
|
|
|
xhttp.DrainBody(r)
|
2021-04-19 10:30:42 -07:00
|
|
|
if err != nil {
|
2023-03-06 17:56:10 +01:00
|
|
|
if _, ok := err.(BackendDown); ok {
|
2021-11-10 22:33:17 -08:00
|
|
|
return err
|
|
|
|
}
|
2021-04-19 10:30:42 -07:00
|
|
|
switch {
|
|
|
|
case isErrBucketNotFound(err):
|
|
|
|
return errTierBucketNotFound
|
|
|
|
case isErrSignatureDoesNotMatch(err):
|
|
|
|
return errTierInvalidCredentials
|
|
|
|
default:
|
|
|
|
return tierPermErr{
|
|
|
|
Op: tierGet,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-28 22:33:24 -07:00
|
|
|
if err = w.Remove(ctx, probeObject, remoteVersionID); err != nil {
|
2023-03-06 17:56:10 +01:00
|
|
|
if _, ok := err.(BackendDown); ok {
|
2021-11-10 22:33:17 -08:00
|
|
|
return err
|
|
|
|
}
|
2021-04-19 10:30:42 -07:00
|
|
|
return tierPermErr{
|
|
|
|
Op: tierDelete,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
type tierOp uint8
|
|
|
|
|
|
|
|
const (
|
|
|
|
_ tierOp = iota
|
|
|
|
tierGet
|
|
|
|
tierPut
|
|
|
|
tierDelete
|
|
|
|
)
|
|
|
|
|
|
|
|
func (op tierOp) String() string {
|
|
|
|
switch op {
|
|
|
|
case tierGet:
|
|
|
|
return "GET"
|
|
|
|
case tierPut:
|
|
|
|
return "PUT"
|
|
|
|
case tierDelete:
|
|
|
|
return "DELETE"
|
|
|
|
}
|
|
|
|
return "UNKNOWN"
|
|
|
|
}
|
|
|
|
|
|
|
|
type tierPermErr struct {
|
|
|
|
Op tierOp
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (te tierPermErr) Error() string {
|
2023-09-14 15:28:20 -07:00
|
|
|
return fmt.Sprintf("failed to perform %s: %v", te.Op, te.Err)
|
2021-04-19 10:30:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func errIsTierPermError(err error) bool {
|
|
|
|
var tpErr tierPermErr
|
|
|
|
return errors.As(err, &tpErr)
|
|
|
|
}
|
|
|
|
|
2021-06-03 14:26:51 -07:00
|
|
|
// remoteVersionID represents the version id of an object in the remote tier.
|
|
|
|
// Its usage is remote tier cloud implementation specific.
|
|
|
|
type remoteVersionID string
|
|
|
|
|
2021-04-19 10:30:42 -07:00
|
|
|
// newWarmBackend instantiates the tier type specific WarmBackend, runs
|
|
|
|
// checkWarmBackend on it.
|
|
|
|
func newWarmBackend(ctx context.Context, tier madmin.TierConfig) (d WarmBackend, err error) {
|
|
|
|
switch tier.Type {
|
|
|
|
case madmin.S3:
|
2023-02-15 22:09:46 -08:00
|
|
|
d, err = newWarmBackendS3(*tier.S3, tier.Name)
|
2021-04-19 10:30:42 -07:00
|
|
|
case madmin.Azure:
|
2023-02-15 22:09:46 -08:00
|
|
|
d, err = newWarmBackendAzure(*tier.Azure, tier.Name)
|
2021-04-19 10:30:42 -07:00
|
|
|
case madmin.GCS:
|
2023-02-15 22:09:46 -08:00
|
|
|
d, err = newWarmBackendGCS(*tier.GCS, tier.Name)
|
2022-04-11 13:24:40 -07:00
|
|
|
case madmin.MinIO:
|
2023-02-15 22:09:46 -08:00
|
|
|
d, err = newWarmBackendMinIO(*tier.MinIO, tier.Name)
|
2021-04-19 10:30:42 -07:00
|
|
|
default:
|
|
|
|
return nil, errTierTypeUnsupported
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, errTierTypeUnsupported
|
|
|
|
}
|
|
|
|
|
|
|
|
err = checkWarmBackend(ctx, d)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return d, nil
|
|
|
|
}
|