mirror of
https://github.com/nadoo/glider.git
synced 2025-02-23 17:35:40 +08:00
136 lines
2.4 KiB
Go
136 lines
2.4 KiB
Go
![]() |
package vmess
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"crypto/cipher"
|
||
|
"encoding/binary"
|
||
|
"io"
|
||
|
)
|
||
|
|
||
|
type aeadWriter struct {
|
||
|
io.Writer
|
||
|
cipher.AEAD
|
||
|
nonce []byte
|
||
|
buf []byte
|
||
|
count uint16
|
||
|
iv []byte
|
||
|
}
|
||
|
|
||
|
// AEADWriter returns a aead writer
|
||
|
func AEADWriter(w io.Writer, aead cipher.AEAD, iv []byte) io.Writer {
|
||
|
return &aeadWriter{
|
||
|
Writer: w,
|
||
|
AEAD: aead,
|
||
|
buf: make([]byte, lenSize+maxChunkSize),
|
||
|
nonce: make([]byte, aead.NonceSize()),
|
||
|
count: 0,
|
||
|
iv: iv,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (w *aeadWriter) Write(b []byte) (int, error) {
|
||
|
n, err := w.ReadFrom(bytes.NewBuffer(b))
|
||
|
return int(n), err
|
||
|
}
|
||
|
|
||
|
func (w *aeadWriter) ReadFrom(r io.Reader) (n int64, err error) {
|
||
|
for {
|
||
|
buf := w.buf
|
||
|
payloadBuf := buf[lenSize : lenSize+defaultChunkSize]
|
||
|
|
||
|
nr, er := r.Read(payloadBuf)
|
||
|
if nr > 0 {
|
||
|
n += int64(nr)
|
||
|
buf = buf[:lenSize+nr+w.Overhead()]
|
||
|
payloadBuf = payloadBuf[:nr]
|
||
|
binary.BigEndian.PutUint16(buf[:lenSize], uint16(nr))
|
||
|
|
||
|
binary.BigEndian.PutUint16(w.nonce[:2], w.count)
|
||
|
copy(w.nonce[2:], w.iv[2:12])
|
||
|
|
||
|
w.Seal(payloadBuf[:0], w.nonce, payloadBuf, nil)
|
||
|
w.count++
|
||
|
|
||
|
_, 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
|
||
|
}
|
||
|
|
||
|
type aeadReader struct {
|
||
|
io.Reader
|
||
|
cipher.AEAD
|
||
|
nonce []byte
|
||
|
buf []byte
|
||
|
leftover []byte
|
||
|
count uint16
|
||
|
iv []byte
|
||
|
}
|
||
|
|
||
|
// AEADReader returns a aead reader
|
||
|
func AEADReader(r io.Reader, aead cipher.AEAD, iv []byte) io.Reader {
|
||
|
return &aeadReader{
|
||
|
Reader: r,
|
||
|
AEAD: aead,
|
||
|
buf: make([]byte, lenSize+maxChunkSize),
|
||
|
nonce: make([]byte, aead.NonceSize()),
|
||
|
count: 0,
|
||
|
iv: iv,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *aeadReader) Read(b []byte) (int, error) {
|
||
|
if len(r.leftover) > 0 {
|
||
|
n := copy(b, r.leftover)
|
||
|
r.leftover = r.leftover[n:]
|
||
|
return n, nil
|
||
|
}
|
||
|
|
||
|
// get length
|
||
|
_, err := io.ReadFull(r.Reader, r.buf[:lenSize])
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
// if length == 0, then this is the end
|
||
|
len := binary.BigEndian.Uint16(r.buf[:lenSize])
|
||
|
if len == 0 {
|
||
|
return 0, nil
|
||
|
}
|
||
|
|
||
|
// get payload
|
||
|
buf := r.buf[:len]
|
||
|
_, err = io.ReadFull(r.Reader, buf)
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
binary.BigEndian.PutUint16(r.nonce[:2], r.count)
|
||
|
copy(r.nonce[2:], r.iv[2:12])
|
||
|
|
||
|
_, err = r.Open(buf[:0], r.nonce, buf, nil)
|
||
|
r.count++
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
m := copy(b, r.buf[:len])
|
||
|
if m < int(len) {
|
||
|
r.leftover = r.buf[m:len]
|
||
|
}
|
||
|
|
||
|
return m, err
|
||
|
}
|