new hook-based system for inserting callbacks to run before and after a query
debugMode() now works through the new system
This commit is contained in:
297
db.class.php
297
db.class.php
@@ -30,19 +30,13 @@ class DB {
|
|||||||
// configure workings
|
// configure workings
|
||||||
public static $param_char = '%';
|
public static $param_char = '%';
|
||||||
public static $named_param_seperator = '_';
|
public static $named_param_seperator = '_';
|
||||||
public static $success_handler = false;
|
|
||||||
public static $error_handler = true;
|
|
||||||
public static $throw_exception_on_error = false;
|
|
||||||
public static $nonsql_error_handler = null;
|
|
||||||
public static $pre_sql_handler = false;
|
|
||||||
public static $throw_exception_on_nonsql_error = false;
|
|
||||||
public static $nested_transactions = false;
|
public static $nested_transactions = false;
|
||||||
public static $ssl = array('key' => '', 'cert' => '', 'ca_cert' => '', 'ca_path' => '', 'cipher' => '');
|
public static $ssl = array('key' => '', 'cert' => '', 'ca_cert' => '', 'ca_path' => '', 'cipher' => '');
|
||||||
public static $connect_options = array(MYSQLI_OPT_CONNECT_TIMEOUT => 30);
|
public static $connect_options = array(MYSQLI_OPT_CONNECT_TIMEOUT => 30);
|
||||||
|
|
||||||
// internal
|
// internal
|
||||||
protected static $mdb = null;
|
protected static $mdb = null;
|
||||||
public static $variables_to_sync = array('param_char', 'named_param_seperator', 'success_handler', 'error_handler', 'throw_exception_on_error', 'nonsql_error_handler', 'pre_sql_handler', 'throw_exception_on_nonsql_error', 'nested_transactions', 'ssl', 'connect_options');
|
public static $variables_to_sync = array('param_char', 'named_param_seperator', 'nested_transactions', 'ssl', 'connect_options');
|
||||||
|
|
||||||
public static function getMDB() {
|
public static function getMDB() {
|
||||||
$mdb = DB::$mdb;
|
$mdb = DB::$mdb;
|
||||||
@@ -65,11 +59,6 @@ class DB {
|
|||||||
|
|
||||||
return call_user_func_array($fn, $args);
|
return call_user_func_array($fn, $args);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function debugMode($handler = true) {
|
|
||||||
DB::$success_handler = $handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -86,12 +75,6 @@ class MeekroDB {
|
|||||||
// configure workings
|
// configure workings
|
||||||
public $param_char = '%';
|
public $param_char = '%';
|
||||||
public $named_param_seperator = '_';
|
public $named_param_seperator = '_';
|
||||||
public $success_handler = false;
|
|
||||||
public $error_handler = true;
|
|
||||||
public $throw_exception_on_error = false;
|
|
||||||
public $nonsql_error_handler = null;
|
|
||||||
public $pre_sql_handler = false;
|
|
||||||
public $throw_exception_on_nonsql_error = false;
|
|
||||||
public $nested_transactions = false;
|
public $nested_transactions = false;
|
||||||
public $ssl = array('key' => '', 'cert' => '', 'ca_cert' => '', 'ca_path' => '', 'cipher' => '');
|
public $ssl = array('key' => '', 'cert' => '', 'ca_cert' => '', 'ca_path' => '', 'cipher' => '');
|
||||||
public $connect_options = array(MYSQLI_OPT_CONNECT_TIMEOUT => 30);
|
public $connect_options = array(MYSQLI_OPT_CONNECT_TIMEOUT => 30);
|
||||||
@@ -105,6 +88,14 @@ class MeekroDB {
|
|||||||
public $current_db = null;
|
public $current_db = null;
|
||||||
public $nested_transactions_count = 0;
|
public $nested_transactions_count = 0;
|
||||||
|
|
||||||
|
protected $hooks = array(
|
||||||
|
'pre_parse' => array(),
|
||||||
|
'pre_run' => array(),
|
||||||
|
'post_run' => array(),
|
||||||
|
'run_success' => array(),
|
||||||
|
'run_failed' => array(),
|
||||||
|
);
|
||||||
|
|
||||||
public function __construct($host=null, $user=null, $password=null, $dbName=null, $port=null, $encoding=null, $socket=null) {
|
public function __construct($host=null, $user=null, $password=null, $dbName=null, $port=null, $encoding=null, $socket=null) {
|
||||||
if ($host === null) $host = DB::$host;
|
if ($host === null) $host = DB::$host;
|
||||||
if ($user === null) $user = DB::$user;
|
if ($user === null) $user = DB::$user;
|
||||||
@@ -155,7 +146,7 @@ class MeekroDB {
|
|||||||
@$mysql->real_connect($this->host, $this->user, $this->password, $this->dbName, $this->port, $this->socket, $connect_flags);
|
@$mysql->real_connect($this->host, $this->user, $this->password, $this->dbName, $this->port, $this->socket, $connect_flags);
|
||||||
|
|
||||||
if ($mysql->connect_error) {
|
if ($mysql->connect_error) {
|
||||||
return $this->nonSQLError('Unable to connect to MySQL server! Error: ' . $mysql->connect_error);
|
throw new MeekroDBException("Unable to connect to MySQL server! Error: {$mysql->connect_error}");
|
||||||
}
|
}
|
||||||
|
|
||||||
$mysql->set_charset($this->encoding);
|
$mysql->set_charset($this->encoding);
|
||||||
@@ -175,22 +166,129 @@ class MeekroDB {
|
|||||||
$this->internal_mysql = null;
|
$this->internal_mysql = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function nonSQLError($message) {
|
function addHook($type, $fn) {
|
||||||
if ($this->throw_exception_on_nonsql_error) {
|
if (! array_key_exists($type, $this->hooks)) {
|
||||||
$e = new MeekroDBException($message);
|
throw new MeekroDBException("Hook type $type is not recognized");
|
||||||
throw $e;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$error_handler = is_callable($this->nonsql_error_handler) ? $this->nonsql_error_handler : 'meekrodb_error_handler';
|
if (! is_callable($fn)) {
|
||||||
|
throw new MeekroDBException("Second arg to addHook() must be callable");
|
||||||
call_user_func($error_handler, array(
|
|
||||||
'type' => 'nonsql',
|
|
||||||
'error' => $message
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function debugMode($handler = true) {
|
$this->hooks[$type][] = $fn;
|
||||||
$this->success_handler = $handler;
|
end($this->hooks[$type]);
|
||||||
|
return key($this->hooks[$type]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeHook($type, $index) {
|
||||||
|
if (! array_key_exists($type, $this->hooks)) {
|
||||||
|
throw new MeekroDBException("Hook type $type is not recognized");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (! array_key_exists($index, $this->hooks[$type])) {
|
||||||
|
throw new MeekroDBException("That hook does not exist");
|
||||||
|
}
|
||||||
|
|
||||||
|
unset($this->hooks[$type][$index]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeHooks($type) {
|
||||||
|
if (! array_key_exists($type, $this->hooks)) {
|
||||||
|
throw new MeekroDBException("Hook type $type is not recognized");
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->hooks[$type] = array();
|
||||||
|
}
|
||||||
|
|
||||||
|
function runHook($type, $args=array()) {
|
||||||
|
if (! array_key_exists($type, $this->hooks)) {
|
||||||
|
throw new MeekroDBException("Hook type $type is not recognized");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($type == 'pre_parse') {
|
||||||
|
$query = $args['query'];
|
||||||
|
$args = $args['args'];
|
||||||
|
|
||||||
|
foreach ($this->hooks[$type] as $hook) {
|
||||||
|
$result = call_user_func($hook, array('query' => $query, 'args' => $args));
|
||||||
|
if ($result) {
|
||||||
|
if (!is_array($result) || count($result) != 2) {
|
||||||
|
throw new MeekroDBException("pre_parse hook must return an array of 2 items");
|
||||||
|
}
|
||||||
|
if (!is_string($result[0])) {
|
||||||
|
throw new MeekroDBException("pre_parse hook must return a string as its first item");
|
||||||
|
}
|
||||||
|
if (!is_array($result[1])) {
|
||||||
|
throw new MeekroDBException("pre_parse hook must return an array as its second item");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = $result[0];
|
||||||
|
$args = $result[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return array($query, $args);
|
||||||
|
}
|
||||||
|
else if ($type == 'pre_run') {
|
||||||
|
$query = $args['query'];
|
||||||
|
|
||||||
|
foreach ($this->hooks[$type] as $hook) {
|
||||||
|
$result = call_user_func($hook, array('query' => $query));
|
||||||
|
if (!is_string($result)) {
|
||||||
|
throw new MeekroDBException("pre_run hook must return a string");
|
||||||
|
}
|
||||||
|
|
||||||
|
$query = $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $query;
|
||||||
|
}
|
||||||
|
else if ($type == 'post_run') {
|
||||||
|
|
||||||
|
foreach ($this->hooks[$type] as $hook) {
|
||||||
|
call_user_func($hook, $args);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if ($type == 'run_success') {
|
||||||
|
|
||||||
|
foreach ($this->hooks[$type] as $hook) {
|
||||||
|
call_user_func($hook, $args);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if ($type == 'run_failed') {
|
||||||
|
|
||||||
|
foreach ($this->hooks[$type] as $hook) {
|
||||||
|
$result = call_user_func($hook, $args);
|
||||||
|
if ($result === false) return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new MeekroDBException("runHook() type $type not recognized");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function debugMode($enable = true) {
|
||||||
|
$fn = function($args) {
|
||||||
|
$results[] = sprintf('QUERY: %s [%s ms]', $args['query'], $args['runtime']);
|
||||||
|
|
||||||
|
if (isset($args['error'])) {
|
||||||
|
$results[] = 'ERROR: ' . $args['error'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (php_sapi_name() == 'cli' && empty($_SERVER['REMOTE_ADDR'])) {
|
||||||
|
echo implode("\n", $results) . "\n";
|
||||||
|
} else {
|
||||||
|
echo implode("<br>\n", $results) . "<br>\n";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if ($enable && !isset($this->debug_mode_hook)) {
|
||||||
|
$this->debug_mode_hook = $this->addHook('post_run', $fn);
|
||||||
|
}
|
||||||
|
else if (!$enable && isset($this->debug_mode_hook)) {
|
||||||
|
$this->removeHook('post_run', $this->debug_mode_hook);
|
||||||
|
unset($this->debug_mode_hook);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function serverVersion() { $this->get(); return $this->server_info; }
|
public function serverVersion() { $this->get(); return $this->server_info; }
|
||||||
@@ -203,14 +301,14 @@ class MeekroDB {
|
|||||||
public function useDB() { $args = func_get_args(); return call_user_func_array(array($this, 'setDB'), $args); }
|
public function useDB() { $args = func_get_args(); return call_user_func_array(array($this, 'setDB'), $args); }
|
||||||
public function setDB($dbName) {
|
public function setDB($dbName) {
|
||||||
$db = $this->get();
|
$db = $this->get();
|
||||||
if (! $db->select_db($dbName)) return $this->nonSQLError("Unable to set database to $dbName");
|
if (! $db->select_db($dbName)) throw new MeekroDBException("Unable to set database to $dbName");
|
||||||
$this->current_db = $dbName;
|
$this->current_db = $dbName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public function startTransaction() {
|
public function startTransaction() {
|
||||||
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
||||||
return $this->nonSQLError("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
throw new MeekroDBException("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$this->nested_transactions || $this->nested_transactions_count == 0) {
|
if (!$this->nested_transactions || $this->nested_transactions_count == 0) {
|
||||||
@@ -226,7 +324,7 @@ class MeekroDB {
|
|||||||
|
|
||||||
public function commit($all=false) {
|
public function commit($all=false) {
|
||||||
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
||||||
return $this->nonSQLError("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
throw new MeekroDBException("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->nested_transactions && $this->nested_transactions_count > 0)
|
if ($this->nested_transactions && $this->nested_transactions_count > 0)
|
||||||
@@ -244,7 +342,7 @@ class MeekroDB {
|
|||||||
|
|
||||||
public function rollback($all=false) {
|
public function rollback($all=false) {
|
||||||
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
if ($this->nested_transactions && $this->serverVersion() < '5.5') {
|
||||||
return $this->nonSQLError("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
throw new MeekroDBException("Nested transactions are only available on MySQL 5.5 and greater. You are using MySQL " . $this->serverVersion());
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->nested_transactions && $this->nested_transactions_count > 0)
|
if ($this->nested_transactions && $this->nested_transactions_count > 0)
|
||||||
@@ -303,7 +401,7 @@ class MeekroDB {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($which != 'INSERT' && $which != 'INSERT IGNORE' && $which != 'REPLACE') {
|
if ($which != 'INSERT' && $which != 'INSERT IGNORE' && $which != 'REPLACE') {
|
||||||
return $this->nonSQLError('insertOrReplace() must be called with one of: INSERT, INSERT IGNORE, REPLACE');
|
throw new MeekroDBException('insertOrReplace() must be called with one of: INSERT, INSERT IGNORE, REPLACE');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isset($options['update']) && is_array($options['update']) && $options['update'] && $which == 'INSERT') {
|
if (isset($options['update']) && is_array($options['update']) && $options['update'] && $which == 'INSERT') {
|
||||||
@@ -338,7 +436,7 @@ class MeekroDB {
|
|||||||
|
|
||||||
if (! isset($args[0])) { // update will have all the data of the insert
|
if (! isset($args[0])) { // update will have all the data of the insert
|
||||||
if (isset($data[0]) && is_array($data[0])) { //multiple insert rows specified -- failing!
|
if (isset($data[0]) && is_array($data[0])) { //multiple insert rows specified -- failing!
|
||||||
return $this->nonSQLError("Badly formatted insertUpdate() query -- you didn't specify the update component!");
|
throw new MeekroDBException("Badly formatted insertUpdate() query -- you didn't specify the update component!");
|
||||||
}
|
}
|
||||||
|
|
||||||
$args[0] = $data;
|
$args[0] = $data;
|
||||||
@@ -504,15 +602,15 @@ class MeekroDB {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($use_named_args && $use_numbered_args) {
|
if ($use_named_args && $use_numbered_args) {
|
||||||
return $this->nonSQLError("You can't mix named and numbered args!");
|
throw new MeekroDBException("You can't mix named and numbered args!");
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($use_named_args && count($args) != 1) {
|
if ($use_named_args && count($args) != 1) {
|
||||||
return $this->nonSQLError("If you use named args, you must pass an assoc array of args!");
|
throw new MeekroDBException("If you use named args, you must pass an assoc array of args!");
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($use_numbered_args && $max_numbered_arg+1 > count($args)) {
|
if ($use_numbered_args && $max_numbered_arg+1 > count($args)) {
|
||||||
return $this->nonSQLError(sprintf('Expected %d args, but only got %d!', $max_numbered_arg+1, count($args)));
|
throw new MeekroDBException(sprintf('Expected %d args, but only got %d!', $max_numbered_arg+1, count($args)));
|
||||||
}
|
}
|
||||||
|
|
||||||
return $queryParts;
|
return $queryParts;
|
||||||
@@ -542,7 +640,7 @@ class MeekroDB {
|
|||||||
if (!is_null($Part['named_arg'])) {
|
if (!is_null($Part['named_arg'])) {
|
||||||
$key = $Part['named_arg'];
|
$key = $Part['named_arg'];
|
||||||
if (! array_key_exists($key, $args[0])) {
|
if (! array_key_exists($key, $args[0])) {
|
||||||
return $this->nonSQLError("Couldn't find named arg {$key}!");
|
throw new MeekroDBException("Couldn't find named arg {$key}!");
|
||||||
}
|
}
|
||||||
|
|
||||||
$val = $args[0][$key];
|
$val = $args[0][$key];
|
||||||
@@ -553,10 +651,10 @@ class MeekroDB {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ($is_array_type && !is_array($val)) {
|
if ($is_array_type && !is_array($val)) {
|
||||||
return $this->nonSQLError("Expected an array for arg $key but didn't get one!");
|
throw new MeekroDBException("Expected an array for arg $key but didn't get one!");
|
||||||
}
|
}
|
||||||
if ($is_array_type && count($val) == 0) {
|
if ($is_array_type && count($val) == 0) {
|
||||||
return $this->nonSQLError("Arg {$key} array can't be empty!");
|
throw new MeekroDBException("Arg {$key} array can't be empty!");
|
||||||
}
|
}
|
||||||
if (!$is_array_type && is_array($val)) {
|
if (!$is_array_type && is_array($val)) {
|
||||||
$val = '';
|
$val = '';
|
||||||
@@ -564,7 +662,7 @@ class MeekroDB {
|
|||||||
|
|
||||||
if (is_object($val) && ($val instanceof WhereClause)) {
|
if (is_object($val) && ($val instanceof WhereClause)) {
|
||||||
if ($Part['type'] != 'l') {
|
if ($Part['type'] != 'l') {
|
||||||
return $this->nonSQLError("WhereClause must be used with l arg, you used {$Part['type']} instead!");
|
throw new MeekroDBException("WhereClause must be used with l arg, you used {$Part['type']} instead!");
|
||||||
}
|
}
|
||||||
|
|
||||||
list($clause_sql, $clause_args) = $val->textAndArgs();
|
list($clause_sql, $clause_args) = $val->textAndArgs();
|
||||||
@@ -604,7 +702,7 @@ class MeekroDB {
|
|||||||
$value = array_values($value);
|
$value = array_values($value);
|
||||||
return '(' . implode(', ', array_map(array($this, 'sanitize'), $value)) . ')';
|
return '(' . implode(', ', array_map(array($this, 'sanitize'), $value)) . ')';
|
||||||
} else {
|
} else {
|
||||||
return $this->nonSQLError("Expected array parameter, got something different!");
|
throw new MeekroDBException("Expected array parameter, got something different!");
|
||||||
}
|
}
|
||||||
} else if ($type == 'doublelist') {
|
} else if ($type == 'doublelist') {
|
||||||
if (is_array($value) && array_values($value) === $value && is_array($value[0])) {
|
if (is_array($value) && array_values($value) === $value && is_array($value[0])) {
|
||||||
@@ -615,7 +713,7 @@ class MeekroDB {
|
|||||||
return implode(', ', $cleanvalues);
|
return implode(', ', $cleanvalues);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
return $this->nonSQLError("Expected double array parameter, got something different!");
|
throw new MeekroDBException("Expected double array parameter, got something different!");
|
||||||
}
|
}
|
||||||
} else if ($type == 'hash') {
|
} else if ($type == 'hash') {
|
||||||
if (is_array($value)) {
|
if (is_array($value)) {
|
||||||
@@ -626,10 +724,10 @@ class MeekroDB {
|
|||||||
|
|
||||||
return implode($hashjoin, $pairs);
|
return implode($hashjoin, $pairs);
|
||||||
} else {
|
} else {
|
||||||
return $this->nonSQLError("Expected hash (associative array) parameter, got something different!");
|
throw new MeekroDBException("Expected hash (associative array) parameter, got something different!");
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return $this->nonSQLError("Invalid type passed to sanitize()!");
|
throw new MeekroDBException("Invalid type passed to sanitize()!");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -661,6 +759,7 @@ class MeekroDB {
|
|||||||
protected function queryHelper() {
|
protected function queryHelper() {
|
||||||
$args = func_get_args();
|
$args = func_get_args();
|
||||||
$type = array_shift($args);
|
$type = array_shift($args);
|
||||||
|
$query = array_shift($args);
|
||||||
$db = $this->get();
|
$db = $this->get();
|
||||||
|
|
||||||
$is_buffered = true;
|
$is_buffered = true;
|
||||||
@@ -685,49 +784,17 @@ class MeekroDB {
|
|||||||
$row_type = 'raw';
|
$row_type = 'raw';
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return $this->nonSQLError('Error -- invalid argument to queryHelper!');
|
throw new MeekroDBException('Invalid argument to queryHelper!');
|
||||||
}
|
}
|
||||||
|
|
||||||
$sql = call_user_func_array(array($this, 'parse'), $args);
|
list($query, $args) = $this->runHook('pre_parse', array('query' => $query, 'args' => $args));
|
||||||
|
$sql = call_user_func_array(array($this, 'parse'), array_merge(array($query), $args));
|
||||||
|
$sql = $this->runHook('pre_run', array('query' => $sql));
|
||||||
|
|
||||||
if ($this->pre_sql_handler !== false && is_callable($this->pre_sql_handler)) {
|
$starttime = microtime(true);
|
||||||
$sql = call_user_func($this->pre_sql_handler, $sql);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($this->success_handler) $starttime = microtime(true);
|
|
||||||
$result = $db->query($sql, $is_buffered ? MYSQLI_STORE_RESULT : MYSQLI_USE_RESULT);
|
$result = $db->query($sql, $is_buffered ? MYSQLI_STORE_RESULT : MYSQLI_USE_RESULT);
|
||||||
if ($this->success_handler) $runtime = microtime(true) - $starttime;
|
$runtime = microtime(true) - $starttime;
|
||||||
else $runtime = 0;
|
|
||||||
|
|
||||||
// ----- BEGIN ERROR HANDLING
|
|
||||||
if (!$sql || $db->error) {
|
|
||||||
if ($this->error_handler) {
|
|
||||||
$error_handler = is_callable($this->error_handler) ? $this->error_handler : 'meekrodb_error_handler';
|
|
||||||
|
|
||||||
call_user_func($error_handler, array(
|
|
||||||
'type' => 'sql',
|
|
||||||
'query' => $sql,
|
|
||||||
'error' => $db->error,
|
|
||||||
'code' => $db->errno
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($this->throw_exception_on_error) {
|
|
||||||
$e = new MeekroDBException($db->error, $sql, $db->errno);
|
|
||||||
throw $e;
|
|
||||||
}
|
|
||||||
} else if ($this->success_handler) {
|
|
||||||
$runtime = sprintf('%f', $runtime * 1000);
|
$runtime = sprintf('%f', $runtime * 1000);
|
||||||
$success_handler = is_callable($this->success_handler) ? $this->success_handler : 'meekrodb_debugmode_handler';
|
|
||||||
|
|
||||||
call_user_func($success_handler, array(
|
|
||||||
'query' => $sql,
|
|
||||||
'runtime' => $runtime,
|
|
||||||
'affected' => $db->affected_rows
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
// ----- END ERROR HANDLING
|
|
||||||
|
|
||||||
$this->insert_id = $db->insert_id;
|
$this->insert_id = $db->insert_id;
|
||||||
$this->affected_rows = $db->affected_rows;
|
$this->affected_rows = $db->affected_rows;
|
||||||
@@ -736,6 +803,37 @@ class MeekroDB {
|
|||||||
if ($is_buffered && ($result instanceof MySQLi_Result)) $this->num_rows = $result->num_rows;
|
if ($is_buffered && ($result instanceof MySQLi_Result)) $this->num_rows = $result->num_rows;
|
||||||
else $this->num_rows = null;
|
else $this->num_rows = null;
|
||||||
|
|
||||||
|
$Exception = null;
|
||||||
|
if (!$sql || $db->error) {
|
||||||
|
$Exception = new MeekroDBException($db->error, $sql, $db->errno);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->runHook('post_run', array(
|
||||||
|
'query' => $sql,
|
||||||
|
'runtime' => $runtime,
|
||||||
|
'affected' => $db->affected_rows,
|
||||||
|
'exception' => $Exception,
|
||||||
|
'error' => $Exception ? $Exception->getMessage() : null,
|
||||||
|
));
|
||||||
|
|
||||||
|
if ($Exception) {
|
||||||
|
$result = $this->runHook('run_failed', array(
|
||||||
|
'query' => $sql,
|
||||||
|
'runtime' => $runtime,
|
||||||
|
'exception' => $Exception,
|
||||||
|
'error' => $Exception->getMessage(),
|
||||||
|
));
|
||||||
|
|
||||||
|
if ($result !== false) throw $Exception;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$this->runHook('run_success', array(
|
||||||
|
'query' => $sql,
|
||||||
|
'runtime' => $runtime,
|
||||||
|
'affected' => $db->affected_rows,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
if ($row_type == 'raw' || !($result instanceof MySQLi_Result)) return $result;
|
if ($row_type == 'raw' || !($result instanceof MySQLi_Result)) return $result;
|
||||||
|
|
||||||
$return = array();
|
$return = array();
|
||||||
@@ -797,6 +895,8 @@ class MeekroDB {
|
|||||||
if ($row == null) return null;
|
if ($row == null) return null;
|
||||||
return $row[0];
|
return $row[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class WhereClause {
|
class WhereClause {
|
||||||
@@ -806,7 +906,7 @@ class WhereClause {
|
|||||||
|
|
||||||
function __construct($type) {
|
function __construct($type) {
|
||||||
$type = strtolower($type);
|
$type = strtolower($type);
|
||||||
if ($type !== 'or' && $type !== 'and') return DB::nonSQLError('you must use either WhereClause(and) or WhereClause(or)');
|
if ($type !== 'or' && $type !== 'and') throw new MeekroDBException('you must use either WhereClause(and) or WhereClause(or)');
|
||||||
$this->type = $type;
|
$this->type = $type;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -951,29 +1051,6 @@ class DBHelper {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function meekrodb_error_handler($params) {
|
|
||||||
if (isset($params['query'])) $out[] = "QUERY: " . $params['query'];
|
|
||||||
if (isset($params['error'])) $out[] = "ERROR: " . $params['error'];
|
|
||||||
$out[] = "";
|
|
||||||
|
|
||||||
if (php_sapi_name() == 'cli' && empty($_SERVER['REMOTE_ADDR'])) {
|
|
||||||
echo implode("\n", $out);
|
|
||||||
} else {
|
|
||||||
echo implode("<br>\n", $out);
|
|
||||||
}
|
|
||||||
|
|
||||||
die;
|
|
||||||
}
|
|
||||||
|
|
||||||
function meekrodb_debugmode_handler($params) {
|
|
||||||
echo "QUERY: " . $params['query'] . " [" . $params['runtime'] . " ms]";
|
|
||||||
if (php_sapi_name() == 'cli' && empty($_SERVER['REMOTE_ADDR'])) {
|
|
||||||
echo "\n";
|
|
||||||
} else {
|
|
||||||
echo "<br>\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class MeekroDBEval {
|
class MeekroDBEval {
|
||||||
public $text = '';
|
public $text = '';
|
||||||
|
|
||||||
|
|||||||
@@ -1,48 +1,52 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
function new_error_callback($params) {
|
function my_error_handler($params) {
|
||||||
global $error_callback_worked;
|
global $error_callback_worked;
|
||||||
|
|
||||||
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $error_callback_worked = 1;
|
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $error_callback_worked = 1;
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function my_debug_handler($params) {
|
function my_success_handler($params) {
|
||||||
global $debug_callback_worked;
|
global $debug_callback_worked;
|
||||||
if (substr_count($params['query'], 'SELECT')) $debug_callback_worked = 1;
|
if (substr_count($params['query'], 'SELECT')) $debug_callback_worked = 1;
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class ErrorTest extends SimpleTest {
|
class ErrorTest extends SimpleTest {
|
||||||
|
static function static_error_callback($params) {
|
||||||
|
global $static_error_callback_worked;
|
||||||
|
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $static_error_callback_worked = 1;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function nonstatic_error_callback($params) {
|
||||||
|
global $nonstatic_error_callback_worked;
|
||||||
|
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $nonstatic_error_callback_worked = 1;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
function test_1_error_handler() {
|
function test_1_error_handler() {
|
||||||
global $error_callback_worked, $static_error_callback_worked, $nonstatic_error_callback_worked;
|
global $error_callback_worked, $static_error_callback_worked, $nonstatic_error_callback_worked;
|
||||||
|
|
||||||
DB::$error_handler = 'new_error_callback';
|
DB::addHook('run_failed', 'my_error_handler');
|
||||||
DB::query("SELET * FROM accounts");
|
DB::query("SELET * FROM accounts");
|
||||||
$this->assert($error_callback_worked === 1);
|
$this->assert($error_callback_worked === 1);
|
||||||
|
|
||||||
DB::$error_handler = array('ErrorTest', 'static_error_callback');
|
DB::removeHooks('run_failed');
|
||||||
|
DB::addHook('run_failed', array('ErrorTest', 'static_error_callback'));
|
||||||
DB::query("SELET * FROM accounts");
|
DB::query("SELET * FROM accounts");
|
||||||
$this->assert($static_error_callback_worked === 1);
|
$this->assert($static_error_callback_worked === 1);
|
||||||
|
|
||||||
DB::$error_handler = array($this, 'nonstatic_error_callback');
|
DB::removeHooks('run_failed');
|
||||||
|
DB::addHook('run_failed', array($this, 'nonstatic_error_callback'));
|
||||||
DB::query("SELET * FROM accounts");
|
DB::query("SELET * FROM accounts");
|
||||||
$this->assert($nonstatic_error_callback_worked === 1);
|
$this->assert($nonstatic_error_callback_worked === 1);
|
||||||
|
DB::removeHooks('run_failed');
|
||||||
}
|
|
||||||
|
|
||||||
public static function static_error_callback($params) {
|
|
||||||
global $static_error_callback_worked;
|
|
||||||
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $static_error_callback_worked = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function nonstatic_error_callback($params) {
|
|
||||||
global $nonstatic_error_callback_worked;
|
|
||||||
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $nonstatic_error_callback_worked = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_2_exception_catch() {
|
function test_2_exception_catch() {
|
||||||
$dbname = DB::$dbName;
|
$dbname = DB::$dbName;
|
||||||
DB::$error_handler = '';
|
|
||||||
DB::$throw_exception_on_error = true;
|
|
||||||
try {
|
try {
|
||||||
DB::query("SELET * FROM accounts");
|
DB::query("SELET * FROM accounts");
|
||||||
} catch(MeekroDBException $e) {
|
} catch(MeekroDBException $e) {
|
||||||
@@ -67,15 +71,29 @@ class ErrorTest extends SimpleTest {
|
|||||||
$this->assert($exception_was_caught === 2);
|
$this->assert($exception_was_caught === 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_3_debugmode_handler() {
|
function test_3_success_handler() {
|
||||||
global $debug_callback_worked;
|
global $debug_callback_worked;
|
||||||
|
|
||||||
DB::debugMode('my_debug_handler');
|
DB::addHook('run_success', 'my_success_handler');
|
||||||
DB::query("SELECT * FROM accounts WHERE username!=%s", "Charlie's Friend");
|
DB::query("SELECT * FROM accounts WHERE username!=%s", "Charlie's Friend");
|
||||||
|
|
||||||
$this->assert($debug_callback_worked === 1);
|
$this->assert($debug_callback_worked === 1);
|
||||||
|
DB::removeHooks('run_success');
|
||||||
|
}
|
||||||
|
|
||||||
DB::debugMode(false);
|
function test_4_error_handler() {
|
||||||
|
global $anonymous_error_callback_worked;
|
||||||
|
|
||||||
|
$error_handler = function($params) {
|
||||||
|
global $anonymous_error_callback_worked;
|
||||||
|
if (substr_count($params['error'], 'You have an error in your SQL syntax')) {
|
||||||
|
$anonymous_error_callback_worked = 1;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
DB::addHook('run_failed', $error_handler);
|
||||||
|
DB::query("SELET * FROM accounts");
|
||||||
|
$this->assert($anonymous_error_callback_worked === 1);
|
||||||
|
DB::removeHooks('run_failed');
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,19 +0,0 @@
|
|||||||
<?php
|
|
||||||
class ErrorTest_53 extends SimpleTest {
|
|
||||||
function test_1_error_handler() {
|
|
||||||
global $anonymous_error_callback_worked;
|
|
||||||
|
|
||||||
DB::$throw_exception_on_error = false;
|
|
||||||
DB::$error_handler = function($params) {
|
|
||||||
global $anonymous_error_callback_worked;
|
|
||||||
if (substr_count($params['error'], 'You have an error in your SQL syntax')) $anonymous_error_callback_worked = 1;
|
|
||||||
};
|
|
||||||
DB::query("SELET * FROM accounts");
|
|
||||||
$this->assert($anonymous_error_callback_worked === 1);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
||||||
@@ -35,7 +35,6 @@ require_once __DIR__ . '/CallTest.php';
|
|||||||
require_once __DIR__ . '/ObjectTest.php';
|
require_once __DIR__ . '/ObjectTest.php';
|
||||||
require_once __DIR__ . '/WhereClauseTest.php';
|
require_once __DIR__ . '/WhereClauseTest.php';
|
||||||
require_once __DIR__ . '/ErrorTest.php';
|
require_once __DIR__ . '/ErrorTest.php';
|
||||||
require_once __DIR__ . '/ErrorTest_53.php';
|
|
||||||
require_once __DIR__ . '/TransactionTest.php';
|
require_once __DIR__ . '/TransactionTest.php';
|
||||||
require_once __DIR__ . '/HelperTest.php';
|
require_once __DIR__ . '/HelperTest.php';
|
||||||
|
|
||||||
@@ -45,7 +44,6 @@ $classes_to_test = array(
|
|||||||
'WhereClauseTest',
|
'WhereClauseTest',
|
||||||
'ObjectTest',
|
'ObjectTest',
|
||||||
'ErrorTest',
|
'ErrorTest',
|
||||||
'ErrorTest_53',
|
|
||||||
'TransactionTest',
|
'TransactionTest',
|
||||||
'HelperTest',
|
'HelperTest',
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user