| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657 |
- package io
- import (
- "hash/fnv"
- "io"
- "github.com/v2ray/v2ray-core/common/alloc"
- "github.com/v2ray/v2ray-core/common/serial"
- "github.com/v2ray/v2ray-core/transport"
- )
- type AuthChunkReader struct {
- reader io.Reader
- }
- func NewAuthChunkReader(reader io.Reader) *AuthChunkReader {
- return &AuthChunkReader{
- reader: reader,
- }
- }
- func (this *AuthChunkReader) Read() (*alloc.Buffer, error) {
- buffer := alloc.NewBuffer()
- if _, err := io.ReadFull(this.reader, buffer.Value[:2]); err != nil {
- buffer.Release()
- return nil, err
- }
- length := serial.BytesLiteral(buffer.Value[:2]).Uint16Value()
- if length <= 4 { // Length of authentication bytes.
- return nil, io.EOF
- }
- if length > 8*1024-16 {
- buffer.Release()
- buffer = alloc.NewLargeBuffer()
- }
- if _, err := io.ReadFull(this.reader, buffer.Value[:length]); err != nil {
- buffer.Release()
- return nil, err
- }
- buffer.Slice(0, int(length))
- fnvHash := fnv.New32a()
- fnvHash.Write(buffer.Value[4:])
- expAuth := serial.BytesLiteral(fnvHash.Sum(nil))
- actualAuth := serial.BytesLiteral(buffer.Value[:4])
- if !actualAuth.Equals(expAuth) {
- buffer.Release()
- return nil, transport.ErrorCorruptedPacket
- }
- buffer.SliceFrom(4)
- return buffer, nil
- }
- func (this *AuthChunkReader) Release() {
- this.reader = nil
- }
|