JsonResponseFormatter.php 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\web;
  8. use Yii;
  9. use yii\base\Component;
  10. use yii\helpers\Json;
  11. /**
  12. * JsonResponseFormatter formats the given data into a JSON or JSONP response content.
  13. *
  14. * It is used by [[Response]] to format response data.
  15. *
  16. * To configure properties like [[encodeOptions]] or [[prettyPrint]], you can configure the `response`
  17. * application component like the following:
  18. *
  19. * ```php
  20. * 'response' => [
  21. * // ...
  22. * 'formatters' => [
  23. * \yii\web\Response::FORMAT_JSON => [
  24. * 'class' => 'yii\web\JsonResponseFormatter',
  25. * 'prettyPrint' => YII_DEBUG, // use "pretty" output in debug mode
  26. * 'keepObjectType' => false, // keep object type for zero-indexed objects
  27. * // ...
  28. * ],
  29. * ],
  30. * ],
  31. * ```
  32. *
  33. * @author Qiang Xue <qiang.xue@gmail.com>
  34. * @since 2.0
  35. */
  36. class JsonResponseFormatter extends Component implements ResponseFormatterInterface
  37. {
  38. /**
  39. * JSON Content Type
  40. * @since 2.0.14
  41. */
  42. const CONTENT_TYPE_JSONP = 'application/javascript; charset=UTF-8';
  43. /**
  44. * JSONP Content Type
  45. * @since 2.0.14
  46. */
  47. const CONTENT_TYPE_JSON = 'application/json; charset=UTF-8';
  48. /**
  49. * HAL JSON Content Type
  50. * @since 2.0.14
  51. */
  52. const CONTENT_TYPE_HAL_JSON = 'application/hal+json; charset=UTF-8';
  53. /**
  54. * @var string|null custom value of the `Content-Type` header of the response.
  55. * When equals `null` default content type will be used based on the `useJsonp` property.
  56. * @since 2.0.14
  57. */
  58. public $contentType;
  59. /**
  60. * @var bool whether to use JSONP response format. When this is true, the [[Response::data|response data]]
  61. * must be an array consisting of `data` and `callback` members. The latter should be a JavaScript
  62. * function name while the former will be passed to this function as a parameter.
  63. */
  64. public $useJsonp = false;
  65. /**
  66. * @var int the encoding options passed to [[Json::encode()]]. For more details please refer to
  67. * <https://www.php.net/manual/en/function.json-encode.php>.
  68. * Default is `JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE`.
  69. * This property has no effect, when [[useJsonp]] is `true`.
  70. * @since 2.0.7
  71. */
  72. public $encodeOptions = 320;
  73. /**
  74. * @var bool whether to format the output in a readable "pretty" format. This can be useful for debugging purpose.
  75. * If this is true, `JSON_PRETTY_PRINT` will be added to [[encodeOptions]].
  76. * Defaults to `false`.
  77. * This property has no effect, when [[useJsonp]] is `true`.
  78. * @since 2.0.7
  79. */
  80. public $prettyPrint = false;
  81. /**
  82. * @var bool Avoids objects with zero-indexed keys to be encoded as array
  83. * Json::encode((object)['test']) will be encoded as an object not array. This matches the behaviour of json_encode().
  84. * Defaults to Json::$keepObjectType value
  85. * @since 2.0.44
  86. */
  87. public $keepObjectType;
  88. /**
  89. * Formats the specified response.
  90. * @param Response $response the response to be formatted.
  91. */
  92. public function format($response)
  93. {
  94. if ($this->contentType === null) {
  95. $this->contentType = $this->useJsonp
  96. ? self::CONTENT_TYPE_JSONP
  97. : self::CONTENT_TYPE_JSON;
  98. } elseif (strpos($this->contentType, 'charset') === false) {
  99. $this->contentType .= '; charset=UTF-8';
  100. }
  101. $response->getHeaders()->set('Content-Type', $this->contentType);
  102. if ($this->useJsonp) {
  103. $this->formatJsonp($response);
  104. } else {
  105. $this->formatJson($response);
  106. }
  107. }
  108. /**
  109. * Formats response data in JSON format.
  110. * @param Response $response
  111. */
  112. protected function formatJson($response)
  113. {
  114. if ($response->data !== null) {
  115. $options = $this->encodeOptions;
  116. if ($this->prettyPrint) {
  117. $options |= JSON_PRETTY_PRINT;
  118. }
  119. $default = Json::$keepObjectType;
  120. if ($this->keepObjectType !== null) {
  121. Json::$keepObjectType = $this->keepObjectType;
  122. }
  123. $response->content = Json::encode($response->data, $options);
  124. // Restore default value to avoid any unexpected behaviour
  125. Json::$keepObjectType = $default;
  126. } elseif ($response->content === null) {
  127. $response->content = 'null';
  128. }
  129. }
  130. /**
  131. * Formats response data in JSONP format.
  132. * @param Response $response
  133. */
  134. protected function formatJsonp($response)
  135. {
  136. if (is_array($response->data)
  137. && isset($response->data['data'], $response->data['callback'])
  138. ) {
  139. $response->content = sprintf(
  140. '%s(%s);',
  141. $response->data['callback'],
  142. Json::htmlEncode($response->data['data'])
  143. );
  144. } elseif ($response->data !== null) {
  145. $response->content = '';
  146. Yii::warning(
  147. "The 'jsonp' response requires that the data be an array consisting of both 'data' and 'callback' elements.",
  148. __METHOD__
  149. );
  150. }
  151. }
  152. }