2018-07-07 11:07:38 +08:00
|
|
|
package vmess
|
|
|
|
|
2018-07-09 23:42:33 +08:00
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"io"
|
2020-10-26 21:43:56 +08:00
|
|
|
"net"
|
2018-07-09 23:42:33 +08:00
|
|
|
)
|
|
|
|
|
2018-07-10 00:53:15 +08:00
|
|
|
const (
|
2020-04-19 23:20:15 +08:00
|
|
|
lenSize = 2
|
2020-10-26 21:43:56 +08:00
|
|
|
chunkSize = 16 << 10
|
2018-07-10 00:53:15 +08:00
|
|
|
)
|
2018-07-09 23:42:33 +08:00
|
|
|
|
|
|
|
type chunkedWriter struct {
|
|
|
|
io.Writer
|
2020-10-26 21:43:56 +08:00
|
|
|
buf [lenSize]byte
|
2018-07-09 23:42:33 +08:00
|
|
|
}
|
|
|
|
|
2020-08-16 12:00:46 +08:00
|
|
|
// ChunkedWriter returns a chunked writer.
|
2018-07-11 00:26:05 +08:00
|
|
|
func ChunkedWriter(w io.Writer) io.Writer {
|
2020-04-19 23:20:15 +08:00
|
|
|
return &chunkedWriter{Writer: w}
|
2018-07-09 23:42:33 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
func (w *chunkedWriter) Write(p []byte) (n int, err error) {
|
|
|
|
var dataLen int
|
|
|
|
for left := len(p); left != 0; {
|
|
|
|
dataLen = left
|
|
|
|
if dataLen > chunkSize {
|
|
|
|
dataLen = chunkSize
|
2018-07-09 23:42:33 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
binary.BigEndian.PutUint16(w.buf[:], uint16(dataLen))
|
|
|
|
if _, err = (&net.Buffers{w.buf[:], p[n : n+dataLen]}).WriteTo(w.Writer); err != nil {
|
2018-07-09 23:42:33 +08:00
|
|
|
break
|
|
|
|
}
|
2020-04-19 23:20:15 +08:00
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
n += dataLen
|
|
|
|
left -= dataLen
|
2018-07-09 23:42:33 +08:00
|
|
|
}
|
2020-04-19 23:20:15 +08:00
|
|
|
return
|
2018-07-09 23:42:33 +08:00
|
|
|
}
|
2018-07-11 00:26:05 +08:00
|
|
|
|
|
|
|
type chunkedReader struct {
|
|
|
|
io.Reader
|
2020-08-25 22:14:08 +08:00
|
|
|
buf [lenSize]byte
|
2020-04-19 23:20:15 +08:00
|
|
|
left int
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-08-16 12:00:46 +08:00
|
|
|
// ChunkedReader returns a chunked reader.
|
2018-07-11 00:26:05 +08:00
|
|
|
func ChunkedReader(r io.Reader) io.Reader {
|
2020-04-19 23:20:15 +08:00
|
|
|
return &chunkedReader{Reader: r}
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
func (r *chunkedReader) Read(p []byte) (int, error) {
|
2020-04-19 23:20:15 +08:00
|
|
|
if r.left == 0 {
|
2018-07-24 00:45:41 +08:00
|
|
|
// get length
|
2020-08-25 22:14:08 +08:00
|
|
|
_, err := io.ReadFull(r.Reader, r.buf[:lenSize])
|
2018-07-24 00:45:41 +08:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-08-25 22:14:08 +08:00
|
|
|
r.left = int(binary.BigEndian.Uint16(r.buf[:lenSize]))
|
2018-07-11 00:26:05 +08:00
|
|
|
|
2020-04-19 23:20:15 +08:00
|
|
|
// if left == 0, then this is the end
|
|
|
|
if r.left == 0 {
|
2018-07-24 00:45:41 +08:00
|
|
|
return 0, nil
|
|
|
|
}
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
readLen := len(p)
|
2020-04-19 23:20:15 +08:00
|
|
|
if readLen > r.left {
|
|
|
|
readLen = r.left
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
n, err := r.Reader.Read(p[:readLen])
|
2018-07-11 00:26:05 +08:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2020-04-19 23:20:15 +08:00
|
|
|
r.left -= n
|
|
|
|
|
|
|
|
return n, err
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|