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.
112 lines
3.8 KiB
112 lines
3.8 KiB
<?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\HttpKernel; |
|
|
|
use Symfony\Component\HttpClient\HttpClient; |
|
use Symfony\Component\HttpFoundation\Request; |
|
use Symfony\Component\HttpFoundation\Response; |
|
use Symfony\Component\HttpFoundation\ResponseHeaderBag; |
|
use Symfony\Component\Mime\Part\AbstractPart; |
|
use Symfony\Component\Mime\Part\DataPart; |
|
use Symfony\Component\Mime\Part\Multipart\FormDataPart; |
|
use Symfony\Component\Mime\Part\TextPart; |
|
use Symfony\Contracts\HttpClient\HttpClientInterface; |
|
|
|
// Help opcache.preload discover always-needed symbols |
|
class_exists(ResponseHeaderBag::class); |
|
|
|
/** |
|
* An implementation of a Symfony HTTP kernel using a "real" HTTP client. |
|
* |
|
* @author Fabien Potencier <fabien@symfony.com> |
|
*/ |
|
final class HttpClientKernel implements HttpKernelInterface |
|
{ |
|
private $client; |
|
|
|
public function __construct(HttpClientInterface $client = null) |
|
{ |
|
if (null === $client && !class_exists(HttpClient::class)) { |
|
throw new \LogicException(sprintf('You cannot use "%s" as the HttpClient component is not installed. Try running "composer require symfony/http-client".', __CLASS__)); |
|
} |
|
|
|
$this->client = $client ?? HttpClient::create(); |
|
} |
|
|
|
public function handle(Request $request, int $type = HttpKernelInterface::MAIN_REQUEST, bool $catch = true): Response |
|
{ |
|
$headers = $this->getHeaders($request); |
|
$body = ''; |
|
if (null !== $part = $this->getBody($request)) { |
|
$headers = array_merge($headers, $part->getPreparedHeaders()->toArray()); |
|
$body = $part->bodyToIterable(); |
|
} |
|
$response = $this->client->request($request->getMethod(), $request->getUri(), [ |
|
'headers' => $headers, |
|
'body' => $body, |
|
] + $request->attributes->get('http_client_options', [])); |
|
|
|
$response = new Response($response->getContent(!$catch), $response->getStatusCode(), $response->getHeaders(!$catch)); |
|
|
|
$response->headers->remove('X-Body-File'); |
|
$response->headers->remove('X-Body-Eval'); |
|
$response->headers->remove('X-Content-Digest'); |
|
|
|
$response->headers = new class($response->headers->all()) extends ResponseHeaderBag { |
|
protected function computeCacheControlValue(): string |
|
{ |
|
return $this->getCacheControlHeader(); // preserve the original value |
|
} |
|
}; |
|
|
|
return $response; |
|
} |
|
|
|
private function getBody(Request $request): ?AbstractPart |
|
{ |
|
if (\in_array($request->getMethod(), ['GET', 'HEAD'])) { |
|
return null; |
|
} |
|
|
|
if (!class_exists(AbstractPart::class)) { |
|
throw new \LogicException('You cannot pass non-empty bodies as the Mime component is not installed. Try running "composer require symfony/mime".'); |
|
} |
|
|
|
if ($content = $request->getContent()) { |
|
return new TextPart($content, 'utf-8', 'plain', '8bit'); |
|
} |
|
|
|
$fields = $request->request->all(); |
|
foreach ($request->files->all() as $name => $file) { |
|
$fields[$name] = DataPart::fromPath($file->getPathname(), $file->getClientOriginalName(), $file->getClientMimeType()); |
|
} |
|
|
|
return new FormDataPart($fields); |
|
} |
|
|
|
private function getHeaders(Request $request): array |
|
{ |
|
$headers = []; |
|
foreach ($request->headers as $key => $value) { |
|
$headers[$key] = $value; |
|
} |
|
$cookies = []; |
|
foreach ($request->cookies->all() as $name => $value) { |
|
$cookies[] = $name.'='.$value; |
|
} |
|
if ($cookies) { |
|
$headers['cookie'] = implode('; ', $cookies); |
|
} |
|
|
|
return $headers; |
|
} |
|
}
|
|
|