* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Bridge\PsrHttpMessage\Factory; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Message\StreamInterface; use Psr\Http\Message\UploadedFileInterface; use Psr\Http\Message\UriInterface; use Symfony\Bridge\PsrHttpMessage\HttpFoundationFactoryInterface; use Symfony\Component\HttpFoundation\Cookie; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpFoundation\StreamedResponse; /** * {@inheritdoc} * * @author Kévin Dunglas */ class HttpFoundationFactory implements HttpFoundationFactoryInterface { /** * @var int The maximum output buffering size for each iteration when sending the response */ private $responseBufferMaxLength; public function __construct(int $responseBufferMaxLength = 16372) { $this->responseBufferMaxLength = $responseBufferMaxLength; } /** * {@inheritdoc} * * @return Request */ public function createRequest(ServerRequestInterface $psrRequest, bool $streamed = false) { $server = []; $uri = $psrRequest->getUri(); if ($uri instanceof UriInterface) { $server['SERVER_NAME'] = $uri->getHost(); $server['SERVER_PORT'] = $uri->getPort() ?: ('https' === $uri->getScheme() ? 443 : 80); $server['REQUEST_URI'] = $uri->getPath(); $server['QUERY_STRING'] = $uri->getQuery(); if ('' !== $server['QUERY_STRING']) { $server['REQUEST_URI'] .= '?'.$server['QUERY_STRING']; } if ('https' === $uri->getScheme()) { $server['HTTPS'] = 'on'; } } $server['REQUEST_METHOD'] = $psrRequest->getMethod(); $server = array_replace($psrRequest->getServerParams(), $server); $parsedBody = $psrRequest->getParsedBody(); $parsedBody = \is_array($parsedBody) ? $parsedBody : []; $request = new Request( $psrRequest->getQueryParams(), $parsedBody, $psrRequest->getAttributes(), $psrRequest->getCookieParams(), $this->getFiles($psrRequest->getUploadedFiles()), $server, $streamed ? $psrRequest->getBody()->detach() : $psrRequest->getBody()->__toString() ); $request->headers->add($psrRequest->getHeaders()); return $request; } /** * Converts to the input array to $_FILES structure. */ private function getFiles(array $uploadedFiles): array { $files = []; foreach ($uploadedFiles as $key => $value) { if ($value instanceof UploadedFileInterface) { $files[$key] = $this->createUploadedFile($value); } else { $files[$key] = $this->getFiles($value); } } return $files; } /** * Creates Symfony UploadedFile instance from PSR-7 ones. */ private function createUploadedFile(UploadedFileInterface $psrUploadedFile): UploadedFile { return new UploadedFile($psrUploadedFile, function () { return $this->getTemporaryPath(); }); } /** * Gets a temporary file path. * * @return string */ protected function getTemporaryPath() { return tempnam(sys_get_temp_dir(), uniqid('symfony', true)); } /** * {@inheritdoc} * * @return Response */ public function createResponse(ResponseInterface $psrResponse, bool $streamed = false) { $cookies = $psrResponse->getHeader('Set-Cookie'); $psrResponse = $psrResponse->withoutHeader('Set-Cookie'); if ($streamed) { $response = new StreamedResponse( $this->createStreamedResponseCallback($psrResponse->getBody()), $psrResponse->getStatusCode(), $psrResponse->getHeaders() ); } else { $response = new Response( $psrResponse->getBody()->__toString(), $psrResponse->getStatusCode(), $psrResponse->getHeaders() ); } $response->setProtocolVersion($psrResponse->getProtocolVersion()); foreach ($cookies as $cookie) { $response->headers->setCookie($this->createCookie($cookie)); } return $response; } /** * Creates a Cookie instance from a cookie string. * * Some snippets have been taken from the Guzzle project: https://github.com/guzzle/guzzle/blob/5.3/src/Cookie/SetCookie.php#L34 * * @throws \InvalidArgumentException */ private function createCookie(string $cookie): Cookie { foreach (explode(';', $cookie) as $part) { $part = trim($part); $data = explode('=', $part, 2); $name = $data[0]; $value = isset($data[1]) ? trim($data[1], " \n\r\t\0\x0B\"") : null; if (!isset($cookieName)) { $cookieName = $name; $cookieValue = $value; continue; } if ('expires' === strtolower($name) && null !== $value) { $cookieExpire = new \DateTime($value); continue; } if ('path' === strtolower($name) && null !== $value) { $cookiePath = $value; continue; } if ('domain' === strtolower($name) && null !== $value) { $cookieDomain = $value; continue; } if ('secure' === strtolower($name)) { $cookieSecure = true; continue; } if ('httponly' === strtolower($name)) { $cookieHttpOnly = true; continue; } if ('samesite' === strtolower($name) && null !== $value) { $samesite = $value; continue; } } if (!isset($cookieName)) { throw new \InvalidArgumentException('The value of the Set-Cookie header is malformed.'); } return new Cookie( $cookieName, $cookieValue, isset($cookieExpire) ? $cookieExpire : 0, isset($cookiePath) ? $cookiePath : '/', isset($cookieDomain) ? $cookieDomain : null, isset($cookieSecure), isset($cookieHttpOnly), true, isset($samesite) ? $samesite : null ); } private function createStreamedResponseCallback(StreamInterface $body): callable { return function () use ($body) { if ($body->isSeekable()) { $body->rewind(); } if (!$body->isReadable()) { echo $body; return; } while (!$body->eof()) { echo $body->read($this->responseBufferMaxLength); } }; } }