123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\HttpClient\Response;
- use Symfony\Component\HttpClient\Chunk\DataChunk;
- use Symfony\Component\HttpClient\Chunk\ErrorChunk;
- use Symfony\Component\HttpClient\Chunk\FirstChunk;
- use Symfony\Component\HttpClient\Chunk\LastChunk;
- use Symfony\Component\HttpClient\Exception\TransportException;
- use Symfony\Component\HttpClient\Internal\ClientState;
- /**
- * Implements common logic for transport-level response classes.
- *
- * @author Nicolas Grekas <p@tchwork.com>
- *
- * @internal
- */
- trait TransportResponseTrait
- {
- private $canary;
- private $headers = [];
- private $info = [
- 'response_headers' => [],
- 'http_code' => 0,
- 'error' => null,
- 'canceled' => false,
- ];
- /** @var object|resource */
- private $handle;
- private $id;
- private $timeout = 0;
- private $inflate;
- private $finalInfo;
- private $logger;
- /**
- * {@inheritdoc}
- */
- public function getStatusCode(): int
- {
- if ($this->initializer) {
- self::initialize($this);
- }
- return $this->info['http_code'];
- }
- /**
- * {@inheritdoc}
- */
- public function getHeaders(bool $throw = true): array
- {
- if ($this->initializer) {
- self::initialize($this);
- }
- if ($throw) {
- $this->checkStatusCode();
- }
- return $this->headers;
- }
- /**
- * {@inheritdoc}
- */
- public function cancel(): void
- {
- $this->info['canceled'] = true;
- $this->info['error'] = 'Response has been canceled.';
- $this->close();
- }
- /**
- * Closes the response and all its network handles.
- */
- protected function close(): void
- {
- $this->canary->cancel();
- $this->inflate = null;
- }
- /**
- * Adds pending responses to the activity list.
- */
- abstract protected static function schedule(self $response, array &$runningResponses): void;
- /**
- * Performs all pending non-blocking operations.
- */
- abstract protected static function perform(ClientState $multi, array &$responses): void;
- /**
- * Waits for network activity.
- */
- abstract protected static function select(ClientState $multi, float $timeout): int;
- private static function addResponseHeaders(array $responseHeaders, array &$info, array &$headers, string &$debug = ''): void
- {
- foreach ($responseHeaders as $h) {
- if (11 <= \strlen($h) && '/' === $h[4] && preg_match('#^HTTP/\d+(?:\.\d+)? (\d\d\d)(?: |$)#', $h, $m)) {
- if ($headers) {
- $debug .= "< \r\n";
- $headers = [];
- }
- $info['http_code'] = (int) $m[1];
- } elseif (2 === \count($m = explode(':', $h, 2))) {
- $headers[strtolower($m[0])][] = ltrim($m[1]);
- }
- $debug .= "< {$h}\r\n";
- $info['response_headers'][] = $h;
- }
- $debug .= "< \r\n";
- }
- /**
- * Ensures the request is always sent and that the response code was checked.
- */
- private function doDestruct()
- {
- $this->shouldBuffer = true;
- if ($this->initializer && null === $this->info['error']) {
- self::initialize($this);
- $this->checkStatusCode();
- }
- }
- /**
- * Implements an event loop based on a buffer activity queue.
- *
- * @param iterable<array-key, self> $responses
- *
- * @internal
- */
- public static function stream(iterable $responses, float $timeout = null): \Generator
- {
- $runningResponses = [];
- foreach ($responses as $response) {
- self::schedule($response, $runningResponses);
- }
- $lastActivity = microtime(true);
- $elapsedTimeout = 0;
- if ($fromLastTimeout = 0.0 === $timeout && '-0' === (string) $timeout) {
- $timeout = null;
- } elseif ($fromLastTimeout = 0 > $timeout) {
- $timeout = -$timeout;
- }
- while (true) {
- $hasActivity = false;
- $timeoutMax = 0;
- $timeoutMin = $timeout ?? \INF;
- /** @var ClientState $multi */
- foreach ($runningResponses as $i => [$multi]) {
- $responses = &$runningResponses[$i][1];
- self::perform($multi, $responses);
- foreach ($responses as $j => $response) {
- $timeoutMax = $timeout ?? max($timeoutMax, $response->timeout);
- $timeoutMin = min($timeoutMin, $response->timeout, 1);
- $chunk = false;
- if ($fromLastTimeout && null !== $multi->lastTimeout) {
- $elapsedTimeout = microtime(true) - $multi->lastTimeout;
- }
- if (isset($multi->handlesActivity[$j])) {
- $multi->lastTimeout = null;
- } elseif (!isset($multi->openHandles[$j])) {
- unset($responses[$j]);
- continue;
- } elseif ($elapsedTimeout >= $timeoutMax) {
- $multi->handlesActivity[$j] = [new ErrorChunk($response->offset, sprintf('Idle timeout reached for "%s".', $response->getInfo('url')))];
- $multi->lastTimeout ?? $multi->lastTimeout = $lastActivity;
- } else {
- continue;
- }
- while ($multi->handlesActivity[$j] ?? false) {
- $hasActivity = true;
- $elapsedTimeout = 0;
- if (\is_string($chunk = array_shift($multi->handlesActivity[$j]))) {
- if (null !== $response->inflate && false === $chunk = @inflate_add($response->inflate, $chunk)) {
- $multi->handlesActivity[$j] = [null, new TransportException(sprintf('Error while processing content unencoding for "%s".', $response->getInfo('url')))];
- continue;
- }
- if ('' !== $chunk && null !== $response->content && \strlen($chunk) !== fwrite($response->content, $chunk)) {
- $multi->handlesActivity[$j] = [null, new TransportException(sprintf('Failed writing %d bytes to the response buffer.', \strlen($chunk)))];
- continue;
- }
- $chunkLen = \strlen($chunk);
- $chunk = new DataChunk($response->offset, $chunk);
- $response->offset += $chunkLen;
- } elseif (null === $chunk) {
- $e = $multi->handlesActivity[$j][0];
- unset($responses[$j], $multi->handlesActivity[$j]);
- $response->close();
- if (null !== $e) {
- $response->info['error'] = $e->getMessage();
- if ($e instanceof \Error) {
- throw $e;
- }
- $chunk = new ErrorChunk($response->offset, $e);
- } else {
- if (0 === $response->offset && null === $response->content) {
- $response->content = fopen('php://memory', 'w+');
- }
- $chunk = new LastChunk($response->offset);
- }
- } elseif ($chunk instanceof ErrorChunk) {
- unset($responses[$j]);
- $elapsedTimeout = $timeoutMax;
- } elseif ($chunk instanceof FirstChunk) {
- if ($response->logger) {
- $info = $response->getInfo();
- $response->logger->info(sprintf('Response: "%s %s"', $info['http_code'], $info['url']));
- }
- $response->inflate = \extension_loaded('zlib') && $response->inflate && 'gzip' === ($response->headers['content-encoding'][0] ?? null) ? inflate_init(\ZLIB_ENCODING_GZIP) : null;
- if ($response->shouldBuffer instanceof \Closure) {
- try {
- $response->shouldBuffer = ($response->shouldBuffer)($response->headers);
- if (null !== $response->info['error']) {
- throw new TransportException($response->info['error']);
- }
- } catch (\Throwable $e) {
- $response->close();
- $multi->handlesActivity[$j] = [null, $e];
- }
- }
- if (true === $response->shouldBuffer) {
- $response->content = fopen('php://temp', 'w+');
- } elseif (\is_resource($response->shouldBuffer)) {
- $response->content = $response->shouldBuffer;
- }
- $response->shouldBuffer = null;
- yield $response => $chunk;
- if ($response->initializer && null === $response->info['error']) {
- // Ensure the HTTP status code is always checked
- $response->getHeaders(true);
- }
- continue;
- }
- yield $response => $chunk;
- }
- unset($multi->handlesActivity[$j]);
- if ($chunk instanceof ErrorChunk && !$chunk->didThrow()) {
- // Ensure transport exceptions are always thrown
- $chunk->getContent();
- }
- }
- if (!$responses) {
- unset($runningResponses[$i]);
- }
- // Prevent memory leaks
- $multi->handlesActivity = $multi->handlesActivity ?: [];
- $multi->openHandles = $multi->openHandles ?: [];
- }
- if (!$runningResponses) {
- break;
- }
- if ($hasActivity) {
- $lastActivity = microtime(true);
- continue;
- }
- if (-1 === self::select($multi, min($timeoutMin, $timeoutMax - $elapsedTimeout))) {
- usleep(min(500, 1E6 * $timeoutMin));
- }
- $elapsedTimeout = microtime(true) - $lastActivity;
- }
- }
- }
|