API
1 year ago
Access
1 year ago
Application
1 year ago
Archive
1 year ago
ArchiveProcessor
1 year ago
Archiver
2 years ago
AssetManager
1 year ago
Auth
1 year ago
Category
2 years ago
Changes
1 year ago
CliMulti
1 year ago
Columns
1 year ago
Concurrency
1 year ago
Config
1 year ago
Container
1 year ago
CronArchive
1 year ago
DataAccess
1 year ago
DataFiles
2 years ago
DataTable
1 year ago
Db
1 year ago
DeviceDetector
1 year ago
Email
2 years ago
Exception
1 year ago
Http
1 year ago
Intl
1 year ago
Log
2 years ago
Mail
1 year ago
Measurable
1 year ago
Menu
1 year ago
Metrics
1 year ago
Notification
1 year ago
Period
1 year ago
Plugin
1 year ago
ProfessionalServices
1 year ago
Report
1 year ago
ReportRenderer
1 year ago
Scheduler
1 year ago
Segment
1 year ago
Session
1 year ago
Settings
1 year ago
Tracker
1 year ago
Translation
1 year ago
Twig
1 year ago
UpdateCheck
1 year ago
Updater
1 year ago
Updates
1 year ago
Validators
1 year ago
View
1 year ago
ViewDataTable
1 year ago
Visualization
1 year ago
Widget
1 year ago
.htaccess
2 years ago
Access.php
1 year ago
Archive.php
1 year ago
ArchiveProcessor.php
1 year ago
AssetManager.php
1 year ago
Auth.php
2 years ago
AuthResult.php
2 years ago
BaseFactory.php
2 years ago
Cache.php
2 years ago
CacheId.php
1 year ago
CliMulti.php
1 year ago
Common.php
1 year ago
Config.php
1 year ago
Console.php
1 year ago
Context.php
2 years ago
Cookie.php
1 year ago
CronArchive.php
1 year ago
DI.php
1 year ago
DataArray.php
1 year ago
DataTable.php
1 year ago
Date.php
1 year ago
Db.php
1 year ago
DbHelper.php
1 year ago
Development.php
1 year ago
ErrorHandler.php
1 year ago
EventDispatcher.php
1 year ago
ExceptionHandler.php
1 year ago
FileIntegrity.php
1 year ago
Filechecks.php
1 year ago
Filesystem.php
1 year ago
FrontController.php
1 year ago
Http.php
1 year ago
IP.php
1 year ago
Log.php
2 years ago
LogDeleter.php
1 year ago
Mail.php
1 year ago
Metrics.php
1 year ago
NoAccessException.php
2 years ago
Nonce.php
1 year ago
Notification.php
1 year ago
NumberFormatter.php
1 year ago
Option.php
1 year ago
Period.php
1 year ago
Piwik.php
1 year ago
Plugin.php
1 year ago
Process.php
1 year ago
Profiler.php
1 year ago
ProxyHeaders.php
2 years ago
ProxyHttp.php
1 year ago
QuickForm2.php
1 year ago
RankingQuery.php
1 year ago
ReportRenderer.php
1 year ago
Request.php
1 year ago
Segment.php
1 year ago
Sequence.php
2 years ago
Session.php
1 year ago
SettingsPiwik.php
1 year ago
SettingsServer.php
1 year ago
Singleton.php
2 years ago
Site.php
1 year ago
SiteContentDetector.php
1 year ago
SupportedBrowser.php
2 years ago
TCPDF.php
1 year ago
Theme.php
1 year ago
Timer.php
2 years ago
Tracker.php
1 year ago
Twig.php
1 year ago
Unzip.php
1 year ago
UpdateCheck.php
1 year ago
Updater.php
1 year ago
UpdaterErrorException.php
2 years ago
Updates.php
1 year ago
Url.php
1 year ago
UrlHelper.php
1 year ago
Version.php
1 year ago
View.php
1 year ago
bootstrap.php
1 year ago
dispatch.php
2 years ago
testMinimumPhpVersion.php
2 years ago
ExceptionHandler.php
168 lines
| 1 | <?php |
| 2 | |
| 3 | /** |
| 4 | * Matomo - free/libre analytics platform |
| 5 | * |
| 6 | * @link https://matomo.org |
| 7 | * @license https://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later |
| 8 | */ |
| 9 | namespace Piwik; |
| 10 | |
| 11 | use Piwik\Exception\DI\DependencyException; |
| 12 | use Exception; |
| 13 | use Piwik\API\Request; |
| 14 | use Piwik\API\ResponseBuilder; |
| 15 | use Piwik\Container\ContainerDoesNotExistException; |
| 16 | use Piwik\Container\StaticContainer; |
| 17 | use Piwik\Exception\IRedirectException; |
| 18 | use Piwik\Plugins\CoreAdminHome\CustomLogo; |
| 19 | use Piwik\Plugins\Monolog\Processor\ExceptionToTextProcessor; |
| 20 | use Piwik\Log\LoggerInterface; |
| 21 | /** |
| 22 | * Contains Piwik's uncaught exception handler. |
| 23 | */ |
| 24 | class ExceptionHandler |
| 25 | { |
| 26 | public static function setUp() |
| 27 | { |
| 28 | set_exception_handler(['Piwik\\ExceptionHandler', 'handleException']); |
| 29 | } |
| 30 | /** |
| 31 | * @param Exception|\Throwable $exception |
| 32 | */ |
| 33 | public static function handleException($exception) |
| 34 | { |
| 35 | if (\Piwik\Common::isPhpCliMode()) { |
| 36 | self::dieWithCliError($exception); |
| 37 | } |
| 38 | self::dieWithHtmlErrorPage($exception); |
| 39 | } |
| 40 | /** |
| 41 | * @param Exception|\Throwable $exception |
| 42 | */ |
| 43 | public static function dieWithCliError($exception) |
| 44 | { |
| 45 | self::logException($exception); |
| 46 | $message = $exception->getMessage(); |
| 47 | if (!method_exists($exception, 'isHtmlMessage') || !$exception->isHtmlMessage()) { |
| 48 | $message = strip_tags(str_replace('<br />', \PHP_EOL, $message)); |
| 49 | } |
| 50 | $message = sprintf("Uncaught exception in %s line %d:\n%s\n", $exception->getFile(), $exception->getLine(), ExceptionToTextProcessor::getMessageAndWholeBacktrace($exception)); |
| 51 | echo $message; |
| 52 | exit(1); |
| 53 | } |
| 54 | /** |
| 55 | * @param Exception|\Throwable $exception |
| 56 | */ |
| 57 | public static function dieWithHtmlErrorPage($exception) |
| 58 | { |
| 59 | // Set an appropriate HTTP response code. |
| 60 | switch (\true) { |
| 61 | case ($exception instanceof \Piwik\Http\HttpCodeException || $exception instanceof \Piwik\Exception\NotSupportedBrowserException) && $exception->getCode() > 0: |
| 62 | // For these exception types, use the exception-provided error code. |
| 63 | http_response_code($exception->getCode()); |
| 64 | break; |
| 65 | case $exception instanceof \Piwik\Exception\NotYetInstalledException: |
| 66 | http_response_code(404); |
| 67 | break; |
| 68 | default: |
| 69 | http_response_code(500); |
| 70 | } |
| 71 | // Log the error with an appropriate loglevel. |
| 72 | switch (\true) { |
| 73 | case $exception instanceof \Piwik\Exception\NotSupportedBrowserException: |
| 74 | // These unsupported browsers are really a client-side problem, so log only at DEBUG level. |
| 75 | self::logException($exception, \Piwik\Log::DEBUG); |
| 76 | break; |
| 77 | default: |
| 78 | self::logException($exception); |
| 79 | } |
| 80 | \Piwik\Common::sendHeader('Content-Type: text/html; charset=utf-8'); |
| 81 | try { |
| 82 | echo self::getErrorResponse($exception); |
| 83 | } catch (Exception $e) { |
| 84 | // When there are failures while generating the HTML error response itself, |
| 85 | // we simply print out the error message instead. |
| 86 | echo $exception->getMessage(); |
| 87 | } |
| 88 | exit(1); |
| 89 | } |
| 90 | /** |
| 91 | * @param Exception|\Throwable $ex |
| 92 | */ |
| 93 | private static function getErrorResponse($ex) |
| 94 | { |
| 95 | $debugTrace = $ex->getTraceAsString(); |
| 96 | $message = $ex->getMessage(); |
| 97 | $isHtmlMessage = method_exists($ex, 'isHtmlMessage') && $ex->isHtmlMessage(); |
| 98 | if (!$isHtmlMessage && Request::isApiRequest($_GET)) { |
| 99 | $outputFormat = strtolower(\Piwik\Common::getRequestVar('format', 'xml', 'string', $_GET + $_POST)); |
| 100 | $response = new ResponseBuilder($outputFormat); |
| 101 | return $response->getResponseException($ex); |
| 102 | } elseif (!$isHtmlMessage) { |
| 103 | $message = \Piwik\Common::sanitizeInputValue($message); |
| 104 | } |
| 105 | $logoHeaderUrl = 'plugins/Morpheus/images/logo.svg'; |
| 106 | $logoFaviconUrl = 'plugins/CoreHome/images/favicon.png'; |
| 107 | try { |
| 108 | $logo = new CustomLogo(); |
| 109 | if ($logo->hasSVGLogo()) { |
| 110 | $logoHeaderUrl = $logo->getSVGLogoUrl(); |
| 111 | } else { |
| 112 | $logoHeaderUrl = $logo->getHeaderLogoUrl(); |
| 113 | } |
| 114 | $logoFaviconUrl = $logo->getPathUserFavicon(); |
| 115 | } catch (Exception $ex) { |
| 116 | try { |
| 117 | \Piwik\Log::debug($ex); |
| 118 | } catch (\Exception $otherEx) { |
| 119 | // DI container may not be setup at this point |
| 120 | } |
| 121 | } |
| 122 | // Unsupported browser errors shouldn't be written to the web server log. At DEBUG logging level this error will |
| 123 | // be written to the application log instead |
| 124 | $writeErrorLog = !$ex instanceof \Piwik\Exception\NotSupportedBrowserException; |
| 125 | $redirectUrl = null; |
| 126 | $countdownToRedirect = null; |
| 127 | if ($ex instanceof IRedirectException) { |
| 128 | $redirectUrl = $ex->getRedirectionUrl(); |
| 129 | $countdownToRedirect = $ex->getCountdown(); |
| 130 | } |
| 131 | $hostname = \Piwik\Url::getRFCValidHostname(); |
| 132 | $hostStr = $hostname ? "[{$hostname}] " : '- '; |
| 133 | $result = Piwik_GetErrorMessagePage($message, $debugTrace, \true, \true, $logoHeaderUrl, $logoFaviconUrl, null, $hostStr, $writeErrorLog, $redirectUrl, $countdownToRedirect); |
| 134 | try { |
| 135 | /** |
| 136 | * Triggered before a Piwik error page is displayed to the user. |
| 137 | * |
| 138 | * This event can be used to modify the content of the error page that is displayed when |
| 139 | * an exception is caught. |
| 140 | * |
| 141 | * @param string &$result The HTML of the error page. |
| 142 | * @param Exception $ex The Exception displayed in the error page. |
| 143 | */ |
| 144 | \Piwik\Piwik::postEvent('FrontController.modifyErrorPage', [&$result, $ex]); |
| 145 | } catch (ContainerDoesNotExistException $ex) { |
| 146 | // this can happen when an error occurs before the Piwik environment is created |
| 147 | } |
| 148 | return $result; |
| 149 | } |
| 150 | private static function logException($exception, $loglevel = \Piwik\Log::ERROR) |
| 151 | { |
| 152 | try { |
| 153 | switch ($loglevel) { |
| 154 | case \Piwik\Log::DEBUG: |
| 155 | StaticContainer::get(LoggerInterface::class)->debug('Uncaught exception: {exception}', ['exception' => $exception, 'ignoreInScreenWriter' => \true]); |
| 156 | break; |
| 157 | case \Piwik\Log::ERROR: |
| 158 | default: |
| 159 | StaticContainer::get(LoggerInterface::class)->error('Uncaught exception: {exception}', ['exception' => $exception, 'ignoreInScreenWriter' => \true]); |
| 160 | } |
| 161 | } catch (DependencyException $ex) { |
| 162 | // ignore (occurs if exception is thrown when resolving DI entries) |
| 163 | } catch (ContainerDoesNotExistException $ex) { |
| 164 | // ignore |
| 165 | } |
| 166 | } |
| 167 | } |
| 168 |