2020-10-26 21:43:56 +08:00
|
|
|
// protocol:
|
|
|
|
// format: [data length] [data]
|
|
|
|
// sizes: 2 bytes, n bytes
|
|
|
|
// max(n): 2^14 bytes
|
|
|
|
// [data]: [encrypted payload] + [Overhead]
|
|
|
|
|
2018-07-11 00:26:05 +08:00
|
|
|
package vmess
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/cipher"
|
|
|
|
"encoding/binary"
|
|
|
|
"io"
|
2020-10-26 21:43:56 +08:00
|
|
|
"net"
|
|
|
|
|
|
|
|
"github.com/nadoo/glider/pool"
|
2018-07-11 00:26:05 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type aeadWriter struct {
|
|
|
|
io.Writer
|
2021-11-28 23:02:46 +08:00
|
|
|
chunkSizeEncoder ChunkSizeEncoder
|
2018-07-11 00:26:05 +08:00
|
|
|
cipher.AEAD
|
2020-10-26 21:43:56 +08:00
|
|
|
nonce [32]byte
|
2018-07-11 00:26:05 +08:00
|
|
|
count uint16
|
|
|
|
}
|
|
|
|
|
2020-08-16 12:00:46 +08:00
|
|
|
// AEADWriter returns a aead writer.
|
2021-11-28 23:02:46 +08:00
|
|
|
func AEADWriter(w io.Writer, aead cipher.AEAD, iv []byte, chunkSizeEncoder ChunkSizeEncoder) io.Writer {
|
|
|
|
aw := &aeadWriter{Writer: w, AEAD: aead, chunkSizeEncoder: chunkSizeEncoder}
|
|
|
|
copy(aw.nonce[2:], iv[2:aead.NonceSize()])
|
2020-10-26 21:43:56 +08:00
|
|
|
return aw
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
func (w *aeadWriter) Write(b []byte) (n int, err error) {
|
|
|
|
buf := pool.GetBuffer(chunkSize)
|
|
|
|
defer pool.PutBuffer(buf)
|
|
|
|
|
2021-11-28 23:02:46 +08:00
|
|
|
lenBuf := make([]byte, w.chunkSizeEncoder.SizeBytes())
|
2020-10-26 21:43:56 +08:00
|
|
|
var writeLen, dataLen int
|
2018-07-11 00:26:05 +08:00
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
nonce := w.nonce[:w.NonceSize()]
|
|
|
|
for left := len(b); left != 0; {
|
|
|
|
writeLen = left + w.Overhead()
|
|
|
|
if writeLen > chunkSize {
|
|
|
|
writeLen = chunkSize
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
2020-10-26 21:43:56 +08:00
|
|
|
dataLen = writeLen - w.Overhead()
|
|
|
|
|
2021-11-28 23:02:46 +08:00
|
|
|
w.chunkSizeEncoder.Encode(uint16(writeLen), lenBuf)
|
2020-10-26 21:43:56 +08:00
|
|
|
binary.BigEndian.PutUint16(nonce[:2], w.count)
|
2018-07-11 00:26:05 +08:00
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
w.Seal(buf[:0], nonce, b[n:n+dataLen], nil)
|
|
|
|
w.count++
|
|
|
|
|
|
|
|
if _, err = (&net.Buffers{lenBuf[:], buf[:writeLen]}).WriteTo(w.Writer); err != nil {
|
2018-07-11 00:26:05 +08:00
|
|
|
break
|
|
|
|
}
|
2020-10-26 21:43:56 +08:00
|
|
|
|
|
|
|
n += dataLen
|
|
|
|
left -= dataLen
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
return
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type aeadReader struct {
|
|
|
|
io.Reader
|
2021-11-28 23:02:46 +08:00
|
|
|
chunkSizeDecoder ChunkSizeDecoder
|
2018-07-11 00:26:05 +08:00
|
|
|
cipher.AEAD
|
2020-10-26 21:43:56 +08:00
|
|
|
nonce [32]byte
|
|
|
|
count uint16
|
|
|
|
buf []byte
|
|
|
|
offset int
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-08-16 12:00:46 +08:00
|
|
|
// AEADReader returns a aead reader.
|
2021-11-28 23:02:46 +08:00
|
|
|
func AEADReader(r io.Reader, aead cipher.AEAD, iv []byte, chunkSizeDecoder ChunkSizeDecoder) io.Reader {
|
|
|
|
ar := &aeadReader{Reader: r, AEAD: aead, chunkSizeDecoder: chunkSizeDecoder}
|
|
|
|
copy(ar.nonce[2:], iv[2:aead.NonceSize()])
|
2020-10-26 21:43:56 +08:00
|
|
|
return ar
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
func (r *aeadReader) read(p []byte) (int, error) {
|
2021-11-28 23:02:46 +08:00
|
|
|
if _, err := io.ReadFull(r.Reader, p[:r.chunkSizeDecoder.SizeBytes()]); err != nil {
|
2020-10-26 21:43:56 +08:00
|
|
|
return 0, err
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2021-11-28 23:02:46 +08:00
|
|
|
size, err := r.chunkSizeDecoder.Decode(p[:r.chunkSizeDecoder.SizeBytes()])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2021-12-24 19:40:36 +08:00
|
|
|
|
2021-12-25 12:28:09 +08:00
|
|
|
if int(size) <= r.Overhead() || int(size) > len(p) {
|
2021-12-24 19:40:36 +08:00
|
|
|
return 0, io.EOF
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
p = p[:size]
|
|
|
|
if _, err := io.ReadFull(r.Reader, p); err != nil {
|
2018-07-11 00:26:05 +08:00
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
binary.BigEndian.PutUint16(r.nonce[:2], r.count)
|
2021-11-28 23:02:46 +08:00
|
|
|
_, err = r.Open(p[:0], r.nonce[:r.NonceSize()], p, nil)
|
2020-10-26 21:43:56 +08:00
|
|
|
r.count++
|
2018-07-11 00:26:05 +08:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2021-11-28 23:02:46 +08:00
|
|
|
return int(size) - r.Overhead(), nil
|
2020-10-26 21:43:56 +08:00
|
|
|
}
|
2018-07-11 00:26:05 +08:00
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
func (r *aeadReader) Read(p []byte) (int, error) {
|
|
|
|
if r.buf == nil {
|
|
|
|
if len(p) >= chunkSize {
|
|
|
|
return r.read(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := pool.GetBuffer(chunkSize)
|
|
|
|
n, err := r.read(buf)
|
|
|
|
if err != nil || n == 0 {
|
|
|
|
pool.PutBuffer(buf)
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
r.buf = buf[:n]
|
|
|
|
r.offset = 0
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
n := copy(p, r.buf[r.offset:])
|
|
|
|
r.offset += n
|
|
|
|
if r.offset == len(r.buf) {
|
|
|
|
pool.PutBuffer(r.buf)
|
|
|
|
r.buf = nil
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:43:56 +08:00
|
|
|
return n, nil
|
2018-07-11 00:26:05 +08:00
|
|
|
}
|