194 lines
5.2 KiB
PHP
194 lines
5.2 KiB
PHP
<?php declare(strict_types = 1);
|
|
/**
|
|
* Duplicate database query output for HTML pages.
|
|
*
|
|
* @package query-monitor
|
|
*/
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit;
|
|
}
|
|
|
|
class QM_Output_Html_DB_Dupes extends QM_Output_Html {
|
|
|
|
/**
|
|
* Collector instance.
|
|
*
|
|
* @var QM_Collector_DB_Dupes Collector.
|
|
*/
|
|
protected $collector;
|
|
|
|
public function __construct( QM_Collector $collector ) {
|
|
parent::__construct( $collector );
|
|
add_filter( 'qm/output/menus', array( $this, 'admin_menu' ), 45 );
|
|
add_filter( 'qm/output/panel_menus', array( $this, 'panel_menu' ), 25 );
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
public function name() {
|
|
return __( 'Duplicate Queries', 'query-monitor' );
|
|
}
|
|
|
|
/**
|
|
* @return void
|
|
*/
|
|
public function output() {
|
|
/** @var QM_Data_DB_Dupes $data */
|
|
$data = $this->collector->get_data();
|
|
|
|
if ( empty( $data->dupes ) ) {
|
|
return;
|
|
}
|
|
|
|
$this->before_tabular_output();
|
|
|
|
echo '<thead>';
|
|
|
|
echo '<tr>';
|
|
echo '<th scope="col">' . esc_html__( 'Query', 'query-monitor' ) . '</th>';
|
|
echo '<th scope="col" class="qm-num">' . esc_html__( 'Count', 'query-monitor' ) . '</th>';
|
|
echo '<th scope="col" class="qm-num">' . esc_html__( 'Time', 'query-monitor' ) . '</th>';
|
|
echo '<th scope="col">' . esc_html__( 'Callers', 'query-monitor' ) . '</th>';
|
|
if ( ! empty( $data->dupe_components ) ) {
|
|
echo '<th scope="col">' . esc_html__( 'Components', 'query-monitor' ) . '</th>';
|
|
}
|
|
echo '<th scope="col">' . esc_html__( 'Potential Troublemakers', 'query-monitor' ) . '</th>';
|
|
echo '</tr>';
|
|
|
|
echo '</thead>';
|
|
|
|
echo '<tbody>';
|
|
|
|
/* translators: %s: Number of calls to a PHP function */
|
|
$call_text = _n_noop( '%s call', '%s calls', 'query-monitor' );
|
|
|
|
foreach ( $data->dupes as $sql => $queries ) {
|
|
|
|
// This should probably happen in the collector's processor
|
|
$type = QM_Util::get_query_type( $sql );
|
|
$sql_out = self::format_sql( $sql );
|
|
$time = $data->dupe_times[ $sql ];
|
|
|
|
if ( 'SELECT' !== $type ) {
|
|
$sql_out = "<span class='qm-nonselectsql'>{$sql_out}</span>";
|
|
}
|
|
|
|
echo '<tr>';
|
|
echo '<td class="qm-row-sql qm-ltr qm-wrap">';
|
|
echo $sql_out; // WPCS: XSS ok;
|
|
echo '</td>';
|
|
echo '<td class="qm-num">';
|
|
echo esc_html( number_format_i18n( count( $queries ), 0 ) );
|
|
echo '</td>';
|
|
echo '<td class="qm-num">';
|
|
echo esc_html( number_format_i18n( $time, 4 ) );
|
|
echo '</td>';
|
|
echo '<td class="qm-row-caller qm-nowrap qm-ltr">';
|
|
foreach ( $data->dupe_callers[ $sql ] as $caller => $calls ) {
|
|
echo self::build_filter_trigger( 'db_queries', 'caller', $caller, '<code>' . esc_html( $caller ) . '</code>' ); // WPCS: XSS ok;
|
|
printf(
|
|
'<br><span class="qm-info qm-supplemental">%s</span><br>',
|
|
esc_html( sprintf(
|
|
translate_nooped_plural( $call_text, $calls, 'query-monitor' ),
|
|
number_format_i18n( $calls )
|
|
) )
|
|
);
|
|
}
|
|
echo '</td>';
|
|
if ( isset( $data->dupe_components[ $sql ] ) ) {
|
|
echo '<td class="qm-row-component qm-nowrap">';
|
|
foreach ( $data->dupe_components[ $sql ] as $component => $calls ) {
|
|
printf(
|
|
'%s<br><span class="qm-info qm-supplemental">%s</span><br>',
|
|
esc_html( $component ),
|
|
esc_html( sprintf(
|
|
translate_nooped_plural( $call_text, $calls, 'query-monitor' ),
|
|
number_format_i18n( $calls )
|
|
) )
|
|
);
|
|
}
|
|
echo '</td>';
|
|
}
|
|
echo '<td class="qm-row-caller qm-nowrap qm-ltr">';
|
|
foreach ( $data->dupe_sources[ $sql ] as $source => $calls ) {
|
|
printf(
|
|
'<code>%s</code><br><span class="qm-info qm-supplemental">%s</span><br>',
|
|
esc_html( $source ),
|
|
esc_html( sprintf(
|
|
translate_nooped_plural( $call_text, $calls, 'query-monitor' ),
|
|
number_format_i18n( $calls )
|
|
) )
|
|
);
|
|
}
|
|
echo '</td>';
|
|
echo '</tr>';
|
|
}
|
|
echo '</tbody>';
|
|
|
|
$this->after_tabular_output();
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed[]> $menu
|
|
* @return array<string, mixed[]>
|
|
*/
|
|
public function admin_menu( array $menu ) {
|
|
/** @var QM_Collector_DB_Dupes|null $dbq */
|
|
$dbq = QM_Collectors::get( 'db_dupes' );
|
|
|
|
if ( $dbq ) {
|
|
/** @var QM_Data_DB_Queries $dbq_data */
|
|
$dbq_data = $dbq->get_data();
|
|
if ( ! empty( $dbq_data->dupes ) ) {
|
|
$count = 0;
|
|
|
|
foreach ( $dbq_data->dupes as $dupe ) {
|
|
$count += count( $dupe );
|
|
}
|
|
|
|
$menu[ $this->collector->id() ] = $this->menu( array(
|
|
'title' => esc_html( sprintf(
|
|
/* translators: %s: Number of duplicate database queries */
|
|
__( 'Duplicate Queries (%s)', 'query-monitor' ),
|
|
number_format_i18n( $count )
|
|
) ),
|
|
) );
|
|
}
|
|
}
|
|
return $menu;
|
|
|
|
}
|
|
|
|
/**
|
|
* @param array<string, mixed[]> $menu
|
|
* @return array<string, mixed[]>
|
|
*/
|
|
public function panel_menu( array $menu ) {
|
|
$id = $this->collector->id();
|
|
if ( isset( $menu[ $id ] ) ) {
|
|
$menu['qm-db_queries']['children'][] = $menu[ $id ];
|
|
unset( $menu[ $id ] );
|
|
}
|
|
|
|
return $menu;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* @param array<string, QM_Output> $output
|
|
* @param QM_Collectors $collectors
|
|
* @return array<string, QM_Output>
|
|
*/
|
|
function register_qm_output_html_db_dupes( array $output, QM_Collectors $collectors ) {
|
|
$collector = QM_Collectors::get( 'db_dupes' );
|
|
if ( $collector ) {
|
|
$output['db_dupes'] = new QM_Output_Html_DB_Dupes( $collector );
|
|
}
|
|
return $output;
|
|
}
|
|
|
|
add_filter( 'qm/outputter/html', 'register_qm_output_html_db_dupes', 45, 2 );
|