vendor/sentry/sentry/src/HttpClient/HttpClientFactory.php line 111

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. namespace Sentry\HttpClient;
  4. use GuzzleHttp\RequestOptions as GuzzleHttpClientOptions;
  5. use Http\Adapter\Guzzle6\Client as GuzzleHttpClient;
  6. use Http\Client\Common\Plugin\AuthenticationPlugin;
  7. use Http\Client\Common\Plugin\DecoderPlugin;
  8. use Http\Client\Common\Plugin\ErrorPlugin;
  9. use Http\Client\Common\Plugin\HeaderSetPlugin;
  10. use Http\Client\Common\Plugin\RetryPlugin;
  11. use Http\Client\Common\PluginClient;
  12. use Http\Client\Curl\Client as CurlHttpClient;
  13. use Http\Client\HttpAsyncClient as HttpAsyncClientInterface;
  14. use Http\Discovery\HttpAsyncClientDiscovery;
  15. use Psr\Http\Client\ClientInterface;
  16. use Psr\Http\Message\ResponseFactoryInterface;
  17. use Psr\Http\Message\StreamFactoryInterface;
  18. use Psr\Http\Message\UriFactoryInterface;
  19. use Sentry\HttpClient\Authentication\SentryAuthentication;
  20. use Sentry\HttpClient\Plugin\GzipEncoderPlugin;
  21. use Sentry\Options;
  22. use Symfony\Component\HttpClient\HttpClient as SymfonyHttpClient;
  23. use Symfony\Component\HttpClient\HttplugClient as SymfonyHttplugClient;
  24. /**
  25.  * Default implementation of the {@HttpClientFactoryInterface} interface that uses
  26.  * Httplug to autodiscover the HTTP client if none is passed by the user.
  27.  */
  28. final class HttpClientFactory implements HttpClientFactoryInterface
  29. {
  30.     /**
  31.      * @var StreamFactoryInterface The PSR-17 stream factory
  32.      */
  33.     private $streamFactory;
  34.     /**
  35.      * @var HttpAsyncClientInterface|null The HTTP client
  36.      */
  37.     private $httpClient;
  38.     /**
  39.      * @var string The name of the SDK
  40.      */
  41.     private $sdkIdentifier;
  42.     /**
  43.      * @var string The version of the SDK
  44.      */
  45.     private $sdkVersion;
  46.     /**
  47.      * Constructor.
  48.      *
  49.      * @param UriFactoryInterface|null      $uriFactory      The PSR-7 URI factory
  50.      * @param ResponseFactoryInterface|null $responseFactory The PSR-7 response factory
  51.      * @param StreamFactoryInterface        $streamFactory   The PSR-17 stream factory
  52.      * @param HttpAsyncClientInterface|null $httpClient      The HTTP client
  53.      * @param string                        $sdkIdentifier   The SDK identifier
  54.      * @param string                        $sdkVersion      The SDK version
  55.      */
  56.     public function __construct(
  57.         ?UriFactoryInterface $uriFactory,
  58.         ?ResponseFactoryInterface $responseFactory,
  59.         StreamFactoryInterface $streamFactory,
  60.         ?HttpAsyncClientInterface $httpClient,
  61.         string $sdkIdentifier,
  62.         string $sdkVersion
  63.     ) {
  64.         $this->streamFactory $streamFactory;
  65.         $this->httpClient $httpClient;
  66.         $this->sdkIdentifier $sdkIdentifier;
  67.         $this->sdkVersion $sdkVersion;
  68.     }
  69.     /**
  70.      * {@inheritdoc}
  71.      */
  72.     public function create(Options $options): HttpAsyncClientInterface
  73.     {
  74.         if (null === $options->getDsn()) {
  75.             throw new \RuntimeException('Cannot create an HTTP client without the Sentry DSN set in the options.');
  76.         }
  77.         if (null !== $this->httpClient && null !== $options->getHttpProxy()) {
  78.             throw new \RuntimeException('The "http_proxy" option does not work together with a custom HTTP client.');
  79.         }
  80.         $httpClient $this->httpClient ?? $this->resolveClient($options);
  81.         $httpClientPlugins = [
  82.             new HeaderSetPlugin(['User-Agent' => $this->sdkIdentifier '/' $this->sdkVersion]),
  83.             new AuthenticationPlugin(new SentryAuthentication($options$this->sdkIdentifier$this->sdkVersion)),
  84.             new RetryPlugin(['retries' => $options->getSendAttempts(false)]),
  85.             new ErrorPlugin(['only_server_exception' => true]),
  86.         ];
  87.         if ($options->isCompressionEnabled()) {
  88.             $httpClientPlugins[] = new GzipEncoderPlugin($this->streamFactory);
  89.             $httpClientPlugins[] = new DecoderPlugin();
  90.         }
  91.         return new PluginClient($httpClient$httpClientPlugins);
  92.     }
  93.     /**
  94.      * @return ClientInterface|HttpAsyncClientInterface
  95.      */
  96.     private function resolveClient(Options $options)
  97.     {
  98.         if (class_exists(SymfonyHttplugClient::class)) {
  99.             $symfonyConfig = [
  100.                 'timeout' => $options->getHttpConnectTimeout(),
  101.                 'max_duration' => $options->getHttpTimeout(),
  102.             ];
  103.             if (null !== $options->getHttpProxy()) {
  104.                 $symfonyConfig['proxy'] = $options->getHttpProxy();
  105.             }
  106.             return new SymfonyHttplugClient(SymfonyHttpClient::create($symfonyConfig));
  107.         }
  108.         if (class_exists(GuzzleHttpClient::class)) {
  109.             $guzzleConfig = [
  110.                 GuzzleHttpClientOptions::TIMEOUT => $options->getHttpTimeout(),
  111.                 GuzzleHttpClientOptions::CONNECT_TIMEOUT => $options->getHttpConnectTimeout(),
  112.             ];
  113.             if (null !== $options->getHttpProxy()) {
  114.                 $guzzleConfig[GuzzleHttpClientOptions::PROXY] = $options->getHttpProxy();
  115.             }
  116.             return GuzzleHttpClient::createWithConfig($guzzleConfig);
  117.         }
  118.         if (class_exists(CurlHttpClient::class)) {
  119.             $curlConfig = [
  120.                 \CURLOPT_TIMEOUT => $options->getHttpTimeout(),
  121.                 \CURLOPT_CONNECTTIMEOUT => $options->getHttpConnectTimeout(),
  122.             ];
  123.             if (null !== $options->getHttpProxy()) {
  124.                 $curlConfig[\CURLOPT_PROXY] = $options->getHttpProxy();
  125.             }
  126.             return new CurlHttpClient(nullnull$curlConfig);
  127.         }
  128.         if (null !== $options->getHttpProxy()) {
  129.             throw new \RuntimeException('The "http_proxy" option requires either the "php-http/curl-client", the "symfony/http-client" or the "php-http/guzzle6-adapter" package to be installed.');
  130.         }
  131.         return HttpAsyncClientDiscovery::find();
  132.     }
  133. }