transactions, err := handler.Handler(cc, &transaction)
if err != nil {
- return err
+ return fmt.Errorf("error handling transaction: %w", err)
}
for _, t := range transactions {
cc.Server.outbox <- t
Size [2]byte // Total size of FileHeader payload
Type [2]byte // 0 for file, 1 for dir
FilePath []byte // encoded file path
+
+ readOffset int // Internal offset to track read progress
}
func NewFileHeader(fileName string, isDir bool) FileHeader {
}
func (fh *FileHeader) Read(p []byte) (int, error) {
- return copy(p, slices.Concat(
+ buf := slices.Concat(
fh.Size[:],
fh.Type[:],
fh.FilePath,
- ),
- ), io.EOF
+ )
+
+ if fh.readOffset >= len(buf) {
+ return 0, io.EOF // All bytes have been read
+ }
+
+ n := copy(p, buf[fh.readOffset:])
+ fh.readOffset += n
+
+ return n, nil
}
_, err = io.Copy(&f.ffo.FlatFileInformationFork, bytes.NewReader(b))
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("error copying FlatFileInformationFork: %w", err)
}
} else {
import (
"encoding/binary"
"errors"
+ "fmt"
"io"
"io/fs"
"os"
func getFileNameList(path string, ignoreList []string) (fields []Field, err error) {
files, err := os.ReadDir(path)
if err != nil {
- return fields, nil
+ return fields, fmt.Errorf("error reading path: %s: %w", path, err)
}
- for _, file := range files {
+ for i, _ := range files {
+ file := files[i]
var fnwi FileNameWithInfo
if ignoreFile(file.Name(), ignoreList) {
fileInfo, err := file.Info()
if err != nil {
- return fields, err
+ return fields, fmt.Errorf("error getting file info: %s: %w", file.Name(), err)
}
// Check if path is a symlink. If so, follow it.
if fileInfo.Mode()&os.ModeSymlink != 0 {
resolvedPath, err := os.Readlink(filepath.Join(path, file.Name()))
if err != nil {
- return fields, err
+ return fields, fmt.Errorf("error following symlink: %s: %w", resolvedPath, err)
}
rFile, err := os.Stat(resolvedPath)
} else if file.IsDir() {
dir, err := os.ReadDir(filepath.Join(path, file.Name()))
if err != nil {
- return fields, err
+ return fields, fmt.Errorf("readDir: %w", err)
}
var c uint32
hlFile, err := newFileWrapper(&OSFileStore{}, path+"/"+file.Name(), 0)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("newFileWrapper: %w", err)
}
copy(fnwi.FileSize[:], hlFile.totalSize())
b, err := io.ReadAll(&fnwi)
if err != nil {
- return nil, err
+ return nil, fmt.Errorf("error io.ReadAll: %w", err)
}
fields = append(fields, NewField(FieldFileNameWithInfo, b))
}
FlatFileInformationFork FlatFileInformationFork
FlatFileDataForkHeader FlatFileForkHeader
FlatFileResForkHeader FlatFileForkHeader
+
+ readOffset int // Internal offset to track read progress
}
// FlatFileHeader is the first section of a "Flattened File Object". All fields have static values.
Name []byte // File name
CommentSize []byte // Length of the comment
Comment []byte // File comment
+
+ readOffset int // Internal offset to track read progress
}
func NewFlatFileInformationFork(fileName string, modifyTime []byte, typeSignature string, creatorSignature string) FlatFileInformationFork {
}
func (ffif *FlatFileInformationFork) Read(p []byte) (int, error) {
- return copy(p,
- slices.Concat(
- ffif.Platform,
- ffif.TypeSignature,
- ffif.CreatorSignature,
- ffif.Flags,
- ffif.PlatformFlags,
- ffif.RSVD,
- ffif.CreateDate,
- ffif.ModifyDate,
- ffif.NameScript,
- ffif.ReadNameSize(),
- ffif.Name,
- ffif.CommentSize,
- ffif.Comment,
- ),
- ), io.EOF
+ buf := slices.Concat(
+ ffif.Platform,
+ ffif.TypeSignature,
+ ffif.CreatorSignature,
+ ffif.Flags,
+ ffif.PlatformFlags,
+ ffif.RSVD,
+ ffif.CreateDate,
+ ffif.ModifyDate,
+ ffif.NameScript,
+ ffif.ReadNameSize(),
+ ffif.Name,
+ ffif.CommentSize,
+ ffif.Comment,
+ )
+
+ if ffif.readOffset >= len(buf) {
+ return 0, io.EOF // All bytes have been read
+ }
+
+ n := copy(p, buf[ffif.readOffset:])
+ ffif.readOffset += n
+
+ return n, nil
}
// Write implements the io.Writer interface for FlatFileInformationFork
if len(p) > int(total) {
ffif.CommentSize = p[total : total+2]
commentLen := binary.BigEndian.Uint16(ffif.CommentSize)
-
commentStartPos := int(total) + 2
commentEndPos := int(total) + 2 + int(commentLen)
total = uint16(commentEndPos)
}
- return int(total), nil
+ return len(p), nil
}
func (ffif *FlatFileInformationFork) UnmarshalBinary(b []byte) error {
// Read implements the io.Reader interface for flattenedFileObject
func (ffo *flattenedFileObject) Read(p []byte) (int, error) {
- return copy(p, slices.Concat(
+ buf := slices.Concat(
ffo.FlatFileHeader.Format[:],
ffo.FlatFileHeader.Version[:],
ffo.FlatFileHeader.RSVD[:],
ffo.FlatFileDataForkHeader.CompressionType[:],
ffo.FlatFileDataForkHeader.RSVD[:],
ffo.FlatFileDataForkHeader.DataSize[:],
- ),
- ), io.EOF
+ )
+
+ if ffo.readOffset >= len(buf) {
+ return 0, io.EOF // All bytes have been read
+ }
+
+ n := copy(p, buf[ffo.readOffset:])
+ ffo.readOffset += n
+
+ return n, nil
}
func (ffo *flattenedFileObject) ReadFrom(r io.Reader) (int64, error) {
PassID [4]byte // Random number generated by the server
Name string // Server Name
Description string // Description of the server
+
+ readOffset int // Internal offset to track read progress
}
// Read implements io.Reader to write tracker registration payload bytes to slice
userCount := make([]byte, 2)
binary.BigEndian.PutUint16(userCount, uint16(tr.UserCount))
- return copy(p, slices.Concat(
+ buf := slices.Concat(
[]byte{0x00, 0x01}, // Magic number, always 1
tr.Port[:],
userCount,
[]byte(tr.Name),
[]byte{uint8(len(tr.Description))},
[]byte(tr.Description),
- )[:]), io.EOF
+ )
+
+ if tr.readOffset >= len(buf) {
+ return 0, io.EOF // All bytes have been read
+ }
+
+ n := copy(p, buf[tr.readOffset:])
+ tr.readOffset += n
+
+ return n, nil
}
func register(tracker string, tr *TrackerRegistration) error {
nil,
)
if err != nil {
- return res, err
+ return res, fmt.Errorf("error reading file path: %w", err)
}
var fp FilePath
if t.GetField(FieldFilePath).Data != nil {
if _, err = fp.Write(t.GetField(FieldFilePath).Data); err != nil {
- return res, err
+ return res, fmt.Errorf("error writing file path: %w", err)
}
}
fileNames, err := getFileNameList(fullPath, cc.Server.Config.IgnoreFiles)
if err != nil {
- return res, err
+ return res, fmt.Errorf("getFileNameList: %w", err)
}
res = append(res, cc.NewReply(t, fileNames...))