IndexAction.php 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  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\rest;
  8. use Yii;
  9. use yii\data\ActiveDataProvider;
  10. use yii\data\DataFilter;
  11. use yii\data\Pagination;
  12. use yii\data\Sort;
  13. use yii\helpers\ArrayHelper;
  14. /**
  15. * IndexAction implements the API endpoint for listing multiple models.
  16. *
  17. * For more details and usage information on IndexAction, see the [guide article on rest controllers](guide:rest-controllers).
  18. *
  19. * @author Qiang Xue <qiang.xue@gmail.com>
  20. * @since 2.0
  21. */
  22. class IndexAction extends Action
  23. {
  24. /**
  25. * @var callable|null a PHP callable that will be called to prepare a data provider that
  26. * should return a collection of the models. If not set, [[prepareDataProvider()]] will be used instead.
  27. * The signature of the callable should be:
  28. *
  29. * ```php
  30. * function (IndexAction $action) {
  31. * // $action is the action object currently running
  32. * }
  33. * ```
  34. *
  35. * The callable should return an instance of [[ActiveDataProvider]].
  36. *
  37. * If [[dataFilter]] is set the result of [[DataFilter::build()]] will be passed to the callable as a second parameter.
  38. * In this case the signature of the callable should be the following:
  39. *
  40. * ```php
  41. * function (IndexAction $action, mixed $filter) {
  42. * // $action is the action object currently running
  43. * // $filter the built filter condition
  44. * }
  45. * ```
  46. */
  47. public $prepareDataProvider;
  48. /**
  49. * @var callable a PHP callable that will be called to prepare query in prepareDataProvider.
  50. * Should return $query.
  51. * For example:
  52. *
  53. * ```php
  54. * function ($query, $requestParams) {
  55. * $query->andFilterWhere(['id' => 1]);
  56. * ...
  57. * return $query;
  58. * }
  59. * ```
  60. *
  61. * @since 2.0.42
  62. */
  63. public $prepareSearchQuery;
  64. /**
  65. * @var DataFilter|null data filter to be used for the search filter composition.
  66. * You must set up this field explicitly in order to enable filter processing.
  67. * For example:
  68. *
  69. * ```php
  70. * [
  71. * 'class' => 'yii\data\ActiveDataFilter',
  72. * 'searchModel' => function () {
  73. * return (new \yii\base\DynamicModel(['id' => null, 'name' => null, 'price' => null]))
  74. * ->addRule('id', 'integer')
  75. * ->addRule('name', 'trim')
  76. * ->addRule('name', 'string')
  77. * ->addRule('price', 'number');
  78. * },
  79. * ]
  80. * ```
  81. *
  82. * @see DataFilter
  83. *
  84. * @since 2.0.13
  85. */
  86. public $dataFilter;
  87. /**
  88. * @var array|Pagination|false The pagination to be used by [[prepareDataProvider()]].
  89. * If this is `false`, it means pagination is disabled.
  90. * Note: if a Pagination object is passed, it's `params` will be set to the request parameters.
  91. * @see Pagination
  92. * @since 2.0.45
  93. */
  94. public $pagination = [];
  95. /**
  96. * @var array|Sort|false The sorting to be used by [[prepareDataProvider()]].
  97. * If this is `false`, it means sorting is disabled.
  98. * Note: if a Sort object is passed, it's `params` will be set to the request parameters.
  99. * @see Sort
  100. * @since 2.0.45
  101. */
  102. public $sort = [];
  103. /**
  104. * @return ActiveDataProvider
  105. */
  106. public function run()
  107. {
  108. if ($this->checkAccess) {
  109. call_user_func($this->checkAccess, $this->id);
  110. }
  111. return $this->prepareDataProvider();
  112. }
  113. /**
  114. * Prepares the data provider that should return the requested collection of the models.
  115. * @return ActiveDataProvider
  116. */
  117. protected function prepareDataProvider()
  118. {
  119. $requestParams = Yii::$app->getRequest()->getBodyParams();
  120. if (empty($requestParams)) {
  121. $requestParams = Yii::$app->getRequest()->getQueryParams();
  122. }
  123. $filter = null;
  124. if ($this->dataFilter !== null) {
  125. $this->dataFilter = Yii::createObject($this->dataFilter);
  126. if ($this->dataFilter->load($requestParams)) {
  127. $filter = $this->dataFilter->build();
  128. if ($filter === false) {
  129. return $this->dataFilter;
  130. }
  131. }
  132. }
  133. if ($this->prepareDataProvider !== null) {
  134. return call_user_func($this->prepareDataProvider, $this, $filter);
  135. }
  136. /* @var $modelClass \yii\db\BaseActiveRecord */
  137. $modelClass = $this->modelClass;
  138. $query = $modelClass::find();
  139. if (!empty($filter)) {
  140. $query->andWhere($filter);
  141. }
  142. if (is_callable($this->prepareSearchQuery)) {
  143. $query = call_user_func($this->prepareSearchQuery, $query, $requestParams);
  144. }
  145. if (is_array($this->pagination)) {
  146. $pagination = ArrayHelper::merge(
  147. [
  148. 'params' => $requestParams,
  149. ],
  150. $this->pagination
  151. );
  152. } else {
  153. $pagination = $this->pagination;
  154. if ($this->pagination instanceof Pagination) {
  155. $pagination->params = $requestParams;
  156. }
  157. }
  158. if (is_array($this->sort)) {
  159. $sort = ArrayHelper::merge(
  160. [
  161. 'params' => $requestParams,
  162. ],
  163. $this->sort
  164. );
  165. } else {
  166. $sort = $this->sort;
  167. if ($this->sort instanceof Sort) {
  168. $sort->params = $requestParams;
  169. }
  170. }
  171. return Yii::createObject([
  172. 'class' => ActiveDataProvider::className(),
  173. 'query' => $query,
  174. 'pagination' => $pagination,
  175. 'sort' => $sort,
  176. ]);
  177. }
  178. }