300 lines
8.8 KiB
PHP
300 lines
8.8 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* Zend Framework
|
||
|
*
|
||
|
* LICENSE
|
||
|
*
|
||
|
* This source file is subject to the new BSD license that is bundled
|
||
|
* with this package in the file LICENSE.txt.
|
||
|
* It is also available through the world-wide-web at this URL:
|
||
|
* http://framework.zend.com/license/new-bsd
|
||
|
* If you did not receive a copy of the license and are unable to
|
||
|
* obtain it through the world-wide-web, please send an email
|
||
|
* to license@zend.com so we can send you a copy immediately.
|
||
|
*
|
||
|
* @category Zend
|
||
|
* @package Zend_Controller
|
||
|
* @subpackage Plugins
|
||
|
* @copyright Copyright (c) 2005-2014 Zend Technologies USA Inc. (http://www.zend.com)
|
||
|
* @license http://framework.zend.com/license/new-bsd New BSD License
|
||
|
*/
|
||
|
|
||
|
/** Zend_Controller_Plugin_Abstract */
|
||
|
|
||
|
/**
|
||
|
* Handle exceptions that bubble up based on missing controllers, actions, or
|
||
|
* application errors, and forward to an error handler.
|
||
|
*
|
||
|
* @uses Zend_Controller_Plugin_Abstract
|
||
|
* @category Zend
|
||
|
* @package Zend_Controller
|
||
|
* @subpackage Plugins
|
||
|
* @copyright Copyright (c) 2005-2014 Zend Technologies USA Inc. (http://www.zend.com)
|
||
|
* @license http://framework.zend.com/license/new-bsd New BSD License
|
||
|
* @version $Id$
|
||
|
*/
|
||
|
class Zend_Controller_Plugin_ErrorHandler extends Zend_Controller_Plugin_Abstract
|
||
|
{
|
||
|
/**
|
||
|
* Const - No controller exception; controller does not exist
|
||
|
*/
|
||
|
const EXCEPTION_NO_CONTROLLER = 'EXCEPTION_NO_CONTROLLER';
|
||
|
|
||
|
/**
|
||
|
* Const - No action exception; controller exists, but action does not
|
||
|
*/
|
||
|
const EXCEPTION_NO_ACTION = 'EXCEPTION_NO_ACTION';
|
||
|
|
||
|
/**
|
||
|
* Const - No route exception; no routing was possible
|
||
|
*/
|
||
|
const EXCEPTION_NO_ROUTE = 'EXCEPTION_NO_ROUTE';
|
||
|
|
||
|
/**
|
||
|
* Const - Other Exception; exceptions thrown by application controllers
|
||
|
*/
|
||
|
const EXCEPTION_OTHER = 'EXCEPTION_OTHER';
|
||
|
|
||
|
/**
|
||
|
* Module to use for errors; defaults to default module in dispatcher
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $_errorModule;
|
||
|
|
||
|
/**
|
||
|
* Controller to use for errors; defaults to 'error'
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $_errorController = 'error';
|
||
|
|
||
|
/**
|
||
|
* Action to use for errors; defaults to 'error'
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $_errorAction = 'error';
|
||
|
|
||
|
/**
|
||
|
* Flag; are we already inside the error handler loop?
|
||
|
* @var bool
|
||
|
*/
|
||
|
protected $_isInsideErrorHandlerLoop = false;
|
||
|
|
||
|
/**
|
||
|
* Exception count logged at first invocation of plugin
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $_exceptionCountAtFirstEncounter = 0;
|
||
|
|
||
|
/**
|
||
|
* Constructor
|
||
|
*
|
||
|
* Options may include:
|
||
|
* - module
|
||
|
* - controller
|
||
|
* - action
|
||
|
*
|
||
|
* @param Array $options
|
||
|
* @return void
|
||
|
*/
|
||
|
public function __construct(Array $options = array())
|
||
|
{
|
||
|
$this->setErrorHandler($options);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* setErrorHandler() - setup the error handling options
|
||
|
*
|
||
|
* @param array $options
|
||
|
* @return Zend_Controller_Plugin_ErrorHandler
|
||
|
*/
|
||
|
public function setErrorHandler(Array $options = array())
|
||
|
{
|
||
|
if (isset($options['module'])) {
|
||
|
$this->setErrorHandlerModule($options['module']);
|
||
|
}
|
||
|
if (isset($options['controller'])) {
|
||
|
$this->setErrorHandlerController($options['controller']);
|
||
|
}
|
||
|
if (isset($options['action'])) {
|
||
|
$this->setErrorHandlerAction($options['action']);
|
||
|
}
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the module name for the error handler
|
||
|
*
|
||
|
* @param string $module
|
||
|
* @return Zend_Controller_Plugin_ErrorHandler
|
||
|
*/
|
||
|
public function setErrorHandlerModule($module)
|
||
|
{
|
||
|
$this->_errorModule = (string) $module;
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve the current error handler module
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getErrorHandlerModule()
|
||
|
{
|
||
|
if (null === $this->_errorModule) {
|
||
|
$this->_errorModule = Zend_Controller_Front::getInstance()->getDispatcher()->getDefaultModule();
|
||
|
}
|
||
|
return $this->_errorModule;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the controller name for the error handler
|
||
|
*
|
||
|
* @param string $controller
|
||
|
* @return Zend_Controller_Plugin_ErrorHandler
|
||
|
*/
|
||
|
public function setErrorHandlerController($controller)
|
||
|
{
|
||
|
$this->_errorController = (string) $controller;
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve the current error handler controller
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getErrorHandlerController()
|
||
|
{
|
||
|
return $this->_errorController;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the action name for the error handler
|
||
|
*
|
||
|
* @param string $action
|
||
|
* @return Zend_Controller_Plugin_ErrorHandler
|
||
|
*/
|
||
|
public function setErrorHandlerAction($action)
|
||
|
{
|
||
|
$this->_errorAction = (string) $action;
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve the current error handler action
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getErrorHandlerAction()
|
||
|
{
|
||
|
return $this->_errorAction;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Route shutdown hook -- Ccheck for router exceptions
|
||
|
*
|
||
|
* @param Zend_Controller_Request_Abstract $request
|
||
|
*/
|
||
|
public function routeShutdown(Zend_Controller_Request_Abstract $request)
|
||
|
{
|
||
|
$this->_handleError($request);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Pre dispatch hook -- check for exceptions and dispatch error handler if
|
||
|
* necessary
|
||
|
*
|
||
|
* @param Zend_Controller_Request_Abstract $request
|
||
|
*/
|
||
|
public function preDispatch(Zend_Controller_Request_Abstract $request)
|
||
|
{
|
||
|
$this->_handleError($request);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Post dispatch hook -- check for exceptions and dispatch error handler if
|
||
|
* necessary
|
||
|
*
|
||
|
* @param Zend_Controller_Request_Abstract $request
|
||
|
*/
|
||
|
public function postDispatch(Zend_Controller_Request_Abstract $request)
|
||
|
{
|
||
|
$this->_handleError($request);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Handle errors and exceptions
|
||
|
*
|
||
|
* If the 'noErrorHandler' front controller flag has been set,
|
||
|
* returns early.
|
||
|
*
|
||
|
* @param Zend_Controller_Request_Abstract $request
|
||
|
* @return void
|
||
|
*/
|
||
|
protected function _handleError(Zend_Controller_Request_Abstract $request)
|
||
|
{
|
||
|
$frontController = Zend_Controller_Front::getInstance();
|
||
|
if ($frontController->getParam('noErrorHandler')) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$response = $this->getResponse();
|
||
|
|
||
|
if ($this->_isInsideErrorHandlerLoop) {
|
||
|
$exceptions = $response->getException();
|
||
|
if (count($exceptions) > $this->_exceptionCountAtFirstEncounter) {
|
||
|
// Exception thrown by error handler; tell the front controller to throw it
|
||
|
$frontController->throwExceptions(true);
|
||
|
throw array_pop($exceptions);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// check for an exception AND allow the error handler controller the option to forward
|
||
|
if (($response->isException()) && (!$this->_isInsideErrorHandlerLoop)) {
|
||
|
$this->_isInsideErrorHandlerLoop = true;
|
||
|
|
||
|
// Get exception information
|
||
|
$error = new ArrayObject(array(), ArrayObject::ARRAY_AS_PROPS);
|
||
|
$exceptions = $response->getException();
|
||
|
$exception = $exceptions[0];
|
||
|
$exceptionType = get_class($exception);
|
||
|
$error->exception = $exception;
|
||
|
switch ($exceptionType) {
|
||
|
case 'Zend_Controller_Router_Exception':
|
||
|
if (404 == $exception->getCode()) {
|
||
|
$error->type = self::EXCEPTION_NO_ROUTE;
|
||
|
} else {
|
||
|
$error->type = self::EXCEPTION_OTHER;
|
||
|
}
|
||
|
break;
|
||
|
case 'Zend_Controller_Dispatcher_Exception':
|
||
|
$error->type = self::EXCEPTION_NO_CONTROLLER;
|
||
|
break;
|
||
|
case 'Zend_Controller_Action_Exception':
|
||
|
if (404 == $exception->getCode()) {
|
||
|
$error->type = self::EXCEPTION_NO_ACTION;
|
||
|
} else {
|
||
|
$error->type = self::EXCEPTION_OTHER;
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
$error->type = self::EXCEPTION_OTHER;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Keep a copy of the original request
|
||
|
$error->request = clone $request;
|
||
|
|
||
|
// get a count of the number of exceptions encountered
|
||
|
$this->_exceptionCountAtFirstEncounter = count($exceptions);
|
||
|
|
||
|
// Forward to the error handler
|
||
|
$request->setParam('error_handler', $error)
|
||
|
->setModuleName($this->getErrorHandlerModule())
|
||
|
->setControllerName($this->getErrorHandlerController())
|
||
|
->setActionName($this->getErrorHandlerAction())
|
||
|
->setDispatched(false);
|
||
|
}
|
||
|
}
|
||
|
}
|