debug.class.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. <?php
  2. /**
  3. * Classe de gestion des fonctionnalités de débogage.
  4. *
  5. * Cette classe fournit des outils pour le débogage, tels que la gestion des journaux,
  6. * l'affichage des erreurs, et la mesure du temps d'exécution.
  7. */
  8. class debug
  9. {
  10. /**
  11. * Tableau des journaux de débogage.
  12. *
  13. * @var array
  14. */
  15. private static $logs = [];
  16. /**
  17. * Temps de début pour mesurer l'exécution.
  18. *
  19. * @var float
  20. */
  21. private static $startTime;
  22. /**
  23. * Temps de fin pour mesurer l'exécution.
  24. *
  25. * @var float
  26. */
  27. private static $closeTime;
  28. /**
  29. * Active l'affichage des erreurs PHP.
  30. */
  31. public static function display_errors(){
  32. ini_set('display_errors', 1);
  33. ini_set('display_startup_errors', 1);
  34. error_reporting(E_ALL);
  35. }
  36. /**
  37. * Génère le type de fichier en fonction d'une chaîne donnée.
  38. *
  39. * @param string $_string Chaîne utilisée pour déterminer le type de fichier.
  40. * @return string Nom du fichier généré.
  41. */
  42. private static function typeFile(string $_string){
  43. switch ($_string) {
  44. case 'maintenance':
  45. return ".lock-maintenance";
  46. break;
  47. case 'debug':
  48. return ".active-debug";
  49. break;
  50. default:
  51. return ".active-debug-".$_string;
  52. break;
  53. }
  54. }
  55. /**
  56. * Ajoute un fichier spécifique dans le répertoire racine.
  57. *
  58. * @param string|null $_string Nom du fichier à ajouter (optionnel).
  59. */
  60. public static function addFile(?string $_string = NULL)
  61. {
  62. if($_string != NULL){
  63. $myfile = fopen(DOCUMENT_ROOT . self::typeFile($_string), "w");
  64. fclose($myfile);
  65. }
  66. }
  67. /**
  68. * Supprime un fichier spécifique du répertoire racine.
  69. *
  70. * @param string|null $_string Nom du fichier à supprimer (optionnel).
  71. */
  72. public static function removeFile(?string $_string = NULL)
  73. {
  74. if($_string != NULL){
  75. unlink(DOCUMENT_ROOT . self::typeFile($_string));
  76. }
  77. }
  78. /**
  79. * Vérifie si un fichier spécifique existe dans le répertoire racine.
  80. *
  81. * @param string|null $_string Nom du fichier à vérifier (optionnel).
  82. * @return bool TRUE si le fichier existe, FALSE sinon.
  83. */
  84. public static function isFile(?string $_string = NULL)
  85. {
  86. return file_exists(DOCUMENT_ROOT . self::typeFile($_string)) ? TRUE : FALSE;
  87. }
  88. /**
  89. * Inclut les ressources de débogage si le mode debug est activé.
  90. */
  91. public static function includeDebug()
  92. {
  93. if (debug::isFile("debug")) {
  94. echo '<link rel="stylesheet" href="' . cache::getFileWithTime("css/debug.css") . '">';
  95. }
  96. }
  97. /**
  98. * Affiche ou retourne une variable formatée pour le débogage.
  99. *
  100. * @param mixed $var Variable à afficher.
  101. * @param string $label Étiquette pour identifier la variable (par défaut 'Dump').
  102. * @param bool $echo Indique si la sortie doit être affichée (TRUE) ou retournée (FALSE).
  103. * @return string|null Contenu formaté si $echo est FALSE, sinon NULL.
  104. */
  105. public static function dump($var, $label = 'Dump', $echo = true)
  106. {
  107. // Start output buffering
  108. ob_start();
  109. echo "<pre class='debug-dump'>";
  110. echo "<strong>" . $label . ":</strong> ";
  111. // Convert array or object to string
  112. if (is_array($var)) {
  113. print_r($var);
  114. } elseif (is_object($var)) {
  115. echo self::objectToString($var);
  116. } else {
  117. var_dump($var);
  118. }
  119. echo "</pre>";
  120. // Get the contents of the buffer
  121. $output = ob_get_clean();
  122. // If echo is true, print the output
  123. if ($echo) {
  124. echo $output;
  125. } else {
  126. return $output;
  127. }
  128. }
  129. /**
  130. * Convertit un objet en chaîne formatée pour le débogage.
  131. *
  132. * @param object $object Objet à convertir.
  133. * @param string|null $_tab Indentation pour le formatage (optionnel).
  134. * @return string Représentation formatée de l'objet.
  135. */
  136. public static function objectToString($object, $_tab = NULL) {
  137. ob_start();
  138. $tab = "&nbsp;&nbsp;&nbsp;&nbsp;" . $_tab;
  139. echo "<span class='debug-console-capsule'>Object (" . get_class($object) . ")\n</span>";
  140. echo $_tab . "<span class='debug-console-capsule'>{\n</span>";
  141. foreach (get_object_vars($object) as $property => $value) {
  142. echo $tab . "<span class='debug-console-capsule'>[<span class='debug-console-key'>$property</span>] => </span>";
  143. if (is_array($value)) {
  144. echo self::arrayToString($value, $tab);
  145. } elseif (is_object($value)) {
  146. echo self::objectToString($value, $tab);
  147. } else {
  148. echo "<span class='debug-console-value'>" . var_export($value, true) . "</span>\n";
  149. }
  150. }
  151. echo $_tab . "<span class='debug-console-capsule'>}\n</span>";
  152. return ob_get_clean();
  153. }
  154. /**
  155. * Convertit un tableau en chaîne formatée pour le débogage.
  156. *
  157. * @param array $array Tableau à convertir.
  158. * @param string|null $_tab Indentation pour le formatage (optionnel).
  159. * @return string Représentation formatée du tableau.
  160. */
  161. public static function arrayToString($array, $_tab = NULL)
  162. {
  163. ob_start();
  164. $tab = "&nbsp;&nbsp;&nbsp;&nbsp;".$_tab;
  165. echo "<span class='debug-console-capsule'>Array\n</span>";
  166. echo $_tab . "<span class='debug-console-capsule'>(\n</span>";
  167. foreach ($array as $key => $value) {
  168. echo $tab . "<span class='debug-console-capsule'>[<span class='debug-console-key'>$key</span>] => </span>";
  169. if (is_array($value)) {
  170. echo self::arrayToString($value, $tab);
  171. } elseif (is_object($value)) {
  172. echo self::objectToString($value, $tab);
  173. } else {
  174. echo "<span class='debug-console-value'>" . var_export($value, true) . "</span>\n";
  175. }
  176. }
  177. echo $_tab . "<span class='debug-console-capsule'>)\n</span>";
  178. return ob_get_clean();
  179. }
  180. /**
  181. * Convertit une variable en chaîne de caractères pour le débogage.
  182. *
  183. * @param mixed $var La variable à convertir.
  184. * @param string $label Une étiquette pour identifier la variable.
  185. * @return string La représentation formatée de la variable.
  186. */
  187. private static function variableToString($var, $label)
  188. {
  189. ob_start();
  190. echo "$label: ";
  191. if (is_array($var) || is_object($var)) {
  192. print_r($var);
  193. } else {
  194. var_dump($var);
  195. }
  196. return ob_get_clean();
  197. }
  198. /**
  199. * Récupère la trace d'exécution sous forme de chaîne.
  200. *
  201. * @return string La trace d'exécution formatée.
  202. */
  203. public static function getTraces(){
  204. $return = "Trace : ";
  205. // Obtenir la trace d'exécution
  206. $backtrace = debug_backtrace();
  207. $nb = count($backtrace)-1;
  208. for ($i=$nb; $i > 0; $i--) {
  209. $return .= ($i != 0) ? "[".$backtrace[$i]["function"]."] " : NULL;
  210. $return .= str_replace(DOCUMENT_ROOT, '', $backtrace[$i]["file"]).":".$backtrace[$i]["line"];
  211. $return .= ($i != 1) ? " >> " : NULL;
  212. }
  213. return $return;
  214. }
  215. /**
  216. * Affiche un tableau formaté pour le débogage.
  217. *
  218. * @param array|null $_array Le tableau à afficher (optionnel).
  219. * @param int|null $_exit Si défini, termine le script après l'affichage (optionnel).
  220. */
  221. public static function print_r(?array $_array = NULL, ?int $_exit = NULL)
  222. {
  223. echo "<div>".debug::getTraces() . "</div>";
  224. if(empty($_array)){
  225. echo "<pre>EMPTY</pre>";
  226. } else {
  227. echo "<pre>";
  228. print_r($_array);
  229. echo "</pre>";
  230. }
  231. ($_exit != NULL) ? exit() : NULL;
  232. }
  233. /**
  234. * Vérifie si une chaîne contient du HTML.
  235. *
  236. * @param string $_string La chaîne à vérifier.
  237. * @return bool TRUE si la chaîne contient du HTML, FALSE sinon.
  238. */
  239. public static function isHtml(string $_string) : bool
  240. {
  241. $stripped_string = strip_tags($_string);
  242. return $_string !== $stripped_string;
  243. }
  244. /**
  245. * Génère un contenu HTML encapsulé dans un iframe.
  246. *
  247. * @param string $htmlContent Le contenu HTML à encapsuler.
  248. * @return string Le contenu HTML complet avec iframe.
  249. */
  250. public static function generateHtmlContent($htmlContent)
  251. {
  252. $id = md5(microtime());
  253. return <<<HTML
  254. <!DOCTYPE html>
  255. <html>
  256. <head>
  257. <title>Exécution de HTML</title>
  258. <style>
  259. .isolated-iframe {
  260. width: 100%;
  261. height: 300px;
  262. border: none;
  263. }
  264. </style>
  265. </head>
  266. <body>
  267. <iframe id="isolatedIframe-$id" class="isolated-iframe"></iframe>
  268. <script>
  269. var iframe = document.getElementById('isolatedIframe-$id');
  270. var doc = iframe.document || iframe.contentDocument || iframe.contentWindow.document;
  271. doc.open();
  272. doc.write(`{$htmlContent}`);
  273. doc.close();
  274. </script>
  275. </body>
  276. </html>
  277. HTML;
  278. }
  279. /**
  280. * Ajoute un message au journal de débogage.
  281. *
  282. * @param mixed $_message Le message à ajouter.
  283. * @param string|null $_mark Une marque pour identifier le message (optionnel).
  284. */
  285. public static function log($_message, $_mark = NULL)
  286. {
  287. $mark = "<div class='debug-head-console'>";
  288. $mark .= ($_mark != NULL) ? "<div style='font-weight: bold;'>". $_mark . "</div>" : NULL;
  289. $mark .= self::getTraces();
  290. $mark .= "</div>";
  291. if($_message != NULL) {
  292. if(is_array($_message)){
  293. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::arrayToString($_message) . "</div>", "nl2br" => TRUE];
  294. } elseif(is_object($_message)){
  295. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::objectToString($_message) . "</div>", "nl2br" => TRUE];
  296. } elseif(self::isHtml($_message)){
  297. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::generateHtmlContent($_message) . "</div>", "nl2br" => FALSE];
  298. } elseif($mark != NULL){
  299. self::$logs[] = ["data" => $mark ."<div class='debug-console' style='color:blue;'>" . $_message . "</div>", "nl2br" => TRUE];
  300. } else {
  301. self::$logs[] = ["data" => $_message, "nl2br" => TRUE];
  302. }
  303. }
  304. }
  305. /**
  306. * Ajoute un message au journal de session de débogage.
  307. *
  308. * @param mixed $_message Le message à ajouter.
  309. * @param string|null $_mark Une marque pour identifier le message (optionnel).
  310. */
  311. public static function logSession($_message, $_mark = NULL)
  312. {
  313. $mark = "<div class='debug-head-console'>";
  314. $mark .= ($_mark != NULL) ? "<div style='font-weight: bold;'>". $_mark . "</div>" : NULL;
  315. $mark .= self::getTraces();
  316. $mark .= "</div>";
  317. if($_message != NULL) {
  318. if(is_array($_message)){
  319. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::arrayToString($_message) . "</div>", "nl2br" => TRUE]);
  320. } elseif(is_object($_message)){
  321. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::objectToString($_message) . "</div>", "nl2br" => TRUE]);
  322. } elseif(self::isHtml($_message)){
  323. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::generateHtmlContent($_message) . "</div>", "nl2br" => false]);
  324. } elseif($mark != NULL){
  325. self::setSession(["data" => $mark ."<div class='debug-console' style='color:blue;'>" . $_message . "</div>", "nl2br" => TRUE]);
  326. } else {
  327. self::setSession(["data" => $_message, "nl2br" => TRUE]);
  328. }
  329. }
  330. }
  331. /**
  332. * Affiche le panneau des journaux du débogueur en HTML.
  333. *
  334. * Cette méthode affiche les versions de PHP et MySQL, le temps d'exécution, ainsi qu'un ensemble de cases à cocher
  335. * pour intercepter différents types d'actions (Submit, requêtes SQL, Emails et mode debug d'envoi d'email).
  336. * Elle affiche ensuite les journaux de débogage collectés et, si disponibles, les journaux de session.
  337. * Enfin, elle charge le JavaScript associé au débogage.
  338. *
  339. * @return void
  340. */
  341. public static function renderLogs()
  342. {
  343. echo "<div id='debugger-logs'>";
  344. echo "<div class='debug-renderLogs-header'>";
  345. echo "PHP <span style=\"font-weight: bold;\">". phpversion() . "</span> | MYSQL <span style=\"font-weight: bold;\">" . db::version() . "</span> | " . number_format(self::$closeTime, 4) . " secondes ";
  346. echo "</div>";
  347. echo "<div class=\"form-check form-switch\" style=\"margin-top: -30px;\">
  348. <div>
  349. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsSubmit\" " . core::checkboxSelecter(self::isFile("submit"), 0) . " >
  350. <label class=\"form-check-label\" for=\"checkIsSubmit\">Intercepter les <span style=\"font-weight: bold;\">Submit</span></label>
  351. </div>
  352. <div>
  353. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsSql\" " . core::checkboxSelecter(self::isFile("sql"), 0) . " >
  354. <label class=\"form-check-label\" for=\"checkIsSql\">Intercepter les <span style=\"font-weight: bold;\">requêtes SQL</span></label>
  355. </div>
  356. <div>
  357. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsEmail\" " . core::checkboxSelecter(self::isFile("email"), 0) . " >
  358. <label class=\"form-check-label\" for=\"checkIsEmail\">Intercepter les <span style=\"font-weight: bold;\">Email</span></label>
  359. </div>
  360. <div>
  361. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkSendIsEmail\" " . core::checkboxSelecter(self::isFile("send-email"), 0) . " >
  362. <label class=\"form-check-label\" for=\"checkSendIsEmail\">Afficher le mode debug lors de l'<span style=\"font-weight: bold;\">envoie d'email</span></label>
  363. </div>
  364. </div>";
  365. foreach (self::$logs as $log) {
  366. if($log != NULL){
  367. echo "<div class='debug-renderLogs-print'>";
  368. echo $log["nl2br"] == TRUE ? nl2br($log["data"]) : $log["data"];
  369. echo "</div>";
  370. }
  371. }
  372. if(self::ifSession()){
  373. foreach (self::getSession() as $logSession) {
  374. echo "<div class='debug-renderLogs-print'>";
  375. echo $logSession["nl2br"] == TRUE ? nl2br($logSession["data"]) : $logSession["data"];
  376. echo "</div>";
  377. }
  378. }
  379. echo "</div>";
  380. get::javascript("debug");
  381. }
  382. /**
  383. * Initialise le débogueur en enregistrant une fonction de fermeture.
  384. */
  385. public static function init()
  386. {
  387. register_shutdown_function(function () {
  388. self::renderLogs();
  389. });
  390. }
  391. /**
  392. * Démarre un chronomètre pour mesurer le temps d'exécution.
  393. */
  394. public static function startTimer()
  395. {
  396. self::$startTime = microtime(true);
  397. }
  398. /**
  399. * Arrête le chronomètre et calcule le temps écoulé.
  400. */
  401. public static function endTimer()
  402. {
  403. self::$closeTime = microtime(true) - self::$startTime;
  404. }
  405. /**
  406. * Génère un badge HTML avec un lien et un libellé.
  407. *
  408. * @param string $_link Lien vers lequel le badge redirige.
  409. * @param string $_label Libellé affiché sur le badge.
  410. * @return string Le code HTML du badge.
  411. */
  412. public static function getBadge(string $_link,string $_label){
  413. return '<a href="'. $_link .'" target="_blank" class="badge debug-badge">'. $_label .'</a>';
  414. }
  415. /**
  416. * Affiche l'environnement actuel si ce n'est pas l'environnement de production.
  417. */
  418. public static function printEnvironnement(){
  419. echo (ENVIRONNEMENT != "PROD") ? " [" . ENVIRONNEMENT . "]" : NULL;
  420. }
  421. /**
  422. * Construit un badge HTML personnalisé à partir des données fournies.
  423. *
  424. * @param array $_data Données pour personnaliser le badge (couleur, texte, lien, etc.).
  425. * @return string Le code HTML du badge.
  426. */
  427. private static function buildBadge(array $_data){
  428. $color = empty($_data["color"]) ? "black" : $_data["color"];
  429. $backgroundColor = empty($_data["background-color"]) ? "orangered" : $_data["background-color"];
  430. $class = empty($_data["class"]) ? NULL : $_data["class"];
  431. $link = empty($_data["link"]) ? "#" : $_data["link"];
  432. $txt = empty($_data["txt"]) ? NULL : $_data["txt"];
  433. $icon = empty($_data["icon"]) ? NULL : "<i class=\"" . $_data["icon"] . "\"></i> ";
  434. return "<a href=\"" . $link . "\"><span class=\"badge " . $class . "\" style=\"background-color:" . $backgroundColor . "; color:" . $color . "; padding: 3px 5px 2px 5px; margin-right: 5px;\">" . $icon . $txt . "</span></a>";
  435. }
  436. /**
  437. * Récupère tous les badges de débogage disponibles.
  438. *
  439. * @return string Les badges HTML générés.
  440. */
  441. public static function getBadges(){
  442. $return = "";
  443. if(debug::isFile("maintenance")){
  444. $return .= self::buildBadge([
  445. "icon" => "bi bi-exclamation-diamond-fill",
  446. "link" => "/parametres.html#parametres-debug",
  447. "background-color" => "red",
  448. "color" => "white",
  449. "txt" => "MODE MAINTENANCE"
  450. ]);
  451. }
  452. if(debug::isFile("debug")){
  453. $return .= self::buildBadge([
  454. "icon" => "bi bi-bug-fill",
  455. "class" => "toggle-logs",
  456. "link" => "#",
  457. "background-color" => "orangered",
  458. "color" => "white",
  459. "txt" => "MODE DEBUG"
  460. ]);
  461. }
  462. if(self::isFile("sql")){
  463. $return .= self::buildBadge([
  464. "icon" => "bi bi-database-fill",
  465. "class" => "toggle-logs",
  466. "link" => "#",
  467. "background-color" => "orange",
  468. "color" => "black",
  469. "txt" => "DEBUG SQL"
  470. ]);
  471. }
  472. if(self::isFile("email")){
  473. $return .= self::buildBadge([
  474. "icon" => "bi bi-envelope-fill",
  475. "class" => "toggle-logs",
  476. "link" => "#",
  477. "background-color" => "orange",
  478. "color" => "black",
  479. "txt" => "DEBUG EMAIL"
  480. ]);
  481. }
  482. if(self::isFile("send-email")){
  483. $return .= self::buildBadge([
  484. "icon" => "bi bi-envelope-fill",
  485. "class" => "toggle-logs",
  486. "link" => "#",
  487. "background-color" => "orange",
  488. "color" => "black",
  489. "txt" => "DEBUG SEND EMAIL"
  490. ]);
  491. }
  492. if(self::isFile("submit")){
  493. $return .= self::buildBadge([
  494. "icon" => "bi bi-send-fill",
  495. "class" => "toggle-logs",
  496. "link" => "#",
  497. "background-color" => "orange",
  498. "color" => "black",
  499. "txt" => "DEBUG SUBMIT"
  500. ]);
  501. }
  502. return $return;
  503. }
  504. /**
  505. * Ajoute des données au tableau de session de débogage.
  506. *
  507. * @param mixed $_data Les données à ajouter à la session.
  508. */
  509. public static function setSession($_data){
  510. $_SESSION["DEBUG"][] = $_data;
  511. }
  512. /**
  513. * Récupère les données de la session de débogage et les réinitialise.
  514. *
  515. * @return array Les données de la session de débogage.
  516. */
  517. public static function getSession(){
  518. $return = $_SESSION["DEBUG"];
  519. self::resetSession();
  520. return $return;
  521. }
  522. /**
  523. * Vérifie si des données de débogage sont présentes dans la session.
  524. *
  525. * @return bool TRUE si des données sont présentes, FALSE sinon.
  526. */
  527. public static function ifSession(){
  528. return empty($_SESSION["DEBUG"]) ? FALSE : TRUE;
  529. }
  530. /**
  531. * Réinitialise les données de la session de débogage.
  532. */
  533. public static function resetSession(){
  534. unset($_SESSION["DEBUG"]);
  535. }
  536. }