class AppendStream
Reads from multiple streams, one after the other.
This is a read-only stream decorator.
Hierarchy
- class \GuzzleHttp\Psr7\AppendStream implements \Psr\Http\Message\StreamInterface
Expanded class hierarchy of AppendStream
File
-
vendor/
guzzlehttp/ psr7/ src/ AppendStream.php, line 14
Namespace
GuzzleHttp\Psr7View source
final class AppendStream implements StreamInterface {
/** @var StreamInterface[] Streams being decorated */
private $streams = [];
/** @var bool */
private $seekable = true;
/** @var int */
private $current = 0;
/** @var int */
private $pos = 0;
/**
* @param StreamInterface[] $streams Streams to decorate. Each stream must
* be readable.
*/
public function __construct(array $streams = []) {
foreach ($streams as $stream) {
$this->addStream($stream);
}
}
public function __toString() : string {
try {
$this->rewind();
return $this->getContents();
} catch (\Throwable $e) {
if (\PHP_VERSION_ID >= 70400) {
throw $e;
}
trigger_error(sprintf('%s::__toString exception: %s', self::class, (string) $e), E_USER_ERROR);
return '';
}
}
/**
* Add a stream to the AppendStream
*
* @param StreamInterface $stream Stream to append. Must be readable.
*
* @throws \InvalidArgumentException if the stream is not readable
*/
public function addStream(StreamInterface $stream) : void {
if (!$stream->isReadable()) {
throw new \InvalidArgumentException('Each stream must be readable');
}
// The stream is only seekable if all streams are seekable
if (!$stream->isSeekable()) {
$this->seekable = false;
}
$this->streams[] = $stream;
}
public function getContents() : string {
return Utils::copyToString($this);
}
/**
* Closes each attached stream.
*/
public function close() : void {
$this->pos = $this->current = 0;
$this->seekable = true;
foreach ($this->streams as $stream) {
$stream->close();
}
$this->streams = [];
}
/**
* Detaches each attached stream.
*
* Returns null as it's not clear which underlying stream resource to return.
*/
public function detach() {
$this->pos = $this->current = 0;
$this->seekable = true;
foreach ($this->streams as $stream) {
$stream->detach();
}
$this->streams = [];
return null;
}
public function tell() : int {
return $this->pos;
}
/**
* Tries to calculate the size by adding the size of each stream.
*
* If any of the streams do not return a valid number, then the size of the
* append stream cannot be determined and null is returned.
*/
public function getSize() : ?int {
$size = 0;
foreach ($this->streams as $stream) {
$s = $stream->getSize();
if ($s === null) {
return null;
}
$size += $s;
}
return $size;
}
public function eof() : bool {
return !$this->streams || $this->current >= count($this->streams) - 1 && $this->streams[$this->current]
->eof();
}
public function rewind() : void {
$this->seek(0);
}
/**
* Attempts to seek to the given position. Only supports SEEK_SET.
*/
public function seek($offset, $whence = SEEK_SET) : void {
if (!$this->seekable) {
throw new \RuntimeException('This AppendStream is not seekable');
}
elseif ($whence !== SEEK_SET) {
throw new \RuntimeException('The AppendStream can only seek with SEEK_SET');
}
$this->pos = $this->current = 0;
// Rewind each stream
foreach ($this->streams as $i => $stream) {
try {
$stream->rewind();
} catch (\Exception $e) {
throw new \RuntimeException('Unable to seek stream ' . $i . ' of the AppendStream', 0, $e);
}
}
// Seek to the actual position by reading from each stream
while ($this->pos < $offset && !$this->eof()) {
$result = $this->read(min(8096, $offset - $this->pos));
if ($result === '') {
break;
}
}
}
/**
* Reads from all of the appended streams until the length is met or EOF.
*/
public function read($length) : string {
$buffer = '';
$total = count($this->streams) - 1;
$remaining = $length;
$progressToNext = false;
while ($remaining > 0) {
// Progress to the next stream if needed.
if ($progressToNext || $this->streams[$this->current]
->eof()) {
$progressToNext = false;
if ($this->current === $total) {
break;
}
++$this->current;
}
$result = $this->streams[$this->current]
->read($remaining);
if ($result === '') {
$progressToNext = true;
continue;
}
$buffer .= $result;
$remaining = $length - strlen($buffer);
}
$this->pos += strlen($buffer);
return $buffer;
}
public function isReadable() : bool {
return true;
}
public function isWritable() : bool {
return false;
}
public function isSeekable() : bool {
return $this->seekable;
}
public function write($string) : int {
throw new \RuntimeException('Cannot write to an AppendStream');
}
/**
* @return mixed
*/
public function getMetadata($key = null) {
return $key ? null : [];
}
}
Members
Title Sort descending | Modifiers | Object type | Summary | Overriden Title |
---|---|---|---|---|
AppendStream::$current | private | property | @var int | |
AppendStream::$pos | private | property | @var int | |
AppendStream::$seekable | private | property | @var bool | |
AppendStream::$streams | private | property | @var StreamInterface[] Streams being decorated | |
AppendStream::addStream | public | function | Add a stream to the AppendStream | |
AppendStream::close | public | function | Closes each attached stream. | Overrides StreamInterface::close |
AppendStream::detach | public | function | Detaches each attached stream. | Overrides StreamInterface::detach |
AppendStream::eof | public | function | Returns true if the stream is at the end of the stream. | Overrides StreamInterface::eof |
AppendStream::getContents | public | function | Returns the remaining contents in a string | Overrides StreamInterface::getContents |
AppendStream::getMetadata | public | function | Overrides StreamInterface::getMetadata | |
AppendStream::getSize | public | function | Tries to calculate the size by adding the size of each stream. | Overrides StreamInterface::getSize |
AppendStream::isReadable | public | function | Returns whether or not the stream is readable. | Overrides StreamInterface::isReadable |
AppendStream::isSeekable | public | function | Returns whether or not the stream is seekable. | Overrides StreamInterface::isSeekable |
AppendStream::isWritable | public | function | Returns whether or not the stream is writable. | Overrides StreamInterface::isWritable |
AppendStream::read | public | function | Reads from all of the appended streams until the length is met or EOF. | Overrides StreamInterface::read |
AppendStream::rewind | public | function | Seek to the beginning of the stream. | Overrides StreamInterface::rewind |
AppendStream::seek | public | function | Attempts to seek to the given position. Only supports SEEK_SET. | Overrides StreamInterface::seek |
AppendStream::tell | public | function | Returns the current position of the file read/write pointer | Overrides StreamInterface::tell |
AppendStream::write | public | function | Write data to the stream. | Overrides StreamInterface::write |
AppendStream::__construct | public | function | ||
AppendStream::__toString | public | function | Reads all data from the stream into a string, from the beginning to end. | Overrides StreamInterface::__toString |