No puede seleccionar más de 25 temas Los temas deben comenzar con una letra o número, pueden incluir guiones ('-') y pueden tener hasta 35 caracteres de largo.

Module.php 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. <?php
  2. /**
  3. * @link http://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license http://www.yiiframework.com/license/
  6. */
  7. namespace yii\debug;
  8. use Yii;
  9. use yii\base\Application;
  10. use yii\base\BootstrapInterface;
  11. use yii\helpers\Url;
  12. use yii\web\View;
  13. use yii\web\ForbiddenHttpException;
  14. /**
  15. * The Yii Debug Module provides the debug toolbar and debugger
  16. *
  17. * @author Qiang Xue <qiang.xue@gmail.com>
  18. * @since 2.0
  19. */
  20. class Module extends \yii\base\Module implements BootstrapInterface
  21. {
  22. /**
  23. * @var array the list of IPs that are allowed to access this module.
  24. * Each array element represents a single IP filter which can be either an IP address
  25. * or an address with wildcard (e.g. 192.168.0.*) to represent a network segment.
  26. * The default value is `['127.0.0.1', '::1']`, which means the module can only be accessed
  27. * by localhost.
  28. */
  29. public $allowedIPs = ['127.0.0.1', '::1'];
  30. /**
  31. * @inheritdoc
  32. */
  33. public $controllerNamespace = 'yii\debug\controllers';
  34. /**
  35. * @var LogTarget
  36. */
  37. public $logTarget;
  38. /**
  39. * @var array list of debug panels. The array keys are the panel IDs, and values are the corresponding
  40. * panel class names or configuration arrays. This will be merged with [[corePanels()]].
  41. * You may reconfigure a core panel via this property by using the same panel ID.
  42. * You may also disable a core panel by setting it to be false in this property.
  43. */
  44. public $panels = [];
  45. /**
  46. * @var string the directory storing the debugger data files. This can be specified using a path alias.
  47. */
  48. public $dataPath = '@runtime/debug';
  49. /**
  50. * @var integer the maximum number of debug data files to keep. If there are more files generated,
  51. * the oldest ones will be removed.
  52. */
  53. public $historySize = 50;
  54. /**
  55. * @var boolean whether to enable message logging for the requests about debug module actions.
  56. * You normally do not want to keep these logs because they may distract you from the logs about your applications.
  57. * You may want to enable the debug logs if you want to investigate how the debug module itself works.
  58. */
  59. public $enableDebugLogs = false;
  60. /**
  61. * Returns Yii logo ready to use in `<img src="`
  62. *
  63. * @return string base64 representation of the image
  64. */
  65. public static function getYiiLogo()
  66. {
  67. return '';
  68. }
  69. /**
  70. * @inheritdoc
  71. */
  72. public function init()
  73. {
  74. parent::init();
  75. $this->dataPath = Yii::getAlias($this->dataPath);
  76. $this->initPanels();
  77. }
  78. /**
  79. * Initializes panels.
  80. */
  81. protected function initPanels()
  82. {
  83. // merge custom panels and core panels so that they are ordered mainly by custom panels
  84. if (empty($this->panels)) {
  85. $this->panels = $this->corePanels();
  86. } else {
  87. $corePanels = $this->corePanels();
  88. foreach ($corePanels as $id => $config) {
  89. if (isset($this->panels[$id])) {
  90. unset($corePanels[$id]);
  91. }
  92. }
  93. $this->panels = array_filter(array_merge($corePanels, $this->panels));
  94. }
  95. foreach ($this->panels as $id => $config) {
  96. if (is_string($config)) {
  97. $config = ['class' => $config];
  98. }
  99. $config['module'] = $this;
  100. $config['id'] = $id;
  101. $this->panels[$id] = Yii::createObject($config);
  102. }
  103. }
  104. /**
  105. * @inheritdoc
  106. */
  107. public function bootstrap($app)
  108. {
  109. $this->logTarget = Yii::$app->getLog()->targets['debug'] = new LogTarget($this);
  110. // delay attaching event handler to the view component after it is fully configured
  111. $app->on(Application::EVENT_BEFORE_REQUEST, function () use ($app) {
  112. $app->getView()->on(View::EVENT_END_BODY, [$this, 'renderToolbar']);
  113. });
  114. $app->getUrlManager()->addRules([
  115. $this->id => $this->id,
  116. $this->id . '/<controller:[\w\-]+>/<action:[\w\-]+>' => $this->id . '/<controller>/<action>',
  117. ], false);
  118. }
  119. /**
  120. * @inheritdoc
  121. */
  122. public function beforeAction($action)
  123. {
  124. if (!$this->enableDebugLogs) {
  125. foreach (Yii::$app->getLog()->targets as $target) {
  126. $target->enabled = false;
  127. }
  128. }
  129. if (!parent::beforeAction($action)) {
  130. return false;
  131. }
  132. // do not display debug toolbar when in debug view mode
  133. Yii::$app->getView()->off(View::EVENT_END_BODY, [$this, 'renderToolbar']);
  134. if ($this->checkAccess()) {
  135. $this->resetGlobalSettings();
  136. return true;
  137. } elseif ($action->id === 'toolbar') {
  138. // Accessing toolbar remotely is normal. Do not throw exception.
  139. return false;
  140. } else {
  141. throw new ForbiddenHttpException('You are not allowed to access this page.');
  142. }
  143. }
  144. /**
  145. * Resets potentially incompatible global settings done in app config.
  146. */
  147. protected function resetGlobalSettings()
  148. {
  149. Yii::$app->assetManager->bundles = [];
  150. }
  151. /**
  152. * Renders mini-toolbar at the end of page body.
  153. *
  154. * @param \yii\base\Event $event
  155. */
  156. public function renderToolbar($event)
  157. {
  158. if (!$this->checkAccess() || Yii::$app->getRequest()->getIsAjax()) {
  159. return;
  160. }
  161. $url = Url::toRoute(['/' . $this->id . '/default/toolbar',
  162. 'tag' => $this->logTarget->tag,
  163. ]);
  164. echo '<div id="yii-debug-toolbar" data-url="' . $url . '" style="display:none"></div>';
  165. /* @var $view View */
  166. $view = $event->sender;
  167. echo '<style>' . $view->renderPhpFile(__DIR__ . '/assets/toolbar.css') . '</style>';
  168. echo '<script>' . $view->renderPhpFile(__DIR__ . '/assets/toolbar.js') . '</script>';
  169. }
  170. /**
  171. * Checks if current user is allowed to access the module
  172. * @return boolean if access is granted
  173. */
  174. protected function checkAccess()
  175. {
  176. $ip = Yii::$app->getRequest()->getUserIP();
  177. foreach ($this->allowedIPs as $filter) {
  178. if ($filter === '*' || $filter === $ip || (($pos = strpos($filter, '*')) !== false && !strncmp($ip, $filter, $pos))) {
  179. return true;
  180. }
  181. }
  182. Yii::warning('Access to debugger is denied due to IP address restriction. The requesting IP address is ' . $ip, __METHOD__);
  183. return false;
  184. }
  185. /**
  186. * @return array default set of panels
  187. */
  188. protected function corePanels()
  189. {
  190. return [
  191. 'config' => ['class' => 'yii\debug\panels\ConfigPanel'],
  192. 'request' => ['class' => 'yii\debug\panels\RequestPanel'],
  193. 'log' => ['class' => 'yii\debug\panels\LogPanel'],
  194. 'profiling' => ['class' => 'yii\debug\panels\ProfilingPanel'],
  195. 'db' => ['class' => 'yii\debug\panels\DbPanel'],
  196. 'assets' => ['class' => 'yii\debug\panels\AssetPanel'],
  197. 'mail' => ['class' => 'yii\debug\panels\MailPanel'],
  198. ];
  199. }
  200. }