reader.go 1.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859
  1. package io // import "github.com/v2ray/v2ray-core/common/io"
  2. import (
  3. "io"
  4. "github.com/v2ray/v2ray-core/common/alloc"
  5. )
  6. // ReadFrom reads from a reader and put all content to a buffer.
  7. // If buffer is nil, ReadFrom creates a new normal buffer.
  8. func ReadFrom(reader io.Reader, buffer *alloc.Buffer) (*alloc.Buffer, error) {
  9. if buffer == nil {
  10. buffer = alloc.NewBuffer()
  11. }
  12. nBytes, err := reader.Read(buffer.Value)
  13. buffer.Slice(0, nBytes)
  14. return buffer, err
  15. }
  16. // Reader extends io.Reader with alloc.Buffer.
  17. type Reader interface {
  18. // Read reads content from underlying reader, and put it into an alloc.Buffer.
  19. Read() (*alloc.Buffer, error)
  20. }
  21. // AdaptiveReader is a Reader that adjusts its reading speed automatically.
  22. type AdaptiveReader struct {
  23. reader io.Reader
  24. allocate func() *alloc.Buffer
  25. isLarge bool
  26. }
  27. // NewAdaptiveReader creates a new AdaptiveReader.
  28. func NewAdaptiveReader(reader io.Reader) *AdaptiveReader {
  29. return &AdaptiveReader{
  30. reader: reader,
  31. allocate: alloc.NewBuffer,
  32. isLarge: false,
  33. }
  34. }
  35. // Read implements Reader.Read().
  36. func (this *AdaptiveReader) Read() (*alloc.Buffer, error) {
  37. buffer, err := ReadFrom(this.reader, this.allocate())
  38. if buffer.IsFull() && !this.isLarge {
  39. this.allocate = alloc.NewLargeBuffer
  40. this.isLarge = true
  41. } else if !buffer.IsFull() {
  42. this.allocate = alloc.NewBuffer
  43. this.isLarge = false
  44. }
  45. if err != nil {
  46. buffer.Release()
  47. return nil, err
  48. }
  49. return buffer, nil
  50. }