template = new Template(); $this->isEnabled = true; $this->scripts = new Scripts(); $this->isMinimal = false; $this->relation = new Relation($GLOBALS['dbi']); } /** * @return array * @psalm-return array{revision: string, revisionUrl: string, branch: string, branchUrl: string}|[] */ private function getGitRevisionInfo(): array { $info = []; if (@file_exists(ROOT_PATH . 'revision-info.php')) { /** @psalm-suppress MissingFile,UnresolvableInclude */ $info = include ROOT_PATH . 'revision-info.php'; } return is_array($info) ? $info : []; } /** * Remove recursions and iterator objects from an object * * @param mixed $object Object to clean * @param array $stack Stack used to keep track of recursion, need not be passed for the first time * * @return mixed Reference passed object */ private static function removeRecursion(&$object, array $stack = []) { if ((is_object($object) || is_array($object)) && $object) { if ($object instanceof Traversable) { $object = '***ITERATOR***'; } elseif (! in_array($object, $stack, true)) { $stack[] = $object; // @phpstan-ignore-next-line foreach ($object as &$subObject) { self::removeRecursion($subObject, $stack); } } else { $object = '***RECURSION***'; } } return $object; } /** * Renders the debug messages */ public function getDebugMessage(): string { $retval = '\'false\''; if ($GLOBALS['cfg']['DBG']['sql'] && empty($_REQUEST['no_debug']) && ! empty($_SESSION['debug'])) { // Remove recursions and iterators from $_SESSION['debug'] self::removeRecursion($_SESSION['debug']); $retval = (string) json_encode($_SESSION['debug']); $_SESSION['debug'] = []; return json_last_error() ? '\'false\'' : $retval; } $_SESSION['debug'] = []; return $retval; } /** * Returns the url of the current page */ public function getSelfUrl(): string { $GLOBALS['server'] = $GLOBALS['server'] ?? null; $params = []; $params['route'] = Common::getRequest()->getRoute(); if (isset($GLOBALS['db']) && strlen($GLOBALS['db']) > 0) { $params['db'] = $GLOBALS['db']; } if (isset($GLOBALS['table']) && strlen($GLOBALS['table']) > 0) { $params['table'] = $GLOBALS['table']; } $params['server'] = $GLOBALS['server']; if (isset($_REQUEST['single_table']) && in_array($_REQUEST['single_table'], [true, false])) { $params['single_table'] = $_REQUEST['single_table']; } return basename(Core::getenv('SCRIPT_NAME')) . Url::getCommonRaw($params); } /** * Renders the link to open a new page */ public function getErrorMessages(): string { $retval = ''; if ($GLOBALS['errorHandler']->hasDisplayErrors()) { $retval .= $GLOBALS['errorHandler']->getDispErrors(); } /** * Report php errors */ $GLOBALS['errorHandler']->reportErrors(); return $retval; } /** * Saves query in history */ private function setHistory(): void { if ( ( isset($_REQUEST['no_history']) && is_scalar($_REQUEST['no_history']) && strlen((string) $_REQUEST['no_history']) > 0 ) || ! empty($GLOBALS['error_message']) || empty($GLOBALS['sql_query']) || ! isset($GLOBALS['dbi']) || ! $GLOBALS['dbi']->isConnected() ) { return; } $this->relation->setHistory( isset($GLOBALS['db']) && is_scalar($GLOBALS['db']) ? (string) $GLOBALS['db'] : '', isset($GLOBALS['table']) && is_scalar($GLOBALS['table']) ? (string) $GLOBALS['table'] : '', $GLOBALS['cfg']['Server']['user'], $GLOBALS['sql_query'] ); } /** * Disables the rendering of the footer */ public function disable(): void { $this->isEnabled = false; } /** * Set the ajax flag to indicate whether * we are servicing an ajax request * * @param bool $isAjax Whether we are servicing an ajax request */ public function setAjax(bool $isAjax): void { $this->isAjax = $isAjax; } /** * Turn on minimal display mode */ public function setMinimal(): void { $this->isMinimal = true; } /** * Returns the Scripts object * * @return Scripts object */ public function getScripts(): Scripts { return $this->scripts; } /** * Renders the footer */ public function getDisplay(): string { $this->setHistory(); if ($this->isEnabled) { if (! $this->isAjax && ! $this->isMinimal) { if (Core::getenv('SCRIPT_NAME')) { $url = $this->getSelfUrl(); } $this->scripts->addCode('window.Console.debugSqlInfo = ' . $this->getDebugMessage() . ';'); $errorMessages = $this->getErrorMessages(); $scripts = $this->scripts->getDisplay(); if ($GLOBALS['cfg']['DBG']['demo']) { $gitRevisionInfo = $this->getGitRevisionInfo(); } $footer = Config::renderFooter(); } return $this->template->render('footer', [ 'is_ajax' => $this->isAjax, 'is_minimal' => $this->isMinimal, 'self_url' => $url ?? null, 'error_messages' => $errorMessages ?? '', 'scripts' => $scripts ?? '', 'is_demo' => $GLOBALS['cfg']['DBG']['demo'], 'git_revision_info' => $gitRevisionInfo ?? [], 'footer' => $footer ?? '', ]); } return ''; } }