reader.go 1.5 KB

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