rclone cat: add --head, --tail, --offset, --count and --discard
Fixes #819
This commit is contained in:
parent
381b845307
commit
d091d4a8bb
3 changed files with 100 additions and 14 deletions
|
@ -1,6 +1,9 @@
|
||||||
package cat
|
package cat
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/ncw/rclone/cmd"
|
"github.com/ncw/rclone/cmd"
|
||||||
|
@ -8,8 +11,22 @@ import (
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Globals
|
||||||
|
var (
|
||||||
|
head = int64(0)
|
||||||
|
tail = int64(0)
|
||||||
|
offset = int64(0)
|
||||||
|
count = int64(-1)
|
||||||
|
discard = false
|
||||||
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cmd.Root.AddCommand(commandDefintion)
|
cmd.Root.AddCommand(commandDefintion)
|
||||||
|
commandDefintion.Flags().Int64VarP(&head, "head", "", head, "Only print the first N characters.")
|
||||||
|
commandDefintion.Flags().Int64VarP(&tail, "tail", "", tail, "Only print the last N characters.")
|
||||||
|
commandDefintion.Flags().Int64VarP(&offset, "offset", "", offset, "Start printing at offset N (or from end if -ve).")
|
||||||
|
commandDefintion.Flags().Int64VarP(&count, "count", "", count, "Only print N characters.")
|
||||||
|
commandDefintion.Flags().BoolVarP(&discard, "discard", "", discard, "Discard the output instead of printing.")
|
||||||
}
|
}
|
||||||
|
|
||||||
var commandDefintion = &cobra.Command{
|
var commandDefintion = &cobra.Command{
|
||||||
|
@ -29,12 +46,48 @@ Or like this to output any file in dir or subdirectories.
|
||||||
Or like this to output any .txt files in dir or subdirectories.
|
Or like this to output any .txt files in dir or subdirectories.
|
||||||
|
|
||||||
rclone --include "*.txt" cat remote:path/to/dir
|
rclone --include "*.txt" cat remote:path/to/dir
|
||||||
|
|
||||||
|
Use the --head flag to print characters only at the start, --tail for
|
||||||
|
the end and --offset and --count to print a section in the middle.
|
||||||
|
Note that if offset is negative it will count from the end, so
|
||||||
|
--offset -1 --count 1 is equivalent to --tail 1.
|
||||||
`,
|
`,
|
||||||
Run: func(command *cobra.Command, args []string) {
|
Run: func(command *cobra.Command, args []string) {
|
||||||
|
usedOffset := offset != 0 || count >= 0
|
||||||
|
usedHead := head > 0
|
||||||
|
usedTail := tail > 0
|
||||||
|
if usedHead && usedTail || usedHead && usedOffset || usedTail && usedOffset {
|
||||||
|
log.Fatalf("Can only use one of --head, --tail or --offset with --count")
|
||||||
|
}
|
||||||
|
if head > 0 {
|
||||||
|
offset = 0
|
||||||
|
count = head
|
||||||
|
}
|
||||||
|
if tail > 0 {
|
||||||
|
offset = -tail
|
||||||
|
count = -1
|
||||||
|
}
|
||||||
cmd.CheckArgs(1, 1, command, args)
|
cmd.CheckArgs(1, 1, command, args)
|
||||||
fsrc := cmd.NewFsSrc(args)
|
fsrc := cmd.NewFsSrc(args)
|
||||||
|
var w io.Writer = os.Stdout
|
||||||
|
if discard {
|
||||||
|
w = ioutil.Discard
|
||||||
|
}
|
||||||
cmd.Run(false, false, command, func() error {
|
cmd.Run(false, false, command, func() error {
|
||||||
return fs.Cat(fsrc, os.Stdout)
|
return fs.Cat(fsrc, w, offset, count)
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Try removing buffering to stop the transfer!!!
|
||||||
|
|
||||||
|
|
||||||
|
Transferred: 2.847 GBytes (2.555 MBytes/s)
|
||||||
|
Errors: 3
|
||||||
|
Checks: 0
|
||||||
|
Transferred: 1844
|
||||||
|
Elapsed time: 19m0.8s
|
||||||
|
Transferring:
|
||||||
|
* 2001test/rogers-wedding/0016_1~1.jpg: 74% done, 0 Bytes/s, ETA: -
|
||||||
|
*/
|
||||||
|
|
|
@ -5,6 +5,7 @@ package fs
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
"mime"
|
"mime"
|
||||||
"path"
|
"path"
|
||||||
|
@ -1236,7 +1237,14 @@ func CleanUp(f Fs) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cat any files to the io.Writer
|
// Cat any files to the io.Writer
|
||||||
func Cat(f Fs, w io.Writer) error {
|
//
|
||||||
|
// if offset == 0 it will be ignored
|
||||||
|
// if offset > 0 then the file will be seeked to that offset
|
||||||
|
// if offset < 0 then the file will be seeked that far from the end
|
||||||
|
//
|
||||||
|
// if count < 0 then it will be ignored
|
||||||
|
// if count >= 0 then only that many characters will be output
|
||||||
|
func Cat(f Fs, w io.Writer, offset, count int64) error {
|
||||||
var mu sync.Mutex
|
var mu sync.Mutex
|
||||||
return ListFn(f, func(o Object) {
|
return ListFn(f, func(o Object) {
|
||||||
var err error
|
var err error
|
||||||
|
@ -1244,14 +1252,24 @@ func Cat(f Fs, w io.Writer) error {
|
||||||
defer func() {
|
defer func() {
|
||||||
Stats.DoneTransferring(o.Remote(), err == nil)
|
Stats.DoneTransferring(o.Remote(), err == nil)
|
||||||
}()
|
}()
|
||||||
mu.Lock()
|
thisOffset := offset
|
||||||
defer mu.Unlock()
|
if thisOffset < 0 {
|
||||||
in, err := o.Open()
|
thisOffset += o.Size()
|
||||||
|
}
|
||||||
|
var options []OpenOption
|
||||||
|
if thisOffset > 0 {
|
||||||
|
options = append(options, &SeekOption{Offset: thisOffset})
|
||||||
|
}
|
||||||
|
in, err := o.Open(options...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
ErrorLog(o, "Failed to open: %v", err)
|
ErrorLog(o, "Failed to open: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
reader := in
|
||||||
|
if count >= 0 {
|
||||||
|
reader = ioutil.NopCloser(&io.LimitedReader{R: in, N: count})
|
||||||
|
}
|
||||||
defer func() {
|
defer func() {
|
||||||
err = in.Close()
|
err = in.Close()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -1259,7 +1277,10 @@ func Cat(f Fs, w io.Writer) error {
|
||||||
ErrorLog(o, "Failed to close: %v", err)
|
ErrorLog(o, "Failed to close: %v", err)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
inAccounted := NewAccountWithBuffer(in, o) // account and buffer the transfer
|
inAccounted := NewAccountWithBuffer(reader, o) // account and buffer the transfer
|
||||||
|
// take the lock just before we output stuff, so at the last possible moment
|
||||||
|
mu.Lock()
|
||||||
|
defer mu.Unlock()
|
||||||
_, err = io.Copy(w, inAccounted)
|
_, err = io.Copy(w, inAccounted)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
Stats.Error()
|
Stats.Error()
|
||||||
|
|
|
@ -667,18 +667,30 @@ func TestDeduplicateRename(t *testing.T) {
|
||||||
func TestCat(t *testing.T) {
|
func TestCat(t *testing.T) {
|
||||||
r := NewRun(t)
|
r := NewRun(t)
|
||||||
defer r.Finalise()
|
defer r.Finalise()
|
||||||
file1 := r.WriteBoth("file1", "aaa", t1)
|
file1 := r.WriteBoth("file1", "ABCDEFGHIJ", t1)
|
||||||
file2 := r.WriteBoth("file2", "bbb", t2)
|
file2 := r.WriteBoth("file2", "012345678", t2)
|
||||||
|
|
||||||
fstest.CheckItems(t, r.fremote, file1, file2)
|
fstest.CheckItems(t, r.fremote, file1, file2)
|
||||||
|
|
||||||
var buf bytes.Buffer
|
for _, test := range []struct {
|
||||||
err := fs.Cat(r.fremote, &buf)
|
offset int64
|
||||||
require.NoError(t, err)
|
count int64
|
||||||
res := buf.String()
|
a string
|
||||||
|
b string
|
||||||
|
}{
|
||||||
|
{0, -1, "ABCDEFGHIJ", "012345678"},
|
||||||
|
{0, 5, "ABCDE", "01234"},
|
||||||
|
{-3, -1, "HIJ", "678"},
|
||||||
|
{1, 3, "BCD", "123"},
|
||||||
|
} {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err := fs.Cat(r.fremote, &buf, test.offset, test.count)
|
||||||
|
require.NoError(t, err)
|
||||||
|
res := buf.String()
|
||||||
|
|
||||||
if res != "aaabbb" && res != "bbbaaa" {
|
if res != test.a+test.b && res != test.b+test.a {
|
||||||
t.Errorf("Incorrect output from Cat: %q", res)
|
t.Errorf("Incorrect output from Cat(%d,%d): %q", test.offset, test.count, res)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue