mirror of
https://github.com/rclone/rclone.git
synced 2024-12-22 23:22:08 +01:00
rclone cat: add --head, --tail, --offset, --count and --discard
Fixes #819
This commit is contained in:
parent
381b845307
commit
d091d4a8bb
@ -1,6 +1,9 @@
|
||||
package cat
|
||||
|
||||
import (
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"os"
|
||||
|
||||
"github.com/ncw/rclone/cmd"
|
||||
@ -8,8 +11,22 @@ import (
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
// Globals
|
||||
var (
|
||||
head = int64(0)
|
||||
tail = int64(0)
|
||||
offset = int64(0)
|
||||
count = int64(-1)
|
||||
discard = false
|
||||
)
|
||||
|
||||
func init() {
|
||||
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{
|
||||
@ -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.
|
||||
|
||||
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) {
|
||||
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)
|
||||
fsrc := cmd.NewFsSrc(args)
|
||||
var w io.Writer = os.Stdout
|
||||
if discard {
|
||||
w = ioutil.Discard
|
||||
}
|
||||
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 (
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"mime"
|
||||
"path"
|
||||
@ -1236,7 +1237,14 @@ func CleanUp(f Fs) error {
|
||||
}
|
||||
|
||||
// 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
|
||||
return ListFn(f, func(o Object) {
|
||||
var err error
|
||||
@ -1244,14 +1252,24 @@ func Cat(f Fs, w io.Writer) error {
|
||||
defer func() {
|
||||
Stats.DoneTransferring(o.Remote(), err == nil)
|
||||
}()
|
||||
mu.Lock()
|
||||
defer mu.Unlock()
|
||||
in, err := o.Open()
|
||||
thisOffset := offset
|
||||
if thisOffset < 0 {
|
||||
thisOffset += o.Size()
|
||||
}
|
||||
var options []OpenOption
|
||||
if thisOffset > 0 {
|
||||
options = append(options, &SeekOption{Offset: thisOffset})
|
||||
}
|
||||
in, err := o.Open(options...)
|
||||
if err != nil {
|
||||
Stats.Error()
|
||||
ErrorLog(o, "Failed to open: %v", err)
|
||||
return
|
||||
}
|
||||
reader := in
|
||||
if count >= 0 {
|
||||
reader = ioutil.NopCloser(&io.LimitedReader{R: in, N: count})
|
||||
}
|
||||
defer func() {
|
||||
err = in.Close()
|
||||
if err != nil {
|
||||
@ -1259,7 +1277,10 @@ func Cat(f Fs, w io.Writer) error {
|
||||
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)
|
||||
if err != nil {
|
||||
Stats.Error()
|
||||
|
@ -667,18 +667,30 @@ func TestDeduplicateRename(t *testing.T) {
|
||||
func TestCat(t *testing.T) {
|
||||
r := NewRun(t)
|
||||
defer r.Finalise()
|
||||
file1 := r.WriteBoth("file1", "aaa", t1)
|
||||
file2 := r.WriteBoth("file2", "bbb", t2)
|
||||
file1 := r.WriteBoth("file1", "ABCDEFGHIJ", t1)
|
||||
file2 := r.WriteBoth("file2", "012345678", t2)
|
||||
|
||||
fstest.CheckItems(t, r.fremote, file1, file2)
|
||||
|
||||
var buf bytes.Buffer
|
||||
err := fs.Cat(r.fremote, &buf)
|
||||
require.NoError(t, err)
|
||||
res := buf.String()
|
||||
for _, test := range []struct {
|
||||
offset int64
|
||||
count int64
|
||||
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" {
|
||||
t.Errorf("Incorrect output from Cat: %q", res)
|
||||
if res != test.a+test.b && res != test.b+test.a {
|
||||
t.Errorf("Incorrect output from Cat(%d,%d): %q", test.offset, test.count, res)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user