2017-06-12 17:40:20 +00:00
|
|
|
package lsjson
|
|
|
|
|
|
|
|
import (
|
2019-06-17 08:34:30 +00:00
|
|
|
"context"
|
2017-06-12 17:40:20 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2019-07-28 17:47:38 +00:00
|
|
|
"github.com/rclone/rclone/cmd"
|
|
|
|
"github.com/rclone/rclone/cmd/ls/lshelp"
|
2020-09-23 16:20:28 +00:00
|
|
|
"github.com/rclone/rclone/fs"
|
2019-10-11 15:55:04 +00:00
|
|
|
"github.com/rclone/rclone/fs/config/flags"
|
2019-07-28 17:47:38 +00:00
|
|
|
"github.com/rclone/rclone/fs/operations"
|
2017-06-12 17:40:20 +00:00
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-09-23 16:20:28 +00:00
|
|
|
opt operations.ListJSONOpt
|
|
|
|
statOnly bool
|
2017-06-12 17:40:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2019-10-11 15:58:11 +00:00
|
|
|
cmd.Root.AddCommand(commandDefinition)
|
2019-10-11 15:55:04 +00:00
|
|
|
cmdFlags := commandDefinition.Flags()
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.Recurse, "recursive", "R", false, "Recurse into the listing.")
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.ShowHash, "hash", "", false, "Include hashes in the output (may take longer).")
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.NoModTime, "no-modtime", "", false, "Don't read the modification time (can speed things up).")
|
2020-01-26 16:38:00 +00:00
|
|
|
flags.BoolVarP(cmdFlags, &opt.NoMimeType, "no-mimetype", "", false, "Don't read the mime type (can speed things up).")
|
2019-10-11 15:55:04 +00:00
|
|
|
flags.BoolVarP(cmdFlags, &opt.ShowEncrypted, "encrypted", "M", false, "Show the encrypted names.")
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.ShowOrigIDs, "original", "", false, "Show the ID of the underlying Object.")
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.FilesOnly, "files-only", "", false, "Show only files in the listing.")
|
|
|
|
flags.BoolVarP(cmdFlags, &opt.DirsOnly, "dirs-only", "", false, "Show only directories in the listing.")
|
2020-04-28 17:44:04 +00:00
|
|
|
flags.StringArrayVarP(cmdFlags, &opt.HashTypes, "hash-type", "", nil, "Show only this hash type (may be repeated).")
|
2020-09-23 16:20:28 +00:00
|
|
|
flags.BoolVarP(cmdFlags, &statOnly, "stat", "", false, "Just return the info for the pointed to file.")
|
2017-06-12 17:40:20 +00:00
|
|
|
}
|
|
|
|
|
2019-10-11 15:58:11 +00:00
|
|
|
var commandDefinition = &cobra.Command{
|
2017-06-12 17:40:20 +00:00
|
|
|
Use: "lsjson remote:path",
|
|
|
|
Short: `List directories and objects in the path in JSON format.`,
|
|
|
|
Long: `List directories and objects in the path in JSON format.
|
|
|
|
|
|
|
|
The output is an array of Items, where each Item looks like this
|
|
|
|
|
|
|
|
{
|
|
|
|
"Hashes" : {
|
|
|
|
"SHA-1" : "f572d396fae9206628714fb2ce00f72e94f2258f",
|
|
|
|
"MD5" : "b1946ac92492d2347c6235b4d2611184",
|
|
|
|
"DropboxHash" : "ecb65bb98f9d905b70458986c39fcbad7715e5f2fcc3b1f07767d7c83e2438cc"
|
|
|
|
},
|
2018-05-13 08:19:06 +00:00
|
|
|
"ID": "y2djkhiujf83u33",
|
2018-07-26 15:47:23 +00:00
|
|
|
"OrigID": "UYOJVTUW00Q1RzTDA",
|
2019-06-07 16:28:15 +00:00
|
|
|
"IsBucket" : false,
|
2017-06-12 17:40:20 +00:00
|
|
|
"IsDir" : false,
|
2018-05-13 09:37:46 +00:00
|
|
|
"MimeType" : "application/octet-stream",
|
2017-06-12 17:40:20 +00:00
|
|
|
"ModTime" : "2017-05-31T16:15:57.034468261+01:00",
|
|
|
|
"Name" : "file.txt",
|
2018-03-09 08:44:02 +00:00
|
|
|
"Encrypted" : "v0qpsdq8anpci8n929v3uu9338",
|
2019-06-07 16:28:15 +00:00
|
|
|
"EncryptedPath" : "kja9098349023498/v0qpsdq8anpci8n929v3uu9338",
|
2017-06-12 17:40:20 +00:00
|
|
|
"Path" : "full/path/goes/here/file.txt",
|
2019-06-07 16:28:15 +00:00
|
|
|
"Size" : 6,
|
|
|
|
"Tier" : "hot",
|
2017-06-12 17:40:20 +00:00
|
|
|
}
|
|
|
|
|
2020-04-28 17:44:04 +00:00
|
|
|
If --hash is not specified the Hashes property won't be emitted. The
|
|
|
|
types of hash can be specified with the --hash-type parameter (which
|
|
|
|
may be repeated). If --hash-type is set then it implies --hash.
|
2017-06-12 17:40:20 +00:00
|
|
|
|
2020-04-28 17:44:04 +00:00
|
|
|
If --no-modtime is specified then ModTime will be blank. This can
|
|
|
|
speed things up on remotes where reading the ModTime takes an extra
|
2020-10-13 21:49:58 +00:00
|
|
|
request (e.g. s3, swift).
|
2020-01-26 16:38:00 +00:00
|
|
|
|
2020-04-28 17:44:04 +00:00
|
|
|
If --no-mimetype is specified then MimeType will be blank. This can
|
|
|
|
speed things up on remotes where reading the MimeType takes an extra
|
2020-10-13 21:49:58 +00:00
|
|
|
request (e.g. s3, swift).
|
2017-06-12 17:40:20 +00:00
|
|
|
|
2018-03-09 08:44:02 +00:00
|
|
|
If --encrypted is not specified the Encrypted won't be emitted.
|
|
|
|
|
2020-04-28 17:44:04 +00:00
|
|
|
If --dirs-only is not specified files in addition to directories are
|
|
|
|
returned
|
2019-04-05 20:24:09 +00:00
|
|
|
|
2020-04-28 17:44:04 +00:00
|
|
|
If --files-only is not specified directories in addition to the files
|
|
|
|
will be returned.
|
2019-04-05 20:24:09 +00:00
|
|
|
|
2020-09-23 16:20:28 +00:00
|
|
|
if --stat is set then a single JSON blob will be returned about the
|
|
|
|
item pointed to. This will return an error if the item isn't found.
|
|
|
|
However on bucket based backends (like s3, gcs, b2, azureblob etc) if
|
|
|
|
the item isn't found it will return an empty directory as it isn't
|
|
|
|
possible to tell empty directories from missing directories there.
|
|
|
|
|
2018-01-31 12:34:23 +00:00
|
|
|
The Path field will only show folders below the remote path being listed.
|
|
|
|
If "remote:path" contains the file "subfolder/file.txt", the Path for "file.txt"
|
|
|
|
will be "subfolder/file.txt", not "remote:path/subfolder/file.txt".
|
|
|
|
When used without --recursive the Path will always be the same as Name.
|
|
|
|
|
2019-06-07 16:28:15 +00:00
|
|
|
If the directory is a bucket in a bucket based backend, then
|
|
|
|
"IsBucket" will be set to true. This key won't be present unless it is
|
|
|
|
"true".
|
|
|
|
|
2019-02-03 11:01:52 +00:00
|
|
|
The time is in RFC3339 format with up to nanosecond precision. The
|
|
|
|
number of decimal digits in the seconds will depend on the precision
|
|
|
|
that the remote can hold the times, so if times are accurate to the
|
2020-10-13 21:49:58 +00:00
|
|
|
nearest millisecond (e.g. Google Drive) then 3 digits will always be
|
2019-02-03 11:01:52 +00:00
|
|
|
shown ("2017-05-31T16:15:57.034+01:00") whereas if the times are
|
2020-10-13 22:07:12 +00:00
|
|
|
accurate to the nearest second (Dropbox, Box, WebDav, etc.) no digits
|
2019-02-03 11:01:52 +00:00
|
|
|
will be shown ("2017-05-31T16:15:57+01:00").
|
2017-06-12 17:40:20 +00:00
|
|
|
|
|
|
|
The whole output can be processed as a JSON blob, or alternatively it
|
|
|
|
can be processed line by line as each item is written one to a line.
|
2018-01-06 17:00:20 +00:00
|
|
|
` + lshelp.Help,
|
2020-09-23 16:20:28 +00:00
|
|
|
RunE: func(command *cobra.Command, args []string) error {
|
2017-06-12 17:40:20 +00:00
|
|
|
cmd.CheckArgs(1, 1, command, args)
|
2020-09-23 16:20:28 +00:00
|
|
|
var fsrc fs.Fs
|
|
|
|
var remote string
|
|
|
|
if statOnly {
|
|
|
|
fsrc, remote = cmd.NewFsFile(args[0])
|
|
|
|
} else {
|
|
|
|
fsrc = cmd.NewFsSrc(args)
|
|
|
|
}
|
2017-06-12 17:40:20 +00:00
|
|
|
cmd.Run(false, false, command, func() error {
|
2020-09-23 16:20:28 +00:00
|
|
|
if statOnly {
|
|
|
|
item, err := operations.StatJSON(context.Background(), fsrc, remote, &opt)
|
2017-06-12 17:40:20 +00:00
|
|
|
if err != nil {
|
2020-09-23 16:20:28 +00:00
|
|
|
return err
|
2017-06-12 17:40:20 +00:00
|
|
|
}
|
2020-09-23 16:20:28 +00:00
|
|
|
out, err := json.MarshalIndent(item, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to marshal list object")
|
2018-10-26 13:50:55 +00:00
|
|
|
}
|
|
|
|
_, err = os.Stdout.Write(out)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to write to output")
|
2017-06-12 17:40:20 +00:00
|
|
|
}
|
|
|
|
fmt.Println()
|
2020-09-23 16:20:28 +00:00
|
|
|
} else {
|
|
|
|
fmt.Println("[")
|
|
|
|
first := true
|
|
|
|
err := operations.ListJSON(context.Background(), fsrc, remote, &opt, func(item *operations.ListJSONItem) error {
|
|
|
|
out, err := json.Marshal(item)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to marshal list object")
|
|
|
|
}
|
|
|
|
if first {
|
|
|
|
first = false
|
|
|
|
} else {
|
|
|
|
fmt.Print(",\n")
|
|
|
|
}
|
|
|
|
_, err = os.Stdout.Write(out)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to write to output")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !first {
|
|
|
|
fmt.Println()
|
|
|
|
}
|
|
|
|
fmt.Println("]")
|
2017-06-12 17:40:20 +00:00
|
|
|
}
|
2020-09-23 16:20:28 +00:00
|
|
|
return nil
|
2017-06-12 17:40:20 +00:00
|
|
|
})
|
2020-09-23 16:20:28 +00:00
|
|
|
return nil
|
2017-06-12 17:40:20 +00:00
|
|
|
},
|
|
|
|
}
|