glider/proxy/vmess/chunk.go

105 lines
1.8 KiB
Go
Raw Normal View History

2018-07-07 11:07:38 +08:00
package vmess
2018-07-09 23:42:33 +08:00
import (
"bytes"
"encoding/binary"
"io"
)
2018-07-10 00:53:15 +08:00
const (
2018-07-11 00:26:05 +08:00
lenSize = 2
2018-07-10 00:53:15 +08:00
maxChunkSize = 1 << 14 // 16384
defaultChunkSize = 1 << 13 // 8192
)
2018-07-09 23:42:33 +08:00
type chunkedWriter struct {
io.Writer
buf []byte
}
2018-07-11 00:26:05 +08:00
// ChunkedWriter returns a chunked writer
func ChunkedWriter(w io.Writer) io.Writer {
2018-07-09 23:42:33 +08:00
return &chunkedWriter{
Writer: w,
2018-07-11 00:26:05 +08:00
buf: make([]byte, lenSize+maxChunkSize),
2018-07-09 23:42:33 +08:00
}
}
func (w *chunkedWriter) Write(b []byte) (int, error) {
n, err := w.ReadFrom(bytes.NewBuffer(b))
return int(n), err
}
func (w *chunkedWriter) ReadFrom(r io.Reader) (n int64, err error) {
for {
buf := w.buf
2018-07-11 00:26:05 +08:00
payloadBuf := buf[lenSize : lenSize+defaultChunkSize]
2018-07-09 23:42:33 +08:00
nr, er := r.Read(payloadBuf)
if nr > 0 {
n += int64(nr)
2018-07-11 00:26:05 +08:00
buf = buf[:lenSize+nr]
2018-07-09 23:42:33 +08:00
payloadBuf = payloadBuf[:nr]
2018-07-11 00:26:05 +08:00
binary.BigEndian.PutUint16(buf[:lenSize], uint16(nr))
2018-07-09 23:42:33 +08:00
_, ew := w.Writer.Write(buf)
if ew != nil {
err = ew
break
}
}
if er != nil {
if er != io.EOF { // ignore EOF as per io.ReaderFrom contract
err = er
}
break
}
}
return n, err
}
2018-07-11 00:26:05 +08:00
type chunkedReader struct {
io.Reader
2018-07-24 00:45:41 +08:00
buf []byte
leftBytes int
2018-07-11 00:26:05 +08:00
}
// ChunkedReader returns a chunked reader
func ChunkedReader(r io.Reader) io.Reader {
return &chunkedReader{
Reader: r,
2018-07-24 00:45:41 +08:00
buf: make([]byte, lenSize), // NOTE: buf only used to save header bytes now
2018-07-11 00:26:05 +08:00
}
}
func (r *chunkedReader) Read(b []byte) (int, error) {
2018-07-24 00:45:41 +08:00
if r.leftBytes == 0 {
// get length
_, err := io.ReadFull(r.Reader, r.buf[:lenSize])
if err != nil {
return 0, err
}
2018-07-25 08:23:58 +08:00
r.leftBytes = int(binary.BigEndian.Uint16(r.buf[:lenSize]))
2018-07-11 00:26:05 +08:00
2018-07-24 00:45:41 +08:00
// if length == 0, then this is the end
if r.leftBytes == 0 {
return 0, nil
}
2018-07-11 00:26:05 +08:00
}
2018-07-24 00:45:41 +08:00
readLen := len(b)
if readLen > r.leftBytes {
readLen = r.leftBytes
2018-07-11 00:26:05 +08:00
}
2018-07-24 00:45:41 +08:00
m, err := r.Reader.Read(b[:readLen])
2018-07-11 00:26:05 +08:00
if err != nil {
return 0, err
}
2018-07-24 00:45:41 +08:00
r.leftBytes -= m
2018-07-11 00:26:05 +08:00
return m, err
}