2015-04-24 13:20:47 +00:00
|
|
|
package wire
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"time"
|
2015-07-12 06:18:58 +00:00
|
|
|
|
2020-07-13 06:31:19 +00:00
|
|
|
"github.com/Evrins/goadb/internal/errors"
|
2015-04-24 13:20:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type SyncSender interface {
|
2015-12-28 09:08:51 +00:00
|
|
|
io.Closer
|
|
|
|
|
2015-04-24 13:20:47 +00:00
|
|
|
// SendOctetString sends a 4-byte string.
|
|
|
|
SendOctetString(string) error
|
|
|
|
SendInt32(int32) error
|
|
|
|
SendFileMode(os.FileMode) error
|
|
|
|
SendTime(time.Time) error
|
|
|
|
|
2015-12-28 09:08:51 +00:00
|
|
|
// Sends len(data) as an octet, followed by the bytes.
|
|
|
|
// If data is bigger than SyncMaxChunkSize, it returns an assertion error.
|
|
|
|
SendBytes(data []byte) error
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type realSyncSender struct {
|
|
|
|
io.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewSyncSender(w io.Writer) SyncSender {
|
|
|
|
return &realSyncSender{w}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *realSyncSender) SendOctetString(str string) error {
|
|
|
|
if len(str) != 4 {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.AssertionErrorf("octet string must be exactly 4 bytes: '%s'", str)
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
2015-07-12 06:18:58 +00:00
|
|
|
|
2016-05-22 17:49:32 +00:00
|
|
|
wrappedErr := errors.WrapErrorf(writeFully(s.Writer, []byte(str)),
|
|
|
|
errors.NetworkError, "error sending octet string on sync sender")
|
2015-07-12 06:18:58 +00:00
|
|
|
|
|
|
|
return wrappedErr
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *realSyncSender) SendInt32(val int32) error {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.WrapErrorf(binary.Write(s.Writer, binary.LittleEndian, val),
|
|
|
|
errors.NetworkError, "error sending int on sync sender")
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *realSyncSender) SendFileMode(mode os.FileMode) error {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.WrapErrorf(binary.Write(s.Writer, binary.LittleEndian, mode),
|
|
|
|
errors.NetworkError, "error sending filemode on sync sender")
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *realSyncSender) SendTime(t time.Time) error {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.WrapErrorf(s.SendInt32(int32(t.Unix())),
|
|
|
|
errors.NetworkError, "error sending time on sync sender")
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
2015-12-28 09:08:51 +00:00
|
|
|
func (s *realSyncSender) SendBytes(data []byte) error {
|
|
|
|
length := len(data)
|
|
|
|
if length > SyncMaxChunkSize {
|
2015-04-24 13:20:47 +00:00
|
|
|
// This limit might not apply to filenames, but it's big enough
|
|
|
|
// that I don't think it will be a problem.
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.AssertionErrorf("data must be <= %d in length", SyncMaxChunkSize)
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.SendInt32(int32(length)); err != nil {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.WrapErrorf(err, errors.NetworkError, "error sending data length on sync sender")
|
2015-12-28 09:08:51 +00:00
|
|
|
}
|
2015-12-30 07:06:40 +00:00
|
|
|
return writeFully(s.Writer, data)
|
2015-12-28 09:08:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *realSyncSender) Close() error {
|
|
|
|
if closer, ok := s.Writer.(io.Closer); ok {
|
2016-05-22 17:49:32 +00:00
|
|
|
return errors.WrapErrorf(closer.Close(), errors.NetworkError, "error closing sync sender")
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|
2015-12-28 09:08:51 +00:00
|
|
|
return nil
|
2015-04-24 13:20:47 +00:00
|
|
|
}
|