mirror of
				https://github.com/yiisoft/yii2.git
				synced 2025-10-31 10:39:59 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			210 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			210 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| Обработка ошибок
 | ||
| ================
 | ||
| 
 | ||
| В состав Yii входит встроенный [[yii\web\ErrorHandler|обработчик ошибок]], делающий работу с ошибками гораздо более
 | ||
| приятным занятием. А именно:
 | ||
| 
 | ||
| * Все не фатальные ошибки PHP (то есть warning, notice) конвертируются в исключения, которые можно перехватывать.
 | ||
| * Исключения и фатальные ошибки PHP отображаются в режиме отладки с детальным стеком вызовов и исходным кодом.
 | ||
| * Можно использовать для отображения ошибок [действие контроллера](structure-controllers.md#actions).
 | ||
| * Поддерживаются различные форматы ответа.
 | ||
| 
 | ||
| По умолчанию [[yii\web\ErrorHandler|обработчик ошибок]] включен. Вы можете выключить его объявив константу
 | ||
| `YII_ENABLE_ERROR_HANDLER` со значением `false` во [входном скрипте](structure-entry-scripts.md) вашего приложения.
 | ||
| 
 | ||
| 
 | ||
| ## Использование обработчика ошибок <span id="using-error-handler"></span>
 | ||
| 
 | ||
| [[yii\web\ErrorHandler|Обработчик ошибок]] регистрируется в качестве [компонента приложения](structure-application-components.md)
 | ||
| с именем `errorHandler`. Вы можете настраивать его следующим образом:
 | ||
| 
 | ||
| ```php
 | ||
| return [
 | ||
|     'components' => [
 | ||
|         'errorHandler' => [
 | ||
|             'maxSourceLines' => 20,
 | ||
|         ],
 | ||
|     ],
 | ||
| ];
 | ||
| ```
 | ||
| 
 | ||
| С приведённой выше конфигурацией на странице ошибки будет отображаться до 20 строк исходного кода.
 | ||
| 
 | ||
| Как уже было упомянуто, обработчик ошибок конвертирует все не фатальные ошибки PHP в перехватываемые исключения.
 | ||
| Это означает что можно поступать с ошибками следующим образом:
 | ||
| 
 | ||
| ```php
 | ||
| use Yii;
 | ||
| use yii\base\ErrorException;
 | ||
| 
 | ||
| try {
 | ||
|     10/0;
 | ||
| } catch (ErrorException $e) {
 | ||
|     Yii::warning("Деление на ноль.");
 | ||
| }
 | ||
| 
 | ||
| // можно продолжать выполнение
 | ||
| ```
 | ||
| 
 | ||
| Если вам необходимо показать пользователю страницу с ошибкой, говорящей ему о том, что его запрос не верен или не
 | ||
| должен был быть сделан, вы можете выкинуть [[yii\web\HttpException|исключение HTTP]], такое как 
 | ||
| [[yii\web\NotFoundHttpException]]. Обработчик ошибок корректно выставит статус код HTTP для ответа и использует
 | ||
| подходящий вид страницы ошибки.
 | ||
| 
 | ||
| ```php
 | ||
| use yii\web\NotFoundHttpException;
 | ||
|  
 | ||
| throw new NotFoundHttpException();
 | ||
| ```
 | ||
| 
 | ||
| ## Настройка отображения ошибок <span id="customizing-error-display"></span>
 | ||
| 
 | ||
| [[yii\web\ErrorHandler|Обработчик ошибок]] меняет отображение ошибок в зависимости от значения константы `YII_DEBUG`.
 | ||
| При `YII_DEBUG` равной `true` (режим отладки), обработчик ошибок будет отображать для облегчения отладки детальный стек
 | ||
| вызовов и исходный код. При `YII_DEBUG` равной `false` отображается только сообщение об ошибке, тем самым не позволяя
 | ||
| получить информацию о внутренностях приложения.
 | ||
| 
 | ||
| > Info: Если исключение является наследником [[yii\base\UserException]], стек вызовов не отображается вне
 | ||
|   зависимости от значения `YII_DEBUG` так как такие исключения считаются ошибками пользователя и исправлять что-либо
 | ||
|   разработчику не требуется.
 | ||
| 
 | ||
| По умолчанию [[yii\web\ErrorHandler|обработчик ошибок]] показывает ошибки используя два [представления](structure-views.md):
 | ||
| 
 | ||
| * `@yii/views/errorHandler/error.php`: используется для отображения ошибок БЕЗ стека вызовов.
 | ||
|   При `YII_DEBUG` равной `false` используется только это преставление.
 | ||
| * `@yii/views/errorHandler/exception.php`: используется для отображения ошибок СО стеком вызовов.
 | ||
|  
 | ||
| Вы можете настроить свойства [[yii\web\ErrorHandler::errorView|errorView]] и [[yii\web\ErrorHandler::exceptionView|exceptionView]]
 | ||
| для того, чтобы использовать свои представления.
 | ||
|   
 | ||
| ### Использование действий для отображения ошибок <span id="using-error-actions"></span>
 | ||
| 
 | ||
| Лучшим способом изменения отображения ошибок является использование [действий](structure-controllers.md) путём
 | ||
| конфигурирования свойства [[yii\web\ErrorHandler::errorAction|errorAction]] компонента `errorHandler`:
 | ||
| 
 | ||
| ```php
 | ||
| // ...
 | ||
| 'components' => [
 | ||
|     // ...
 | ||
|     'errorHandler' => [
 | ||
|         'errorAction' => 'site/error',
 | ||
|     ],
 | ||
| ]
 | ||
| ```
 | ||
| 
 | ||
| Свойство [[yii\web\ErrorHandler::errorAction|errorAction]] принимает [маршрут](structure-controllers.md#routes)
 | ||
| действия. Конфигурация выше означает, что для отображения ошибки без стека вызовов будет использовано действие `site/error`.
 | ||
| 
 | ||
| Само действие можно реализовать следующим образом:
 | ||
| 
 | ||
| ```php
 | ||
| namespace app\controllers;
 | ||
| 
 | ||
| use Yii;
 | ||
| use yii\web\Controller;
 | ||
| 
 | ||
| class SiteController extends Controller
 | ||
| {
 | ||
|     public function actions()
 | ||
|     {
 | ||
|         return [
 | ||
|             'error' => [
 | ||
|                 'class' => 'yii\web\ErrorAction',
 | ||
|             ],
 | ||
|         ];
 | ||
|     }
 | ||
| }
 | ||
| ```
 | ||
| 
 | ||
| Приведённый выше код задаёт действие `error` используя класс [[yii\web\ErrorAction]], который рендерит ошибку используя
 | ||
| отображение `error`.
 | ||
| 
 | ||
| Вместо использования [[yii\web\ErrorAction]] вы можете создать действие `error` как обычный метод:
 | ||
| 
 | ||
| ```php
 | ||
| public function actionError()
 | ||
| {
 | ||
|     $exception = Yii::$app->errorHandler->exception;
 | ||
|     if ($exception !== null) {
 | ||
|         return $this->render('error', ['exception' => $exception]);
 | ||
|     }
 | ||
| }
 | ||
| ```
 | ||
| 
 | ||
| Вы должны создать файл представления `views/site/error.php`. В этом файле, если используется [[yii\web\ErrorAction]],
 | ||
| вам доступны следующие переменные:
 | ||
| 
 | ||
| * `name`: имя ошибки;
 | ||
| * `message`: текст ошибки;
 | ||
| * `exception`: объект исключения, из которого можно получить дополнительную информацию, такую как статус HTTP,
 | ||
|   код ошибки, стек вызовов и т.д.
 | ||
|  
 | ||
| > Info: Если вы используете шаблоны приложения [basic](start-installation.md) или [advanced](tutorial-advanced-app.md),
 | ||
|   действие error и файл представления уже созданы за вас.
 | ||
|   
 | ||
| ### Изменение формата ответа <span id="error-format"></span>
 | ||
| 
 | ||
| Обработчик ошибок отображает ошибки в соответствии с выбранным форматом [ответа](runtime-responses.md).
 | ||
| Если [[yii\web\Response::format|формат ответа]] задан как `html`, будут использованы представления для ошибок и
 | ||
| исключений, как описывалось ранее. Для остальных форматов ответа обработчик ошибок присваивает массив данных,
 | ||
| представляющий ошибку свойству [[yii\web\Response::data]]. Оно далее конвертируется в необходимый формат. Например,
 | ||
| если используется формат ответа `json`, вы получите подобный ответ:
 | ||
| 
 | ||
| ```
 | ||
| HTTP/1.1 404 Not Found
 | ||
| Date: Sun, 02 Mar 2014 05:31:43 GMT
 | ||
| Server: Apache/2.2.26 (Unix) DAV/2 PHP/5.4.20 mod_ssl/2.2.26 OpenSSL/0.9.8y
 | ||
| Transfer-Encoding: chunked
 | ||
| Content-Type: application/json; charset=UTF-8
 | ||
| 
 | ||
| {
 | ||
|     "name": "Not Found Exception",
 | ||
|     "message": "The requested resource was not found.",
 | ||
|     "code": 0,
 | ||
|     "status": 404
 | ||
| }
 | ||
| ```
 | ||
| 
 | ||
| Изменить формат можно в обработчике события `beforeSend` компонента `response` в конфигурации приложения:
 | ||
| 
 | ||
| ```php
 | ||
| return [
 | ||
|     // ...
 | ||
|     'components' => [
 | ||
|         'response' => [
 | ||
|             'class' => 'yii\web\Response',
 | ||
|             'on beforeSend' => function ($event) {
 | ||
|                 $response = $event->sender;
 | ||
|                 if ($response->data !== null) {
 | ||
|                     $response->data = [
 | ||
|                         'success' => $response->isSuccessful,
 | ||
|                         'data' => $response->data,
 | ||
|                     ];
 | ||
|                     $response->statusCode = 200;
 | ||
|                 }
 | ||
|             },
 | ||
|         ],
 | ||
|     ],
 | ||
| ];
 | ||
| ```
 | ||
| 
 | ||
| Приведённый код изменит формат ответа на подобный:
 | ||
| 
 | ||
| ```
 | ||
| HTTP/1.1 200 OK
 | ||
| Date: Sun, 02 Mar 2014 05:31:43 GMT
 | ||
| Server: Apache/2.2.26 (Unix) DAV/2 PHP/5.4.20 mod_ssl/2.2.26 OpenSSL/0.9.8y
 | ||
| Transfer-Encoding: chunked
 | ||
| Content-Type: application/json; charset=UTF-8
 | ||
| 
 | ||
| {
 | ||
|     "success": false,
 | ||
|     "data": {
 | ||
|         "name": "Not Found Exception",
 | ||
|         "message": "The requested resource was not found.",
 | ||
|         "code": 0,
 | ||
|         "status": 404
 | ||
|     }
 | ||
| }
 | ||
| ```
 | 
