_tmpArguments = $arguments; parent::__construct($message, $code, $previous); $this->_rawMessage = $message; $this->message = @vsprintf($message, $this->getArguments()); return; } /** * Get the backtrace. * Do not use \Exception::getTrace() any more. * * @return array */ public function getBacktrace() { if (null === $this->_trace) { $this->_trace = $this->getTrace(); } return $this->_trace; } /** * Get previous. * Do not use \Exception::getPrevious() any more. * * @return \Exception */ public function getPreviousThrow() { if (null === $this->_previous) { $this->_previous = $this->getPrevious(); } return $this->_previous; } /** * Get arguments for the message. * * @return array */ public function getArguments() { if (null === $this->_arguments) { $arguments = $this->_tmpArguments; if (!is_array($arguments)) { $arguments = [$arguments]; } foreach ($arguments as &$value) { if (null === $value) { $value = '(null)'; } } $this->_arguments = $arguments; unset($this->_tmpArguments); } return $this->_arguments; } /** * Get the raw message. * * @return string */ public function getRawMessage() { return $this->_rawMessage; } /** * Get the message already formatted. * * @return string */ public function getFormattedMessage() { return $this->getMessage(); } /** * Get the source of the exception (class, method, function, main etc.). * * @return string */ public function getFrom() { $trace = $this->getBacktrace(); $from = '{main}'; if (!empty($trace)) { $t = $trace[0]; $from = ''; if (isset($t['class'])) { $from .= $t['class'] . '::'; } if (isset($t['function'])) { $from .= $t['function'] . '()'; } } return $from; } /** * Raise an exception as a string. * * @param bool $previous Whether raise previous exception if exists. * @return string */ public function raise($previous = false) { $message = $this->getFormattedMessage(); $trace = $this->getBacktrace(); $file = '/dev/null'; $line = -1; $pre = $this->getFrom(); if (!empty($trace)) { $file = isset($trace['file']) ? $trace['file'] : null; $line = isset($trace['line']) ? $trace['line'] : null; } $pre .= ': '; try { $out = $pre . '(' . $this->getCode() . ') ' . $message . "\n" . 'in ' . $this->getFile() . ' at line ' . $this->getLine() . '.'; } catch (\Exception $e) { $out = $pre . '(' . $this->getCode() . ') ' . $message . "\n" . 'in ' . $file . ' around line ' . $line . '.'; } if (true === $previous && null !== $previous = $this->getPreviousThrow()) { $out .= "\n\n" . ' ⬇' . "\n\n" . 'Nested exception (' . get_class($previous) . '):' . "\n" . ($previous instanceof self ? $previous->raise(true) : $previous->getMessage()); } return $out; } /** * Catch uncaught exception (only \Hoa\Exception\Idle and children). * * @param \Throwable $exception The exception. * @return void * @throws \Throwable */ public static function uncaught($exception) { if (!($exception instanceof self)) { throw $exception; } while (0 < ob_get_level()) { ob_end_flush(); } echo 'Uncaught exception (' . get_class($exception) . '):' . "\n" . $exception->raise(true); return; } /** * String representation of object. * * @return string */ public function __toString() { return $this->raise(); } /** * Enable uncaught exception handler. * This is restricted to Hoa's exceptions only. * * @param bool $enable Enable. * @return mixed */ public static function enableUncaughtHandler($enable = true) { if (false === $enable) { return restore_exception_handler(); } return set_exception_handler(function ($exception) { return self::uncaught($exception); }); } }