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 mixed $data    The response data
  32.      * @param int   $status  The response status code
  33.      * @param array $headers An array of response headers
  34.      * @param bool  $json    If the data is already a JSON string
  35.      */
  36.     public function __construct($data nullint $status 200, array $headers = [], bool $json false)
  37.     {
  38.         parent::__construct(''$status$headers);
  39.         if ($json && !\is_string($data) && !is_numeric($data) && !\is_callable([$data'__toString'])) {
  40.             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)));
  41.         }
  42.         if (null === $data) {
  43.             $data = new \ArrayObject();
  44.         }
  45.         $json $this->setJson($data) : $this->setData($data);
  46.     }
  47.     /**
  48.      * Factory method for chainability.
  49.      *
  50.      * Example:
  51.      *
  52.      *     return JsonResponse::create(['key' => 'value'])
  53.      *         ->setSharedMaxAge(300);
  54.      *
  55.      * @param mixed $data    The JSON response data
  56.      * @param int   $status  The response status code
  57.      * @param array $headers An array of response headers
  58.      *
  59.      * @return static
  60.      */
  61.     public static function create($data null$status 200$headers = [])
  62.     {
  63.         return new static($data$status$headers);
  64.     }
  65.     /**
  66.      * Factory method for chainability.
  67.      *
  68.      * Example:
  69.      *
  70.      *     return JsonResponse::fromJsonString('{"key": "value"}')
  71.      *         ->setSharedMaxAge(300);
  72.      *
  73.      * @param string $data    The JSON response string
  74.      * @param int    $status  The response status code
  75.      * @param array  $headers An array of response headers
  76.      *
  77.      * @return static
  78.      */
  79.     public static function fromJsonString($data$status 200$headers = [])
  80.     {
  81.         return new static($data$status$headerstrue);
  82.     }
  83.     /**
  84.      * Sets the JSONP callback.
  85.      *
  86.      * @param string|null $callback The JSONP callback or null to use none
  87.      *
  88.      * @return $this
  89.      *
  90.      * @throws \InvalidArgumentException When the callback name is not valid
  91.      */
  92.     public function setCallback($callback null)
  93.     {
  94.         if (null !== $callback) {
  95.             // partially taken from https://geekality.net/2011/08/03/valid-javascript-identifier/
  96.             // partially taken from https://github.com/willdurand/JsonpCallbackValidator
  97.             //      JsonpCallbackValidator is released under the MIT License. See https://github.com/willdurand/JsonpCallbackValidator/blob/v1.1.0/LICENSE for details.
  98.             //      (c) William Durand <william.durand1@gmail.com>
  99.             $pattern '/^[$_\p{L}][$_\p{L}\p{Mn}\p{Mc}\p{Nd}\p{Pc}\x{200C}\x{200D}]*(?:\[(?:"(?:\\\.|[^"\\\])*"|\'(?:\\\.|[^\'\\\])*\'|\d+)\])*?$/u';
  100.             $reserved = [
  101.                 'break''do''instanceof''typeof''case''else''new''var''catch''finally''return''void''continue''for''switch''while',
  102.                 'debugger''function''this''with''default''if''throw''delete''in''try''class''enum''extends''super',  'const''export',
  103.                 'import''implements''let''private''public''yield''interface''package''protected''static''null''true''false',
  104.             ];
  105.             $parts explode('.'$callback);
  106.             foreach ($parts as $part) {
  107.                 if (!preg_match($pattern$part) || \in_array($part$reservedtrue)) {
  108.                     throw new \InvalidArgumentException('The callback name is not valid.');
  109.                 }
  110.             }
  111.         }
  112.         $this->callback $callback;
  113.         return $this->update();
  114.     }
  115.     /**
  116.      * Sets a raw string containing a JSON document to be sent.
  117.      *
  118.      * @param string $json
  119.      *
  120.      * @return $this
  121.      */
  122.     public function setJson($json)
  123.     {
  124.         $this->data $json;
  125.         return $this->update();
  126.     }
  127.     /**
  128.      * Sets the data to be sent as JSON.
  129.      *
  130.      * @param mixed $data
  131.      *
  132.      * @return $this
  133.      *
  134.      * @throws \InvalidArgumentException
  135.      */
  136.     public function setData($data = [])
  137.     {
  138.         try {
  139.             $data json_encode($data$this->encodingOptions);
  140.         } catch (\Exception $e) {
  141.             if ('Exception' === \get_class($e) && str_starts_with($e->getMessage(), 'Failed calling ')) {
  142.                 throw $e->getPrevious() ?: $e;
  143.             }
  144.             throw $e;
  145.         }
  146.         if (\PHP_VERSION_ID >= 70300 && (\JSON_THROW_ON_ERROR $this->encodingOptions)) {
  147.             return $this->setJson($data);
  148.         }
  149.         if (\JSON_ERROR_NONE !== json_last_error()) {
  150.             throw new \InvalidArgumentException(json_last_error_msg());
  151.         }
  152.         return $this->setJson($data);
  153.     }
  154.     /**
  155.      * Returns options used while encoding data to JSON.
  156.      *
  157.      * @return int
  158.      */
  159.     public function getEncodingOptions()
  160.     {
  161.         return $this->encodingOptions;
  162.     }
  163.     /**
  164.      * Sets options used while encoding data to JSON.
  165.      *
  166.      * @param int $encodingOptions
  167.      *
  168.      * @return $this
  169.      */
  170.     public function setEncodingOptions($encodingOptions)
  171.     {
  172.         $this->encodingOptions = (int) $encodingOptions;
  173.         return $this->setData(json_decode($this->data));
  174.     }
  175.     /**
  176.      * Updates the content and headers according to the JSON data and callback.
  177.      *
  178.      * @return $this
  179.      */
  180.     protected function update()
  181.     {
  182.         if (null !== $this->callback) {
  183.             // Not using application/javascript for compatibility reasons with older browsers.
  184.             $this->headers->set('Content-Type''text/javascript');
  185.             return $this->setContent(sprintf('/**/%s(%s);'$this->callback$this->data));
  186.         }
  187.         // Only set the header when there is none or when it equals 'text/javascript' (from a previous update with callback)
  188.         // in order to not overwrite a custom definition.
  189.         if (!$this->headers->has('Content-Type') || 'text/javascript' === $this->headers->get('Content-Type')) {
  190.             $this->headers->set('Content-Type''application/json');
  191.         }
  192.         return $this->setContent($this->data);
  193.     }
  194. }