X-Git-Url: https://git.r.bdr.sh/rbdr/mobius/blobdiff_plain/00d1ef67636df59460bd4e060f6da4b0c9bcb24c..c6a3fa2cdbf7eb78964a5df7622482cdd8782d74:/hotline/file_path.go?ds=sidebyside diff --git a/hotline/file_path.go b/hotline/file_path.go index d6c05bc..460af03 100644 --- a/hotline/file_path.go +++ b/hotline/file_path.go @@ -4,11 +4,11 @@ import ( "bytes" "encoding/binary" "errors" - "path" + "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") // 00 00 // 09 @@ -18,56 +18,82 @@ type FilePathItem struct { Name []byte } -func NewFilePathItem(b []byte) FilePathItem { - return FilePathItem{ - Len: b[2], - Name: b[3:], - } -} - type FilePath struct { ItemCount [2]byte Items []FilePathItem } -const minFilePathLen = 2 func (fp *FilePath) UnmarshalBinary(b []byte) error { - if len(b) < minFilePathLen { - return errors.New("insufficient bytes") - } - err := binary.Read(bytes.NewReader(b[0:2]), binary.BigEndian, &fp.ItemCount) - if err != nil { + reader := bytes.NewReader(b) + err := binary.Read(reader, binary.BigEndian, &fp.ItemCount) + if err != nil && !errors.Is(err, io.EOF) { return err } + if errors.Is(err, io.EOF) { + return nil + } - 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:] + // skip two bytes for the file path delimiter + _, _ = reader.Seek(2, io.SeekCurrent) + + // read the length of the next pathItem + segLen, err := reader.ReadByte() + if err != nil { + return err + } + + pBytes := make([]byte, segLen) + + _, err = reader.Read(pBytes) + if err != nil && !errors.Is(err, io.EOF) { + return err + } + + fp.Items = append(fp.Items, FilePathItem{Len: segLen, Name: pBytes}) } return nil } -func (fp *FilePath) Len() uint16 { - return binary.BigEndian.Uint16(fp.ItemCount[:]) +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) String() string { - out := []string{"/"} - for _, i := range fp.Items { - out = append(out, string(i.Name)) +func (fp *FilePath) IsUploadDir() bool { + if fp.Len() == 0 { + return false } - return path.Join(out...) + return strings.Contains(strings.ToLower(string(fp.Items[fp.Len()-1].Name)), "upload") } -func ReadFilePath(filePathFieldData []byte) string { +func (fp *FilePath) Len() uint16 { + return binary.BigEndian.Uint16(fp.ItemCount[:]) +} + +func readPath(fileRoot string, filePath, fileName []byte) (fullPath string, err error) { var fp FilePath - err := fp.UnmarshalBinary(filePathFieldData) - if err != nil { - // TODO + if filePath != nil { + if err = fp.UnmarshalBinary(filePath); err != nil { + return "", err + } + } + + var subPath string + for _, pathItem := range fp.Items { + subPath = filepath.Join("/", subPath, string(pathItem.Name)) } - return fp.String() + + fullPath = filepath.Join( + fileRoot, + subPath, + filepath.Join("/", string(fileName)), + ) + + return fullPath, nil }