glider/proxy/vmess/aead.go

128 lines
2.3 KiB
Go
Raw Normal View History

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
cipher.AEAD
2020-10-26 21:43:56 +08:00
nonce [32]byte
2018-07-11 00:26:05 +08:00
count uint16
}
// AEADWriter returns a aead writer.
2018-07-11 00:26:05 +08:00
func AEADWriter(w io.Writer, aead cipher.AEAD, iv []byte) io.Writer {
2020-10-26 21:43:56 +08:00
aw := &aeadWriter{Writer: w, AEAD: aead}
copy(aw.nonce[2:], iv[2:12])
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)
var lenBuf [lenSize]byte
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()
binary.BigEndian.PutUint16(lenBuf[:], uint16(writeLen))
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
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
}
// AEADReader returns a aead reader.
2018-07-11 00:26:05 +08:00
func AEADReader(r io.Reader, aead cipher.AEAD, iv []byte) io.Reader {
2020-10-26 21:43:56 +08:00
ar := &aeadReader{Reader: r, AEAD: aead}
copy(ar.nonce[2:], iv[2:12])
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) {
if _, err := io.ReadFull(r.Reader, p[:lenSize]); err != nil {
return 0, err
2018-07-11 00:26:05 +08:00
}
2020-10-26 21:43:56 +08:00
size := int(binary.BigEndian.Uint16(p[:lenSize]))
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)
_, err := r.Open(p[:0], r.nonce[:r.NonceSize()], p, nil)
r.count++
2018-07-11 00:26:05 +08:00
if err != nil {
return 0, err
}
2020-10-26 21:43:56 +08:00
return size - r.Overhead(), nil
}
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
}