287 lines
6.5 KiB
PHP
287 lines
6.5 KiB
PHP
<?php declare(strict_types = 1);
|
|
/**
|
|
* Database query collector.
|
|
*
|
|
* @package query-monitor
|
|
*/
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit;
|
|
}
|
|
|
|
if ( ! defined( 'SAVEQUERIES' ) ) {
|
|
define( 'SAVEQUERIES', true );
|
|
}
|
|
if ( ! defined( 'QM_DB_EXPENSIVE' ) ) {
|
|
define( 'QM_DB_EXPENSIVE', 0.05 );
|
|
}
|
|
|
|
if ( SAVEQUERIES && property_exists( $GLOBALS['wpdb'], 'save_queries' ) ) {
|
|
$GLOBALS['wpdb']->save_queries = true;
|
|
}
|
|
|
|
/**
|
|
* @extends QM_DataCollector<QM_Data_DB_Queries>
|
|
*/
|
|
class QM_Collector_DB_Queries extends QM_DataCollector {
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
public $id = 'db_queries';
|
|
|
|
/**
|
|
* @var wpdb
|
|
*/
|
|
public $wpdb;
|
|
|
|
public function get_storage(): QM_Data {
|
|
return new QM_Data_DB_Queries();
|
|
}
|
|
|
|
/**
|
|
* @return mixed[]|false
|
|
*/
|
|
public function get_errors() {
|
|
if ( ! empty( $this->data->errors ) ) {
|
|
return $this->data->errors;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @return mixed[]|false
|
|
*/
|
|
public function get_expensive() {
|
|
if ( ! empty( $this->data->expensive ) ) {
|
|
return $this->data->expensive;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed> $row
|
|
* @return bool
|
|
*/
|
|
public static function is_expensive( array $row ) {
|
|
return $row['ltime'] > QM_DB_EXPENSIVE;
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function process() {
|
|
$this->data->total_qs = 0;
|
|
$this->data->total_time = 0;
|
|
$this->data->errors = array();
|
|
$this->process_db_object();
|
|
}
|
|
|
|
/**
|
|
* @param string $caller
|
|
* @param float $ltime
|
|
* @param string $type
|
|
* @return void
|
|
*/
|
|
protected function log_caller( $caller, $ltime, $type ) {
|
|
|
|
if ( ! isset( $this->data->times[ $caller ] ) ) {
|
|
$this->data->times[ $caller ] = array(
|
|
'caller' => $caller,
|
|
'ltime' => 0,
|
|
'types' => array(),
|
|
);
|
|
}
|
|
|
|
$this->data->times[ $caller ]['ltime'] += $ltime;
|
|
|
|
if ( isset( $this->data->times[ $caller ]['types'][ $type ] ) ) {
|
|
$this->data->times[ $caller ]['types'][ $type ]++;
|
|
} else {
|
|
$this->data->times[ $caller ]['types'][ $type ] = 1;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function process_db_object() {
|
|
global $wp_the_query, $wpdb;
|
|
|
|
$this->wpdb = $wpdb;
|
|
|
|
// With SAVEQUERIES defined as false, `wpdb::queries` is empty but `wpdb::num_queries` is not.
|
|
if ( empty( $wpdb->queries ) ) {
|
|
$this->data->total_qs += $wpdb->num_queries;
|
|
return;
|
|
}
|
|
|
|
$rows = array();
|
|
$types = array();
|
|
$total_time = 0;
|
|
$has_result = false;
|
|
$has_trace = false;
|
|
$i = 0;
|
|
$request = trim( $wp_the_query->request ?: '' );
|
|
|
|
if ( method_exists( $wpdb, 'remove_placeholder_escape' ) ) {
|
|
$request = $wpdb->remove_placeholder_escape( $request );
|
|
}
|
|
|
|
/**
|
|
* @phpstan-var array{
|
|
* 0: string,
|
|
* 1: float,
|
|
* 2: string,
|
|
* trace?: QM_Backtrace,
|
|
* result?: int|bool|WP_Error,
|
|
* }|array{
|
|
* query: string,
|
|
* elapsed: float,
|
|
* debug: string,
|
|
* } $query
|
|
*/
|
|
foreach ( $wpdb->queries as $query ) {
|
|
$has_trace = false;
|
|
$has_result = false;
|
|
$callers = array();
|
|
|
|
if ( isset( $query['query'], $query['elapsed'], $query['debug'] ) ) {
|
|
// WordPress.com VIP.
|
|
$sql = $query['query'];
|
|
$ltime = $query['elapsed'];
|
|
$stack = $query['debug'];
|
|
} elseif ( isset( $query[0], $query[1], $query[2] ) ) {
|
|
// Standard WP.
|
|
$sql = $query[0];
|
|
$ltime = $query[1];
|
|
$stack = $query[2];
|
|
|
|
// Query Monitor db.php drop-in.
|
|
$has_trace = isset( $query['trace'] );
|
|
$has_result = isset( $query['result'] );
|
|
} else {
|
|
// ¯\_(ツ)_/¯
|
|
continue;
|
|
}
|
|
|
|
// @TODO: decide what I want to do with this:
|
|
// phpcs:ignore WordPress.Security.NonceVerification.Recommended
|
|
if ( false !== strpos( $stack, 'wp_admin_bar' ) && ! isset( $_REQUEST['qm_display_admin_bar'] ) ) {
|
|
continue;
|
|
}
|
|
|
|
$result = $query['result'] ?? null;
|
|
$total_time += $ltime;
|
|
|
|
if ( isset( $query['trace'] ) ) {
|
|
|
|
$trace = $query['trace'];
|
|
$component = $query['trace']->get_component();
|
|
$caller = $query['trace']->get_caller();
|
|
$caller_name = $caller['display'] ?? 'Unknown';
|
|
$caller = $caller['display'] ?? 'Unknown';
|
|
|
|
} else {
|
|
|
|
$trace = null;
|
|
$component = null;
|
|
$callers = array_reverse( explode( ',', $stack ) );
|
|
$callers = array_map( 'trim', $callers );
|
|
$callers = QM_Backtrace::get_filtered_stack( $callers );
|
|
$caller = reset( $callers );
|
|
$caller_name = $caller;
|
|
|
|
}
|
|
|
|
$sql = trim( $sql );
|
|
$type = QM_Util::get_query_type( $sql );
|
|
|
|
$this->log_type( $type );
|
|
$this->log_caller( $caller_name, $ltime, $type );
|
|
|
|
$this->maybe_log_dupe( $sql, $i );
|
|
|
|
if ( $component ) {
|
|
$this->log_component( $component, $ltime, $type );
|
|
}
|
|
|
|
if ( ! isset( $types[ $type ]['total'] ) ) {
|
|
$types[ $type ]['total'] = 1;
|
|
} else {
|
|
$types[ $type ]['total']++;
|
|
}
|
|
|
|
if ( ! isset( $types[ $type ]['callers'][ $caller ] ) ) {
|
|
$types[ $type ]['callers'][ $caller ] = 1;
|
|
} else {
|
|
$types[ $type ]['callers'][ $caller ]++;
|
|
}
|
|
|
|
$is_main_query = ( $request === $sql && ( false !== strpos( $stack, ' WP->main,' ) ) );
|
|
|
|
$row = compact( 'caller', 'caller_name', 'sql', 'ltime', 'result', 'type', 'component', 'trace', 'is_main_query' );
|
|
|
|
if ( ! isset( $trace ) ) {
|
|
$row['stack'] = $callers;
|
|
}
|
|
|
|
// @TODO these should store a reference ($i) instead of the whole row
|
|
if ( $result instanceof WP_Error ) {
|
|
$this->data->errors[] = $row;
|
|
}
|
|
|
|
// @TODO these should store a reference ($i) instead of the whole row
|
|
if ( self::is_expensive( $row ) ) {
|
|
$this->data->expensive[] = $row;
|
|
}
|
|
|
|
$rows[ $i ] = $row;
|
|
$i++;
|
|
|
|
}
|
|
|
|
$total_qs = count( $rows );
|
|
|
|
$this->data->total_qs += $total_qs;
|
|
$this->data->total_time += $total_time;
|
|
|
|
$has_main_query = wp_list_filter( $rows, array(
|
|
'is_main_query' => true,
|
|
) );
|
|
|
|
# @TODO put errors in here too:
|
|
# @TODO proper class instead of (object)
|
|
$this->data->wpdb = (object) compact( 'rows', 'types', 'has_result', 'has_trace', 'total_time', 'total_qs', 'has_main_query' );
|
|
|
|
}
|
|
|
|
/**
|
|
* @param string $sql
|
|
* @param int $i
|
|
* @return void
|
|
*/
|
|
protected function maybe_log_dupe( $sql, $i ) {
|
|
$sql = str_replace( array( "\r\n", "\r", "\n" ), ' ', $sql );
|
|
$sql = str_replace( array( "\t", '`' ), '', $sql );
|
|
$sql = preg_replace( '/ +/', ' ', $sql );
|
|
$sql = trim( $sql );
|
|
$sql = rtrim( $sql, ';' );
|
|
|
|
$this->data->dupes[ $sql ][] = $i;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param array<string, QM_Collector> $collectors
|
|
* @param QueryMonitor $qm
|
|
* @return array<string, QM_Collector>
|
|
*/
|
|
function register_qm_collector_db_queries( array $collectors, QueryMonitor $qm ) {
|
|
$collectors['db_queries'] = new QM_Collector_DB_Queries();
|
|
return $collectors;
|
|
}
|
|
|
|
add_filter( 'qm/collectors', 'register_qm_collector_db_queries', 10, 2 );
|