1
0
mirror of https://github.com/minio/mc.git synced 2025-04-18 10:04:03 +03:00
mc/cmd/put-url.go
2024-02-21 10:10:34 -08:00

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()
}