%PDF- %PDF-
Direktori : /usr/share/php/ |
Current File : //usr/share/php/DibiNettePanel.php |
<?php /** * This file is part of the "dibi" - smart database abstraction layer. * * Copyright (c) 2005 David Grudl (http://davidgrudl.com) * * For the full copyright and license information, please view * the file license.txt that was distributed with this source code. */ if (interface_exists('Nette\Diagnostics\IBarPanel')) { class_alias('Nette\Diagnostics\IBarPanel', 'IBarPanel'); } /** * Dibi panel for Nette\Diagnostics. * * @author David Grudl * @package dibi\nette */ class DibiNettePanel extends DibiObject implements IBarPanel { /** @var int maximum SQL length */ static public $maxLength = 1000; /** @var bool explain queries? */ public $explain; /** @var int */ public $filter; /** @var array */ private $events = array(); public function __construct($explain = TRUE, $filter = NULL) { $this->filter = $filter ? (int) $filter : DibiEvent::QUERY; $this->explain = $explain; } public function register(DibiConnection $connection) { if (is_callable('Nette\Diagnostics\Debugger::enable') && !class_exists('NDebugger')) { class_alias('Nette\Diagnostics\Debugger', 'NDebugger'); // PHP 5.2 code compatibility } if (is_callable('NDebugger::enable') && is_callable('NDebugger::getBlueScreen')) { // Nette Framework 2.1 NDebugger::getBar()->addPanel($this); NDebugger::getBlueScreen()->addPanel(array(__CLASS__, 'renderException')); $connection->onEvent[] = array($this, 'logEvent'); } elseif (is_callable('NDebugger::enable')) { // Nette Framework 2.0 (for PHP 5.3 or PHP 5.2 prefixed) NDebugger::$bar && NDebugger::$bar->addPanel($this); NDebugger::$blueScreen && NDebugger::$blueScreen->addPanel(array(__CLASS__, 'renderException'), __CLASS__); $connection->onEvent[] = array($this, 'logEvent'); } elseif (is_callable('Debugger::enable') && !is_callable('Debugger::getBlueScreen')) { // Nette Framework 2.0 for PHP 5.2 non-prefixed Debugger::$bar && Debugger::$bar->addPanel($this); Debugger::$blueScreen && Debugger::$blueScreen->addPanel(array(__CLASS__, 'renderException'), __CLASS__); $connection->onEvent[] = array($this, 'logEvent'); } } /** * After event notification. * @return void */ public function logEvent(DibiEvent $event) { if (($event->type & $this->filter) === 0) { return; } $this->events[] = $event; } /** * Returns blue-screen custom tab. * @return mixed */ public static function renderException($e) { if ($e instanceof DibiException && $e->getSql()) { return array( 'tab' => 'SQL', 'panel' => dibi::dump($e->getSql(), TRUE), ); } } /** * Returns HTML code for custom tab. (Nette\Diagnostics\IBarPanel) * @return mixed */ public function getTab() { $totalTime = 0; foreach ($this->events as $event) { $totalTime += $event->time; } return '<span title="dibi"><img src="" />' . count($this->events) . ' queries' . ($totalTime ? ' / ' . sprintf('%0.1f', $totalTime * 1000) . 'ms' : '') . '</span>'; } /** * Returns HTML code for custom panel. (Nette\Diagnostics\IBarPanel) * @return mixed */ public function getPanel() { $totalTime = $s = NULL; $h = 'htmlSpecialChars'; foreach ($this->events as $event) { $totalTime += $event->time; $explain = NULL; // EXPLAIN is called here to work SELECT FOUND_ROWS() if ($this->explain && $event->type === DibiEvent::SELECT) { try { $backup = array($event->connection->onEvent, dibi::$numOfQueries, dibi::$totalTime); $event->connection->onEvent = NULL; $cmd = is_string($this->explain) ? $this->explain : ($event->connection->getConfig('driver') === 'oracle' ? 'EXPLAIN PLAN' : 'EXPLAIN'); $explain = dibi::dump($event->connection->nativeQuery("$cmd $event->sql"), TRUE); } catch (DibiException $e) {} list($event->connection->onEvent, dibi::$numOfQueries, dibi::$totalTime) = $backup; } $s .= '<tr><td>' . sprintf('%0.3f', $event->time * 1000); if ($explain) { static $counter; $counter++; $s .= "<br /><a href='#nette-debug-DibiProfiler-row-$counter' class='nette-toggler nette-toggle-collapsed' rel='#nette-debug-DibiProfiler-row-$counter'>explain</a>"; } $s .= '</td><td class="nette-DibiProfiler-sql">' . dibi::dump(strlen($event->sql) > self::$maxLength ? substr($event->sql, 0, self::$maxLength) . '...' : $event->sql, TRUE); if ($explain) { $s .= "<div id='nette-debug-DibiProfiler-row-$counter' class='nette-collapsed'>{$explain}</div>"; } if ($event->source) { $helpers = 'Nette\Diagnostics\Helpers'; if (!class_exists($helpers)) { $helpers = class_exists('NDebugHelpers') ? 'NDebugHelpers' : 'DebugHelpers'; } $s .= call_user_func(array($helpers, 'editorLink'), $event->source[0], $event->source[1])->class('nette-DibiProfiler-source'); } $s .= "</td><td>{$event->count}</td><td>{$h($event->connection->getConfig('driver') . '/' . $event->connection->getConfig('name'))}</td></tr>"; } return empty($this->events) ? '' : '<style> #nette-debug td.nette-DibiProfiler-sql { background: white !important } #nette-debug .nette-DibiProfiler-source { color: #999 !important } #nette-debug nette-DibiProfiler tr table { margin: 8px 0; max-height: 150px; overflow:auto } </style> <h1>Queries: ' . count($this->events) . ($totalTime === NULL ? '' : ', time: ' . sprintf('%0.3f', $totalTime * 1000) . ' ms') . '</h1> <div class="nette-inner nette-DibiProfiler"> <table> <tr><th>Time ms</th><th>SQL Statement</th><th>Rows</th><th>Connection</th></tr>' . $s . ' </table> </div>'; } }