reader.go 1.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253
  1. package io
  2. import (
  3. "hash/fnv"
  4. "io"
  5. "github.com/v2ray/v2ray-core/common/alloc"
  6. "github.com/v2ray/v2ray-core/common/serial"
  7. "github.com/v2ray/v2ray-core/transport"
  8. )
  9. type AuthChunkReader struct {
  10. reader io.Reader
  11. }
  12. func NewAuthChunkReader(reader io.Reader) *AuthChunkReader {
  13. return &AuthChunkReader{
  14. reader: reader,
  15. }
  16. }
  17. func (this *AuthChunkReader) Read() (*alloc.Buffer, error) {
  18. buffer := alloc.NewBuffer()
  19. if _, err := io.ReadFull(this.reader, buffer.Value[:2]); err != nil {
  20. buffer.Release()
  21. return nil, err
  22. }
  23. length := serial.BytesLiteral(buffer.Value[:2]).Uint16Value()
  24. if length == 4 { // Length of authentication bytes.
  25. return nil, io.EOF
  26. }
  27. if _, err := io.ReadFull(this.reader, buffer.Value[:length]); err != nil {
  28. buffer.Release()
  29. return nil, err
  30. }
  31. buffer.Slice(0, int(length))
  32. fnvHash := fnv.New32a()
  33. fnvHash.Write(buffer.Value[4:])
  34. expAuth := serial.BytesLiteral(fnvHash.Sum(nil))
  35. actualAuth := serial.BytesLiteral(buffer.Value[:4])
  36. if !actualAuth.Equals(expAuth) {
  37. buffer.Release()
  38. return nil, transport.ErrorCorruptedPacket
  39. }
  40. buffer.SliceFrom(4)
  41. return buffer, nil
  42. }
  43. func (this *AuthChunkReader) Release() {
  44. this.reader = nil
  45. }