287 lines
6.5 KiB
PHP
Raw Permalink Normal View History

2024-05-20 15:37:46 +03:00
<?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 );