forked from TrueCloudLab/rclone
3fcff32524
...unless had some errors or stats interval requested. Add fs.ErrorLog to differentiate between Logs which should be suppressed and errors which shouldn't.
393 lines
8.6 KiB
Go
393 lines
8.6 KiB
Go
// Sync files and directories to and from local and remote object stores
|
|
//
|
|
// Nick Craig-Wood <nick@craig-wood.com>
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"runtime"
|
|
"runtime/pprof"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/ogier/pflag"
|
|
|
|
"github.com/ncw/rclone/fs"
|
|
// Active file systems
|
|
_ "github.com/ncw/rclone/drive"
|
|
_ "github.com/ncw/rclone/dropbox"
|
|
_ "github.com/ncw/rclone/googlecloudstorage"
|
|
_ "github.com/ncw/rclone/local"
|
|
_ "github.com/ncw/rclone/s3"
|
|
_ "github.com/ncw/rclone/swift"
|
|
)
|
|
|
|
// Globals
|
|
var (
|
|
// Flags
|
|
cpuprofile = pflag.StringP("cpuprofile", "", "", "Write cpu profile to file")
|
|
statsInterval = pflag.DurationP("stats", "", time.Minute*1, "Interval to print stats (0 to disable)")
|
|
version = pflag.BoolP("version", "V", false, "Print the version number")
|
|
logFile = pflag.StringP("log-file", "", "", "Log everything to this file")
|
|
)
|
|
|
|
type Command struct {
|
|
Name string
|
|
Help string
|
|
ArgsHelp string
|
|
Run func(fdst, fsrc fs.Fs)
|
|
MinArgs int
|
|
MaxArgs int
|
|
NoStats bool
|
|
}
|
|
|
|
// checkArgs checks there are enough arguments and prints a message if not
|
|
func (cmd *Command) checkArgs(args []string) {
|
|
if len(args) < cmd.MinArgs {
|
|
syntaxError()
|
|
fmt.Fprintf(os.Stderr, "Command %s needs %d arguments mininum\n", cmd.Name, cmd.MinArgs)
|
|
os.Exit(1)
|
|
} else if len(args) > cmd.MaxArgs {
|
|
syntaxError()
|
|
fmt.Fprintf(os.Stderr, "Command %s needs %d arguments maximum\n", cmd.Name, cmd.MaxArgs)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
var Commands = []Command{
|
|
{
|
|
Name: "copy",
|
|
ArgsHelp: "source:path dest:path",
|
|
Help: `
|
|
Copy the source to the destination. Doesn't transfer
|
|
unchanged files, testing by size and modification time or
|
|
MD5SUM. Doesn't delete files from the destination.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Sync(fdst, fsrc, false)
|
|
if err != nil {
|
|
log.Fatalf("Failed to copy: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 2,
|
|
MaxArgs: 2,
|
|
},
|
|
{
|
|
Name: "sync",
|
|
ArgsHelp: "source:path dest:path",
|
|
Help: `
|
|
Sync the source to the destination, changing the destination
|
|
only. Doesn't transfer unchanged files, testing by size and
|
|
modification time or MD5SUM. Destination is updated to match
|
|
source, including deleting files if necessary. Since this can
|
|
cause data loss, test first with the --dry-run flag.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Sync(fdst, fsrc, true)
|
|
if err != nil {
|
|
log.Fatalf("Failed to sync: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 2,
|
|
MaxArgs: 2,
|
|
},
|
|
{
|
|
Name: "ls",
|
|
ArgsHelp: "[remote:path]",
|
|
Help: `
|
|
List all the objects in the the path with size and path.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.List(fdst, os.Stdout)
|
|
if err != nil {
|
|
log.Fatalf("Failed to list: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "lsd",
|
|
ArgsHelp: "[remote:path]",
|
|
Help: `
|
|
List all directories/containers/buckets in the the path.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.ListDir(fdst, os.Stdout)
|
|
if err != nil {
|
|
log.Fatalf("Failed to listdir: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "lsl",
|
|
ArgsHelp: "[remote:path]",
|
|
Help: `
|
|
List all the objects in the the path with modification time,
|
|
size and path.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.ListLong(fdst, os.Stdout)
|
|
if err != nil {
|
|
log.Fatalf("Failed to list long: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "md5sum",
|
|
ArgsHelp: "[remote:path]",
|
|
Help: `
|
|
Produces an md5sum file for all the objects in the path. This
|
|
is in the same format as the standard md5sum tool produces.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Md5sum(fdst, os.Stdout)
|
|
if err != nil {
|
|
log.Fatalf("Failed to list: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "mkdir",
|
|
ArgsHelp: "remote:path",
|
|
Help: `
|
|
Make the path if it doesn't already exist`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Mkdir(fdst)
|
|
if err != nil {
|
|
log.Fatalf("Failed to mkdir: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "rmdir",
|
|
ArgsHelp: "remote:path",
|
|
Help: `
|
|
Remove the path. Note that you can't remove a path with
|
|
objects in it, use purge for that.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Rmdir(fdst)
|
|
if err != nil {
|
|
log.Fatalf("Failed to rmdir: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "purge",
|
|
ArgsHelp: "remote:path",
|
|
Help: `
|
|
Remove the path and all of its contents.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Purge(fdst)
|
|
if err != nil {
|
|
log.Fatalf("Failed to purge: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 1,
|
|
MaxArgs: 1,
|
|
},
|
|
{
|
|
Name: "check",
|
|
ArgsHelp: "source:path dest:path",
|
|
Help: `
|
|
Checks the files in the source and destination match. It
|
|
compares sizes and MD5SUMs and prints a report of files which
|
|
don't match. It doesn't alter the source or destination.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
err := fs.Check(fdst, fsrc)
|
|
if err != nil {
|
|
log.Fatalf("Failed to check: %v", err)
|
|
}
|
|
},
|
|
MinArgs: 2,
|
|
MaxArgs: 2,
|
|
},
|
|
{
|
|
Name: "config",
|
|
Help: `
|
|
Enter an interactive configuration session.`,
|
|
Run: func(fdst, fsrc fs.Fs) {
|
|
fs.EditConfig()
|
|
},
|
|
NoStats: true,
|
|
},
|
|
{
|
|
Name: "help",
|
|
Help: `
|
|
This help.`,
|
|
NoStats: true,
|
|
},
|
|
}
|
|
|
|
// syntaxError prints the syntax
|
|
func syntaxError() {
|
|
fmt.Fprintf(os.Stderr, `Sync files and directories to and from local and remote object stores - %s.
|
|
|
|
Syntax: [options] subcommand <parameters> <parameters...>
|
|
|
|
Subcommands:
|
|
|
|
`, fs.Version)
|
|
for i := range Commands {
|
|
cmd := &Commands[i]
|
|
fmt.Fprintf(os.Stderr, " %s %s\n", cmd.Name, cmd.ArgsHelp)
|
|
fmt.Fprintf(os.Stderr, "%s\n\n", cmd.Help)
|
|
}
|
|
|
|
fmt.Fprintf(os.Stderr, "Options:\n")
|
|
pflag.PrintDefaults()
|
|
fmt.Fprintf(os.Stderr, `
|
|
It is only necessary to use a unique prefix of the subcommand, eg 'up'
|
|
for 'upload'.
|
|
`)
|
|
}
|
|
|
|
// Exit with the message
|
|
func fatal(message string, args ...interface{}) {
|
|
syntaxError()
|
|
fmt.Fprintf(os.Stderr, message, args...)
|
|
os.Exit(1)
|
|
}
|
|
|
|
// Parse the command line flags
|
|
func ParseFlags() {
|
|
pflag.Usage = syntaxError
|
|
pflag.Parse()
|
|
runtime.GOMAXPROCS(runtime.NumCPU())
|
|
fs.LoadConfig()
|
|
|
|
// Setup profiling if desired
|
|
if *cpuprofile != "" {
|
|
f, err := os.Create(*cpuprofile)
|
|
if err != nil {
|
|
fs.Stats.Error()
|
|
log.Fatal(err)
|
|
}
|
|
err = pprof.StartCPUProfile(f)
|
|
if err != nil {
|
|
fs.Stats.Error()
|
|
log.Fatal(err)
|
|
}
|
|
defer pprof.StopCPUProfile()
|
|
}
|
|
}
|
|
|
|
// Parse the command from the command line
|
|
func ParseCommand() (*Command, []string) {
|
|
args := pflag.Args()
|
|
if len(args) < 1 {
|
|
fatal("No command supplied\n")
|
|
}
|
|
|
|
cmd := strings.ToLower(args[0])
|
|
args = args[1:]
|
|
|
|
// Find the command doing a prefix match
|
|
var command *Command
|
|
for i := range Commands {
|
|
trialCommand := &Commands[i]
|
|
// exact command name found - use that
|
|
if trialCommand.Name == cmd {
|
|
command = trialCommand
|
|
break
|
|
} else if strings.HasPrefix(trialCommand.Name, cmd) {
|
|
if command != nil {
|
|
fs.Stats.Error()
|
|
log.Fatalf("Not unique - matches multiple commands %q", cmd)
|
|
}
|
|
command = trialCommand
|
|
}
|
|
}
|
|
if command == nil {
|
|
fs.Stats.Error()
|
|
log.Fatalf("Unknown command %q", cmd)
|
|
}
|
|
if command.Run == nil {
|
|
syntaxError()
|
|
}
|
|
command.checkArgs(args)
|
|
return command, args
|
|
}
|
|
|
|
// Create a Fs from a name
|
|
func NewFs(remote string) fs.Fs {
|
|
f, err := fs.NewFs(remote)
|
|
if err != nil {
|
|
fs.Stats.Error()
|
|
log.Fatalf("Failed to create file system for %q: %v", remote, err)
|
|
}
|
|
return f
|
|
}
|
|
|
|
// Print the stats every statsInterval
|
|
func StartStats() {
|
|
if *statsInterval <= 0 {
|
|
return
|
|
}
|
|
go func() {
|
|
ch := time.Tick(*statsInterval)
|
|
for {
|
|
<-ch
|
|
fs.Stats.Log()
|
|
}
|
|
}()
|
|
}
|
|
|
|
func main() {
|
|
ParseFlags()
|
|
if *version {
|
|
fmt.Printf("rclone %s\n", fs.Version)
|
|
os.Exit(0)
|
|
}
|
|
command, args := ParseCommand()
|
|
|
|
// Log file output
|
|
if *logFile != "" {
|
|
f, err := os.OpenFile(*logFile, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0640)
|
|
if err != nil {
|
|
log.Fatalf("Failed to open log file: %v", err)
|
|
}
|
|
f.Seek(0, os.SEEK_END)
|
|
log.SetOutput(f)
|
|
redirectStderr(f)
|
|
}
|
|
|
|
// Make source and destination fs
|
|
var fdst, fsrc fs.Fs
|
|
if len(args) >= 1 {
|
|
fdst = NewFs(args[0])
|
|
}
|
|
if len(args) >= 2 {
|
|
fsrc = fdst
|
|
fdst = NewFs(args[1])
|
|
}
|
|
|
|
fs.CalculateModifyWindow(fdst, fsrc)
|
|
|
|
if !command.NoStats {
|
|
StartStats()
|
|
}
|
|
|
|
// Run the actual command
|
|
if command.Run != nil {
|
|
command.Run(fdst, fsrc)
|
|
if !command.NoStats && (!fs.Config.Quiet || fs.Stats.Errored() || *statsInterval > 0) {
|
|
fmt.Fprintln(os.Stderr, fs.Stats)
|
|
}
|
|
if fs.Config.Verbose {
|
|
fs.Debug(nil, "Go routines at exit %d\n", runtime.NumGoroutine())
|
|
}
|
|
if fs.Stats.Errored() {
|
|
os.Exit(1)
|
|
}
|
|
os.Exit(0)
|
|
} else {
|
|
}
|
|
|
|
}
|