-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathstream_decoder.go
115 lines (96 loc) · 1.94 KB
/
stream_decoder.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
package y3
import (
"bytes"
"errors"
"fmt"
"io"
"github.com/yomorun/y3/encoding"
)
// StreamReader read an Y3 packet from a io.Reader, and return
// the ValReader after decode out Tag and Len
type StreamReader struct {
src io.Reader
// Tag of a y3 packet
Tag byte
// Len of a y3 packet
Len int
// Val of a y3 packet
Val io.Reader
}
// NewStreamReader create a new y3 StreamReader
func NewStreamParser(reader io.Reader) *StreamReader {
return &StreamReader{
src: reader,
}
}
func (sr *StreamReader) GetValBuffer() ([]byte, error) {
buf, err := io.ReadAll(sr.Val)
return buf, err
}
// Do must run in a goroutine
func (sr *StreamReader) Do() error {
if sr.src == nil {
return errors.New("y3: nil source reader")
}
tag, err := readByte(sr.src)
if err != nil {
return err
}
// the first byte is y3.Tag
sr.Tag = tag
// read y3.Length bytes, a varint format
lenbuf := bytes.Buffer{}
for {
b, err := readByte(sr.src)
if err != nil {
return err
}
lenbuf.WriteByte(b)
if b&0x80 != 0x80 {
break
}
}
// parse to y3.Length
var length int32
codec := encoding.VarCodec{}
err = codec.DecodePVarInt32(lenbuf.Bytes(), &length)
if err != nil {
return err
}
// validate len decoded from stream
if length < 0 {
return fmt.Errorf("y3: streamParse() get lenbuf=(%# x), decode len=(%v)", lenbuf.Bytes(), length)
}
sr.Len = int(length)
// read next {len} bytes as y3.Value
sr.Val = &valR{
length: int(length),
src: sr.src,
}
return nil
}
type valR struct {
length int
off int
src io.Reader
}
func (r *valR) Read(p []byte) (n int, err error) {
if r.src == nil {
return 0, nil
}
if r.off >= r.length {
return 0, io.EOF
}
bound := len(p)
if len(p) > r.length-r.off {
bound = r.length - r.off
}
// update readed
r.off, err = r.src.Read(p[0:bound])
return r.off, err
}
func StreamReadPacket(reader io.Reader) (*StreamReader, error) {
sp := NewStreamParser(reader)
err := sp.Do()
return sp, err
}