* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\HttpClient; use Psr\Log\LoggerAwareInterface; use Psr\Log\LoggerInterface; use Symfony\Component\HttpClient\Response\TraceableResponse; use Symfony\Contracts\HttpClient\HttpClientInterface; use Symfony\Contracts\HttpClient\ResponseInterface; use Symfony\Contracts\HttpClient\ResponseStreamInterface; use Symfony\Contracts\Service\ResetInterface; /** * @author Jérémy Romey */ final class TraceableHttpClient implements HttpClientInterface, ResetInterface, LoggerAwareInterface { private $client; private $tracedRequests = []; public function __construct(HttpClientInterface $client) { $this->client = $client; } /** * {@inheritdoc} */ public function request(string $method, string $url, array $options = []): ResponseInterface { $content = null; $traceInfo = []; $this->tracedRequests[] = [ 'method' => $method, 'url' => $url, 'options' => $options, 'info' => &$traceInfo, 'content' => &$content, ]; $onProgress = $options['on_progress'] ?? null; $options['on_progress'] = function (int $dlNow, int $dlSize, array $info) use (&$traceInfo, $onProgress) { $traceInfo = $info; if (null !== $onProgress) { $onProgress($dlNow, $dlSize, $info); } }; return new TraceableResponse($this->client, $this->client->request($method, $url, $options), $content); } /** * {@inheritdoc} */ public function stream($responses, float $timeout = null): ResponseStreamInterface { if ($responses instanceof TraceableResponse) { $responses = [$responses]; } elseif (!is_iterable($responses)) { throw new \TypeError(sprintf('"%s()" expects parameter 1 to be an iterable of TraceableResponse objects, "%s" given.', __METHOD__, get_debug_type($responses))); } return $this->client->stream(\Closure::bind(static function () use ($responses) { foreach ($responses as $k => $r) { if (!$r instanceof TraceableResponse) { throw new \TypeError(sprintf('"%s()" expects parameter 1 to be an iterable of TraceableResponse objects, "%s" given.', __METHOD__, get_debug_type($r))); } yield $k => $r->response; } }, null, TraceableResponse::class)(), $timeout); } public function getTracedRequests(): array { return $this->tracedRequests; } public function reset() { if ($this->client instanceof ResetInterface) { $this->client->reset(); } $this->tracedRequests = []; } /** * {@inheritdoc} */ public function setLogger(LoggerInterface $logger): void { if ($this->client instanceof LoggerAwareInterface) { $this->client->setLogger($logger); } } }