vendor/symfony/http-foundation/JsonResponse.php line 25

Open in your IDE?
  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\HttpFoundation;
  11. /**
  12.  * Response represents an HTTP response in JSON format.
  13.  *
  14.  * Note that this class does not force the returned JSON content to be an
  15.  * object. It is however recommended that you do return an object as it
  16.  * protects yourself against XSSI and JSON-JavaScript Hijacking.
  17.  *
  18.  * @see https://github.com/OWASP/CheatSheetSeries/blob/master/cheatsheets/AJAX_Security_Cheat_Sheet.md#always-return-json-with-an-object-on-the-outside
  19.  *
  20.  * @author Igor Wiedler <igor@wiedler.ch>
  21.  */
  22. class JsonResponse extends Response
  23. {
  24.     protected $data;
  25.     protected $callback;
  26.     // Encode <, >, ', &, and " characters in the JSON, making it also safe to be embedded into HTML.
  27.     // 15 === JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_AMP | JSON_HEX_QUOT
  28.     public const DEFAULT_ENCODING_OPTIONS 15;
  29.     protected $encodingOptions self::DEFAULT_ENCODING_OPTIONS;
  30.     /**
  31.      * @param bool $json If the data is already a JSON string
  32.      */
  33.     public function __construct(mixed $data nullint $status 200, array $headers = [], bool $json false)
  34.     {
  35.         parent::__construct(''$status$headers);
  36.         if ($json && !\is_string($data) && !is_numeric($data) && !\is_callable([$data'__toString'])) {
  37.             throw new \TypeError(sprintf('"%s": If $json is set to true, argument $data must be a string or object implementing __toString(), "%s" given.'__METHOD__get_debug_type($data)));
  38.         }
  39.         if (null === $data) {
  40.             $data = new \ArrayObject();
  41.         }
  42.         $json $this->setJson($data) : $this->setData($data);
  43.     }
  44.     /**
  45.      * Factory method for chainability.
  46.      *
  47.      * Example:
  48.      *
  49.      *     return JsonResponse::fromJsonString('{"key": "value"}')
  50.      *         ->setSharedMaxAge(300);
  51.      *
  52.      * @param string $data    The JSON response string
  53.      * @param int    $status  The response status code
  54.      * @param array  $headers An array of response headers
  55.      */
  56.     public static function fromJsonString(string $dataint $status 200, array $headers = []): static
  57.     {
  58.         return new static($data$status$headerstrue);
  59.     }
  60.     /**
  61.      * Sets the JSONP callback.
  62.      *
  63.      * @param string|null $callback The JSONP callback or null to use none
  64.      *
  65.      * @return $this
  66.      *
  67.      * @throws \InvalidArgumentException When the callback name is not valid
  68.      */
  69.     public function setCallback(string $callback null): static
  70.     {
  71.         if (null !== $callback) {
  72.             // partially taken from https://geekality.net/2011/08/03/valid-javascript-identifier/
  73.             // partially taken from https://github.com/willdurand/JsonpCallbackValidator
  74.             //      JsonpCallbackValidator is released under the MIT License. See https://github.com/willdurand/JsonpCallbackValidator/blob/v1.1.0/LICENSE for details.
  75.             //      (c) William Durand <william.durand1@gmail.com>
  76.             $pattern '/^[$_\p{L}][$_\p{L}\p{Mn}\p{Mc}\p{Nd}\p{Pc}\x{200C}\x{200D}]*(?:\[(?:"(?:\\\.|[^"\\\])*"|\'(?:\\\.|[^\'\\\])*\'|\d+)\])*?$/u';
  77.             $reserved = [
  78.                 'break''do''instanceof''typeof''case''else''new''var''catch''finally''return''void''continue''for''switch''while',
  79.                 'debugger''function''this''with''default''if''throw''delete''in''try''class''enum''extends''super',  'const''export',
  80.                 'import''implements''let''private''public''yield''interface''package''protected''static''null''true''false',
  81.             ];
  82.             $parts explode('.'$callback);
  83.             foreach ($parts as $part) {
  84.                 if (!preg_match($pattern$part) || \in_array($part$reservedtrue)) {
  85.                     throw new \InvalidArgumentException('The callback name is not valid.');
  86.                 }
  87.             }
  88.         }
  89.         $this->callback $callback;
  90.         return $this->update();
  91.     }
  92.     /**
  93.      * Sets a raw string containing a JSON document to be sent.
  94.      *
  95.      * @return $this
  96.      */
  97.     public function setJson(string $json): static
  98.     {
  99.         $this->data $json;
  100.         return $this->update();
  101.     }
  102.     /**
  103.      * Sets the data to be sent as JSON.
  104.      *
  105.      * @return $this
  106.      *
  107.      * @throws \InvalidArgumentException
  108.      */
  109.     public function setData(mixed $data = []): static
  110.     {
  111.         try {
  112.             $data json_encode($data$this->encodingOptions);
  113.         } catch (\Exception $e) {
  114.             if ('Exception' === \get_class($e) && str_starts_with($e->getMessage(), 'Failed calling ')) {
  115.                 throw $e->getPrevious() ?: $e;
  116.             }
  117.             throw $e;
  118.         }
  119.         if (\JSON_THROW_ON_ERROR $this->encodingOptions) {
  120.             return $this->setJson($data);
  121.         }
  122.         if (\JSON_ERROR_NONE !== json_last_error()) {
  123.             throw new \InvalidArgumentException(json_last_error_msg());
  124.         }
  125.         return $this->setJson($data);
  126.     }
  127.     /**
  128.      * Returns options used while encoding data to JSON.
  129.      */
  130.     public function getEncodingOptions(): int
  131.     {
  132.         return $this->encodingOptions;
  133.     }
  134.     /**
  135.      * Sets options used while encoding data to JSON.
  136.      *
  137.      * @return $this
  138.      */
  139.     public function setEncodingOptions(int $encodingOptions): static
  140.     {
  141.         $this->encodingOptions $encodingOptions;
  142.         return $this->setData(json_decode($this->data));
  143.     }
  144.     /**
  145.      * Updates the content and headers according to the JSON data and callback.
  146.      *
  147.      * @return $this
  148.      */
  149.     protected function update(): static
  150.     {
  151.         if (null !== $this->callback) {
  152.             // Not using application/javascript for compatibility reasons with older browsers.
  153.             $this->headers->set('Content-Type''text/javascript');
  154.             return $this->setContent(sprintf('/**/%s(%s);'$this->callback$this->data));
  155.         }
  156.         // Only set the header when there is none or when it equals 'text/javascript' (from a previous update with callback)
  157.         // in order to not overwrite a custom definition.
  158.         if (!$this->headers->has('Content-Type') || 'text/javascript' === $this->headers->get('Content-Type')) {
  159.             $this->headers->set('Content-Type''application/json');
  160.         }
  161.         return $this->setContent($this->data);
  162.     }
  163. }