mirror of
https://github.com/minio/mc.git
synced 2025-04-18 10:04:03 +03:00
136 lines
4.0 KiB
Go
136 lines
4.0 KiB
Go
// Copyright (c) 2015-2024 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 (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/minio/mc/pkg/probe"
|
|
"github.com/minio/minio-go/v7"
|
|
)
|
|
|
|
// preparePutURLs - prepares target and source clientURLs for copying.
|
|
func preparePutURLs(ctx context.Context, o prepareCopyURLsOpts) chan URLs {
|
|
copyURLsCh := make(chan URLs)
|
|
go func(o prepareCopyURLsOpts) {
|
|
defer close(copyURLsCh)
|
|
copyURLsContent, err := guessPutURLType(ctx, o)
|
|
if err != nil {
|
|
copyURLsCh <- URLs{Error: err}
|
|
return
|
|
}
|
|
|
|
switch copyURLsContent.copyType {
|
|
case copyURLsTypeA:
|
|
copyURLsCh <- prepareCopyURLsTypeA(ctx, *copyURLsContent, o)
|
|
case copyURLsTypeB:
|
|
copyURLsCh <- prepareCopyURLsTypeB(ctx, *copyURLsContent, o)
|
|
default:
|
|
copyURLsCh <- URLs{Error: errInvalidArgument().Trace(o.sourceURLs...)}
|
|
}
|
|
}(o)
|
|
|
|
finalCopyURLsCh := make(chan URLs)
|
|
go func() {
|
|
defer close(finalCopyURLsCh)
|
|
for cpURLs := range copyURLsCh {
|
|
if cpURLs.Error != nil {
|
|
finalCopyURLsCh <- cpURLs
|
|
continue
|
|
}
|
|
finalCopyURLsCh <- cpURLs
|
|
}
|
|
}()
|
|
|
|
return finalCopyURLsCh
|
|
}
|
|
|
|
// guessPutURLType guesses the type of clientURL. This approach all allows prepareURL
|
|
// functions to accurately report failure causes.
|
|
func guessPutURLType(ctx context.Context, o prepareCopyURLsOpts) (*copyURLsContent, *probe.Error) {
|
|
cc := new(copyURLsContent)
|
|
|
|
// Extract alias before fiddling with the clientURL.
|
|
cc.sourceURL = o.sourceURLs[0]
|
|
cc.sourceAlias, _, _ = mustExpandAlias(cc.sourceURL)
|
|
// Find alias and expanded clientURL.
|
|
cc.targetAlias, cc.targetURL, _ = mustExpandAlias(o.targetURL)
|
|
|
|
if len(o.sourceURLs) == 1 { // 1 Source, 1 Target
|
|
var err *probe.Error
|
|
var client Client
|
|
client, cc.sourceContent, err = url2Stat(ctx, url2StatOptions{urlStr: cc.sourceURL, versionID: o.versionID, fileAttr: false, encKeyDB: o.encKeyDB, timeRef: o.timeRef, isZip: o.isZip, ignoreBucketExistsCheck: false})
|
|
if err != nil {
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, err
|
|
}
|
|
_, ok := client.(*fsClient)
|
|
if !ok {
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, probe.NewError(fmt.Errorf("Source is not local filepath."))
|
|
}
|
|
// If recursion is ON, it is type C.
|
|
// If source is a folder, it is Type C.
|
|
if cc.sourceContent.Type.IsDir() {
|
|
cc.copyType = copyURLsTypeC
|
|
return cc, nil
|
|
}
|
|
client, err = newClient(o.targetURL)
|
|
if err != nil {
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, err
|
|
}
|
|
s3clnt, ok := client.(*S3Client)
|
|
if !ok {
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, probe.NewError(fmt.Errorf("Target is not s3."))
|
|
}
|
|
bucket, path := s3clnt.url2BucketAndObject()
|
|
if bucket == "" {
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, probe.NewError(fmt.Errorf("Bucket should not be empty."))
|
|
}
|
|
cc.targetContent = s3clnt.objectInfo2ClientContent(bucket, minio.ObjectInfo{
|
|
Key: bucket,
|
|
})
|
|
// If target is a folder, it is Type B.
|
|
var isDir bool
|
|
if path == "" {
|
|
isDir = true
|
|
} else {
|
|
isDir = strings.HasSuffix(path, string(cc.targetContent.URL.Separator))
|
|
}
|
|
|
|
if isDir {
|
|
cc.copyType = copyURLsTypeB
|
|
cc.sourceVersionID = cc.sourceContent.VersionID
|
|
return cc, nil
|
|
}
|
|
|
|
// else Type A.
|
|
cc.copyType = copyURLsTypeA
|
|
cc.sourceVersionID = cc.sourceContent.VersionID
|
|
return cc, nil
|
|
}
|
|
|
|
cc.copyType = copyURLsTypeInvalid
|
|
return cc, errInvalidArgument().Trace()
|
|
}
|