46a3242ed4
Change-Id: I9c251028d9f72572f4d42815de31d44517cd5525
144 lines
3.1 KiB
Go
144 lines
3.1 KiB
Go
// Copyright (C) 2021 Storj Labs, Inc.
|
|
// See LICENSE for copying information.
|
|
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
|
|
progressbar "github.com/cheggaaa/pb/v3"
|
|
"github.com/zeebo/clingy"
|
|
"github.com/zeebo/errs"
|
|
)
|
|
|
|
type cmdCp struct {
|
|
projectProvider
|
|
|
|
recursive bool
|
|
dryrun bool
|
|
|
|
source Location
|
|
dest Location
|
|
}
|
|
|
|
func (c *cmdCp) Setup(a clingy.Arguments, f clingy.Flags) {
|
|
c.projectProvider.Setup(a, f)
|
|
|
|
c.recursive = f.New("recursive", "Peform a recursive copy", false,
|
|
clingy.Short('r'),
|
|
clingy.Transform(strconv.ParseBool),
|
|
).(bool)
|
|
c.dryrun = f.New("dryrun", "Print what operations would happen but don't execute them", false,
|
|
clingy.Transform(strconv.ParseBool),
|
|
).(bool)
|
|
|
|
c.source = a.New("source", "Source to copy", clingy.Transform(parseLocation)).(Location)
|
|
c.dest = a.New("dest", "Desination to copy", clingy.Transform(parseLocation)).(Location)
|
|
}
|
|
|
|
func (c *cmdCp) Execute(ctx clingy.Context) error {
|
|
fs, err := c.OpenFilesystem(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() { _ = fs.Close() }()
|
|
|
|
if c.recursive {
|
|
return c.copyRecursive(ctx, fs)
|
|
}
|
|
return c.copyFile(ctx, fs, c.source, c.dest, true)
|
|
}
|
|
|
|
func (c *cmdCp) copyRecursive(ctx clingy.Context, fs filesystem) error {
|
|
if c.source.Std() || c.dest.Std() {
|
|
return errs.New("cannot recursively copy to stdin/stdout")
|
|
}
|
|
|
|
var anyFailed bool
|
|
|
|
iter, err := fs.ListObjects(ctx, c.source, true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for iter.Next() {
|
|
rel, err := c.source.RelativeTo(iter.Item().Loc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
source := iter.Item().Loc
|
|
dest := c.dest.AppendKey(rel)
|
|
|
|
if err := c.copyFile(ctx, fs, source, dest, false); err != nil {
|
|
fmt.Fprintln(ctx.Stderr(), copyVerb(source, dest), "failed:", err.Error())
|
|
anyFailed = true
|
|
}
|
|
}
|
|
|
|
if err := iter.Err(); err != nil {
|
|
return errs.Wrap(err)
|
|
} else if anyFailed {
|
|
return errs.New("some downloads failed")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *cmdCp) copyFile(ctx clingy.Context, fs filesystem, source, dest Location, progress bool) error {
|
|
if isDir := fs.IsLocalDir(ctx, dest); isDir {
|
|
base, ok := source.Base()
|
|
if !ok {
|
|
return errs.New("destination is a directory and cannot find base name for %q", source)
|
|
}
|
|
dest = dest.AppendKey(base)
|
|
}
|
|
|
|
if !source.Std() && !dest.Std() {
|
|
fmt.Println(copyVerb(source, dest), source, "to", dest)
|
|
}
|
|
|
|
if c.dryrun {
|
|
return nil
|
|
}
|
|
|
|
rh, err := fs.Open(ctx, source)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() { _ = rh.Close() }()
|
|
|
|
wh, err := fs.Create(ctx, dest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer func() { _ = wh.Abort() }()
|
|
|
|
var bar *progressbar.ProgressBar
|
|
var writer io.Writer = wh
|
|
|
|
if length := rh.Info().ContentLength; progress && length >= 0 && !c.dest.Std() {
|
|
bar = progressbar.New64(length).SetWriter(ctx.Stdout())
|
|
writer = bar.NewProxyWriter(writer)
|
|
bar.Start()
|
|
defer bar.Finish()
|
|
}
|
|
|
|
if _, err := io.Copy(writer, rh); err != nil {
|
|
return errs.Combine(err, wh.Abort())
|
|
}
|
|
return errs.Wrap(wh.Commit())
|
|
}
|
|
|
|
func copyVerb(source, dest Location) string {
|
|
switch {
|
|
case dest.remote:
|
|
return "upload"
|
|
case source.remote:
|
|
return "download"
|
|
default:
|
|
return "copy"
|
|
}
|
|
}
|