HttpKernelBrowser.php 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\HttpKernel;
  11. use Symfony\Component\BrowserKit\AbstractBrowser;
  12. use Symfony\Component\BrowserKit\CookieJar;
  13. use Symfony\Component\BrowserKit\History;
  14. use Symfony\Component\BrowserKit\Request as DomRequest;
  15. use Symfony\Component\BrowserKit\Response as DomResponse;
  16. use Symfony\Component\HttpFoundation\File\UploadedFile;
  17. use Symfony\Component\HttpFoundation\Request;
  18. use Symfony\Component\HttpFoundation\Response;
  19. /**
  20. * Simulates a browser and makes requests to an HttpKernel instance.
  21. *
  22. * @author Fabien Potencier <fabien@symfony.com>
  23. *
  24. * @method Request getRequest()
  25. * @method Response getResponse()
  26. */
  27. class HttpKernelBrowser extends AbstractBrowser
  28. {
  29. protected $kernel;
  30. private $catchExceptions = true;
  31. /**
  32. * @param array $server The server parameters (equivalent of $_SERVER)
  33. */
  34. public function __construct(HttpKernelInterface $kernel, array $server = [], ?History $history = null, ?CookieJar $cookieJar = null)
  35. {
  36. // These class properties must be set before calling the parent constructor, as it may depend on it.
  37. $this->kernel = $kernel;
  38. $this->followRedirects = false;
  39. parent::__construct($server, $history, $cookieJar);
  40. }
  41. /**
  42. * Sets whether to catch exceptions when the kernel is handling a request.
  43. */
  44. public function catchExceptions(bool $catchExceptions)
  45. {
  46. $this->catchExceptions = $catchExceptions;
  47. }
  48. /**
  49. * {@inheritdoc}
  50. *
  51. * @param Request $request
  52. *
  53. * @return Response
  54. */
  55. protected function doRequest(object $request)
  56. {
  57. $response = $this->kernel->handle($request, HttpKernelInterface::MAIN_REQUEST, $this->catchExceptions);
  58. if ($this->kernel instanceof TerminableInterface) {
  59. $this->kernel->terminate($request, $response);
  60. }
  61. return $response;
  62. }
  63. /**
  64. * {@inheritdoc}
  65. *
  66. * @param Request $request
  67. *
  68. * @return string
  69. */
  70. protected function getScript(object $request)
  71. {
  72. $kernel = var_export(serialize($this->kernel), true);
  73. $request = var_export(serialize($request), true);
  74. $errorReporting = error_reporting();
  75. $requires = '';
  76. foreach (get_declared_classes() as $class) {
  77. if (0 === strpos($class, 'ComposerAutoloaderInit')) {
  78. $r = new \ReflectionClass($class);
  79. $file = \dirname($r->getFileName(), 2).'/autoload.php';
  80. if (file_exists($file)) {
  81. $requires .= 'require_once '.var_export($file, true).";\n";
  82. }
  83. }
  84. }
  85. if (!$requires) {
  86. throw new \RuntimeException('Composer autoloader not found.');
  87. }
  88. $code = <<<EOF
  89. <?php
  90. error_reporting($errorReporting);
  91. $requires
  92. \$kernel = unserialize($kernel);
  93. \$request = unserialize($request);
  94. EOF;
  95. return $code.$this->getHandleScript();
  96. }
  97. protected function getHandleScript()
  98. {
  99. return <<<'EOF'
  100. $response = $kernel->handle($request);
  101. if ($kernel instanceof Symfony\Component\HttpKernel\TerminableInterface) {
  102. $kernel->terminate($request, $response);
  103. }
  104. echo serialize($response);
  105. EOF;
  106. }
  107. /**
  108. * {@inheritdoc}
  109. *
  110. * @return Request
  111. */
  112. protected function filterRequest(DomRequest $request)
  113. {
  114. $httpRequest = Request::create($request->getUri(), $request->getMethod(), $request->getParameters(), $request->getCookies(), $request->getFiles(), $server = $request->getServer(), $request->getContent());
  115. if (!isset($server['HTTP_ACCEPT'])) {
  116. $httpRequest->headers->remove('Accept');
  117. }
  118. foreach ($this->filterFiles($httpRequest->files->all()) as $key => $value) {
  119. $httpRequest->files->set($key, $value);
  120. }
  121. return $httpRequest;
  122. }
  123. /**
  124. * Filters an array of files.
  125. *
  126. * This method created test instances of UploadedFile so that the move()
  127. * method can be called on those instances.
  128. *
  129. * If the size of a file is greater than the allowed size (from php.ini) then
  130. * an invalid UploadedFile is returned with an error set to UPLOAD_ERR_INI_SIZE.
  131. *
  132. * @see UploadedFile
  133. *
  134. * @return array
  135. */
  136. protected function filterFiles(array $files)
  137. {
  138. $filtered = [];
  139. foreach ($files as $key => $value) {
  140. if (\is_array($value)) {
  141. $filtered[$key] = $this->filterFiles($value);
  142. } elseif ($value instanceof UploadedFile) {
  143. if ($value->isValid() && $value->getSize() > UploadedFile::getMaxFilesize()) {
  144. $filtered[$key] = new UploadedFile(
  145. '',
  146. $value->getClientOriginalName(),
  147. $value->getClientMimeType(),
  148. \UPLOAD_ERR_INI_SIZE,
  149. true
  150. );
  151. } else {
  152. $filtered[$key] = new UploadedFile(
  153. $value->getPathname(),
  154. $value->getClientOriginalName(),
  155. $value->getClientMimeType(),
  156. $value->getError(),
  157. true
  158. );
  159. }
  160. }
  161. }
  162. return $filtered;
  163. }
  164. /**
  165. * {@inheritdoc}
  166. *
  167. * @param Response $response
  168. *
  169. * @return DomResponse
  170. */
  171. protected function filterResponse(object $response)
  172. {
  173. // this is needed to support StreamedResponse
  174. ob_start();
  175. $response->sendContent();
  176. $content = ob_get_clean();
  177. return new DomResponse($content, $response->getStatusCode(), $response->headers->all());
  178. }
  179. }