酒店预订平台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

171 lines
4.0 KiB

  1. <?php
  2. namespace GuzzleHttp\Psr7;
  3. use Psr\Http\Message\StreamInterface;
  4. /**
  5. * Provides a read only stream that pumps data from a PHP callable.
  6. *
  7. * When invoking the provided callable, the PumpStream will pass the amount of
  8. * data requested to read to the callable. The callable can choose to ignore
  9. * this value and return fewer or more bytes than requested. Any extra data
  10. * returned by the provided callable is buffered internally until drained using
  11. * the read() function of the PumpStream. The provided callable MUST return
  12. * false when there is no more data to read.
  13. *
  14. * @final
  15. */
  16. class PumpStream implements StreamInterface
  17. {
  18. /** @var callable */
  19. private $source;
  20. /** @var int */
  21. private $size;
  22. /** @var int */
  23. private $tellPos = 0;
  24. /** @var array */
  25. private $metadata;
  26. /** @var BufferStream */
  27. private $buffer;
  28. /**
  29. * @param callable $source Source of the stream data. The callable MAY
  30. * accept an integer argument used to control the
  31. * amount of data to return. The callable MUST
  32. * return a string when called, or false on error
  33. * or EOF.
  34. * @param array $options Stream options:
  35. * - metadata: Hash of metadata to use with stream.
  36. * - size: Size of the stream, if known.
  37. */
  38. public function __construct(callable $source, array $options = [])
  39. {
  40. $this->source = $source;
  41. $this->size = isset($options['size']) ? $options['size'] : null;
  42. $this->metadata = isset($options['metadata']) ? $options['metadata'] : [];
  43. $this->buffer = new BufferStream();
  44. }
  45. public function __toString()
  46. {
  47. try {
  48. return Utils::copyToString($this);
  49. } catch (\Exception $e) {
  50. return '';
  51. }
  52. }
  53. public function close()
  54. {
  55. $this->detach();
  56. }
  57. public function detach()
  58. {
  59. $this->tellPos = false;
  60. $this->source = null;
  61. return null;
  62. }
  63. public function getSize()
  64. {
  65. return $this->size;
  66. }
  67. public function tell()
  68. {
  69. return $this->tellPos;
  70. }
  71. public function eof()
  72. {
  73. return !$this->source;
  74. }
  75. public function isSeekable()
  76. {
  77. return false;
  78. }
  79. public function rewind()
  80. {
  81. $this->seek(0);
  82. }
  83. public function seek($offset, $whence = SEEK_SET)
  84. {
  85. throw new \RuntimeException('Cannot seek a PumpStream');
  86. }
  87. public function isWritable()
  88. {
  89. return false;
  90. }
  91. public function write($string)
  92. {
  93. throw new \RuntimeException('Cannot write to a PumpStream');
  94. }
  95. public function isReadable()
  96. {
  97. return true;
  98. }
  99. public function read($length)
  100. {
  101. $data = $this->buffer->read($length);
  102. $readLen = strlen($data);
  103. $this->tellPos += $readLen;
  104. $remaining = $length - $readLen;
  105. if ($remaining) {
  106. $this->pump($remaining);
  107. $data .= $this->buffer->read($remaining);
  108. $this->tellPos += strlen($data) - $readLen;
  109. }
  110. return $data;
  111. }
  112. public function getContents()
  113. {
  114. $result = '';
  115. while (!$this->eof()) {
  116. $result .= $this->read(1000000);
  117. }
  118. return $result;
  119. }
  120. public function getMetadata($key = null)
  121. {
  122. if (!$key) {
  123. return $this->metadata;
  124. }
  125. return isset($this->metadata[$key]) ? $this->metadata[$key] : null;
  126. }
  127. private function pump($length)
  128. {
  129. if ($this->source) {
  130. do {
  131. $data = call_user_func($this->source, $length);
  132. if ($data === false || $data === null) {
  133. $this->source = null;
  134. return;
  135. }
  136. $this->buffer->write($data);
  137. $length -= strlen($data);
  138. } while ($length > 0);
  139. }
  140. }
  141. }