-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpacker.go
88 lines (78 loc) · 2.7 KB
/
packer.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
package easytcp
import (
"encoding/binary"
"fmt"
"github.com/weilinks/gotcp/message"
"github.com/spf13/cast"
"io"
)
//go:generate mockgen -destination internal/mock/packer_mock.go -package mock . Packer
// Packer is a generic interface to pack and unpack message packet.
type Packer interface {
// Pack packs Message into the packet to be written.
Pack(entry *message.Entry) ([]byte, error)
// Unpack unpacks the message packet from reader,
// returns the message.Entry, and error if error occurred.
Unpack(reader io.Reader) (*message.Entry, error)
}
var _ Packer = &DefaultPacker{}
// NewDefaultPacker create a *DefaultPacker with initial field value.
func NewDefaultPacker() *DefaultPacker {
return &DefaultPacker{
MaxDataSize: 1 << 10 << 10, // 1MB
}
}
// DefaultPacker is the default Packer used in session.
// Treats the packet with the format:
//
// dataSize(4)|id(4)|data(n)
//
// | segment | type | size | remark |
// | ---------- | ------ | ------- | ----------------------- |
// | `dataSize` | uint32 | 4 | the size of `data` only |
// | `id` | uint32 | 4 | |
// | `data` | []byte | dynamic | |
// .
type DefaultPacker struct {
// MaxDataSize represents the max size of `data`
MaxDataSize int
}
func (d *DefaultPacker) bytesOrder() binary.ByteOrder {
return binary.LittleEndian
}
// Pack implements the Packer Pack method.
func (d *DefaultPacker) Pack(entry *message.Entry) ([]byte, error) {
dataSize := len(entry.Data)
buffer := make([]byte, 4+4+dataSize)
d.bytesOrder().PutUint32(buffer[:4], uint32(dataSize)) // write dataSize
id, err := cast.ToUint32E(entry.ID)
if err != nil {
return nil, fmt.Errorf("invalid type of entry.ID: %s", err)
}
d.bytesOrder().PutUint32(buffer[4:8], id) // write id
copy(buffer[8:], entry.Data) // write data
return buffer, nil
}
// Unpack implements the Packer Unpack method.
// Unpack returns the entry whose ID is type of int.
// So we need use int id to register routes.
func (d *DefaultPacker) Unpack(reader io.Reader) (*message.Entry, error) {
headerBuffer := make([]byte, 4+4)
if _, err := io.ReadFull(reader, headerBuffer); err != nil {
return nil, fmt.Errorf("read size and id err: %s", err)
}
dataSize := d.bytesOrder().Uint32(headerBuffer[:4])
if d.MaxDataSize > 0 && int(dataSize) > d.MaxDataSize {
return nil, fmt.Errorf("the dataSize %d is beyond the max: %d", dataSize, d.MaxDataSize)
}
id := d.bytesOrder().Uint32(headerBuffer[4:8])
data := make([]byte, dataSize)
if _, err := io.ReadFull(reader, data); err != nil {
return nil, fmt.Errorf("read data err: %s", err)
}
entry := &message.Entry{
ID: int(id),
Data: data,
}
return entry, nil
}