v2ray-core/proxy/vmess/io/reader.go

117 lines
2.3 KiB
Go
Raw Normal View History

2016-02-01 11:22:29 +00:00
package io
import (
2016-11-19 13:38:13 +00:00
"errors"
2016-05-04 21:41:24 +00:00
"hash"
2016-02-01 11:22:29 +00:00
"hash/fnv"
"io"
2016-08-20 18:55:45 +00:00
"v2ray.com/core/common/alloc"
"v2ray.com/core/common/serial"
2016-02-01 11:22:29 +00:00
)
2016-08-24 09:17:42 +00:00
// Private: Visible for testing.
2016-05-04 21:41:24 +00:00
type Validator struct {
actualAuth hash.Hash32
expectedAuth uint32
}
func NewValidator(expectedAuth uint32) *Validator {
return &Validator{
actualAuth: fnv.New32a(),
expectedAuth: expectedAuth,
}
}
2016-11-27 20:39:09 +00:00
func (v *Validator) Consume(b []byte) {
v.actualAuth.Write(b)
2016-05-04 21:41:24 +00:00
}
2016-11-27 20:39:09 +00:00
func (v *Validator) Validate() bool {
return v.actualAuth.Sum32() == v.expectedAuth
2016-05-04 21:41:24 +00:00
}
2016-02-01 11:22:29 +00:00
type AuthChunkReader struct {
2016-05-04 21:41:24 +00:00
reader io.Reader
last *alloc.Buffer
chunkLength int
validator *Validator
2016-02-01 11:22:29 +00:00
}
func NewAuthChunkReader(reader io.Reader) *AuthChunkReader {
return &AuthChunkReader{
2016-05-04 21:41:24 +00:00
reader: reader,
chunkLength: -1,
2016-02-01 11:22:29 +00:00
}
}
2016-11-27 20:39:09 +00:00
func (v *AuthChunkReader) Read() (*alloc.Buffer, error) {
2016-05-04 21:41:24 +00:00
var buffer *alloc.Buffer
2016-11-27 20:39:09 +00:00
if v.last != nil {
buffer = v.last
v.last = nil
2016-05-04 21:41:24 +00:00
} else {
2016-11-19 00:50:09 +00:00
buffer = alloc.NewBuffer().Clear()
2016-05-04 21:41:24 +00:00
}
2016-11-27 20:39:09 +00:00
if v.chunkLength == -1 {
2016-05-04 21:41:24 +00:00
for buffer.Len() < 6 {
2016-11-27 20:39:09 +00:00
_, err := buffer.FillFrom(v.reader)
2016-05-04 21:41:24 +00:00
if err != nil {
buffer.Release()
2016-06-02 19:20:58 +00:00
return nil, io.ErrUnexpectedEOF
2016-05-04 21:41:24 +00:00
}
}
2016-05-24 20:09:22 +00:00
length := serial.BytesToUint16(buffer.Value[:2])
2016-11-27 20:39:09 +00:00
v.chunkLength = int(length) - 4
v.validator = NewValidator(serial.BytesToUint32(buffer.Value[2:6]))
2016-05-04 21:41:24 +00:00
buffer.SliceFrom(6)
2016-11-27 20:39:09 +00:00
if buffer.Len() < v.chunkLength && v.chunkLength <= 2048 {
_, err := buffer.FillFrom(v.reader)
2016-07-03 20:32:28 +00:00
if err != nil {
buffer.Release()
return nil, io.ErrUnexpectedEOF
}
}
2016-11-27 20:39:09 +00:00
} else if buffer.Len() < v.chunkLength {
_, err := buffer.FillFrom(v.reader)
2016-05-04 22:24:18 +00:00
if err != nil {
buffer.Release()
2016-06-02 19:20:58 +00:00
return nil, io.ErrUnexpectedEOF
2016-05-04 22:24:18 +00:00
}
2016-04-28 19:14:00 +00:00
}
2016-05-04 21:41:24 +00:00
2016-11-27 20:39:09 +00:00
if v.chunkLength == 0 {
2016-02-01 11:22:29 +00:00
buffer.Release()
2016-06-02 00:20:53 +00:00
return nil, io.EOF
2016-02-01 11:22:29 +00:00
}
2016-11-27 20:39:09 +00:00
if buffer.Len() < v.chunkLength {
v.validator.Consume(buffer.Value)
v.chunkLength -= buffer.Len()
2016-05-04 21:41:24 +00:00
} else {
2016-11-27 20:39:09 +00:00
v.validator.Consume(buffer.Value[:v.chunkLength])
if !v.validator.Validate() {
2016-05-04 21:41:24 +00:00
buffer.Release()
2016-11-19 13:38:13 +00:00
return nil, errors.New("VMess|AuthChunkReader: Invalid auth.")
2016-05-04 21:41:24 +00:00
}
2016-11-27 20:39:09 +00:00
leftLength := buffer.Len() - v.chunkLength
2016-05-13 00:20:07 +00:00
if leftLength > 0 {
2016-11-27 20:39:09 +00:00
v.last = alloc.NewBuffer().Clear()
v.last.Append(buffer.Value[v.chunkLength:])
buffer.Slice(0, v.chunkLength)
2016-05-13 00:20:07 +00:00
}
2016-05-04 21:41:24 +00:00
2016-11-27 20:39:09 +00:00
v.chunkLength = -1
v.validator = nil
2016-02-01 11:22:29 +00:00
}
2016-05-04 21:41:24 +00:00
2016-02-01 11:22:29 +00:00
return buffer, nil
}
2016-03-11 22:51:58 +00:00
2016-11-27 20:39:09 +00:00
func (v *AuthChunkReader) Release() {
v.reader = nil
v.last.Release()
v.last = nil
v.validator = nil
2016-03-11 22:51:58 +00:00
}