X-Git-Url: https://git.r.bdr.sh/rbdr/mobius/blobdiff_plain/ba29c43bb23de83c7715271e0830cb9f00e9e1c1..c2f62e9d34248432ee31144378ba9c6900ac974a:/hotline/file_transfer.go diff --git a/hotline/file_transfer.go b/hotline/file_transfer.go index 4a93390..ba4a1e1 100644 --- a/hotline/file_transfer.go +++ b/hotline/file_transfer.go @@ -3,7 +3,10 @@ package hotline import ( "encoding/binary" "fmt" - "strings" + "math" + "math/rand" + "path/filepath" + "sync" ) // File transfer types @@ -12,26 +15,90 @@ const ( FileUpload = 1 FolderDownload = 2 FolderUpload = 3 + bannerDownload = 4 ) type FileTransfer struct { - FileName []byte - FilePath []byte - ReferenceNumber []byte - Type int - TransferSize []byte // total size of all items in the folder. Only used in FolderUpload action - FolderItemCount []byte - BytesSent int - clientID uint16 - fileResumeData *FileResumeData - options []byte + FileName []byte + FilePath []byte + ReferenceNumber []byte + refNum [4]byte + Type int + TransferSize []byte + FolderItemCount []byte + fileResumeData *FileResumeData + options []byte + bytesSentCounter *WriteCounter + ClientConn *ClientConn } +// WriteCounter counts the number of bytes written to it. +type WriteCounter struct { + mux sync.Mutex + Total int64 // Total # of bytes written +} + +// Write implements the io.Writer interface. +// +// Always completes and never returns an error. +func (wc *WriteCounter) Write(p []byte) (int, error) { + wc.mux.Lock() + defer wc.mux.Unlock() + n := len(p) + wc.Total += int64(n) + return n, nil +} + +func (cc *ClientConn) newFileTransfer(transferType int, fileName, filePath, size []byte) *FileTransfer { + var transactionRef [4]byte + rand.Read(transactionRef[:]) + + ft := &FileTransfer{ + FileName: fileName, + FilePath: filePath, + ReferenceNumber: transactionRef[:], + refNum: transactionRef, + Type: transferType, + TransferSize: size, + ClientConn: cc, + bytesSentCounter: &WriteCounter{}, + } + + cc.transfersMU.Lock() + defer cc.transfersMU.Unlock() + cc.transfers[transferType][transactionRef] = ft + + cc.Server.mux.Lock() + defer cc.Server.mux.Unlock() + cc.Server.fileTransfers[transactionRef] = ft + + return ft +} + +// String returns a string representation of a file transfer and its progress for display in the GetInfo window +// Example: +// MasterOfOrionII1.4.0. 0% 197.9M func (ft *FileTransfer) String() string { - percentComplete := 10 - out := fmt.Sprintf("%s\t %v", ft.FileName, percentComplete) + trunc := fmt.Sprintf("%.21s", ft.FileName) + return fmt.Sprintf("%-21s %.3s%% %6s\n", trunc, ft.percentComplete(), ft.formattedTransferSize()) +} - return out +func (ft *FileTransfer) percentComplete() string { + ft.bytesSentCounter.mux.Lock() + defer ft.bytesSentCounter.mux.Unlock() + return fmt.Sprintf( + "%v", + math.RoundToEven(float64(ft.bytesSentCounter.Total)/float64(binary.BigEndian.Uint32(ft.TransferSize))*100), + ) +} + +func (ft *FileTransfer) formattedTransferSize() string { + sizeInKB := float32(binary.BigEndian.Uint32(ft.TransferSize)) / 1024 + if sizeInKB > 1024 { + return fmt.Sprintf("%.1fM", sizeInKB/1024) + } else { + return fmt.Sprintf("%.0fK", sizeInKB) + } } func (ft *FileTransfer) ItemCount() int { @@ -51,11 +118,12 @@ func (fu *folderUpload) FormattedPath() string { var pathSegments []string pathData := fu.FileNamePath + // TODO: implement scanner interface instead? for i := uint16(0); i < pathItemLen; i++ { segLen := pathData[2] pathSegments = append(pathSegments, string(pathData[3:3+segLen])) pathData = pathData[3+segLen:] } - return strings.Join(pathSegments, pathSeparator) + return filepath.Join(pathSegments...) }