2
0

debug.class.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. <?php
  2. class debug
  3. {
  4. private static $logs = [];
  5. private static $startTime;
  6. private static $closeTime;
  7. private static function typeFile(string $_string){
  8. switch ($_string) {
  9. case 'maintenance':
  10. return ".lock-maintenance";
  11. break;
  12. case 'debug':
  13. return ".active-debug";
  14. break;
  15. default:
  16. return ".active-debug-".$_string;
  17. break;
  18. }
  19. }
  20. public static function addFile(?string $_string = NULL)
  21. {
  22. if($_string != NULL){
  23. $myfile = fopen(DOCUMENT_ROOT . self::typeFile($_string), "w");
  24. fclose($myfile);
  25. }
  26. }
  27. public static function removeFile(?string $_string = NULL)
  28. {
  29. if($_string != NULL){
  30. unlink(DOCUMENT_ROOT . self::typeFile($_string));
  31. }
  32. }
  33. public static function isFile(?string $_string = NULL)
  34. {
  35. return file_exists(DOCUMENT_ROOT . self::typeFile($_string)) ? TRUE : FALSE;
  36. }
  37. public static function includeDebug()
  38. {
  39. if (debug::isFile("debug")) {
  40. echo '<link rel="stylesheet" href="' . cache::getFileWithTime("css/debug.css") . '">';
  41. }
  42. }
  43. public static function dump($var, $label = 'Dump', $echo = true)
  44. {
  45. // Start output buffering
  46. ob_start();
  47. echo "<pre class='debug-dump'>";
  48. echo "<strong>" . $label . ":</strong> ";
  49. // Convert array or object to string
  50. if (is_array($var)) {
  51. print_r($var);
  52. } elseif (is_object($var)) {
  53. echo self::objectToString($var);
  54. } else {
  55. var_dump($var);
  56. }
  57. echo "</pre>";
  58. // Get the contents of the buffer
  59. $output = ob_get_clean();
  60. // If echo is true, print the output
  61. if ($echo) {
  62. echo $output;
  63. } else {
  64. return $output;
  65. }
  66. }
  67. public static function objectToString($object, $_tab = NULL) {
  68. ob_start();
  69. $tab = "&nbsp;&nbsp;&nbsp;&nbsp;" . $_tab;
  70. echo "<span class='debug-console-capsule'>Object (" . get_class($object) . ")\n</span>";
  71. echo $_tab . "<span class='debug-console-capsule'>{\n</span>";
  72. foreach (get_object_vars($object) as $property => $value) {
  73. echo $tab . "<span class='debug-console-capsule'>[<span class='debug-console-key'>$property</span>] => </span>";
  74. if (is_array($value)) {
  75. echo self::arrayToString($value, $tab);
  76. } elseif (is_object($value)) {
  77. echo self::objectToString($value, $tab);
  78. } else {
  79. echo "<span class='debug-console-value'>" . var_export($value, true) . "</span>\n";
  80. }
  81. }
  82. echo $_tab . "<span class='debug-console-capsule'>}\n</span>";
  83. return ob_get_clean();
  84. }
  85. public static function arrayToString($array, $_tab = NULL)
  86. {
  87. ob_start();
  88. $tab = "&nbsp;&nbsp;&nbsp;&nbsp;".$_tab;
  89. echo "<span class='debug-console-capsule'>Array\n</span>";
  90. echo $_tab . "<span class='debug-console-capsule'>(\n</span>";
  91. foreach ($array as $key => $value) {
  92. echo $tab . "<span class='debug-console-capsule'>[<span class='debug-console-key'>$key</span>] => </span>";
  93. if (is_array($value)) {
  94. echo self::arrayToString($value, $tab);
  95. } elseif (is_object($value)) {
  96. echo self::objectToString($value, $tab);
  97. } else {
  98. echo "<span class='debug-console-value'>" . var_export($value, true) . "</span>\n";
  99. }
  100. }
  101. echo $_tab . "<span class='debug-console-capsule'>)\n</span>";
  102. return ob_get_clean();
  103. }
  104. private static function variableToString($var, $label)
  105. {
  106. ob_start();
  107. echo "$label: ";
  108. if (is_array($var) || is_object($var)) {
  109. print_r($var);
  110. } else {
  111. var_dump($var);
  112. }
  113. return ob_get_clean();
  114. }
  115. public static function getTraces(){
  116. $return = "Trace : ";
  117. // Obtenir la trace d'exécution
  118. $backtrace = debug_backtrace();
  119. $nb = count($backtrace)-1;
  120. for ($i=$nb; $i > 0; $i--) {
  121. $return .= ($i != 0) ? "[".$backtrace[$i]["function"]."] " : NULL;
  122. $return .= str_replace(DOCUMENT_ROOT, '', $backtrace[$i]["file"]).":".$backtrace[$i]["line"];
  123. $return .= ($i != 1) ? " >> " : NULL;
  124. }
  125. return $return;
  126. }
  127. public static function print_r(?array $_array = NULL, ?int $_exit = NULL)
  128. {
  129. echo "<div>".debug::getTraces() . "</div>";
  130. if(empty($_array)){
  131. echo "<pre>EMPTY</pre>";
  132. } else {
  133. echo "<pre>";
  134. print_r($_array);
  135. echo "</pre>";
  136. }
  137. ($_exit != NULL) ? exit() : NULL;
  138. }
  139. public static function isHtml(string $_string) : bool
  140. {
  141. $stripped_string = strip_tags($_string);
  142. return $_string !== $stripped_string;
  143. }
  144. public static function generateHtmlContent($htmlContent)
  145. {
  146. $id = md5(microtime());
  147. return <<<HTML
  148. <!DOCTYPE html>
  149. <html>
  150. <head>
  151. <title>Exécution de HTML</title>
  152. <style>
  153. .isolated-iframe {
  154. width: 100%;
  155. height: 300px;
  156. border: none;
  157. }
  158. </style>
  159. </head>
  160. <body>
  161. <iframe id="isolatedIframe-$id" class="isolated-iframe"></iframe>
  162. <script>
  163. var iframe = document.getElementById('isolatedIframe-$id');
  164. var doc = iframe.document || iframe.contentDocument || iframe.contentWindow.document;
  165. doc.open();
  166. doc.write(`{$htmlContent}`);
  167. doc.close();
  168. </script>
  169. </body>
  170. </html>
  171. HTML;
  172. }
  173. public static function log($_message, $_mark = NULL)
  174. {
  175. $mark = "<div class='debug-head-console'>";
  176. $mark .= ($_mark != NULL) ? "<div style='font-weight: bold;'>". $_mark . "</div>" : NULL;
  177. $mark .= self::getTraces();
  178. $mark .= "</div>";
  179. if($_message != NULL) {
  180. if(is_array($_message)){
  181. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::arrayToString($_message) . "</div>", "nl2br" => TRUE];
  182. } elseif(is_object($_message)){
  183. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::objectToString($_message) . "</div>", "nl2br" => TRUE];
  184. } elseif(self::isHtml($_message)){
  185. self::$logs[] = ["data" => $mark ."<div class='debug-console'>" . self::generateHtmlContent($_message) . "</div>", "nl2br" => FALSE];
  186. } elseif($mark != NULL){
  187. self::$logs[] = ["data" => $mark ."<div class='debug-console' style='color:blue;'>" . $_message . "</div>", "nl2br" => TRUE];
  188. } else {
  189. self::$logs[] = ["data" => $_message, "nl2br" => TRUE];
  190. }
  191. }
  192. }
  193. public static function logSession($_message, $_mark = NULL)
  194. {
  195. $mark = "<div class='debug-head-console'>";
  196. $mark .= ($_mark != NULL) ? "<div style='font-weight: bold;'>". $_mark . "</div>" : NULL;
  197. $mark .= self::getTraces();
  198. $mark .= "</div>";
  199. if($_message != NULL) {
  200. if(is_array($_message)){
  201. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::arrayToString($_message) . "</div>", "nl2br" => TRUE]);
  202. } elseif(is_object($_message)){
  203. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::objectToString($_message) . "</div>", "nl2br" => TRUE]);
  204. } elseif(self::isHtml($_message)){
  205. self::setSession(["data" => $mark ."<div class='debug-console'>" . self::generateHtmlContent($_message) . "</div>", "nl2br" => false]);
  206. } elseif($mark != NULL){
  207. self::setSession(["data" => $mark ."<div class='debug-console' style='color:blue;'>" . $_message . "</div>", "nl2br" => TRUE]);
  208. } else {
  209. self::setSession(["data" => $_message, "nl2br" => TRUE]);
  210. }
  211. }
  212. }
  213. public static function renderLogs()
  214. {
  215. echo "<div id='debugger-logs'>";
  216. echo "<div class='debug-renderLogs-header'>";
  217. echo "PHP <span style=\"font-weight: bold;\">". phpversion() . "</span> | MYSQL <span style=\"font-weight: bold;\">" . db::version() . "</span> | " . number_format(self::$closeTime, 4) . " secondes ";
  218. echo "</div>";
  219. echo "<div class=\"form-check form-switch\" style=\"margin-top: -30px;\">
  220. <div>
  221. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsSubmit\" " . core::checkboxSelecter(self::isFile("submit"), 0) . " >
  222. <label class=\"form-check-label\" for=\"checkIsSubmit\">Intercepter les <span style=\"font-weight: bold;\">Submit</span></label>
  223. </div>
  224. <div>
  225. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsSql\" " . core::checkboxSelecter(self::isFile("sql"), 0) . " >
  226. <label class=\"form-check-label\" for=\"checkIsSql\">Intercepter les <span style=\"font-weight: bold;\">requêtes SQL</span></label>
  227. </div>
  228. <div>
  229. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkIsEmail\" " . core::checkboxSelecter(self::isFile("email"), 0) . " >
  230. <label class=\"form-check-label\" for=\"checkIsEmail\">Intercepter les <span style=\"font-weight: bold;\">Email</span></label>
  231. </div>
  232. <div>
  233. <input class=\"form-check-input\" type=\"checkbox\" id=\"checkSendIsEmail\" " . core::checkboxSelecter(self::isFile("send-email"), 0) . " >
  234. <label class=\"form-check-label\" for=\"checkSendIsEmail\">Afficher le mode debug lors de l'<span style=\"font-weight: bold;\">envoie d'email</span></label>
  235. </div>
  236. </div>";
  237. foreach (self::$logs as $log) {
  238. if($log != NULL){
  239. echo "<div class='debug-renderLogs-print'>";
  240. echo $log["nl2br"] == TRUE ? nl2br($log["data"]) : $log["data"];
  241. echo "</div>";
  242. }
  243. }
  244. if(self::ifSession()){
  245. foreach (self::getSession() as $logSession) {
  246. echo "<div class='debug-renderLogs-print'>";
  247. echo $logSession["nl2br"] == TRUE ? nl2br($logSession["data"]) : $logSession["data"];
  248. echo "</div>";
  249. }
  250. }
  251. echo "</div>";
  252. get::javascript("debug");
  253. }
  254. public static function init()
  255. {
  256. // Register shutdown function to render logs at the end of the script execution
  257. register_shutdown_function(function () {
  258. self::renderLogs();
  259. });
  260. }
  261. public static function startTimer()
  262. {
  263. self::$startTime = microtime(true);
  264. }
  265. public static function endTimer()
  266. {
  267. self::$closeTime = microtime(true) - self::$startTime;
  268. }
  269. public static function getBadge(string $_link,string $_label){
  270. return '<a href="'. $_link .'" target="_blank" class="badge debug-badge">'. $_label .'</a>';
  271. }
  272. public static function printEnvironnement(){
  273. echo (ENVIRONNEMENT != "PROD") ? " [" . ENVIRONNEMENT . "]" : NULL;
  274. }
  275. private static function buildBadge(array $_data){
  276. $color = empty($_data["color"]) ? "black" : $_data["color"];
  277. $backgroundColor = empty($_data["background-color"]) ? "orangered" : $_data["background-color"];
  278. $class = empty($_data["class"]) ? NULL : $_data["class"];
  279. $link = empty($_data["link"]) ? "#" : $_data["link"];
  280. $txt = empty($_data["txt"]) ? NULL : $_data["txt"];
  281. $icon = empty($_data["icon"]) ? NULL : "<i class=\"" . $_data["icon"] . "\"></i> ";
  282. 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>";
  283. }
  284. public static function getBadges(){
  285. $return = "";
  286. if(debug::isFile("maintenance")){
  287. $return .= self::buildBadge([
  288. "icon" => "bi bi-exclamation-diamond-fill",
  289. "link" => "/parametres.html#parametres-debug",
  290. "background-color" => "red",
  291. "color" => "white",
  292. "txt" => "MODE MAINTENANCE"
  293. ]);
  294. }
  295. if(debug::isFile("debug")){
  296. $return .= self::buildBadge([
  297. "icon" => "bi bi-bug-fill",
  298. "class" => "toggle-logs",
  299. "link" => "#",
  300. "background-color" => "orangered",
  301. "color" => "white",
  302. "txt" => "MODE DEBUG"
  303. ]);
  304. }
  305. if(self::isFile("sql")){
  306. $return .= self::buildBadge([
  307. "icon" => "bi bi-database-fill",
  308. "class" => "toggle-logs",
  309. "link" => "#",
  310. "background-color" => "orange",
  311. "color" => "black",
  312. "txt" => "DEBUG SQL"
  313. ]);
  314. }
  315. if(self::isFile("email")){
  316. $return .= self::buildBadge([
  317. "icon" => "bi bi-envelope-fill",
  318. "class" => "toggle-logs",
  319. "link" => "#",
  320. "background-color" => "orange",
  321. "color" => "black",
  322. "txt" => "DEBUG EMAIL"
  323. ]);
  324. }
  325. if(self::isFile("send-email")){
  326. $return .= self::buildBadge([
  327. "icon" => "bi bi-envelope-fill",
  328. "class" => "toggle-logs",
  329. "link" => "#",
  330. "background-color" => "orange",
  331. "color" => "black",
  332. "txt" => "DEBUG SEND EMAIL"
  333. ]);
  334. }
  335. if(self::isFile("submit")){
  336. $return .= self::buildBadge([
  337. "icon" => "bi bi-send-fill",
  338. "class" => "toggle-logs",
  339. "link" => "#",
  340. "background-color" => "orange",
  341. "color" => "black",
  342. "txt" => "DEBUG SUBMIT"
  343. ]);
  344. }
  345. return $return;
  346. }
  347. public static function setSession($_data){
  348. $_SESSION["DEBUG"][] = $_data;
  349. }
  350. public static function getSession(){
  351. $return = $_SESSION["DEBUG"];
  352. self::resetSession();
  353. return $return;
  354. }
  355. public static function ifSession(){
  356. return empty($_SESSION["DEBUG"]) ? FALSE : TRUE;
  357. }
  358. public static function resetSession(){
  359. unset($_SESSION["DEBUG"]);
  360. }
  361. }