forked from zhiqiangxu/qrpc
-
Notifications
You must be signed in to change notification settings - Fork 0
/
framereader.go
123 lines (101 loc) · 3.03 KB
/
framereader.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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package qrpc
import (
"context"
"encoding/binary"
"errors"
"net"
"unsafe"
"go.uber.org/zap"
)
var (
// ErrInvalidFrameSize when invalid size
ErrInvalidFrameSize = errors.New("invalid frame size")
// ErrFrameTooLarge when frame size too large
ErrFrameTooLarge = errors.New("frame size too large")
)
// defaultFrameReader is responsible for read frames
// should create one instance per connection
type defaultFrameReader struct {
*Reader
rbuf [16]byte // for header
ctx context.Context
maxFrameSize int
codec CompressorCodec
}
// newFrameReader creates a FrameWriter instance to read frames
func newFrameReader(ctx context.Context, rwc net.Conn, timeout int, codec CompressorCodec) *defaultFrameReader {
return newFrameReaderWithMFS(ctx, rwc, timeout, codec, 0)
}
func newFrameReaderWithMFS(ctx context.Context, rwc net.Conn, timeout int, codec CompressorCodec, maxFrameSize int) *defaultFrameReader {
return &defaultFrameReader{Reader: NewReaderWithTimeout(ctx, rwc, timeout), ctx: ctx, codec: codec, maxFrameSize: maxFrameSize}
}
// ReadFrame will only return the first frame in stream
func (dfr *defaultFrameReader) ReadFrame(cs *ConnStreams) (*Frame, error) {
start:
f, err := dfr.readFrame()
if err != nil {
return f, err
}
requestID := f.RequestID
flags := f.Flags
// ReadFrame is not threadsafe, so below need not be atomic
// handle Rst
if flags.IsRst() {
s := cs.GetStream(requestID, flags)
if s != nil {
s.ResetByPeer()
}
goto start
}
s, loaded := cs.CreateOrGetStream(dfr.ctx, requestID, flags)
if !loaded {
l.Debug("defaultFrameReader new stream", zap.Uintptr("cs", uintptr(unsafe.Pointer(cs))), zap.Uint64("requestID", requestID), zap.Uint8("flags", uint8(flags)), zap.Uint32("cmd", uint32(f.Cmd)))
}
if s.TryBind(f) {
return f, nil
}
ok := s.AddInFrame(f)
if !ok {
<-s.Done()
}
goto start
}
func (dfr *defaultFrameReader) readFrame() (*Frame, error) {
header := dfr.rbuf[:]
err := dfr.ReadBytes(header)
if err != nil {
return nil, err
}
size := binary.BigEndian.Uint32(header)
requestID := binary.BigEndian.Uint64(header[4:])
cmdAndFlags := binary.BigEndian.Uint32(header[12:])
cmd := Cmd(cmdAndFlags & 0xffffff)
flags := FrameFlag(cmdAndFlags >> 24)
l.Debug("readFrame", zap.Uint32("size", size), zap.Uint64("requestID", requestID), zap.Uint32("cmd", uint32(cmd)), zap.Uint8("flags", uint8(flags)))
if dfr.maxFrameSize > 0 && size > uint32(dfr.maxFrameSize) {
l.Error("ErrFrameTooLarge", zap.Uint32("size", size), zap.Uint32("cmd", uint32(cmd)), zap.Uint64("requestID", requestID))
return nil, ErrFrameTooLarge
}
if size < 12 {
return nil, ErrInvalidFrameSize
}
frame := &Frame{RequestID: requestID, Cmd: cmd, Flags: flags}
if size > 12 {
payload := make([]byte, size-12)
err = dfr.ReadBytes(payload)
if err != nil {
return nil, err
}
if flags.IsCodec() {
if dfr.codec == nil {
return nil, ErrNoCodec
}
payload, err = dfr.codec.Decode(payload)
if err != nil {
return nil, err
}
}
frame.Payload = payload
}
return frame, nil
}