X-Git-Url: https://git.r.bdr.sh/rbdr/mobius/blobdiff_plain/72dd37f1abb2b550aaaac48eac677403d5664797..8ff2b66d1583b38298dfa415588df89631023695:/hotline/file_path.go diff --git a/hotline/file_path.go b/hotline/file_path.go index ae96e52..46ef97c 100644 --- a/hotline/file_path.go +++ b/hotline/file_path.go @@ -1,54 +1,114 @@ package hotline import ( + "bufio" + "bytes" "encoding/binary" + "errors" + "io" + "path/filepath" "strings" ) -const pathSeparator = "/" // File path separator TODO: make configurable to support Windows - // FilePathItem represents the file or directory portion of a delimited file path (e.g. foo and bar in "/foo/bar") +// Example bytes: // 00 00 // 09 -// 73 75 62 66 6f 6c 64 65 72 // "subfolder" +// 73 75 62 66 6f 6c 64 65 72 "subfolder" type FilePathItem struct { Len byte Name []byte } -func NewFilePathItem(b []byte) FilePathItem { - return FilePathItem{ - Len: b[2], - Name: b[3:], +const fileItemMinLen = 3 + +// fileItemScanner implements bufio.SplitFunc for parsing incoming byte slices into complete tokens +func fileItemScanner(data []byte, _ bool) (advance int, token []byte, err error) { + if len(data) < fileItemMinLen { + return 0, nil, nil } + + advance = fileItemMinLen + int(data[2]) + return advance, data[0:advance], nil +} + +// Write implements the io.Writer interface for FilePathItem +func (fpi *FilePathItem) Write(b []byte) (n int, err error) { + fpi.Len = b[2] + fpi.Name = b[fileItemMinLen : fpi.Len+fileItemMinLen] + + return int(fpi.Len) + fileItemMinLen, nil } type FilePath struct { - ItemCount []byte + ItemCount [2]byte Items []FilePathItem } -func (fp *FilePath) UnmarshalBinary(b []byte) error { - fp.ItemCount = b[0:2] +func (fp *FilePath) Write(b []byte) (n int, err error) { + reader := bytes.NewReader(b) + err = binary.Read(reader, binary.BigEndian, &fp.ItemCount) + if err != nil && !errors.Is(err, io.EOF) { + return n, err + } + if errors.Is(err, io.EOF) { + return n, nil + } + + scanner := bufio.NewScanner(reader) + scanner.Split(fileItemScanner) - pathData := b[2:] - for i := uint16(0); i < fp.Len(); i++ { - segLen := pathData[2] - fp.Items = append(fp.Items, NewFilePathItem(pathData[:segLen+3])) - pathData = pathData[3+segLen:] + for i := 0; i < int(binary.BigEndian.Uint16(fp.ItemCount[:])); i++ { + var fpi FilePathItem + scanner.Scan() + if _, err := fpi.Write(scanner.Bytes()); err != nil { + return n, err + } + fp.Items = append(fp.Items, fpi) } - return nil + return n, nil +} + +// IsDropbox checks if a FilePath matches the special drop box folder type +func (fp *FilePath) IsDropbox() bool { + if fp.Len() == 0 { + return false + } + + return strings.Contains(strings.ToLower(string(fp.Items[fp.Len()-1].Name)), "drop box") +} + +func (fp *FilePath) IsUploadDir() bool { + if fp.Len() == 0 { + return false + } + + return strings.Contains(strings.ToLower(string(fp.Items[fp.Len()-1].Name)), "upload") } func (fp *FilePath) Len() uint16 { - return binary.BigEndian.Uint16(fp.ItemCount) + return binary.BigEndian.Uint16(fp.ItemCount[:]) } -func (fp *FilePath) String() string { - var out []string - for _, i := range fp.Items { - out = append(out, string(i.Name)) +func readPath(fileRoot string, filePath, fileName []byte) (fullPath string, err error) { + var fp FilePath + if filePath != nil { + if _, err = fp.Write(filePath); err != nil { + return "", err + } + } + + var subPath string + for _, pathItem := range fp.Items { + subPath = filepath.Join("/", subPath, string(pathItem.Name)) } - return strings.Join(out, pathSeparator) + + fullPath = filepath.Join( + fileRoot, + subPath, + filepath.Join("/", string(fileName)), + ) + + return fullPath, nil }