]> git.mxchange.org Git - friendica.git/blobdiff - src/Module/Special/HTTPException.php
Remove DI dependency from Module\Contact\Profile
[friendica.git] / src / Module / Special / HTTPException.php
index 36c770fba78d356493cdd809f2f4230d02c8c1c4..2cac142fca80e731f39ee2a499ba41d3b8acee5b 100644 (file)
@@ -1,12 +1,33 @@
 <?php
-
+/**
+ * @copyright Copyright (C) 2010-2023, the Friendica project
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as
+ * published by the Free Software Foundation, either version 3 of the
+ * License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program.  If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
 
 namespace Friendica\Module\Special;
 
+use Friendica\App\Arguments;
+use Friendica\App\Request;
 use Friendica\Core\L10n;
-use Friendica\Core\Logger;
 use Friendica\Core\Renderer;
+use Friendica\Core\Session\Model\UserSession;
 use Friendica\Core\System;
+use Psr\Log\LoggerInterface;
 
 /**
  * This special module displays HTTPException when they are thrown in modules.
@@ -15,77 +36,110 @@ use Friendica\Core\System;
  */
 class HTTPException
 {
+       /** @var L10n */
+       protected $l10n;
+       /** @var LoggerInterface */
+       protected $logger;
+       /** @var Arguments */
+       protected $args;
+       /** @var bool */
+       protected $isSiteAdmin;
+       /** @var array */
+       protected $server;
+       /** @var string */
+       protected $requestId;
+
+       public function __construct(L10n $l10n, LoggerInterface $logger, Arguments $args, UserSession $session, Request $request, array $server = [])
+       {
+               $this->logger      = $logger;
+               $this->l10n        = $l10n;
+               $this->args        = $args;
+               $this->isSiteAdmin = $session->isSiteAdmin();
+               $this->server      = $server;
+               $this->requestId   = $request->getRequestId();
+       }
+
        /**
         * Generates the necessary template variables from the caught HTTPException.
         *
         * Fills in the blanks if title or descriptions aren't provided by the exception.
         *
         * @param \Friendica\Network\HTTPException $e
+        *
         * @return array ['$title' => ..., '$description' => ...]
         */
-       private static function getVars(\Friendica\Network\HTTPException $e)
+       private function getVars(\Friendica\Network\HTTPException $e)
        {
-               $message = $e->getMessage();
-
-               $titles = [
-                       200 => 'OK',
-                       400 => L10n::t('Bad Request'),
-                       401 => L10n::t('Unauthorized'),
-                       403 => L10n::t('Forbidden'),
-                       404 => L10n::t('Not Found'),
-                       500 => L10n::t('Internal Server Error'),
-                       503 => L10n::t('Service Unavailable'),
+               // Explanations are mostly taken from https://en.wikipedia.org/wiki/List_of_HTTP_status_codes
+               $vars = [
+                       '$title'       => $e->getDescription() ?: 'Error ' . $e->getCode(),
+                       '$message'     => $e->getMessage() ?: $e->getExplanation(),
+                       '$back'        => $this->l10n->t('Go back'),
+                       '$stack_trace' => $this->l10n->t('Stack trace:'),
+                       '$request_id'  => $this->requestId,
                ];
-               $title = ($titles[$e->getCode()] ?? '') ?: 'Error ' . $e->getCode();
-
-               if (empty($message)) {
-                       // Explanations are taken from https://en.wikipedia.org/wiki/List_of_HTTP_status_codes
-                       $explanation = [
-                               400 => L10n::t('The server cannot or will not process the request due to an apparent client error.'),
-                               401 => L10n::t('Authentication is required and has failed or has not yet been provided.'),
-                               403 => L10n::t('The request was valid, but the server is refusing action. The user might not have the necessary permissions for a resource, or may need an account.'),
-                               404 => L10n::t('The requested resource could not be found but may be available in the future.'),
-                               500 => L10n::t('An unexpected condition was encountered and no more specific message is suitable.'),
-                               503 => L10n::t('The server is currently unavailable (because it is overloaded or down for maintenance). Please try again later.'),
-                       ];
 
-                       $message = $explanation[$e->getCode()] ?? '';
+               if ($this->isSiteAdmin) {
+                       $vars['$thrown'] = $this->l10n->t('Exception thrown in %s:%d', $e->getFile(), $e->getLine());
+                       $vars['$trace']  = $e->getTraceAsString();
                }
 
-               return ['$title' => $title, '$message' => $message, '$back' => L10n::t('Go back')];
+               return $vars;
        }
 
        /**
         * Displays a bare message page with no theming at all.
         *
         * @param \Friendica\Network\HTTPException $e
+        *
         * @throws \Exception
         */
-       public static function rawContent(\Friendica\Network\HTTPException $e)
+       public function rawContent(\Friendica\Network\HTTPException $e)
        {
                $content = '';
 
                if ($e->getCode() >= 400) {
-                       $tpl = Renderer::getMarkupTemplate('http_status.tpl');
-                       $content = Renderer::replaceMacros($tpl, self::getVars($e));
+                       $vars = $this->getVars($e);
+                       try {
+                               $tpl     = Renderer::getMarkupTemplate('http_status.tpl');
+                               $content = Renderer::replaceMacros($tpl, $vars);
+                       } catch (\Exception $e) {
+                               $vars = array_map('htmlentities', $vars);
+                               $content = "<h1>{$vars['$title']}</h1><p>{$vars['$message']}</p>";
+                               if ($this->isSiteAdmin) {
+                                       $content .= "<p>{$vars['$thrown']}</p>";
+                                       $content .= "<pre>{$vars['$trace']}</pre>";
+                               }
+                       }
                }
 
-               System::httpExit($e->getCode(), $e->httpdesc, $content);
+               System::httpError($e->getCode(), $e->getDescription(), $content);
        }
 
        /**
         * Returns a content string that can be integrated in the current theme.
         *
         * @param \Friendica\Network\HTTPException $e
+        *
         * @return string
         * @throws \Exception
         */
-       public static function content(\Friendica\Network\HTTPException $e)
+       public function content(\Friendica\Network\HTTPException $e): string
        {
-               header($_SERVER["SERVER_PROTOCOL"] . ' ' . $e->getCode() . ' ' . $e->httpdesc);
+               if ($e->getCode() >= 400) {
+                       $this->logger->debug('Exit with error',
+                               [
+                                       'code'        => $e->getCode(),
+                                       'description' => $e->getDescription(),
+                                       'query'       => $this->args->getQueryString(),
+                                       'callstack'   => System::callstack(20),
+                                       'method'      => $this->args->getMethod(),
+                                       'agent'       => $this->server['HTTP_USER_AGENT'] ?? ''
+                               ]);
+               }
 
                $tpl = Renderer::getMarkupTemplate('exception.tpl');
 
-               return Renderer::replaceMacros($tpl, self::getVars($e));
+               return Renderer::replaceMacros($tpl, $this->getVars($e));
        }
 }