PluginProbe ʕ •ᴥ•ʔ
Matomo Analytics – Powerful, Privacy-First Insights for WordPress / 5.2.2
Matomo Analytics – Powerful, Privacy-First Insights for WordPress v5.2.2
5.11.1 5.11.0 5.10.2 5.10.1 trunk 1.0.2 1.0.3 1.0.4 1.0.5 1.0.6 1.1.0 1.1.1 1.1.2 1.1.3 1.2.0 1.3.0 1.3.1 1.3.2 4.0.0 4.0.1 4.0.2 4.0.3 4.0.4 4.1.0 4.1.1 4.1.2 4.1.3 4.10.0 4.11.0 4.12.0 4.13.0 4.13.2 4.13.3 4.13.4 4.13.5 4.14.0 4.14.1 4.14.2 4.15.0 4.15.1 4.15.2 4.15.3 4.2.0 4.3.0 4.3.1 4.4.1 4.4.2 4.5.0 4.6.0 5.0.1 5.0.2 5.0.3 5.0.4 5.0.5 5.0.6 5.0.7 5.0.8 5.1.0 5.1.1 5.1.2 5.1.3 5.1.4 5.1.5 5.1.6 5.1.7 5.10.0 5.2.0 5.2.1 5.2.2 5.3.0 5.3.1 5.3.2 5.3.3 5.6.0 5.6.1 5.7.0 5.7.1 5.8.0 5.8.1 5.8.2
matomo / app / core / ExceptionHandler.php
matomo / app / core Last commit date
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