reader.go 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869
  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. // Read reads content from underlying reader, and put it into an alloc.Buffer.
  20. Read() (*alloc.Buffer, error)
  21. }
  22. type ReleasableReader interface {
  23. Reader
  24. common.Releasable
  25. }
  26. // AdaptiveReader is a Reader that adjusts its reading speed automatically.
  27. type AdaptiveReader struct {
  28. reader io.Reader
  29. allocate func() *alloc.Buffer
  30. isLarge bool
  31. }
  32. // NewAdaptiveReader creates a new AdaptiveReader.
  33. func NewAdaptiveReader(reader io.Reader) *AdaptiveReader {
  34. return &AdaptiveReader{
  35. reader: reader,
  36. allocate: alloc.NewBuffer,
  37. isLarge: false,
  38. }
  39. }
  40. // Read implements Reader.Read().
  41. func (this *AdaptiveReader) Read() (*alloc.Buffer, error) {
  42. buffer, err := ReadFrom(this.reader, this.allocate())
  43. if buffer.IsFull() && !this.isLarge {
  44. this.allocate = alloc.NewLargeBuffer
  45. this.isLarge = true
  46. } else if !buffer.IsFull() {
  47. this.allocate = alloc.NewBuffer
  48. this.isLarge = false
  49. }
  50. if err != nil {
  51. buffer.Release()
  52. return nil, err
  53. }
  54. return buffer, nil
  55. }
  56. func (this *AdaptiveReader) Release() {
  57. this.reader = nil
  58. }