notify/envelope.go

185 lines
3.6 KiB
Go
Raw Permalink Normal View History

package notify
import (
"b612.me/notify/internal/timeutil"
crand "crypto/rand"
"encoding/binary"
"errors"
"fmt"
"os"
"path/filepath"
"sync/atomic"
)
type EnvelopeKind uint8
const (
EnvelopeSignal EnvelopeKind = iota
EnvelopeSignalAck
EnvelopeStreamData
EnvelopeFileMeta
EnvelopeFileChunk
EnvelopeFileEnd
EnvelopeFileAbort
EnvelopeAck
)
type Envelope struct {
Kind EnvelopeKind
ID uint64
Body []byte
Stream StreamPacket
File FilePacket
}
type StreamPacket struct {
StreamID string
Chunk []byte
}
type FilePacket struct {
FileID string
Name string
Size int64
Mode uint32
ModTime int64
Offset int64
Chunk []byte
Checksum string
Error string
Stage string
}
func wrapTransferMsgEnvelope(msg TransferMsg, enFn func(interface{}) ([]byte, error)) (Envelope, error) {
body, err := enFn(msg)
if err != nil {
return Envelope{}, err
}
return Envelope{
Kind: EnvelopeSignal,
ID: msg.ID,
Body: body,
}, nil
}
func unwrapTransferMsgEnvelope(env Envelope, deFn func([]byte) (interface{}, error)) (TransferMsg, error) {
if env.Kind != EnvelopeSignal {
return TransferMsg{}, errors.New("envelope kind is not signal")
}
data, err := deFn(env.Body)
if err != nil {
return TransferMsg{}, err
}
msg, ok := data.(TransferMsg)
if !ok {
return TransferMsg{}, errors.New("invalid signal envelope payload")
}
return msg, nil
}
func newSignalAckEnvelope(signalID uint64) Envelope {
return Envelope{
Kind: EnvelopeSignalAck,
ID: signalID,
}
}
func newStreamDataEnvelope(streamID string, chunk []byte) Envelope {
return Envelope{
Kind: EnvelopeStreamData,
Stream: StreamPacket{
StreamID: streamID,
Chunk: chunk,
},
}
}
func newFileMetaEnvelope(fileID string, fileName string, fileSize int64, checksum string, mode uint32, modTime int64) Envelope {
return Envelope{
Kind: EnvelopeFileMeta,
File: FilePacket{
FileID: fileID,
Name: filepath.Base(fileName),
Size: fileSize,
Mode: mode,
ModTime: modTime,
Checksum: checksum,
},
}
}
func newFileChunkEnvelope(fileID string, offset int64, chunk []byte) Envelope {
return Envelope{
Kind: EnvelopeFileChunk,
File: FilePacket{
FileID: fileID,
Offset: offset,
Chunk: chunk,
},
}
}
func newFileEndEnvelope(fileID string) Envelope {
return Envelope{
Kind: EnvelopeFileEnd,
File: FilePacket{
FileID: fileID,
},
}
}
func newFileAbortEnvelope(fileID string, stage string, offset int64, errMsg string) Envelope {
return Envelope{
Kind: EnvelopeFileAbort,
File: FilePacket{
FileID: fileID,
Stage: stage,
Offset: offset,
Error: errMsg,
},
}
}
func newFileAckEnvelope(fileID string, stage string, offset int64, errMsg string) Envelope {
return Envelope{
Kind: EnvelopeAck,
File: FilePacket{
FileID: fileID,
Stage: stage,
Offset: offset,
Error: errMsg,
},
}
}
var fileIDSerial uint64
func buildFileID(fileName string) string {
base := fileIDBaseName(fileName)
ts := uint64(timeutil.NowUnixNano())
pid := uint64(os.Getpid())
seq := atomic.AddUint64(&fileIDSerial, 1)
rnd := uint64(randomFileIDSuffix())
return fmt.Sprintf("%s-%x-%x-%x-%x", base, ts, pid, seq, rnd)
}
func fileIDBaseName(fileName string) string {
base := sanitizeFileName(filepath.Base(fileName))
switch base {
case "", ".", "/", "\\":
return "unnamed"
default:
return base
}
}
func randomFileIDSuffix() uint32 {
var buf [4]byte
if _, err := crand.Read(buf[:]); err == nil {
return binary.BigEndian.Uint32(buf[:])
}
seq := atomic.LoadUint64(&fileIDSerial)
mix := uint64(timeutil.NowUnixNano()) ^ (seq << 1) ^ uint64(os.Getpid())
return uint32(mix ^ (mix >> 32))
}