2015-04-12 20:34:20 +00:00
|
|
|
package goadb
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
|
2015-12-28 23:28:53 +00:00
|
|
|
"github.com/zach-klippenstein/goadb/util"
|
2015-04-12 20:34:20 +00:00
|
|
|
"github.com/zach-klippenstein/goadb/wire"
|
|
|
|
)
|
|
|
|
|
|
|
|
// syncFileReader wraps a SyncConn that has requested to receive a file.
|
|
|
|
type syncFileReader struct {
|
|
|
|
// Reader used to read data from the adb connection.
|
|
|
|
scanner wire.SyncScanner
|
|
|
|
|
|
|
|
// Reader for the current chunk only.
|
|
|
|
chunkReader io.Reader
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ io.ReadCloser = &syncFileReader{}
|
|
|
|
|
2015-12-28 23:28:53 +00:00
|
|
|
func newSyncFileReader(s wire.SyncScanner) (r io.ReadCloser, err error) {
|
|
|
|
r = &syncFileReader{
|
2015-04-12 20:34:20 +00:00
|
|
|
scanner: s,
|
|
|
|
}
|
2015-12-28 23:28:53 +00:00
|
|
|
|
|
|
|
// Read the header for the first chunk to consume any errors.
|
|
|
|
if _, err = r.Read([]byte{}); err != nil {
|
|
|
|
r.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return
|
2015-04-12 20:34:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *syncFileReader) Read(buf []byte) (n int, err error) {
|
|
|
|
if r.chunkReader == nil {
|
|
|
|
chunkReader, err := readNextChunk(r.scanner)
|
|
|
|
if err != nil {
|
|
|
|
// If this is EOF, we've read the last chunk.
|
|
|
|
// Either way, we want to pass it up to the caller.
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
r.chunkReader = chunkReader
|
|
|
|
}
|
|
|
|
|
2015-12-28 23:28:53 +00:00
|
|
|
if len(buf) == 0 {
|
|
|
|
// Read can be called with an empty buffer to read the next chunk and check for errors.
|
|
|
|
// However, net.Conn.Read seems to return EOF when given an empty buffer, so we need to
|
|
|
|
// handle that case ourselves.
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
2015-04-12 20:34:20 +00:00
|
|
|
n, err = r.chunkReader.Read(buf)
|
|
|
|
if err == io.EOF {
|
|
|
|
// End of current chunk, don't return an error, the next chunk will be
|
|
|
|
// read on the next call to this method.
|
|
|
|
r.chunkReader = nil
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *syncFileReader) Close() error {
|
|
|
|
return r.scanner.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// readNextChunk creates an io.LimitedReader for the next chunk of data,
|
|
|
|
// and returns io.EOF if the last chunk has been read.
|
|
|
|
func readNextChunk(r wire.SyncScanner) (io.Reader, error) {
|
2015-12-28 23:28:53 +00:00
|
|
|
status, err := r.ReadStatus("read-chunk")
|
2015-04-12 20:34:20 +00:00
|
|
|
if err != nil {
|
2015-12-28 23:28:53 +00:00
|
|
|
if wire.IsAdbServerErrorMatching(err, readFileNotFoundPredicate) {
|
|
|
|
return nil, util.Errorf(util.FileNoExistError, "no such file or directory")
|
|
|
|
}
|
2015-04-12 20:34:20 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2015-12-28 23:28:53 +00:00
|
|
|
switch status {
|
|
|
|
case wire.StatusSyncData:
|
2015-04-12 20:34:20 +00:00
|
|
|
return r.ReadBytes()
|
2015-12-28 23:28:53 +00:00
|
|
|
case wire.StatusSyncDone:
|
2015-04-12 20:34:20 +00:00
|
|
|
return nil, io.EOF
|
|
|
|
default:
|
2015-12-28 23:28:53 +00:00
|
|
|
return nil, fmt.Errorf("expected chunk id '%s' or '%s', but got '%s'",
|
|
|
|
wire.StatusSyncData, wire.StatusSyncDone, []byte(status))
|
2015-04-12 20:34:20 +00:00
|
|
|
}
|
|
|
|
}
|
2015-12-28 23:28:53 +00:00
|
|
|
|
|
|
|
// readFileNotFoundPredicate returns true if s is the adb server error message returned
|
|
|
|
// when trying to open a file that doesn't exist.
|
|
|
|
func readFileNotFoundPredicate(s string) bool {
|
|
|
|
return s == "No such file or directory"
|
|
|
|
}
|