mirror of
https://github.com/minio/minio.git
synced 2025-01-11 15:03:22 -05:00
s3: Make/Delete buckets to use error quorum per pool (#17467)
This commit is contained in:
parent
bf8a68879c
commit
d8dad5c9ea
@ -359,6 +359,9 @@ func createServerEndpoints(serverAddr string, args ...string) (
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, -1, err
|
return nil, -1, err
|
||||||
}
|
}
|
||||||
|
for i := range endpointList {
|
||||||
|
endpointList[i].SetPool(0)
|
||||||
|
}
|
||||||
endpointServerPools = append(endpointServerPools, PoolEndpoints{
|
endpointServerPools = append(endpointServerPools, PoolEndpoints{
|
||||||
Legacy: true,
|
Legacy: true,
|
||||||
SetCount: len(setArgs),
|
SetCount: len(setArgs),
|
||||||
|
@ -38,6 +38,7 @@ import (
|
|||||||
"github.com/minio/minio/internal/mountinfo"
|
"github.com/minio/minio/internal/mountinfo"
|
||||||
"github.com/minio/pkg/env"
|
"github.com/minio/pkg/env"
|
||||||
xnet "github.com/minio/pkg/net"
|
xnet "github.com/minio/pkg/net"
|
||||||
|
"golang.org/x/exp/slices"
|
||||||
)
|
)
|
||||||
|
|
||||||
// EndpointType - enum for endpoint type.
|
// EndpointType - enum for endpoint type.
|
||||||
@ -58,9 +59,17 @@ type ProxyEndpoint struct {
|
|||||||
Transport http.RoundTripper
|
Transport http.RoundTripper
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Node holds information about a node in this cluster
|
||||||
|
type Node struct {
|
||||||
|
*url.URL
|
||||||
|
Pools []int
|
||||||
|
IsLocal bool
|
||||||
|
}
|
||||||
|
|
||||||
// Endpoint - any type of endpoint.
|
// Endpoint - any type of endpoint.
|
||||||
type Endpoint struct {
|
type Endpoint struct {
|
||||||
*url.URL
|
*url.URL
|
||||||
|
Pool int
|
||||||
IsLocal bool
|
IsLocal bool
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,6 +106,11 @@ func (endpoint *Endpoint) UpdateIsLocal() (err error) {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetPool sets a specific pool number to this node
|
||||||
|
func (endpoint *Endpoint) SetPool(i int) {
|
||||||
|
endpoint.Pool = i
|
||||||
|
}
|
||||||
|
|
||||||
// NewEndpoint - returns new endpoint based on given arguments.
|
// NewEndpoint - returns new endpoint based on given arguments.
|
||||||
func NewEndpoint(arg string) (ep Endpoint, e error) {
|
func NewEndpoint(arg string) (ep Endpoint, e error) {
|
||||||
// isEmptyPath - check whether given path is not empty.
|
// isEmptyPath - check whether given path is not empty.
|
||||||
@ -209,6 +223,35 @@ type PoolEndpoints struct {
|
|||||||
// EndpointServerPools - list of list of endpoints
|
// EndpointServerPools - list of list of endpoints
|
||||||
type EndpointServerPools []PoolEndpoints
|
type EndpointServerPools []PoolEndpoints
|
||||||
|
|
||||||
|
// GetNodes returns a sorted list of nodes in this cluster
|
||||||
|
func (l EndpointServerPools) GetNodes() (nodes []Node) {
|
||||||
|
nodesMap := make(map[string]Node)
|
||||||
|
for _, pool := range l {
|
||||||
|
for _, ep := range pool.Endpoints {
|
||||||
|
node, ok := nodesMap[ep.Host]
|
||||||
|
if !ok {
|
||||||
|
node.IsLocal = ep.IsLocal
|
||||||
|
node.URL = &url.URL{
|
||||||
|
Scheme: ep.Scheme,
|
||||||
|
Host: ep.Host,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !slices.Contains(node.Pools, ep.Pool) {
|
||||||
|
node.Pools = append(node.Pools, ep.Pool)
|
||||||
|
}
|
||||||
|
nodesMap[ep.Host] = node
|
||||||
|
}
|
||||||
|
}
|
||||||
|
nodes = make([]Node, 0, len(nodesMap))
|
||||||
|
for _, v := range nodesMap {
|
||||||
|
nodes = append(nodes, v)
|
||||||
|
}
|
||||||
|
sort.Slice(nodes, func(i, j int) bool {
|
||||||
|
return nodes[i].Host < nodes[j].Host
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// GetPoolIdx return pool index
|
// GetPoolIdx return pool index
|
||||||
func (l EndpointServerPools) GetPoolIdx(pool string) int {
|
func (l EndpointServerPools) GetPoolIdx(pool string) int {
|
||||||
for id, ep := range globalEndpoints {
|
for id, ep := range globalEndpoints {
|
||||||
@ -768,6 +811,8 @@ func CreatePoolEndpoints(serverAddr string, poolArgs ...[][]string) ([]Endpoints
|
|||||||
return nil, setupType, config.ErrInvalidEndpoint(nil).Msg("use path style endpoint for single node setup")
|
return nil, setupType, config.ErrInvalidEndpoint(nil).Msg("use path style endpoint for single node setup")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
endpoint.SetPool(0)
|
||||||
|
|
||||||
var endpoints Endpoints
|
var endpoints Endpoints
|
||||||
endpoints = append(endpoints, endpoint)
|
endpoints = append(endpoints, endpoint)
|
||||||
setupType = ErasureSDSetupType
|
setupType = ErasureSDSetupType
|
||||||
@ -781,7 +826,7 @@ func CreatePoolEndpoints(serverAddr string, poolArgs ...[][]string) ([]Endpoints
|
|||||||
return poolEndpoints, setupType, nil
|
return poolEndpoints, setupType, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
for i, args := range poolArgs {
|
for poolIdx, args := range poolArgs {
|
||||||
var endpoints Endpoints
|
var endpoints Endpoints
|
||||||
for _, iargs := range args {
|
for _, iargs := range args {
|
||||||
// Convert args to endpoints
|
// Convert args to endpoints
|
||||||
@ -795,6 +840,10 @@ func CreatePoolEndpoints(serverAddr string, poolArgs ...[][]string) ([]Endpoints
|
|||||||
return nil, setupType, config.ErrInvalidErasureEndpoints(nil).Msg(err.Error())
|
return nil, setupType, config.ErrInvalidErasureEndpoints(nil).Msg(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for i := range eps {
|
||||||
|
eps[i].SetPool(poolIdx)
|
||||||
|
}
|
||||||
|
|
||||||
endpoints = append(endpoints, eps...)
|
endpoints = append(endpoints, eps...)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -802,7 +851,7 @@ func CreatePoolEndpoints(serverAddr string, poolArgs ...[][]string) ([]Endpoints
|
|||||||
return nil, setupType, config.ErrInvalidErasureEndpoints(nil).Msg("invalid number of endpoints")
|
return nil, setupType, config.ErrInvalidErasureEndpoints(nil).Msg("invalid number of endpoints")
|
||||||
}
|
}
|
||||||
|
|
||||||
poolEndpoints[i] = endpoints
|
poolEndpoints[poolIdx] = endpoints
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, endpoints := range poolEndpoints {
|
for _, endpoints := range poolEndpoints {
|
||||||
@ -960,7 +1009,7 @@ func CreateEndpoints(serverAddr string, args ...[]string) (Endpoints, SetupType,
|
|||||||
|
|
||||||
_, serverAddrPort := mustSplitHostPort(serverAddr)
|
_, serverAddrPort := mustSplitHostPort(serverAddr)
|
||||||
|
|
||||||
// For single arg, return FS setup.
|
// For single arg, return single drive setup.
|
||||||
if len(args) == 1 && len(args[0]) == 1 {
|
if len(args) == 1 && len(args[0]) == 1 {
|
||||||
var endpoint Endpoint
|
var endpoint Endpoint
|
||||||
endpoint, err = NewEndpoint(args[0][0])
|
endpoint, err = NewEndpoint(args[0][0])
|
||||||
|
@ -224,7 +224,10 @@ var (
|
|||||||
// registered listeners
|
// registered listeners
|
||||||
globalConsoleSys *HTTPConsoleLoggerSys
|
globalConsoleSys *HTTPConsoleLoggerSys
|
||||||
|
|
||||||
|
// All unique drives for this deployment
|
||||||
globalEndpoints EndpointServerPools
|
globalEndpoints EndpointServerPools
|
||||||
|
// All unique nodes for this deployment
|
||||||
|
globalNodes []Node
|
||||||
|
|
||||||
// The name of this local node, fetched from arguments
|
// The name of this local node, fetched from arguments
|
||||||
globalLocalNodeName string
|
globalLocalNodeName string
|
||||||
@ -237,8 +240,6 @@ var (
|
|||||||
// The global callhome config
|
// The global callhome config
|
||||||
globalCallhomeConfig callhome.Config
|
globalCallhomeConfig callhome.Config
|
||||||
|
|
||||||
globalRemoteEndpoints map[string]Endpoint
|
|
||||||
|
|
||||||
// Global server's network statistics
|
// Global server's network statistics
|
||||||
globalConnStats = newConnStats()
|
globalConnStats = newConnStats()
|
||||||
|
|
||||||
|
@ -21,37 +21,81 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"encoding/gob"
|
"encoding/gob"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
xhttp "github.com/minio/minio/internal/http"
|
xhttp "github.com/minio/minio/internal/http"
|
||||||
"github.com/minio/minio/internal/logger"
|
|
||||||
"github.com/minio/minio/internal/rest"
|
"github.com/minio/minio/internal/rest"
|
||||||
xnet "github.com/minio/pkg/net"
|
|
||||||
"github.com/minio/pkg/sync/errgroup"
|
"github.com/minio/pkg/sync/errgroup"
|
||||||
|
"golang.org/x/exp/slices"
|
||||||
)
|
)
|
||||||
|
|
||||||
var errPeerOffline = errors.New("peer is offline")
|
var errPeerOffline = errors.New("peer is offline")
|
||||||
|
|
||||||
|
type peerS3Client interface {
|
||||||
|
ListBuckets(ctx context.Context, opts BucketOptions) ([]BucketInfo, error)
|
||||||
|
GetBucketInfo(ctx context.Context, bucket string, opts BucketOptions) (BucketInfo, error)
|
||||||
|
MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error
|
||||||
|
DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error
|
||||||
|
|
||||||
|
GetHost() string
|
||||||
|
SetPools([]int)
|
||||||
|
GetPools() []int
|
||||||
|
}
|
||||||
|
|
||||||
|
type localPeerS3Client struct {
|
||||||
|
host string
|
||||||
|
pools []int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *localPeerS3Client) GetHost() string {
|
||||||
|
return l.host
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *localPeerS3Client) SetPools(p []int) {
|
||||||
|
l.pools = make([]int, len(p))
|
||||||
|
copy(l.pools, p)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l localPeerS3Client) GetPools() []int {
|
||||||
|
return l.pools
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l localPeerS3Client) ListBuckets(ctx context.Context, opts BucketOptions) ([]BucketInfo, error) {
|
||||||
|
return listBucketsLocal(ctx, opts)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l localPeerS3Client) GetBucketInfo(ctx context.Context, bucket string, opts BucketOptions) (BucketInfo, error) {
|
||||||
|
return getBucketInfoLocal(ctx, bucket, opts)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l localPeerS3Client) MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error {
|
||||||
|
return makeBucketLocal(ctx, bucket, opts)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l localPeerS3Client) DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error {
|
||||||
|
return deleteBucketLocal(ctx, bucket, opts)
|
||||||
|
}
|
||||||
|
|
||||||
// client to talk to peer Nodes.
|
// client to talk to peer Nodes.
|
||||||
type peerS3Client struct {
|
type remotePeerS3Client struct {
|
||||||
host *xnet.Host
|
host string
|
||||||
|
pools []int
|
||||||
restClient *rest.Client
|
restClient *rest.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrapper to restClient.Call to handle network errors, in case of network error the connection is marked disconnected
|
// Wrapper to restClient.Call to handle network errors, in case of network error the connection is marked disconnected
|
||||||
// permanently. The only way to restore the connection is at the xl-sets layer by xlsets.monitorAndConnectEndpoints()
|
// permanently. The only way to restore the connection is at the xl-sets layer by xlsets.monitorAndConnectEndpoints()
|
||||||
// after verifying format.json
|
// after verifying format.json
|
||||||
func (client *peerS3Client) call(method string, values url.Values, body io.Reader, length int64) (respBody io.ReadCloser, err error) {
|
func (client *remotePeerS3Client) call(method string, values url.Values, body io.Reader, length int64) (respBody io.ReadCloser, err error) {
|
||||||
return client.callWithContext(GlobalContext, method, values, body, length)
|
return client.callWithContext(GlobalContext, method, values, body, length)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrapper to restClient.Call to handle network errors, in case of network error the connection is marked disconnected
|
// Wrapper to restClient.Call to handle network errors, in case of network error the connection is marked disconnected
|
||||||
// permanently. The only way to restore the connection is at the xl-sets layer by xlsets.monitorAndConnectEndpoints()
|
// permanently. The only way to restore the connection is at the xl-sets layer by xlsets.monitorAndConnectEndpoints()
|
||||||
// after verifying format.json
|
// after verifying format.json
|
||||||
func (client *peerS3Client) callWithContext(ctx context.Context, method string, values url.Values, body io.Reader, length int64) (respBody io.ReadCloser, err error) {
|
func (client *remotePeerS3Client) callWithContext(ctx context.Context, method string, values url.Values, body io.Reader, length int64) (respBody io.ReadCloser, err error) {
|
||||||
if values == nil {
|
if values == nil {
|
||||||
values = make(url.Values)
|
values = make(url.Values)
|
||||||
}
|
}
|
||||||
@ -67,16 +111,15 @@ func (client *peerS3Client) callWithContext(ctx context.Context, method string,
|
|||||||
|
|
||||||
// S3PeerSys - S3 peer call system.
|
// S3PeerSys - S3 peer call system.
|
||||||
type S3PeerSys struct {
|
type S3PeerSys struct {
|
||||||
peerClients []*peerS3Client // Excludes self
|
peerClients []peerS3Client // Excludes self
|
||||||
allPeerClients []*peerS3Client // Includes nil client for self
|
poolsCount int
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewS3PeerSys - creates new S3 peer calls.
|
// NewS3PeerSys - creates new S3 peer calls.
|
||||||
func NewS3PeerSys(endpoints EndpointServerPools) *S3PeerSys {
|
func NewS3PeerSys(endpoints EndpointServerPools) *S3PeerSys {
|
||||||
remote, all := newPeerS3Clients(endpoints)
|
|
||||||
return &S3PeerSys{
|
return &S3PeerSys{
|
||||||
peerClients: remote,
|
peerClients: newPeerS3Clients(endpoints.GetNodes()),
|
||||||
allPeerClients: all,
|
poolsCount: len(endpoints),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,14 +127,8 @@ func NewS3PeerSys(endpoints EndpointServerPools) *S3PeerSys {
|
|||||||
func (sys *S3PeerSys) ListBuckets(ctx context.Context, opts BucketOptions) (result []BucketInfo, err error) {
|
func (sys *S3PeerSys) ListBuckets(ctx context.Context, opts BucketOptions) (result []BucketInfo, err error) {
|
||||||
g := errgroup.WithNErrs(len(sys.peerClients))
|
g := errgroup.WithNErrs(len(sys.peerClients))
|
||||||
|
|
||||||
localBuckets, err := listBucketsLocal(ctx, opts)
|
nodeBuckets := make([][]BucketInfo, len(sys.peerClients))
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
nodeBuckets := make([][]BucketInfo, len(sys.peerClients)+1)
|
|
||||||
errs := []error{nil}
|
errs := []error{nil}
|
||||||
nodeBuckets[0] = localBuckets
|
|
||||||
|
|
||||||
for idx, client := range sys.peerClients {
|
for idx, client := range sys.peerClients {
|
||||||
idx := idx
|
idx := idx
|
||||||
@ -104,14 +141,14 @@ func (sys *S3PeerSys) ListBuckets(ctx context.Context, opts BucketOptions) (resu
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
nodeBuckets[idx+1] = localBuckets
|
nodeBuckets[idx] = localBuckets
|
||||||
return nil
|
return nil
|
||||||
}, idx)
|
}, idx)
|
||||||
}
|
}
|
||||||
|
|
||||||
errs = append(errs, g.Wait()...)
|
errs = append(errs, g.Wait()...)
|
||||||
|
|
||||||
quorum := (len(sys.allPeerClients) / 2)
|
quorum := len(sys.peerClients)/2 + 1
|
||||||
if err = reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, quorum); err != nil {
|
if err = reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, quorum); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -156,11 +193,7 @@ func (sys *S3PeerSys) GetBucketInfo(ctx context.Context, bucket string, opts Buc
|
|||||||
|
|
||||||
errs := g.Wait()
|
errs := g.Wait()
|
||||||
|
|
||||||
bucketInfo, err := getBucketInfoLocal(ctx, bucket, opts)
|
quorum := len(sys.peerClients)/2 + 1
|
||||||
errs = append(errs, err)
|
|
||||||
bucketInfos = append(bucketInfos, bucketInfo)
|
|
||||||
|
|
||||||
quorum := (len(sys.allPeerClients) / 2)
|
|
||||||
if err = reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, quorum); err != nil {
|
if err = reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, quorum); err != nil {
|
||||||
return BucketInfo{}, toObjectErr(err, bucket)
|
return BucketInfo{}, toObjectErr(err, bucket)
|
||||||
}
|
}
|
||||||
@ -174,7 +207,7 @@ func (sys *S3PeerSys) GetBucketInfo(ctx context.Context, bucket string, opts Buc
|
|||||||
return BucketInfo{}, toObjectErr(errVolumeNotFound, bucket)
|
return BucketInfo{}, toObjectErr(errVolumeNotFound, bucket)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *peerS3Client) ListBuckets(ctx context.Context, opts BucketOptions) ([]BucketInfo, error) {
|
func (client *remotePeerS3Client) ListBuckets(ctx context.Context, opts BucketOptions) ([]BucketInfo, error) {
|
||||||
v := url.Values{}
|
v := url.Values{}
|
||||||
v.Set(peerS3BucketDeleted, strconv.FormatBool(opts.Deleted))
|
v.Set(peerS3BucketDeleted, strconv.FormatBool(opts.Deleted))
|
||||||
|
|
||||||
@ -190,7 +223,7 @@ func (client *peerS3Client) ListBuckets(ctx context.Context, opts BucketOptions)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// GetBucketInfo returns bucket stat info from a peer
|
// GetBucketInfo returns bucket stat info from a peer
|
||||||
func (client *peerS3Client) GetBucketInfo(ctx context.Context, bucket string, opts BucketOptions) (BucketInfo, error) {
|
func (client *remotePeerS3Client) GetBucketInfo(ctx context.Context, bucket string, opts BucketOptions) (BucketInfo, error) {
|
||||||
v := url.Values{}
|
v := url.Values{}
|
||||||
v.Set(peerS3Bucket, bucket)
|
v.Set(peerS3Bucket, bucket)
|
||||||
v.Set(peerS3BucketDeleted, strconv.FormatBool(opts.Deleted))
|
v.Set(peerS3BucketDeleted, strconv.FormatBool(opts.Deleted))
|
||||||
@ -209,7 +242,6 @@ func (client *peerS3Client) GetBucketInfo(ctx context.Context, bucket string, op
|
|||||||
// MakeBucket creates bucket across all peers
|
// MakeBucket creates bucket across all peers
|
||||||
func (sys *S3PeerSys) MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error {
|
func (sys *S3PeerSys) MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error {
|
||||||
g := errgroup.WithNErrs(len(sys.peerClients))
|
g := errgroup.WithNErrs(len(sys.peerClients))
|
||||||
|
|
||||||
for idx, client := range sys.peerClients {
|
for idx, client := range sys.peerClients {
|
||||||
client := client
|
client := client
|
||||||
g.Go(func() error {
|
g.Go(func() error {
|
||||||
@ -219,30 +251,24 @@ func (sys *S3PeerSys) MakeBucket(ctx context.Context, bucket string, opts MakeBu
|
|||||||
return client.MakeBucket(ctx, bucket, opts)
|
return client.MakeBucket(ctx, bucket, opts)
|
||||||
}, idx)
|
}, idx)
|
||||||
}
|
}
|
||||||
|
|
||||||
errs := g.Wait()
|
errs := g.Wait()
|
||||||
errs = append(errs, makeBucketLocal(ctx, bucket, opts))
|
|
||||||
|
|
||||||
quorum := (len(sys.allPeerClients) / 2) + 1
|
for poolIdx := 0; poolIdx < sys.poolsCount; poolIdx++ {
|
||||||
err := reduceWriteQuorumErrs(ctx, errs, bucketOpIgnoredErrs, quorum)
|
perPoolErrs := make([]error, 0, len(sys.peerClients))
|
||||||
|
for i, client := range sys.peerClients {
|
||||||
// Perform MRF on missing buckets for temporary errors.
|
if slices.Contains(client.GetPools(), poolIdx) {
|
||||||
for _, err := range errs {
|
perPoolErrs = append(perPoolErrs, errs[i])
|
||||||
if err == nil {
|
}
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
if errors.Is(err, errPeerOffline) || errors.Is(err, errDiskNotFound) ||
|
if poolErr := reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, len(perPoolErrs)/2+1); poolErr != nil {
|
||||||
isNetworkError(err) {
|
return toObjectErr(poolErr, bucket)
|
||||||
globalMRFState.addPartialOp(partialOperation{
|
|
||||||
bucket: bucket,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return toObjectErr(err, bucket)
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// MakeBucket creates a bucket on a peer
|
// MakeBucket creates a bucket on a peer
|
||||||
func (client *peerS3Client) MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error {
|
func (client *remotePeerS3Client) MakeBucket(ctx context.Context, bucket string, opts MakeBucketOptions) error {
|
||||||
v := url.Values{}
|
v := url.Values{}
|
||||||
v.Set(peerS3Bucket, bucket)
|
v.Set(peerS3Bucket, bucket)
|
||||||
v.Set(peerS3BucketForceCreate, strconv.FormatBool(opts.ForceCreate))
|
v.Set(peerS3BucketForceCreate, strconv.FormatBool(opts.ForceCreate))
|
||||||
@ -259,7 +285,6 @@ func (client *peerS3Client) MakeBucket(ctx context.Context, bucket string, opts
|
|||||||
// DeleteBucket deletes bucket across all peers
|
// DeleteBucket deletes bucket across all peers
|
||||||
func (sys *S3PeerSys) DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error {
|
func (sys *S3PeerSys) DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error {
|
||||||
g := errgroup.WithNErrs(len(sys.peerClients))
|
g := errgroup.WithNErrs(len(sys.peerClients))
|
||||||
|
|
||||||
for idx, client := range sys.peerClients {
|
for idx, client := range sys.peerClients {
|
||||||
client := client
|
client := client
|
||||||
g.Go(func() error {
|
g.Go(func() error {
|
||||||
@ -269,32 +294,26 @@ func (sys *S3PeerSys) DeleteBucket(ctx context.Context, bucket string, opts Dele
|
|||||||
return client.DeleteBucket(ctx, bucket, opts)
|
return client.DeleteBucket(ctx, bucket, opts)
|
||||||
}, idx)
|
}, idx)
|
||||||
}
|
}
|
||||||
|
|
||||||
errs := g.Wait()
|
errs := g.Wait()
|
||||||
errs = append(errs, deleteBucketLocal(ctx, bucket, opts))
|
|
||||||
|
|
||||||
var errReturn error
|
for poolIdx := 0; poolIdx < sys.poolsCount; poolIdx++ {
|
||||||
for _, err := range errs {
|
perPoolErrs := make([]error, 0, len(sys.peerClients))
|
||||||
if errReturn == nil && err != nil {
|
for i, client := range sys.peerClients {
|
||||||
// always return first error
|
if slices.Contains(client.GetPools(), poolIdx) {
|
||||||
errReturn = toObjectErr(err, bucket)
|
perPoolErrs = append(perPoolErrs, errs[i])
|
||||||
break
|
}
|
||||||
}
|
}
|
||||||
}
|
if poolErr := reduceReadQuorumErrs(ctx, errs, bucketOpIgnoredErrs, len(perPoolErrs)/2+1); poolErr != nil {
|
||||||
|
|
||||||
for _, err := range errs {
|
|
||||||
if err == nil && errReturn != nil {
|
|
||||||
// re-create successful deletes, since we are return an error.
|
// re-create successful deletes, since we are return an error.
|
||||||
sys.MakeBucket(ctx, bucket, MakeBucketOptions{})
|
sys.MakeBucket(ctx, bucket, MakeBucketOptions{})
|
||||||
break
|
return toObjectErr(poolErr, bucket)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
return errReturn
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteBucket deletes bucket on a peer
|
// DeleteBucket deletes bucket on a peer
|
||||||
func (client *peerS3Client) DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error {
|
func (client *remotePeerS3Client) DeleteBucket(ctx context.Context, bucket string, opts DeleteBucketOptions) error {
|
||||||
v := url.Values{}
|
v := url.Values{}
|
||||||
v.Set(peerS3Bucket, bucket)
|
v.Set(peerS3Bucket, bucket)
|
||||||
v.Set(peerS3BucketForceDelete, strconv.FormatBool(opts.Force))
|
v.Set(peerS3BucketForceDelete, strconv.FormatBool(opts.Force))
|
||||||
@ -308,33 +327,35 @@ func (client *peerS3Client) DeleteBucket(ctx context.Context, bucket string, opt
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (client remotePeerS3Client) GetHost() string {
|
||||||
|
return client.host
|
||||||
|
}
|
||||||
|
|
||||||
|
func (client remotePeerS3Client) GetPools() []int {
|
||||||
|
return client.pools
|
||||||
|
}
|
||||||
|
|
||||||
|
func (client *remotePeerS3Client) SetPools(p []int) {
|
||||||
|
client.pools = make([]int, len(p))
|
||||||
|
copy(client.pools, p)
|
||||||
|
}
|
||||||
|
|
||||||
// newPeerS3Clients creates new peer clients.
|
// newPeerS3Clients creates new peer clients.
|
||||||
// The two slices will point to the same clients,
|
func newPeerS3Clients(nodes []Node) (peers []peerS3Client) {
|
||||||
// but 'all' will contain nil entry for local client.
|
peers = make([]peerS3Client, len(nodes))
|
||||||
// The 'all' slice will be in the same order across the cluster.
|
for i, node := range nodes {
|
||||||
func newPeerS3Clients(endpoints EndpointServerPools) (remote, all []*peerS3Client) {
|
if node.IsLocal {
|
||||||
if !globalIsDistErasure {
|
peers[i] = &localPeerS3Client{host: node.Host}
|
||||||
// Only useful in distributed setups
|
} else {
|
||||||
return nil, nil
|
peers[i] = newPeerS3Client(node.Host)
|
||||||
}
|
|
||||||
hosts := endpoints.hostsSorted()
|
|
||||||
remote = make([]*peerS3Client, 0, len(hosts))
|
|
||||||
all = make([]*peerS3Client, len(hosts))
|
|
||||||
for i, host := range hosts {
|
|
||||||
if host == nil {
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
all[i] = newPeerS3Client(host)
|
peers[i].SetPools(node.Pools)
|
||||||
remote = append(remote, all[i])
|
|
||||||
}
|
}
|
||||||
if len(all) != len(remote)+1 {
|
return
|
||||||
logger.LogIf(context.Background(), fmt.Errorf("WARNING: Expected number of all hosts (%v) to be remote +1 (%v)", len(all), len(remote)))
|
|
||||||
}
|
|
||||||
return remote, all
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns a peer S3 client.
|
// Returns a peer S3 client.
|
||||||
func newPeerS3Client(peer *xnet.Host) *peerS3Client {
|
func newPeerS3Client(peer string) peerS3Client {
|
||||||
scheme := "http"
|
scheme := "http"
|
||||||
if globalIsTLS {
|
if globalIsTLS {
|
||||||
scheme = "https"
|
scheme = "https"
|
||||||
@ -342,7 +363,7 @@ func newPeerS3Client(peer *xnet.Host) *peerS3Client {
|
|||||||
|
|
||||||
serverURL := &url.URL{
|
serverURL := &url.URL{
|
||||||
Scheme: scheme,
|
Scheme: scheme,
|
||||||
Host: peer.String(),
|
Host: peer,
|
||||||
Path: peerS3Path,
|
Path: peerS3Path,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -360,5 +381,5 @@ func newPeerS3Client(peer *xnet.Host) *peerS3Client {
|
|||||||
return !isNetworkError(err)
|
return !isNetworkError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return &peerS3Client{host: peer, restClient: restClient}
|
return &remotePeerS3Client{host: peer, restClient: restClient}
|
||||||
}
|
}
|
||||||
|
@ -246,23 +246,19 @@ func serverHandleCmdArgs(ctx *cli.Context) {
|
|||||||
|
|
||||||
globalEndpoints, setupType, err = createServerEndpoints(globalMinioAddr, serverCmdArgs(ctx)...)
|
globalEndpoints, setupType, err = createServerEndpoints(globalMinioAddr, serverCmdArgs(ctx)...)
|
||||||
logger.FatalIf(err, "Invalid command line arguments")
|
logger.FatalIf(err, "Invalid command line arguments")
|
||||||
|
globalNodes = globalEndpoints.GetNodes()
|
||||||
|
|
||||||
globalLocalNodeName = GetLocalPeer(globalEndpoints, globalMinioHost, globalMinioPort)
|
globalLocalNodeName = GetLocalPeer(globalEndpoints, globalMinioHost, globalMinioPort)
|
||||||
nodeNameSum := sha256.Sum256([]byte(globalLocalNodeName))
|
nodeNameSum := sha256.Sum256([]byte(globalLocalNodeName))
|
||||||
globalLocalNodeNameHex = hex.EncodeToString(nodeNameSum[:])
|
globalLocalNodeNameHex = hex.EncodeToString(nodeNameSum[:])
|
||||||
globalNodeNamesHex = make(map[string]struct{})
|
globalNodeNamesHex = make(map[string]struct{})
|
||||||
globalRemoteEndpoints = make(map[string]Endpoint)
|
for _, n := range globalNodes {
|
||||||
for _, z := range globalEndpoints {
|
nodeName := n.Host
|
||||||
for _, ep := range z.Endpoints {
|
if n.IsLocal {
|
||||||
if ep.IsLocal {
|
nodeName = globalLocalNodeName
|
||||||
globalRemoteEndpoints[globalLocalNodeName] = ep
|
|
||||||
} else {
|
|
||||||
globalRemoteEndpoints[ep.Host] = ep
|
|
||||||
}
|
|
||||||
nodeNameSum := sha256.Sum256([]byte(ep.Host))
|
|
||||||
globalNodeNamesHex[hex.EncodeToString(nodeNameSum[:])] = struct{}{}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
nodeNameSum := sha256.Sum256([]byte(nodeName))
|
||||||
|
globalNodeNamesHex[hex.EncodeToString(nodeNameSum[:])] = struct{}{}
|
||||||
}
|
}
|
||||||
|
|
||||||
// allow transport to be HTTP/1.1 for proxying.
|
// allow transport to be HTTP/1.1 for proxying.
|
||||||
|
Loading…
Reference in New Issue
Block a user