/* * Minio Client (C) 2017 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 ( "encoding/json" "fmt" "path/filepath" "strings" "time" humanize "github.com/dustin/go-humanize" "github.com/minio/mc/pkg/console" "github.com/minio/mc/pkg/probe" ) // contentMessage container for content message structure. type statMessage struct { Status string `json:"status"` Key string `json:"name"` Date time.Time `json:"lastModified"` Size int64 `json:"size"` ETag string `json:"etag"` Type string `json:"type"` EncryptionHeaders map[string]string `json:"encryption,omitempty"` Metadata map[string]string `json:"metadata"` } // String colorized string message. func printStat(stat statMessage) { // Format properly for alignment based on maxKey length stat.Key = fmt.Sprintf("%-10s: %s", "Name", stat.Key) console.Println(console.Colorize("Name", stat.Key)) console.Println(fmt.Sprintf("%-10s: %s ", "Date", stat.Date.Format(printDate))) console.Println(fmt.Sprintf("%-10s: %-6s ", "Size", humanize.IBytes(uint64(stat.Size)))) if stat.ETag != "" { console.Println(fmt.Sprintf("%-10s: %s ", "ETag", stat.ETag)) } console.Println(fmt.Sprintf("%-10s: %s ", "Type", stat.Type)) var maxKey = 0 for k := range stat.Metadata { if len(k) > maxKey { maxKey = len(k) } } if len(stat.Metadata) > 0 { console.Println(fmt.Sprintf("%-10s:", "Metadata")) for k, v := range stat.Metadata { console.Println(fmt.Sprintf(" %-*.*s: %s ", maxKey, maxKey, k, v)) } } maxKey = 0 for k := range stat.EncryptionHeaders { if len(k) > maxKey { maxKey = len(k) } } if len(stat.EncryptionHeaders) > 0 { console.Println(fmt.Sprintf("%-10s:", "Encrypted")) for k, v := range stat.EncryptionHeaders { console.Println(fmt.Sprintf(" %-*.*s: %s ", maxKey, maxKey, k, v)) } } console.Println() } // JSON jsonified content message. func (c statMessage) JSON() string { c.Status = "success" jsonMessageBytes, e := json.Marshal(c) fatalIf(probe.NewError(e), "Unable to marshal into JSON.") return string(jsonMessageBytes) } // parseStat parses client Content container into statMessage struct. func parseStat(targetAlias string, c *clientContent) statMessage { content := statMessage{} content.Date = c.Time.Local() // guess file type. content.Type = func() string { if c.Type.IsDir() { return "folder" } return "file" }() content.Size = c.Size content.Key = getKey(c) content.Metadata = c.Metadata content.ETag = strings.TrimPrefix(c.ETag, "\"") content.ETag = strings.TrimSuffix(content.ETag, "\"") content.EncryptionHeaders = c.EncryptionHeaders return content } // doStat - list all entities inside a folder. func doStat(clnt Client, isRecursive bool, targetAlias, targetURL string) error { if !strings.HasSuffix(targetURL, string(clnt.GetURL().Separator)) { targetURL = targetURL + string(clnt.GetURL().Separator) } clnt, err := newClient(targetURL) fatalIf(err.Trace(targetURL), "Unable to initialize target `"+targetURL+"`.") prefixPath := clnt.GetURL().Path separator := string(clnt.GetURL().Separator) if !strings.HasSuffix(prefixPath, separator) { prefixPath = prefixPath[:strings.LastIndex(prefixPath, separator)+1] } var cErr error isIncomplete := false var prevStat *clientContent for content := range clnt.List(isRecursive, isIncomplete, DirNone) { if content.Err != nil { switch content.Err.ToGoError().(type) { // handle this specifically for filesystem related errors. case BrokenSymlink: errorIf(content.Err.Trace(clnt.GetURL().String()), "Unable to list broken link.") continue case TooManyLevelsSymlink: errorIf(content.Err.Trace(clnt.GetURL().String()), "Unable to list too many levels link.") continue case PathNotFound: errorIf(content.Err.Trace(clnt.GetURL().String()), "Unable to list folder.") continue case PathInsufficientPermission: errorIf(content.Err.Trace(clnt.GetURL().String()), "Unable to list folder.") continue case ObjectOnGlacier: errorIf(content.Err.Trace(clnt.GetURL().String()), "") continue } errorIf(content.Err.Trace(clnt.GetURL().String()), "Unable to list folder.") cErr = exitStatus(globalErrorExitStatus) // Set the exit status. continue } url := targetAlias + getKey(content) if !isAliasURLDir(targetURL) { url = content.URL.Path } _, stat, err := url2StatWithMetadata(url, true) if err != nil { stat = content } // Convert any os specific delimiters to "/". contentURL := filepath.ToSlash(stat.URL.Path) prefixPath = filepath.ToSlash(prefixPath) // Trim prefix path from the content path. contentURL = strings.TrimPrefix(contentURL, prefixPath) stat.URL.Path = contentURL if stat != prevStat { st := parseStat(targetAlias, stat) if !globalJSON { printStat(st) } else { console.Println(st.JSON()) } } else { } prevStat = stat } return cErr }