package hotline
import (
+ "bufio"
"bytes"
"encoding/binary"
"errors"
return len(b), nil
}
+const fileCopyBufSize = 524288 // 512k
func receiveFile(conn io.Reader, targetFile io.Writer, resForkFile io.Writer) error {
ffhBuf := make([]byte, 24)
- if _, err := conn.Read(ffhBuf); err != nil {
+ if _, err := io.ReadFull(conn, ffhBuf); err != nil {
return err
}
}
ffifhBuf := make([]byte, 16)
- if _, err := conn.Read(ffifhBuf); err != nil {
+ if _, err := io.ReadFull(conn, ffifhBuf); err != nil {
return err
}
dataLen := binary.BigEndian.Uint32(ffifh.DataSize[:])
ffifBuf := make([]byte, dataLen)
- if _, err := conn.Read(ffifBuf); err != nil {
+ if _, err := io.ReadFull(conn, ffifBuf); err != nil {
return err
}
if err := ffif.UnmarshalBinary(ffifBuf); err != nil {
var ffdfh FlatFileDataForkHeader
ffdfhBuf := make([]byte, 16)
- if _, err := conn.Read(ffdfhBuf); err != nil {
+ if _, err := io.ReadFull(conn, ffdfhBuf); err != nil {
return err
}
err = binary.Read(bytes.NewReader(ffdfhBuf), binary.BigEndian, &ffdfh)
// this will be zero if the file only has a resource fork
fileSize := int(binary.BigEndian.Uint32(ffdfh.DataSize[:]))
- // Copy N bytes from conn to upload file
- _, err = io.CopyN(targetFile, conn, int64(fileSize))
+ bw := bufio.NewWriterSize(targetFile, fileCopyBufSize)
+ _, err = io.CopyN(bw, conn, int64(fileSize))
if err != nil {
return err
}
+ if err := bw.Flush(); err != nil {
+ return err
+ }
if ffh.ForkCount == [2]byte{0, 3} {
var resForkHeader FlatFileDataForkHeader
- resForkBuf := make([]byte, 16)
+ if _, err := io.ReadFull(conn, resForkHeader.ForkType[:]); err != nil {
+ return err
+ }
- if _, err := conn.Read(resForkBuf); err != nil {
+ if _, err := io.ReadFull(conn, resForkHeader.CompressionType[:]); err != nil {
return err
}
- err = binary.Read(bytes.NewReader(resForkBuf), binary.BigEndian, &resForkHeader)
- if err != nil {
+
+ if _, err := io.ReadFull(conn, resForkHeader.RSVD[:]); err != nil {
return err
}
- fileSize = int(binary.BigEndian.Uint32(resForkHeader.DataSize[:]))
+ if _, err := io.ReadFull(conn, resForkHeader.DataSize[:]); err != nil {
+ return err
+ }
- _, err = io.CopyN(resForkFile, conn, int64(fileSize))
+ bw = bufio.NewWriterSize(resForkFile, fileCopyBufSize)
+ _, err = io.CopyN(resForkFile, conn, int64(binary.BigEndian.Uint32(resForkHeader.DataSize[:])))
if err != nil {
return err
}
+ if err := bw.Flush(); err != nil {
+ return err
+ }
}
return nil
}