The version of vichan running on lainchan.org
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

162 řádky
4.3KB

  1. <?php
  2. /*
  3. * Copyright (c) 2010-2013 Tinyboard Development Group
  4. */
  5. defined('TINYBOARD') or exit;
  6. class PreparedQueryDebug {
  7. protected $query, $explain_query = false;
  8. public function __construct($query) {
  9. global $pdo, $config;
  10. $query = preg_replace("/[\n\t]+/", ' ', $query);
  11. $this->query = $pdo->prepare($query);
  12. if ($config['debug'] && $config['debug_explain'] && preg_match('/^(SELECT|INSERT|UPDATE|DELETE) /i', $query))
  13. $this->explain_query = $pdo->prepare("EXPLAIN $query");
  14. }
  15. public function __call($function, $args) {
  16. global $config, $debug;
  17. if ($config['debug'] && $function == 'execute') {
  18. if ($this->explain_query) {
  19. $this->explain_query->execute() or error(db_error($this->explain_query));
  20. }
  21. $start = microtime(true);
  22. }
  23. if ($this->explain_query && $function == 'bindValue')
  24. call_user_func_array(array($this->explain_query, $function), $args);
  25. $return = call_user_func_array(array($this->query, $function), $args);
  26. if ($config['debug'] && $function == 'execute') {
  27. $time = microtime(true) - $start;
  28. $debug['sql'][] = array(
  29. 'query' => $this->query->queryString,
  30. 'rows' => $this->query->rowCount(),
  31. 'explain' => $this->explain_query ? $this->explain_query->fetchAll(PDO::FETCH_ASSOC) : null,
  32. 'time' => '~' . round($time * 1000, 2) . 'ms'
  33. );
  34. $debug['time']['db_queries'] += $time;
  35. }
  36. return $return;
  37. }
  38. }
  39. function sql_open() {
  40. global $pdo, $config, $debug;
  41. if ($pdo)
  42. return true;
  43. if ($config['debug'])
  44. $start = microtime(true);
  45. if (isset($config['db']['server'][0]) && $config['db']['server'][0] == ':')
  46. $unix_socket = substr($config['db']['server'], 1);
  47. else
  48. $unix_socket = false;
  49. $dsn = $config['db']['type'] . ':' .
  50. ($unix_socket ? 'unix_socket=' . $unix_socket : 'host=' . $config['db']['server']) .
  51. ';dbname=' . $config['db']['database'];
  52. if (!empty($config['db']['dsn']))
  53. $dsn .= ';' . $config['db']['dsn'];
  54. try {
  55. $options = array(
  56. PDO::ATTR_TIMEOUT => $config['db']['timeout'],
  57. PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => true
  58. );
  59. if ($config['db']['persistent'])
  60. $options[PDO::ATTR_PERSISTENT] = true;
  61. $pdo = new PDO($dsn, $config['db']['user'], $config['db']['password'], $options);
  62. if ($config['debug'])
  63. $debug['time']['db_connect'] = '~' . round((microtime(true) - $start) * 1000, 2) . 'ms';
  64. if (mysql_version() >= 50503)
  65. query('SET NAMES utf8mb4') or error(db_error());
  66. else
  67. query('SET NAMES utf8') or error(db_error());
  68. return $pdo;
  69. } catch(PDOException $e) {
  70. $message = $e->getMessage();
  71. // Remove any sensitive information
  72. $message = str_replace($config['db']['user'], '<em>hidden</em>', $message);
  73. $message = str_replace($config['db']['password'], '<em>hidden</em>', $message);
  74. // Print error
  75. error(_('Database error: ') . $message);
  76. }
  77. }
  78. // 5.6.10 becomes 50610
  79. function mysql_version() {
  80. global $pdo;
  81. $version = $pdo->getAttribute(PDO::ATTR_SERVER_VERSION);
  82. $v = explode('.', $version);
  83. if (count($v) != 3)
  84. return false;
  85. return (int) sprintf("%02d%02d%02d", $v[0], $v[1], $v[2]);
  86. }
  87. function prepare($query) {
  88. global $pdo, $debug, $config;
  89. $query = preg_replace('/``('.$config['board_regex'].')``/u', '`' . $config['db']['prefix'] . '$1`', $query);
  90. sql_open();
  91. if ($config['debug'])
  92. return new PreparedQueryDebug($query);
  93. return $pdo->prepare($query);
  94. }
  95. function query($query) {
  96. global $pdo, $debug, $config;
  97. $query = preg_replace('/``('.$config['board_regex'].')``/u', '`' . $config['db']['prefix'] . '$1`', $query);
  98. sql_open();
  99. if ($config['debug']) {
  100. if ($config['debug_explain'] && preg_match('/^(SELECT|INSERT|UPDATE|DELETE) /i', $query)) {
  101. $explain = $pdo->query("EXPLAIN $query") or error(db_error());
  102. }
  103. $start = microtime(true);
  104. $query = $pdo->query($query);
  105. if (!$query)
  106. return false;
  107. $time = microtime(true) - $start;
  108. $debug['sql'][] = array(
  109. 'query' => $query->queryString,
  110. 'rows' => $query->rowCount(),
  111. 'explain' => isset($explain) ? $explain->fetchAll(PDO::FETCH_ASSOC) : null,
  112. 'time' => '~' . round($time * 1000, 2) . 'ms'
  113. );
  114. $debug['time']['db_queries'] += $time;
  115. return $query;
  116. }
  117. return $pdo->query($query);
  118. }
  119. function db_error($PDOStatement = null) {
  120. global $pdo, $db_error;
  121. if (isset($PDOStatement)) {
  122. $db_error = $PDOStatement->errorInfo();
  123. return $db_error[2];
  124. }
  125. $db_error = $pdo->errorInfo();
  126. return $db_error[2];
  127. }