1
0
mirror of https://github.com/minio/mc.git synced 2025-04-18 10:04:03 +03:00
mc/cmd/stat-main.go
Harshavardhana 32b7b72193
conditionally disallow list() when object does not exist (#5016)
allow users to avoid list() call when an object does not
exist when following flags are used with `mc stat`

```
--no-list (HEAD the latest version)
--version-id <vid>
```

Also avoid superflous url2Stat() calls while processing
the args, there is no real reason to do that.
2024-08-12 01:52:27 -07:00

197 lines
6.2 KiB
Go

// Copyright (c) 2015-2022 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"
"path/filepath"
"strings"
"time"
"github.com/fatih/color"
"github.com/minio/cli"
"github.com/minio/pkg/v3/console"
)
// stat specific flags.
var (
statFlags = []cli.Flag{
cli.StringFlag{
Name: "rewind",
Usage: "stat on older version(s)",
},
cli.BoolFlag{
Name: "versions",
Usage: "stat all versions",
},
cli.StringFlag{
Name: "version-id, vid",
Usage: "stat a specific object version",
},
cli.BoolFlag{
Name: "recursive, r",
Usage: "stat all objects recursively",
},
cli.BoolFlag{
Name: "verbose, v",
Usage: "show extended bucket(s) stat",
},
cli.BoolFlag{
Name: "no-list",
Usage: "disable all LIST operations for stat",
},
}
)
// show object metadata
var statCmd = cli.Command{
Name: "stat",
Usage: "show object metadata",
Action: mainStat,
OnUsageError: onUsageError,
Before: setGlobalsFromContext,
Flags: append(append(statFlags, encCFlag), globalFlags...),
CustomHelpTemplate: `NAME:
{{.HelpName}} - {{.Usage}}
USAGE:
{{.HelpName}} [FLAGS] TARGET [TARGET ...]
FLAGS:
{{range .VisibleFlags}}{{.}}
{{end}}
EXAMPLES:
1. Stat all contents of mybucket on Amazon S3 cloud storage.
{{.Prompt}} {{.HelpName}} s3/mybucket/
2. Stat all contents of all buckets on Amazon S3 cloud storage.
{{.Prompt}} {{.HelpName}} s3 --verbose
3. Stat all contents of mybucket on Amazon S3 cloud storage on Microsoft Windows.
{{.Prompt}} {{.HelpName}} s3\mybucket\
4. Stat files recursively on a local filesystem on Microsoft Windows.
{{.Prompt}} {{.HelpName}} --recursive C:\Users\mydocuments\
5. Stat encrypted files on Amazon S3 cloud storage. In case the encryption key contains non-printable character like tab, pass the
base64 encoded string as key.
{{.Prompt}} {{.HelpName}} --enc-c "s3/personal-document/=MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIzNDU2Nzg5MDA" s3/personal-document/2019-account_report.docx
6. Stat a specific object version.
{{.Prompt}} {{.HelpName}} --version-id "CL3sWgdSN2pNntSf6UnZAuh2kcu8E8si" s3/personal-docs/2018-account_report.docx
7. Stat all objects versions recursively created before 1st January 2020.
{{.Prompt}} {{.HelpName}} --versions --rewind 2020.01.01T00:00 s3/personal-docs/
`,
}
// parseAndCheckStatSyntax - parse and validate all the passed arguments
func parseAndCheckStatSyntax(ctx context.Context, cliCtx *cli.Context) ([]string, bool, string, time.Time, bool) {
if !cliCtx.Args().Present() {
showCommandHelpAndExit(cliCtx, 1) // last argument is exit code
}
args := cliCtx.Args()
for _, arg := range args {
if strings.TrimSpace(arg) == "" {
fatalIf(errInvalidArgument().Trace(args...), "Unable to validate empty argument.")
}
}
recursive := cliCtx.Bool("recursive")
versionID := cliCtx.String("version-id")
withVersions := cliCtx.Bool("versions")
headOnly := cliCtx.Bool("no-list")
rewind := parseRewindFlag(cliCtx.String("rewind"))
// extract URLs.
URLs := cliCtx.Args()
if versionID != "" && len(args) > 1 {
fatalIf(errInvalidArgument().Trace(args...), "You cannot specify --version-id with multiple arguments.")
}
if versionID != "" && (recursive || withVersions || !rewind.IsZero()) {
fatalIf(errInvalidArgument().Trace(args...), "You cannot specify --version-id with either --rewind, --versions or --recursive.")
}
if (recursive || withVersions) && headOnly {
fatalIf(errInvalidArgument().Trace(args...), "You cannot specify --no-list with either --versions or --recursive.")
}
var targetUrls []string
for _, url := range URLs {
_, path := url2Alias(url)
if path != "" || !cliCtx.Bool("verbose") {
targetUrls = append(targetUrls, url)
continue
}
clnt, err := newClient(url)
fatalIf(err.Trace(args...), "Unable to initialize `"+url+"`.")
buckets, e := clnt.ListBuckets(ctx)
if e != nil || len(buckets) == 0 {
targetUrls = append(targetUrls, url)
continue
}
for _, bucket := range buckets {
targetUrls = append(targetUrls, filepath.Join(url, bucket.BucketName))
}
}
return targetUrls, recursive, versionID, rewind, withVersions
}
// mainStat - is a handler for mc stat command
func mainStat(cliCtx *cli.Context) error {
ctx, cancelStat := context.WithCancel(globalContext)
defer cancelStat()
// Additional command specific theme customization.
console.SetColor("Name", color.New(color.Bold, color.FgCyan))
console.SetColor("Date", color.New(color.FgWhite))
console.SetColor("Size", color.New(color.FgWhite))
console.SetColor("ETag", color.New(color.FgWhite))
console.SetColor("Metadata", color.New(color.FgWhite))
// theme specific to stat bucket
console.SetColor("Key", color.New(color.FgCyan))
console.SetColor("Value", color.New(color.FgYellow))
console.SetColor("Unset", color.New(color.FgRed))
console.SetColor("Set", color.New(color.FgGreen))
console.SetColor("Title", color.New(color.Bold, color.FgBlue))
console.SetColor("Count", color.New(color.FgGreen))
// Parse encryption keys per command.
encKeyDB, err := validateAndCreateEncryptionKeys(cliCtx)
fatalIf(err, "Unable to parse encryption keys.")
// check 'stat' cli arguments.
args, isRecursive, versionID, rewind, withVersions := parseAndCheckStatSyntax(ctx, cliCtx)
// mimic operating system tool behavior.
if len(args) == 0 {
args = []string{"."}
}
headOnly := cliCtx.Bool("no-list")
for _, targetURL := range args {
fatalIf(statURL(ctx, targetURL, versionID, rewind, withVersions, false, isRecursive, headOnly, encKeyDB), "Unable to stat `"+targetURL+"`.")
}
return nil
}