runtime-responses.md 18.3 KB
Newer Older
1
Ответы
2
======
3 4

Когда приложение заканчивает обработку [запроса](runtime-requests.md), оно генерирует объект [[yii\web\Response|ответа]]
5 6
и отправляет его пользователю. Объект ответа содержит такие данные, как HTTP-код состояния, HTTP-заголовки и тело ответа.
Конечная цель разработки Web-приложения состоит в создании объектов ответа на различные запросы.
7

8 9 10
В большинстве случаев вам придется иметь дело с [компонентом приложения](structure-application-components.md) `response`,
который по умолчанию является экземпляром класса [[yii\web\Response]]. Однако Yii также позволяет вам создавать собственные
объекты ответа и отправлять их пользователям. Это будет рассмотрено ниже.
11

12
В данном разделе мы опишем, как составлять ответы и отправлять их пользователям. 
13 14


15
## Код состояния <span id="status-code"></span>
16

17 18
Первое, что вы делаете при построении ответа, — определяете, был ли успешно обработан запрос. Это реализуется заданием
свойству [[yii\web\Response::statusCode]] значения, которое может быть одним из валидных
19
[HTTP-кодов состояния](http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html). Например, чтобы показать, что запрос был
20
успешно обработан, вы можете установить значение кода состояния равным 200:
21 22 23 24 25

```php
Yii::$app->response->statusCode = 200;
```

26 27 28
Однако в большинстве случаев явная установка не требуется так как значение [[yii\web\Response::statusCode]] 
по умолчанию равно 200. Если же вам нужно показать, что запрос не удался, вы можете выбросить соответствующее
HTTP-исключение:
29 30 31 32 33 34 35

```php
throw new \yii\web\NotFoundHttpException;
```

Когда [обработчик ошибок](runtime-handling-errors.md) поймает исключение, он извлечёт код состояния 
из исключения и назначит его ответу. Исключение [[yii\web\NotFoundHttpException]] в коде выше 
36
представляет HTTP-код состояния 404. В Yii предопределены следующие HTTP-исключения:
37 38 39 40 41 42 43 44 45 46 47 48 49

* [[yii\web\BadRequestHttpException]]: код состояния 400.
* [[yii\web\ConflictHttpException]]: код состояния 409.
* [[yii\web\ForbiddenHttpException]]: код состояния 403.
* [[yii\web\GoneHttpException]]: код состояния 410.
* [[yii\web\MethodNotAllowedHttpException]]: код состояния 405.
* [[yii\web\NotAcceptableHttpException]]: код состояния 406. 
* [[yii\web\NotFoundHttpException]]: код состояния 404.
* [[yii\web\ServerErrorHttpException]]: код состояния 500.
* [[yii\web\TooManyRequestsHttpException]]: код состояния 429.
* [[yii\web\UnauthorizedHttpException]]: код состояния 401.
* [[yii\web\UnsupportedMediaTypeHttpException]]: код состояния 415.

50
Если в приведённом выше списке нет исключения, которое вы хотите выбросить, вы можете создать его, расширив класс
51 52 53 54 55 56 57
[[yii\web\HttpException]], или выбросить его напрямую с кодом состояния, например:
 
```php
throw new \yii\web\HttpException(402);
```


58
## HTTP-заголовки <span id="http-headers"></span> 
59

60
Вы можете отправлять HTTP-заголовки, работая с [[yii\web\Response::headers|коллекцией заголовков]] компонента `response`:
61 62 63 64 65 66 67 68 69 70

```php
$headers = Yii::$app->response->headers;

// добавить заголовок Pragma. Уже имеющиеся Pragma-заголовки НЕ будут перезаписаны.
$headers->add('Pragma', 'no-cache');

// установить заголовок Pragma. Любые уже имеющиеся Pragma-заголовки будут сброшены.
$headers->set('Pragma', 'no-cache');

71
// удалить заголовок (или заголовки) Pragma и вернуть их значения массивом
72 73 74
$values = $headers->remove('Pragma');
```

75 76
> Информация: названия заголовков не чувствительны к регистру символов. Заново зарегистрированные заголовки не отсылаются
  пользователю до вызова [[yii\web\Response::send()]].
77 78


79
## Тело ответа <span id="response-body"></span>
80

81
Большинство ответов должны иметь тело, содержащее то, что вы хотите показать пользователям.
82

83 84
Если у вас уже имеется отформатированная строка для тела, вы можете присвоить её свойству [[yii\web\Response::content]] 
объекта запроса:
85 86 87 88 89 90

```php
Yii::$app->response->content = 'hello world!';
```

Если ваши данные перед отправкой конечным пользователям нужно привести к определённому формату, вам следует установить значения 
91 92
двух свойств: [[yii\web\Response::format|format]] и [[yii\web\Response::data|data]]. Свойство [[yii\web\Response::format|format]]
определяет, в каком формате следует возвращать данные из [[yii\web\Response::data|data]]. Например:
93 94 95 96 97 98 99 100

```php
$response = Yii::$app->response;
$response->format = \yii\web\Response::FORMAT_JSON;
$response->data = ['message' => 'hello world'];
```

Yii из коробки имеет поддержку следующих форматов, каждый из которых реализован классом [[yii\web\ResponseFormatterInterface|форматтера]].
101
Вы можете настроить эти форматтеры или добавить новые через свойство [[yii\web\Response::formatters]].
102 103 104 105 106 107

* [[yii\web\Response::FORMAT_HTML|HTML]]: реализуется классом [[yii\web\HtmlResponseFormatter]].
* [[yii\web\Response::FORMAT_XML|XML]]: реализуется классом [[yii\web\XmlResponseFormatter]].
* [[yii\web\Response::FORMAT_JSON|JSON]]: реализуется классом [[yii\web\JsonResponseFormatter]].
* [[yii\web\Response::FORMAT_JSONP|JSONP]]: реализуется классом [[yii\web\JsonResponseFormatter]].

108
Хотя тело запроса может быть явно установлено показанным выше способом, в большинстве случаев вы можете задавать его неявно 
109 110 111 112 113 114 115 116 117 118
через возвращаемое значение методов [действий](structure-controllers.md). Типичный пример использования:
 
```php
public function actionIndex()
{
    return $this->render('index');
}
```

Действие `index` в коде выше возвращает результат рендеринга представления `index`. Возвращаемое значение будет взято 
119
компонентом `response`, отформатировано и затем отправлено пользователям.
120

121 122
Так как по умолчанию форматом ответа является [[yii\web\Response::FORMAT_HTML|HTML]], в методе действия следует
вернуть строку. Если вы хотите использовать другой формат ответа, необходимо настроить его перед отправкой данных:
123 124 125 126 127 128 129 130 131 132 133 134

```php
public function actionInfo()
{
    \Yii::$app->response->format = \yii\web\Response::FORMAT_JSON;
    return [
        'message' => 'hello world',
        'code' => 100,
    ];
}
```

135 136
Как уже было сказано, кроме использования стандартного компонента приложения `response` вы также можете создавать свои
объекты ответа и отправлять их конечным пользователям. Вы можете сделать это, возвращая такой объект в методе действия:
137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152

```php
public function actionInfo()
{
    return \Yii::createObject([
        'class' => 'yii\web\Response',
        'format' => \yii\web\Response::FORMAT_JSON,
        'data' => [
            'message' => 'hello world',
            'code' => 100,
        ],
    ]);
}
```

> Примечание: создавая собственные объекты ответов, вы не сможете воспользоваться конфигурацией компонента `response`, 
153
  настроенной вами в конфигурации приложения. Тем не менее, вы можете воспользоваться 
154 155 156
  [внедрением зависимости](concept-di-container.md), чтобы применить общую конфигурацию к вашим новым объектам ответа.


157
## Перенаправление браузера <span id="browser-redirection"></span>
158

159 160
Перенаправление браузера основано на отправке HTTP-заголовка `Location`. Так как данная возможность широко применяется,
Yii имеет средства для её использования.
161 162

Вы можете перенаправить браузер пользователя на URL-адрес, вызвав метод [[yii\web\Response::redirect()]]. Этот метод
163
использует указанный URL-адрес в качестве значения заголовка `Location` и возвращает сам объект ответа. В методе действия
164 165 166 167 168 169 170 171 172
вы можете вызвать короткую версию этого метода — [[yii\web\Controller::redirect()]]. Например:

```php
public function actionOld()
{
    return $this->redirect('http://example.com/new', 301);
}
```

173 174
В приведённом выше коде метод действия возвращает результат `redirect()`. Как говорилось выше, объект ответа,
возвращаемый методом действия, будет использоваться в качестве ответа конечным пользователям.
175

176 177 178
В коде, находящемся вне методов действий, следует использовать [[yii\web\Response::redirect()]] и непосредственно после
него — метод [[yii\web\Response::send()]]. Так можно быть уверенным, что к ответу не будет добавлено нежелательное
содержимое.
179 180 181 182 183 184 185 186 187 188

```php
\Yii::$app->response->redirect('http://example.com/new', 301)->send();
```

> Информация: По умолчанию метод [[yii\web\Response::redirect()]] устанавливает код состояния ответа равным 302, сообщая
  браузеру, что запрашиваемый ресурс *временно* находится по другому URI-адресу. Вы можете передать код состояния 
  301, чтобы сообщить браузеру, что ресурс перемещён *навсегда*.

Если текущий запрос является AJAX-запросом, отправка заголовка `Location` не заставит браузер автоматически 
189 190 191
осуществить перенаправление. Чтобы решить эту задачу, метод [[yii\web\Response::redirect()]] устанавливает значение
заголовка `X-Redirect` равным URL для перенаправления. На стороне клиента вы можете написать JavaScript-код для чтения
значения этого заголовка и перенаправления браузера соответственно.
192

193 194 195 196
> Информация: Yii поставляется с JavaScript-файлом `yii.js`, который предоставляет набор часто используемых
  JavaScript-утилит, включая и перенаправление браузера на основе заголовка `X-Redirect`. Следовательно, если вы
  используете этот JavaScript-файл (зарегистрировав пакет ресурсов [[yii\web\YiiAsset]]), вам не нужно писать
  дополнительный код для поддержки AJAX-перенаправления.
197 198


199
## Отправка файлов <span id="sending-files"></span>
200

201 202
Как и перенаправление браузера, отправка файлов является ещё одной возможностью, основанной на определённых HTTP-заголовках.
Yii предоставляет набор методов для решения различных задач по отправке файлов. Все они поддерживают HTTP-заголовок range.
203 204

* [[yii\web\Response::sendFile()]]: отправляет клиенту существующий файл.
205
* [[yii\web\Response::sendContentAsFile()]]: отправляет клиенту строку как файл.
206 207
* [[yii\web\Response::sendStreamAsFile()]]: отправляет клиенту существующий файловый поток как файл.

208 209 210
Эти методы имеют одинаковую сигнатуру и возвращают объект ответа. Если отправляемый файл очень велик, следует
использовать [[yii\web\Response::sendStreamAsFile()]], так как он более эффективно использует оперативную память.
Следующий пример показывает, как отправить файл в действии контроллера:
211 212 213 214 215 216 217 218

```php
public function actionDownload()
{
    return \Yii::$app->response->sendFile('path/to/file.txt');
}
```

219 220
При вызове метода отправки файла вне методов действий чтобы быть уверенным, что к ответу не будет добавлено никакое
нежелательное содержимое, следует вызвать сразу после него [[yii\web\Response::send()]].
221 222 223 224 225 226

```php
\Yii::$app->response->sendFile('path/to/file.txt')->send();
```

Некоторые Web-серверы поддерживают особый режим отправки файлов, который называется *X-Sendfile*. Идея в том, чтобы 
227 228 229 230
перенаправить запрос файла Web-серверу, который отдаст файл пользователю самостоятельно. В результате Web-приложение
может завершиться раньше, пока Web-сервер ещё пересылает файл. Чтобы использовать эту возможность, воспользуйтесь
методом [[yii\web\Response::xSendFile()]]. Далее приведены ссылки на то, как включить `X-Sendfile` для популярных
Web-серверов:
231 232 233 234 235 236 237 238

- Apache: [X-Sendfile](http://tn123.org/mod_xsendfile)
- Lighttpd v1.4: [X-LIGHTTPD-send-file](http://redmine.lighttpd.net/projects/lighttpd/wiki/X-LIGHTTPD-send-file)
- Lighttpd v1.5: [X-Sendfile](http://redmine.lighttpd.net/projects/lighttpd/wiki/X-LIGHTTPD-send-file)
- Nginx: [X-Accel-Redirect](http://wiki.nginx.org/XSendfile)
- Cherokee: [X-Sendfile and X-Accel-Redirect](http://www.cherokee-project.com/doc/other_goodies.html#x-sendfile)


239
## Отправка ответа <span id="sending-response"></span>
240

241 242 243
Содержимое ответа не отправляется пользователю до вызова метода [[yii\web\Response::send()]]. По умолчанию он вызывается
автоматически в конце метода [[yii\base\Application::run()]]. Однако, чтобы ответ был отправлен немедленно, вы можете
вызвать этот метод явно.
244

245
Для отправки ответа метод [[yii\web\Response::send()]] выполняет следующие шаги:
246 247

1. Инициируется событие [[yii\web\Response::EVENT_BEFORE_SEND]].
248 249
2. Для форматирования [[yii\web\Response::data|данных ответа]] в [[yii\web\Response::content|содержимое ответа]]
   вызывается метод [[yii\web\Response::prepare()]] .
250
3. Инициируется событие [[yii\web\Response::EVENT_AFTER_PREPARE]].
251 252
4. Для отправки зарегистрированных HTTP-заголовков вызывается метод [[yii\web\Response::sendHeaders()]].
5. Для отправки тела ответа вызывается метод [[yii\web\Response::sendContent()]].
253 254
6. Инициируется событие [[yii\web\Response::EVENT_AFTER_SEND]].

255 256
Повторный вызов [[yii\web\Response::send()]] игнорируется. Это означает, что если ответ уже отправлен, то к нему уже
ничего не добавить.
257 258

Как видно, метод [[yii\web\Response::send()]] инициирует несколько полезных событий. Реагируя на
259
эти события, можно настраивать или декорировать ответ.