mirror of
https://github.com/minio/mc.git
synced 2025-11-10 13:42:32 +03:00
app.Before receives a cli.Context but without flags parsed. There is no point calling ctx.Bool() or ctx.IsSet() at that stage. However, flags are parsed in commands, so minio initialization and setting global variables (globalJSON, globalQuiet, etc..) can be moved to the Before function of all commands. Avoid setting command.Before for non leaf commands, otherwise Before function will be called multiples times until it reaches the leaf command.
219 lines
6.7 KiB
Go
219 lines
6.7 KiB
Go
/*
|
|
* MinIO Client (C) 2015-2019 MinIO, Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/fatih/color"
|
|
"github.com/minio/cli"
|
|
json "github.com/minio/mc/pkg/colorjson"
|
|
"github.com/minio/mc/pkg/probe"
|
|
"github.com/minio/minio/pkg/console"
|
|
)
|
|
|
|
// diff specific flags.
|
|
var (
|
|
diffFlags = []cli.Flag{}
|
|
)
|
|
|
|
// Compute differences in object name, size, and date between two buckets.
|
|
var diffCmd = cli.Command{
|
|
Name: "diff",
|
|
Usage: "list differences in object name, size, and date between two buckets",
|
|
Action: mainDiff,
|
|
Before: initBeforeRunningCmd,
|
|
Flags: append(diffFlags, globalFlags...),
|
|
CustomHelpTemplate: `NAME:
|
|
{{.HelpName}} - {{.Usage}}
|
|
|
|
USAGE:
|
|
{{.HelpName}} [FLAGS] FIRST SECOND
|
|
|
|
FLAGS:
|
|
{{range .VisibleFlags}}{{.}}
|
|
{{end}}
|
|
DESCRIPTION:
|
|
Diff only calculates differences in object name, size and time. It *DOES NOT* compare objects' contents.
|
|
|
|
LEGEND:
|
|
< - object is only in source.
|
|
> - object is only in destination.
|
|
! - newer object is in source.
|
|
|
|
EXAMPLES:
|
|
1. Compare a local folder with a folder on Amazon S3 cloud storage.
|
|
{{.Prompt}} {{.HelpName}} ~/Photos s3/mybucket/Photos
|
|
|
|
2. Compare two folders on a local filesystem.
|
|
{{.Prompt}} {{.HelpName}} ~/Photos /Media/Backup/Photos
|
|
`,
|
|
}
|
|
|
|
// diffMessage json container for diff messages
|
|
type diffMessage struct {
|
|
Status string `json:"status"`
|
|
FirstURL string `json:"first"`
|
|
SecondURL string `json:"second"`
|
|
Diff differType `json:"diff"`
|
|
Error *probe.Error `json:"error,omitempty"`
|
|
firstContent *ClientContent
|
|
secondContent *ClientContent
|
|
}
|
|
|
|
// String colorized diff message
|
|
func (d diffMessage) String() string {
|
|
msg := ""
|
|
switch d.Diff {
|
|
case differInFirst:
|
|
msg = console.Colorize("DiffOnlyInFirst", "< "+d.FirstURL)
|
|
case differInSecond:
|
|
msg = console.Colorize("DiffOnlyInSecond", "> "+d.SecondURL)
|
|
case differInType:
|
|
msg = console.Colorize("DiffType", "! "+d.SecondURL)
|
|
case differInSize:
|
|
msg = console.Colorize("DiffSize", "! "+d.SecondURL)
|
|
case differInMetadata:
|
|
msg = console.Colorize("DiffMetadata", "! "+d.SecondURL)
|
|
case differInAASourceMTime:
|
|
msg = console.Colorize("DiffMMSourceMTime", "! "+d.SecondURL)
|
|
default:
|
|
fatalIf(errDummy().Trace(d.FirstURL, d.SecondURL),
|
|
"Unhandled difference between `"+d.FirstURL+"` and `"+d.SecondURL+"`.")
|
|
}
|
|
return msg
|
|
|
|
}
|
|
|
|
// JSON jsonified diff message
|
|
func (d diffMessage) JSON() string {
|
|
d.Status = "success"
|
|
diffJSONBytes, e := json.MarshalIndent(d, "", " ")
|
|
fatalIf(probe.NewError(e),
|
|
"Unable to marshal diff message `"+d.FirstURL+"`, `"+d.SecondURL+"` and `"+string(d.Diff)+"`.")
|
|
return string(diffJSONBytes)
|
|
}
|
|
|
|
func checkDiffSyntax(ctx context.Context, cliCtx *cli.Context, encKeyDB map[string][]prefixSSEPair) {
|
|
if len(cliCtx.Args()) != 2 {
|
|
cli.ShowCommandHelpAndExit(cliCtx, "diff", 1) // last argument is exit code
|
|
}
|
|
for _, arg := range cliCtx.Args() {
|
|
if strings.TrimSpace(arg) == "" {
|
|
fatalIf(errInvalidArgument().Trace(cliCtx.Args()...), "Unable to validate empty argument.")
|
|
}
|
|
}
|
|
URLs := cliCtx.Args()
|
|
firstURL := URLs[0]
|
|
secondURL := URLs[1]
|
|
|
|
// Diff only works between two directories, verify them below.
|
|
|
|
// Verify if firstURL is accessible.
|
|
_, firstContent, err := url2Stat(ctx, firstURL, false, encKeyDB)
|
|
if err != nil {
|
|
fatalIf(err.Trace(firstURL), fmt.Sprintf("Unable to stat '%s'.", firstURL))
|
|
}
|
|
|
|
// Verify if its a directory.
|
|
if !firstContent.Type.IsDir() {
|
|
fatalIf(errInvalidArgument().Trace(firstURL), fmt.Sprintf("`%s` is not a folder.", firstURL))
|
|
}
|
|
|
|
// Verify if secondURL is accessible.
|
|
_, secondContent, err := url2Stat(ctx, secondURL, false, encKeyDB)
|
|
if err != nil {
|
|
fatalIf(err.Trace(secondURL), fmt.Sprintf("Unable to stat '%s'.", secondURL))
|
|
}
|
|
|
|
// Verify if its a directory.
|
|
if !secondContent.Type.IsDir() {
|
|
fatalIf(errInvalidArgument().Trace(secondURL), fmt.Sprintf("`%s` is not a folder.", secondURL))
|
|
}
|
|
}
|
|
|
|
// doDiffMain runs the diff.
|
|
func doDiffMain(ctx context.Context, firstURL, secondURL string) error {
|
|
// Source and targets are always directories
|
|
sourceSeparator := string(newClientURL(firstURL).Separator)
|
|
if !strings.HasSuffix(firstURL, sourceSeparator) {
|
|
firstURL = firstURL + sourceSeparator
|
|
}
|
|
targetSeparator := string(newClientURL(secondURL).Separator)
|
|
if !strings.HasSuffix(secondURL, targetSeparator) {
|
|
secondURL = secondURL + targetSeparator
|
|
}
|
|
|
|
// Expand aliased urls.
|
|
firstAlias, firstURL, _ := mustExpandAlias(firstURL)
|
|
secondAlias, secondURL, _ := mustExpandAlias(secondURL)
|
|
|
|
firstClient, err := newClientFromAlias(firstAlias, firstURL)
|
|
if err != nil {
|
|
fatalIf(err.Trace(firstAlias, firstURL, secondAlias, secondURL),
|
|
fmt.Sprintf("Failed to diff '%s' and '%s'", firstURL, secondURL))
|
|
}
|
|
|
|
secondClient, err := newClientFromAlias(secondAlias, secondURL)
|
|
if err != nil {
|
|
fatalIf(err.Trace(firstAlias, firstURL, secondAlias, secondURL),
|
|
fmt.Sprintf("Failed to diff '%s' and '%s'", firstURL, secondURL))
|
|
}
|
|
|
|
// Diff first and second urls.
|
|
for diffMsg := range objectDifference(ctx, firstClient, secondClient, firstURL, secondURL, true) {
|
|
if diffMsg.Error != nil {
|
|
errorIf(diffMsg.Error, "Unable to calculate objects difference.")
|
|
// Ignore error and proceed to next object.
|
|
continue
|
|
}
|
|
printMsg(diffMsg)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// mainDiff main for 'diff'.
|
|
func mainDiff(cliCtx *cli.Context) error {
|
|
ctx, cancelDiff := context.WithCancel(globalContext)
|
|
defer cancelDiff()
|
|
|
|
// Parse encryption keys per command.
|
|
encKeyDB, err := getEncKeys(cliCtx)
|
|
fatalIf(err, "Unable to parse encryption keys.")
|
|
|
|
// check 'diff' cli arguments.
|
|
checkDiffSyntax(ctx, cliCtx, encKeyDB)
|
|
|
|
// Additional command specific theme customization.
|
|
console.SetColor("DiffMessage", color.New(color.FgGreen, color.Bold))
|
|
console.SetColor("DiffOnlyInFirst", color.New(color.FgRed))
|
|
console.SetColor("DiffOnlyInSecond", color.New(color.FgGreen))
|
|
console.SetColor("DiffType", color.New(color.FgMagenta))
|
|
console.SetColor("DiffSize", color.New(color.FgYellow, color.Bold))
|
|
console.SetColor("DiffMetadata", color.New(color.FgYellow, color.Bold))
|
|
console.SetColor("DiffMMSourceMTime", color.New(color.FgYellow, color.Bold))
|
|
|
|
URLs := cliCtx.Args()
|
|
firstURL := URLs.Get(0)
|
|
secondURL := URLs.Get(1)
|
|
|
|
return doDiffMain(ctx, firstURL, secondURL)
|
|
}
|