mirror of
https://github.com/minio/mc.git
synced 2025-11-10 13:42:32 +03:00
220 lines
7.8 KiB
Go
220 lines
7.8 KiB
Go
// 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 (
|
|
"context"
|
|
"fmt"
|
|
"runtime"
|
|
"time"
|
|
|
|
"github.com/minio/cli"
|
|
"github.com/minio/mc/pkg/probe"
|
|
"github.com/minio/pkg/console"
|
|
)
|
|
|
|
func checkCopySyntax(ctx context.Context, cliCtx *cli.Context, encKeyDB map[string][]prefixSSEPair, isMvCmd bool) {
|
|
if len(cliCtx.Args()) < 2 {
|
|
if isMvCmd {
|
|
showCommandHelpAndExit(cliCtx, 1) // last argument is exit code.
|
|
}
|
|
showCommandHelpAndExit(cliCtx, 1) // last argument is exit code.
|
|
}
|
|
|
|
// extract URLs.
|
|
URLs := cliCtx.Args()
|
|
if len(URLs) < 2 {
|
|
fatalIf(errDummy().Trace(cliCtx.Args()...), "Unable to parse source and target arguments.")
|
|
}
|
|
|
|
srcURLs := URLs[:len(URLs)-1]
|
|
tgtURL := URLs[len(URLs)-1]
|
|
isRecursive := cliCtx.Bool("recursive")
|
|
isZip := cliCtx.Bool("zip")
|
|
timeRef := parseRewindFlag(cliCtx.String("rewind"))
|
|
versionID := cliCtx.String("version-id")
|
|
|
|
if versionID != "" && len(srcURLs) > 1 {
|
|
fatalIf(errDummy().Trace(cliCtx.Args()...), "Unable to pass --version flag with multiple copy sources arguments.")
|
|
}
|
|
|
|
if isZip && cliCtx.String("rewind") != "" {
|
|
fatalIf(errDummy().Trace(cliCtx.Args()...), "--zip and --rewind cannot be used together")
|
|
}
|
|
|
|
// Verify if source(s) exists.
|
|
for _, srcURL := range srcURLs {
|
|
var err *probe.Error
|
|
if !isRecursive {
|
|
_, _, err = url2Stat(ctx, srcURL, versionID, false, encKeyDB, timeRef, isZip)
|
|
} else {
|
|
_, _, err = firstURL2Stat(ctx, srcURL, timeRef, isZip)
|
|
}
|
|
if err != nil {
|
|
msg := "Unable to validate source `" + srcURL + "`"
|
|
if versionID != "" {
|
|
msg += " (" + versionID + ")"
|
|
}
|
|
msg += "."
|
|
console.Fatalln(msg)
|
|
}
|
|
}
|
|
|
|
// Check if bucket name is passed for URL type arguments.
|
|
url := newClientURL(tgtURL)
|
|
if url.Host != "" {
|
|
if url.Path == string(url.Separator) {
|
|
fatalIf(errInvalidArgument().Trace(), fmt.Sprintf("Target `%s` does not contain bucket name.", tgtURL))
|
|
}
|
|
}
|
|
|
|
if cliCtx.String(rdFlag) != "" && cliCtx.String(rmFlag) == "" {
|
|
fatalIf(errInvalidArgument().Trace(), fmt.Sprintf("Both object retention flags `--%s` and `--%s` are required.\n", rdFlag, rmFlag))
|
|
}
|
|
|
|
if cliCtx.String(rdFlag) == "" && cliCtx.String(rmFlag) != "" {
|
|
fatalIf(errInvalidArgument().Trace(), fmt.Sprintf("Both object retention flags `--%s` and `--%s` are required.\n", rdFlag, rmFlag))
|
|
}
|
|
|
|
operation := "copy"
|
|
if isMvCmd {
|
|
operation = "move"
|
|
}
|
|
|
|
// Guess CopyURLsType based on source and target URLs.
|
|
opts := prepareCopyURLsOpts{
|
|
sourceURLs: srcURLs,
|
|
targetURL: tgtURL,
|
|
isRecursive: isRecursive,
|
|
encKeyDB: encKeyDB,
|
|
olderThan: "",
|
|
newerThan: "",
|
|
timeRef: timeRef,
|
|
versionID: versionID,
|
|
isZip: isZip,
|
|
}
|
|
copyURLsType, _, err := guessCopyURLType(ctx, opts)
|
|
if err != nil {
|
|
fatalIf(errInvalidArgument().Trace(), "Unable to guess the type of "+operation+" operation.")
|
|
}
|
|
|
|
switch copyURLsType {
|
|
case copyURLsTypeA: // File -> File.
|
|
// Check source.
|
|
if len(srcURLs) != 1 {
|
|
fatalIf(errInvalidArgument().Trace(), "Invalid number of source arguments.")
|
|
}
|
|
checkCopySyntaxTypeA(ctx, srcURLs[0], versionID, tgtURL, encKeyDB, isMvCmd, timeRef)
|
|
case copyURLsTypeB: // File -> Folder.
|
|
// Check source.
|
|
if len(srcURLs) != 1 {
|
|
fatalIf(errInvalidArgument().Trace(), "Invalid number of source arguments.")
|
|
}
|
|
checkCopySyntaxTypeB(ctx, srcURLs[0], versionID, tgtURL, encKeyDB, isMvCmd, timeRef)
|
|
case copyURLsTypeC: // Folder... -> Folder.
|
|
checkCopySyntaxTypeC(ctx, srcURLs, tgtURL, isRecursive, isZip, encKeyDB, isMvCmd, timeRef)
|
|
case copyURLsTypeD: // File1...FileN -> Folder.
|
|
checkCopySyntaxTypeD(ctx, srcURLs, tgtURL, encKeyDB, isMvCmd, timeRef)
|
|
default:
|
|
fatalIf(errInvalidArgument().Trace(), "Unable to guess the type of "+operation+" operation.")
|
|
}
|
|
|
|
// Preserve functionality not supported for windows
|
|
if cliCtx.Bool("preserve") && runtime.GOOS == "windows" {
|
|
fatalIf(errInvalidArgument().Trace(), "Permissions are not preserved on windows platform.")
|
|
}
|
|
}
|
|
|
|
// checkCopySyntaxTypeA verifies if the source and target are valid file arguments.
|
|
func checkCopySyntaxTypeA(ctx context.Context, srcURL, versionID string, tgtURL string, keys map[string][]prefixSSEPair, isMvCmd bool, timeRef time.Time) {
|
|
_, srcContent, err := url2Stat(ctx, srcURL, versionID, false, keys, timeRef, false)
|
|
fatalIf(err.Trace(srcURL), "Unable to stat source `"+srcURL+"`.")
|
|
|
|
if !srcContent.Type.IsRegular() {
|
|
fatalIf(errInvalidArgument().Trace(), "Source `"+srcURL+"` is not a file.")
|
|
}
|
|
}
|
|
|
|
// checkCopySyntaxTypeB verifies if the source is a valid file and target is a valid folder.
|
|
func checkCopySyntaxTypeB(ctx context.Context, srcURL, versionID string, tgtURL string, keys map[string][]prefixSSEPair, isMvCmd bool, timeRef time.Time) {
|
|
_, srcContent, err := url2Stat(ctx, srcURL, versionID, false, keys, timeRef, false)
|
|
fatalIf(err.Trace(srcURL), "Unable to stat source `"+srcURL+"`.")
|
|
|
|
if !srcContent.Type.IsRegular() {
|
|
fatalIf(errInvalidArgument().Trace(srcURL), "Source `"+srcURL+"` is not a file.")
|
|
}
|
|
|
|
// Check target.
|
|
if _, tgtContent, err := url2Stat(ctx, tgtURL, "", false, keys, timeRef, false); err == nil {
|
|
if !tgtContent.Type.IsDir() {
|
|
fatalIf(errInvalidArgument().Trace(tgtURL), "Target `"+tgtURL+"` is not a folder.")
|
|
}
|
|
}
|
|
}
|
|
|
|
// checkCopySyntaxTypeC verifies if the source is a valid recursive dir and target is a valid folder.
|
|
func checkCopySyntaxTypeC(ctx context.Context, srcURLs []string, tgtURL string, isRecursive, isZip bool, keys map[string][]prefixSSEPair, isMvCmd bool, timeRef time.Time) {
|
|
// Check source.
|
|
if len(srcURLs) != 1 {
|
|
fatalIf(errInvalidArgument().Trace(), "Invalid number of source arguments.")
|
|
}
|
|
|
|
// Check target.
|
|
if _, tgtContent, err := url2Stat(ctx, tgtURL, "", false, keys, timeRef, false); err == nil {
|
|
if !tgtContent.Type.IsDir() {
|
|
fatalIf(errInvalidArgument().Trace(tgtURL), "Target `"+tgtURL+"` is not a folder.")
|
|
}
|
|
}
|
|
|
|
for _, srcURL := range srcURLs {
|
|
c, srcContent, err := url2Stat(ctx, srcURL, "", false, keys, timeRef, isZip)
|
|
fatalIf(err.Trace(srcURL), "Unable to stat source `"+srcURL+"`.")
|
|
|
|
if srcContent.Type.IsDir() {
|
|
// Require --recursive flag if we are copying a directory
|
|
if !isRecursive {
|
|
operation := "copy"
|
|
if isMvCmd {
|
|
operation = "move"
|
|
}
|
|
fatalIf(errInvalidArgument().Trace(srcURL), fmt.Sprintf("To %v a folder requires --recursive flag.", operation))
|
|
}
|
|
|
|
// Check if we are going to copy a directory into itself
|
|
if isURLContains(srcURL, tgtURL, string(c.GetURL().Separator)) {
|
|
operation := "Copying"
|
|
if isMvCmd {
|
|
operation = "Moving"
|
|
}
|
|
fatalIf(errInvalidArgument().Trace(), fmt.Sprintf("%v a folder into itself is not allowed.", operation))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// checkCopySyntaxTypeD verifies if the source is a valid list of files and target is a valid folder.
|
|
func checkCopySyntaxTypeD(ctx context.Context, srcURLs []string, tgtURL string, keys map[string][]prefixSSEPair, isMvCmd bool, timeRef time.Time) {
|
|
// Source can be anything: file, dir, dir...
|
|
// Check target if it is a dir
|
|
if _, tgtContent, err := url2Stat(ctx, tgtURL, "", false, keys, timeRef, false); err == nil {
|
|
if !tgtContent.Type.IsDir() {
|
|
fatalIf(errInvalidArgument().Trace(tgtURL), "Target `"+tgtURL+"` is not a folder.")
|
|
}
|
|
}
|
|
}
|