one-click-accessibility/modules/remediation/components/remediation-runner.php
VasylD 4c1546784b
[APP-1747] add parent selector (#360)
* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* [APP-1747] add parent selector

* Add mixpanel events

* Add mixpanel events

* Add mixpanel events

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Update modules/remediation/assets/js/actions/styles.js

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* Update modules/scanner/assets/js/hooks/use-color-contrast-form.js

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>

* Exclude adminbar from css rule

* Exclude adminbar from css rule

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix quota

* fix linter

---------

Co-authored-by: gitstream-cm[bot] <111687743+gitstream-cm[bot]@users.noreply.github.com>
2025-08-05 15:27:40 +02:00

263 lines
7.8 KiB
PHP

<?php
namespace EA11y\Modules\Remediation\Components;
use DOMDocument;
use EA11y\Classes\Logger;
use EA11y\Modules\Remediation\Classes\Utils;
use EA11y\Modules\Remediation\Database\Page_Entry;
use EA11y\Modules\Remediation\Database\Page_Table;
use EA11y\Modules\Remediation\Database\Remediation_Entry;
use Throwable;
if ( ! defined( 'ABSPATH' ) ) {
exit; // Exit if accessed directly
}
/**
* Class Remediation_Runner
*/
class Remediation_Runner {
public Page_Entry $page;
public array $page_remediations = [];
public ?string $page_html = '';
public array $front_end_remediations = [];
public function get_remediation_classes() : array {
static $classes = null;
if ( null === $classes ) {
$classes = apply_filters( 'ea11y_remediations_classes', [
'ATTRIBUTE' => 'Attribute',
'ELEMENT' => 'Element',
'REPLACE' => 'Replace',
'STYLES' => 'Styles',
] );
}
return $classes;
}
/**
* Detect AJAX requests that go through template_redirect hook
*
* Only detects frontend AJAX requests that would interfere with template_redirect,
* not admin-ajax.php requests which bypass the main query cycle.
*
* @return bool True if a template_redirect-affecting AJAX request is detected
*/
private function is_template_redirect_ajax_request(): bool {
global $wp_query;
// Skip admin-ajax.php requests - they don't go through template_redirect
$request_uri = $_SERVER['REQUEST_URI'] ?? '';
if ( strpos( $request_uri, '/wp-admin/admin-ajax.php' ) !== false ) {
return false;
}
// WooCommerce frontend AJAX (wc-ajax parameter)
// These requests go through the main query cycle and template_redirect
if ( ! empty( $_REQUEST['wc-ajax'] ) ) {
return true;
}
// Check if WP_Query has wc-ajax set (WooCommerce frontend AJAX)
if ( is_object( $wp_query ) && $wp_query->get( 'wc-ajax' ) ) {
return true;
}
// WooCommerce AJAX constant for frontend requests
if ( defined( 'WC_DOING_AJAX' ) && constant( 'WC_DOING_AJAX' ) ) {
return true;
}
// REST API requests that go through template_redirect
if ( defined( 'REST_REQUEST' ) && constant( 'REST_REQUEST' ) ) {
return true;
}
// Check URL patterns for REST API (these go through template_redirect)
if ( strpos( $request_uri, '/wp-json/' ) !== false ||
strpos( $request_uri, '?rest_route=' ) !== false ) {
return true;
}
// WooCommerce Store API (frontend cart/checkout AJAX)
if ( strpos( $request_uri, '/wc/store/' ) !== false ) {
return true;
}
// Elementor frontend AJAX/preview that affects template loading
if ( ! empty( $_REQUEST['elementor-preview'] ) ||
! empty( $_GET['elementor-preview'] ) ) {
return true;
}
// Check for AJAX header on frontend requests (not admin)
// Only if it's not an admin request and has AJAX header
if ( ! is_admin() &&
! empty( $_SERVER['HTTP_X_REQUESTED_WITH'] ) &&
strtolower( $_SERVER['HTTP_X_REQUESTED_WITH'] ) === 'xmlhttprequest' ) {
// Additional check: make sure it's not a heartbeat or other admin request
$action = $_REQUEST['action'] ?? '';
if ( $action !== 'heartbeat' && strpos( $action, 'wp_ajax_' ) !== 0 ) {
return true;
}
}
// Custom AJAX parameters that indicate frontend AJAX
$frontend_ajax_params = [
'ajax_request',
'is_ajax',
'doing_ajax'
];
foreach ( $frontend_ajax_params as $param ) {
if ( ! empty( $_REQUEST[ $param ] ) ) {
return true;
}
}
// Query string patterns for frontend AJAX
$query_string = $_SERVER['QUERY_STRING'] ?? '';
$frontend_ajax_patterns = [
'ajax=true',
'is_ajax=1',
'ajax_request=1'
];
foreach ( $frontend_ajax_patterns as $pattern ) {
if ( strpos( $query_string, $pattern ) !== false ) {
return true;
}
}
return false;
}
private function should_run_remediation(): bool {
// Skip remediation during template_redirect AJAX requests
if ( $this->is_template_redirect_ajax_request() ) {
return false;
}
try {
$current_url = Utils::get_current_page_url();
$this->page = new Page_Entry([
'by' => 'url',
'value' => $current_url,
]);
$this->page_html = $this->page->get_page_html();
$this->page_remediations = Remediation_Entry::get_page_remediations( $current_url );
$status = $this->page->__get( Page_Table::STATUS );
if ( empty( $this->page_remediations ) || Page_Table::STATUSES['ACTIVE'] !== $status ) {
return false;
}
return true;
} catch ( Throwable $t ) {
Logger::error( $t->getMessage() );
return false;
}
}
public function start() {
ob_start( [ $this, 'run_remediations' ] );
}
private function get_remediation_class_name( $type ) {
$classes = $this->get_remediation_classes();
$type = strtoupper( $type );
if ( ! isset( $classes[ $type ] ) ) {
return false;
}
$class = 'EA11y\\Modules\\Remediation\\Actions\\' . $classes[ $type ];
return $class;
}
public function run_remediations( $buffer ): string {
if ( ! is_user_logged_in() && $this->page_html && $this->page->is_valid_hash() ) {
return $this->page_html;
}
$dom = $this->generate_remediation_dom( $buffer );
if ( ! is_user_logged_in() ) {
$this->page->update_html( $dom );
}
return $dom;
}
private function generate_remediation_dom( $buffer ): string {
$dom = new DOMDocument('1.0', 'UTF-8');
$dom->loadHTML( $buffer, LIBXML_HTML_NOIMPLIED | LIBXML_HTML_NODEFDTD | LIBXML_NOERROR );
//Remove admin-bar for correct replace
$admin_bar = $dom->getElementById( 'wpadminbar' );
if ( $admin_bar ) {
$parent = $admin_bar->parentNode;
$next_sibling = $admin_bar->nextSibling;
$removed_admin_bar = $parent->removeChild( $admin_bar );
}
$classes = $this->get_remediation_classes();
foreach ( $this->page_remediations as $item ) {
$remediation = json_decode( $item->content, true );
if ( ! isset( $classes[ strtoupper( $remediation['type'] ) ] ) ) {
continue;
}
$remediation_class_name = $this->get_remediation_class_name( $remediation['type'] );
$remediation_class = new $remediation_class_name( $dom, $remediation );
if ( $remediation_class->use_frontend ) {
$this->add_frontend_remediation( $remediation );
continue;
}
if ( $remediation_class->dom ) {
$remediation_class->dom = $dom;
}
}
if ( isset( $removed_admin_bar, $parent ) ) {
isset( $next_sibling ) && $next_sibling
? $parent->insertBefore( $removed_admin_bar, $next_sibling )
: $parent->appendChild( $removed_admin_bar );
}
// Add frontend remediations for dynamic content
// We don't use Enqueue_Script because we need to add the script to
// The <head> section and WP has already enqueued head scripts by now.
if ( ! empty( $this->front_end_remediations ) ) {
// Create a new <script> element
$script_data = $dom->createElement( 'script' );
$script_data->setAttribute( 'type', 'text/javascript' );
$script_data->textContent = 'window.AllyRemediations = ' . wp_json_encode( [
'remediations' => $this->front_end_remediations,
], JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_QUOT | JSON_HEX_AMP ) . ';';
// Create a new <script> element for the module.js
$module_script = $dom->createElement( 'script' );
$module_script->setAttribute( 'type', 'text/javascript' ); // Optional: specify the type
$module_script->setAttribute( 'src', \EA11Y_ASSETS_URL . 'build/remediation-module.js' ); // Set the script source
// Append the <script> elements to the <head> section
$head = $dom->getElementsByTagName( 'head' )->item( 0 );
if ( $head ) {
$head->appendChild( $script_data );
$head->appendChild( $module_script );
}
}
return $dom->saveHTML();
}
private function add_frontend_remediation( $remediation ) {
$this->front_end_remediations[] = $remediation;
}
public function __construct() {
if ( is_admin() ) {
return;
}
if ( $this->should_run_remediation() ) {
add_action( 'template_redirect', [ $this, 'start' ], -9999 );
}
}
}