mirror of
https://github.com/minio/minio.git
synced 2025-12-08 16:53:11 -05:00
Validate if replication config being added is self referential (#17142)
This commit is contained in:
@@ -26,6 +26,7 @@ import (
|
||||
"math"
|
||||
"math/rand"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"path"
|
||||
"reflect"
|
||||
"strings"
|
||||
@@ -138,6 +139,13 @@ func validateReplicationDestination(ctx context.Context, bucket string, rCfg *re
|
||||
// validate replication ARN against target endpoint
|
||||
c := globalBucketTargetSys.GetRemoteTargetClient(ctx, arnStr)
|
||||
if c != nil {
|
||||
if err := checkRemoteEndpoint(ctx, c.EndpointURL()); err != nil {
|
||||
switch err.(type) {
|
||||
case BucketRemoteIdenticalToSource:
|
||||
return true, errorCodes.ToAPIErrWithErr(ErrBucketRemoteIdenticalToSource, fmt.Errorf("remote target endpoint %s is self referential", c.EndpointURL().String()))
|
||||
default:
|
||||
}
|
||||
}
|
||||
if c.EndpointURL().String() == clnt.EndpointURL().String() {
|
||||
selfTarget, _ := isLocalHost(clnt.EndpointURL().Hostname(), clnt.EndpointURL().Port(), globalMinioPort)
|
||||
if !sameTarget {
|
||||
@@ -154,6 +162,45 @@ func validateReplicationDestination(ctx context.Context, bucket string, rCfg *re
|
||||
return sameTarget, toAPIError(ctx, nil)
|
||||
}
|
||||
|
||||
// performs a http request to remote endpoint to check if deployment id of remote endpoint is same as
|
||||
// local cluster deployment id. This is to prevent replication to self, especially in case of a loadbalancer
|
||||
// in front of MinIO.
|
||||
func checkRemoteEndpoint(ctx context.Context, epURL *url.URL) error {
|
||||
reqURL := &url.URL{
|
||||
Scheme: epURL.Scheme,
|
||||
Host: epURL.Host,
|
||||
Path: healthCheckReadinessPath,
|
||||
}
|
||||
|
||||
req, err := http.NewRequestWithContext(ctx, http.MethodGet, reqURL.String(), nil)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
client := &http.Client{
|
||||
Transport: NewHTTPTransport(),
|
||||
Timeout: 10 * time.Second,
|
||||
}
|
||||
|
||||
resp, err := client.Do(req)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err == nil {
|
||||
// Drain the connection.
|
||||
xhttp.DrainBody(resp.Body)
|
||||
}
|
||||
if resp != nil {
|
||||
amzid := resp.Header.Get(xhttp.AmzRequestHostID)
|
||||
if _, ok := globalNodeNamesHex[amzid]; ok {
|
||||
return BucketRemoteIdenticalToSource{
|
||||
Endpoint: epURL.String(),
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type mustReplicateOptions struct {
|
||||
meta map[string]string
|
||||
status replication.StatusType
|
||||
|
||||
Reference in New Issue
Block a user