_dbi = $dbi; $this->_registry = $registry; $this->_registry->get( 'controller.shutdown' )->register( array( $this, 'shutdown' ) ); add_action( 'ai1ec_loaded', array( $this, 'check_debug' ), PHP_INT_MAX ); $this->set_timezone(); } /** * Set timezone to UTC to avoid conversion errors. * * @return void */ public function set_timezone() { $this->_dbi->query( "SET time_zone = '+0:00'" ); } /** * Call explicitly when debug output must be disabled. * * @return void Method is not meant to return. */ public function disable_debug() { $this->_log_enabled = false; } /** * Only attempt to enable debug after all add-ons are loaded. * * @wp_hook ai1ec_loaded * * @uses apply_filters ai1ec_dbi_debug * * @return void */ public function check_debug() { $this->_log_enabled = apply_filters( 'ai1ec_dbi_debug', ( false !== AI1EC_DEBUG ) ); } /** * Perform a MySQL database query, using current database connection. * * @param string $sql_query Database query * * @return int|false Number of rows affected/selected or false on error */ public function query( $sql_query ) { $this->_query_profile( $sql_query ); $result = $this->_dbi->query( $sql_query ); $this->_query_profile( $result ); return $result; } /** * Retrieve one column from the database. * * Executes a SQL query and returns the column from the SQL result. * If the SQL result contains more than one column, this function returns the column specified. * If $query is null, this function returns the specified column from the previous SQL result. * * @param string|null $query Optional. SQL query. Defaults to previous query. * @param int $col Optional. Column to return. Indexed from 0. * * @return array Database query result. Array indexed from 0 by SQL result row number. */ public function get_col( $query = null , $col = 0 ) { $this->_query_profile( $query ); $result = $this->_dbi->get_col( $query, $col ); $this->_query_profile( count( $result ) ); return $result; } /** * Check if the terms variable is set in the Wpdb object */ public function are_terms_set() { return isset( $this->_dbi->terms ); } /** * Prepares a SQL query for safe execution. Uses sprintf()-like syntax. * * The following directives can be used in the query format string: * %d (integer) * %f (float) * %s (string) * %% (literal percentage sign - no argument needed) * * All of %d, %f, and %s are to be left unquoted in the query string and they need an argument passed for them. * Literals (%) as parts of the query must be properly written as %%. * * This function only supports a small subset of the sprintf syntax; it only supports %d (integer), %f (float), and %s (string). * Does not support sign, padding, alignment, width or precision specifiers. * Does not support argument numbering/swapping. * * May be called like {@link http://php.net/sprintf sprintf()} or like {@link http://php.net/vsprintf vsprintf()}. * * Both %d and %s should be left unquoted in the query string. * * @param string $query Query statement with sprintf()-like placeholders * @param array|mixed $args The array of variables to substitute into the query's placeholders if being called like * {@link http://php.net/vsprintf vsprintf()}, or the first variable to substitute into the query's placeholders if * being called like {@link http://php.net/sprintf sprintf()}. * @param mixed $args,... further variables to substitute into the query's placeholders if being called like * {@link http://php.net/sprintf sprintf()}. * * @return null|false|string Sanitized query string, null if there is no query, false if there is an error and string * if there was something to prepare */ public function prepare( $query, $args ) { if ( null === $query ) { return null; } $args = func_get_args(); array_shift( $args ); // If args were passed as an array (as in vsprintf), move them up if ( isset( $args[0] ) && is_array( $args[0] ) ) { $args = $args[0]; } $query = str_replace( "'%s'", '%s', $query ); // in case someone mistakenly already singlequoted it $query = str_replace( '"%s"', '%s', $query ); // doublequote unquoting $query = preg_replace( '|(?_dbi, 'escape_by_ref' ) ); return @vsprintf( $query, $args ); } /** * Retrieve an entire SQL result set from the database (i.e., many rows) * * Executes a SQL query and returns the entire SQL result. * * @param string $query SQL query. * @param string $output Optional. Any of ARRAY_A | ARRAY_N | OBJECT | OBJECT_K constants. With one of the first three, return an array of rows indexed from 0 by SQL result row number. * Each row is an associative array (column => value, ...), a numerically indexed array (0 => value, ...), or an object. ( ->column = value ), respectively. * With OBJECT_K, return an associative array of row objects keyed by the value of each row's first column's value. Duplicate keys are discarded. * * @return mixed Database query results */ public function get_results( $query, $output = OBJECT ){ $this->_query_profile( $query ); $result = $this->_dbi->get_results( $query, $output ); $this->_query_profile( count( $result ) ); return $result; } /** * Retrieve one variable from the database. * * Executes a SQL query and returns the value from the SQL result. * If the SQL result contains more than one column and/or more than one row, this function returns the value in the column and row specified. * If $query is null, this function returns the value in the specified column and row from the previous SQL result. * * @param string|null $query SQL query. Defaults to null, use the result from the previous query. * @param int $col Column of value to return. Indexed from 0. * @param int $row Row of value to return. Indexed from 0. * * @return string|null Database query result (as string), or null on failure */ public function get_var( $query = null, $col = 0, $row = 0 ) { $this->_query_profile( $query ); $result = $this->_dbi->get_var( $query, $col, $row ); $this->_query_profile( null !== $result ); return $result; } /** * Retrieve one row from the database. * * Executes a SQL query and returns the row from the SQL result * * @param string|null $query SQL query. * @param string $output Optional. one of ARRAY_A | ARRAY_N | OBJECT constants. Return an associative array (column => value, ...), * a numerically indexed array (0 => value, ...) or an object ( ->column = value ), respectively. * @param int $row Optional. Row to return. Indexed from 0. * * @return mixed Database query result in format specified by $output or null on failure */ public function get_row( $query = null, $output = OBJECT, $row = 0 ) { $this->_query_profile( $query ); $result = $this->_dbi->get_row( $query, $output, $row ); $this->_query_profile( null !== $result ); return $result; } /** * Insert a row into a table. * * @param string $table table name * @param array $data Data to insert (in column => value pairs). Both $data columns and $data values should be "raw" (neither should be SQL escaped). * @param array|string $format Optional. An array of formats to be mapped to each of the value in $data. If string, that format will be used for all of the values in $data. * A format is one of '%d', '%f', '%s' (integer, float, string). If omitted, all values in $data will be treated as strings unless otherwise specified in wpdb::$field_types. * * @return int|false The number of rows inserted, or false on error. */ public function insert( $table, $data, $format = null ) { $this->_query_profile( 'INSERT INTO ' . $table . '; data: ' . json_encode( $data ) ); $result = $this->_dbi->insert( $this->get_table_name( $table ), $data, $format ); $this->_query_profile( $result ); return $result; } /** * Perform removal from table. * * @param string $table Table to remove from. * @param array $where Where conditions * @param array $format Format entities for where. * * @return int|false Number of rows deleted or false. */ public function delete( $table, $where, $format = null ) { $this->_query_profile( 'DELETE FROM ' . $table . '; conditions: ' . json_encode( $where ) ); $result = $this->_dbi->delete( $this->get_table_name( $table ), $where, $format ); $this->_query_profile( $result ); return $result; } /** * Update a row in the table * * @param string $table table name * @param array $data Data to update (in column => value pairs). Both $data columns and $data values should be "raw" (neither should be SQL escaped). * @param array $where A named array of WHERE clauses (in column => value pairs). Multiple clauses will be joined with ANDs. Both $where columns and $where values should be "raw". * @param array|string $format Optional. An array of formats to be mapped to each of the values in $data. If string, that format will be used for all of the values in $data. * A format is one of '%d', '%f', '%s' (integer, float, string). If omitted, all values in $data will be treated as strings unless otherwise specified in wpdb::$field_types. * @param array|string $where_format Optional. An array of formats to be mapped to each of the values in $where. If string, that format will be used for all of the items in $where. A format is one of '%d', '%f', '%s' (integer, float, string). If omitted, all values in $where will be treated as strings. * * @return int|false The number of rows updated, or false on error. */ public function update( $table, $data, $where, $format = null, $where_format = null ) { $this->_query_profile( 'UPDATE ' . $table . ': ' . implode( '//', $data ) ); $result = $this->_dbi->update( $table, $data, $where, $format, $where_format ); $this->_query_profile( $result ); return $result; } /** * Retrieve all results from given table. * * @param string $table Name of table. * @param array $columns List of columns to retrieve. * @param string $output See {@see self::get_results()} $output for more. * * @return array Collection. */ public function select( $table, array $columns, $output = OBJECT ) { $sql_query = 'SELECT `' . implode( '`, `', $columns ) . '` FROM `' . $this->get_table_name( $table ) . '`'; return $this->get_results( $sql_query, $output ); } /** * The database version number. * * @return false|string false on failure, version number on success */ public function db_version() { return $this->_dbi->db_version(); } /** * Return the id of last `insert` operation. * * @return int Returns integer optionally zero when no insert was performed. */ public function get_insert_id() { return $this->_dbi->insert_id; } /** * Return the full name for the table. * * @param string $table Table name. * * @return string Full table name for the table requested. */ public function get_table_name( $table = '' ) { static $prefix_len = null; if ( ! isset( $this->_dbi->{$table} ) ) { if ( null === $prefix_len ) { $prefix_len = strlen( $this->_dbi->prefix ); } if ( 0 === strncmp( $this->_dbi->prefix, $table, $prefix_len ) ) { return $table; } return $this->_dbi->prefix . $table; } return $this->_dbi->{$table}; } /** * Return escaped value. * * @param string $input Value to be escaped. * * @return string Escaped value. */ public function escape( $input ) { $this->_dbi->escape_by_ref( $input ); return $input; } /** * In debug mode prints DB queries table. * * @return void */ public function shutdown() { if ( ! $this->_log_enabled ) { return false; } echo '
'; $i = 0; $time = 0; foreach ( $this->_queries as $query ) { $time += $query['d']; echo ''; } echo '
N. Query Duration, ms Row Count
', ++$i, ' ', $query['q'], ' ', round( $query['d'] * 1000, 2 ), ' ', (int)$query['r'], '
Total time, ms: ', round( $time * 1000, 2 ), '
'; return true; } /** * Method aiding query profiling. * * How to use: * - on method resulting in query start call _query_profiler( 'SQL query' ) * - on it's end call _query_profiler( (int)number_of_rows|(bool)false ) * * @param mixed $query_or_result Query on first call, result on second. * * @return void */ protected function _query_profile( $query_or_result ) { static $last = null; if ( null === $last ) { $last = array( 'd' => microtime( true ), 'q' => $query_or_result, ); } else { if ( count( $this->_queries ) > 200 ) { array_shift( $this->_queries ); } $this->_queries[] = array( 'd' => microtime( true ) - $last['d'], 'q' => $last['q'], 'r' => $query_or_result, ); $last = null; } } }