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 );
 |