Leitgedanken/msd2/wordpress/wp-content/plugins/amp/includes/validation/class-amp-validation-manager.php
2023-01-23 11:03:31 +01:00

2004 lines
67 KiB
PHP

<?php
/**
* Class AMP_Validation_Manager
*
* @package AMP
*/
/**
* Class AMP_Validation_Manager
*
* @since 0.7
*/
class AMP_Validation_Manager {
/**
* Query var that triggers validation.
*
* @var string
*/
const VALIDATE_QUERY_VAR = 'amp_validate';
/**
* Query var for containing the number of validation errors on the frontend after redirection when invalid.
*
* @var string
*/
const VALIDATION_ERRORS_QUERY_VAR = 'amp_validation_errors';
/**
* Query var for passing status preview/update for validation error.
*
* @var string
*/
const VALIDATION_ERROR_TERM_STATUS_QUERY_VAR = 'amp_validation_error_term_status';
/**
* Query var for cache-busting.
*
* @var string
*/
const CACHE_BUST_QUERY_VAR = 'amp_cache_bust';
/**
* Transient key to store validation errors when activating a plugin.
*
* @var string
*/
const PLUGIN_ACTIVATION_VALIDATION_ERRORS_TRANSIENT_KEY = 'amp_plugin_activation_validation_errors';
/**
* The name of the REST API field with the AMP validation results.
*
* @var string
*/
const VALIDITY_REST_FIELD_NAME = 'amp_validity';
/**
* The errors encountered when validating.
*
* @var array[][] {
* @type array $error Error code.
* @type bool $sanitized Whether sanitized.
* @type string $slug Hash of the error.
* }
*/
public static $validation_results = array();
/**
* Sources that enqueue each script.
*
* @var array
*/
public static $enqueued_script_sources = array();
/**
* Sources that enqueue each style.
*
* @var array
*/
public static $enqueued_style_sources = array();
/**
* Post IDs for posts that have been updated which need to be re-validated.
*
* Keys are post IDs and values are whether the post has been re-validated.
*
* @var bool[]
*/
public static $posts_pending_frontend_validation = array();
/**
* Current sources gathered for a given hook currently being run.
*
* @see AMP_Validation_Manager::wrap_hook_callbacks()
* @see AMP_Validation_Manager::decorate_filter_source()
* @var array[]
*/
protected static $current_hook_source_stack = array();
/**
* Index for where block appears in a post's content.
*
* @var int
*/
protected static $block_content_index = 0;
/**
* Hook source stack.
*
* This has to be public for the sake of PHP 5.3.
*
* @since 0.7
* @var array[]
*/
public static $hook_source_stack = array();
/**
* Whether validation error sources should be located.
*
* @var bool
*/
public static $should_locate_sources = false;
/**
* Overrides for validation errors.
*
* @var array
*/
public static $validation_error_status_overrides = array();
/**
* Whether the admin bar item was added for AMP.
*
* @var bool
*/
protected static $amp_admin_bar_item_added = false;
/**
* Add the actions.
*
* @param array $args {
* Args.
*
* @type bool $should_locate_sources Whether to locate sources.
* }
* @return void
*/
public static function init( $args = array() ) {
$args = array_merge(
array(
'should_locate_sources' => self::should_validate_response(),
),
$args
);
self::$should_locate_sources = $args['should_locate_sources'];
AMP_Validated_URL_Post_Type::register();
AMP_Validation_Error_Taxonomy::register();
// Short-circuit if AMP is not supported as only the post types should be available.
if ( ! current_theme_supports( AMP_Theme_Support::SLUG ) ) {
return;
}
add_action( 'save_post', array( __CLASS__, 'handle_save_post_prompting_validation' ) );
add_action( 'enqueue_block_editor_assets', array( __CLASS__, 'enqueue_block_validation' ) );
add_action( 'edit_form_top', array( __CLASS__, 'print_edit_form_validation_status' ), 10, 2 );
add_action( 'all_admin_notices', array( __CLASS__, 'print_plugin_notice' ) );
add_action( 'rest_api_init', array( __CLASS__, 'add_rest_api_fields' ) );
// Actions and filters involved in validation.
add_action(
'activate_plugin',
function() {
if ( ! has_action( 'shutdown', array( __CLASS__, 'validate_after_plugin_activation' ) ) ) {
add_action( 'shutdown', array( __CLASS__, 'validate_after_plugin_activation' ) ); // Shutdown so all plugins will have been activated.
}
}
);
// Prevent query vars from persisting after redirect.
add_filter(
'removable_query_args',
function( $query_vars ) {
$query_vars[] = AMP_Validation_Manager::VALIDATION_ERRORS_QUERY_VAR;
return $query_vars;
}
);
add_action( 'admin_bar_menu', array( __CLASS__, 'add_admin_bar_menu_items' ), 100 );
// Add filter to auto-accept tree shaking validation error.
if ( AMP_Options_Manager::get_option( 'accept_tree_shaking' ) || AMP_Options_Manager::get_option( 'auto_accept_sanitization' ) ) {
add_filter( 'amp_validation_error_sanitized', array( __CLASS__, 'filter_tree_shaking_validation_error_as_accepted' ), 10, 2 );
}
if ( self::$should_locate_sources ) {
self::add_validation_error_sourcing();
}
}
/**
* Determine whether AMP theme support is forced via the amp_validate query param.
*
* @since 1.0
*
* @return bool Whether theme support forced.
*/
public static function is_theme_support_forced() {
return (
isset( $_GET[ self::VALIDATE_QUERY_VAR ] ) // phpcs:ignore WordPress.Security.NonceVerification.Recommended
&&
( self::has_cap() || self::get_amp_validate_nonce() === $_GET[ self::VALIDATE_QUERY_VAR ] ) // phpcs:ignore WordPress.Security.NonceVerification.Recommended
);
}
/**
* Return whether sanitization is forcibly accepted, whether because in native mode or via user option.
*
* @return bool Whether sanitization is forcibly accepted.
*/
public static function is_sanitization_auto_accepted() {
return amp_is_canonical() || AMP_Options_Manager::get_option( 'auto_accept_sanitization' );
}
/**
* Filter a tree-shaking validation error as accepted for sanitization.
*
* @param bool $sanitized Sanitized.
* @param array $error Error.
* @return bool Sanitized.
*/
public static function filter_tree_shaking_validation_error_as_accepted( $sanitized, $error ) {
if ( AMP_Style_Sanitizer::TREE_SHAKING_ERROR_CODE === $error['code'] ) {
$sanitized = true;
}
return $sanitized;
}
/**
* Add menu items to admin bar for AMP.
*
* When on a non-AMP response (transitional mode), then the admin bar item should include:
* - Icon: LINK SYMBOL when AMP not known to be invalid and sanitization is not forced, or CROSS MARK when AMP is known to be valid.
* - Parent admin item and first submenu item: link to AMP version.
* - Second submenu item: link to validate the URL.
*
* When on transitional AMP response:
* - Icon: CHECK MARK if no unaccepted validation errors on page, or WARNING SIGN if there are unaccepted validation errors which are being forcibly sanitized.
* Otherwise, if there are unsanitized validation errors then a redirect to the non-AMP version will be done.
* - Parent admin item and first submenu item: link to non-AMP version.
* - Second submenu item: link to validate the URL.
*
* When on native AMP response:
* - Icon: CHECK MARK if no unaccepted validation errors on page, or WARNING SIGN if there are unaccepted validation errors.
* - Parent admin and first submenu item: link to validate the URL.
*
* @see AMP_Validation_Manager::finalize_validation() Where the emoji is updated.
*
* @param WP_Admin_Bar $wp_admin_bar Admin bar.
*/
public static function add_admin_bar_menu_items( $wp_admin_bar ) {
if ( is_admin() || ! self::has_cap() ) {
self::$amp_admin_bar_item_added = false;
return;
}
$availability = AMP_Theme_Support::get_template_availability();
if ( ! $availability['supported'] ) {
self::$amp_admin_bar_item_added = false;
return;
}
$amp_validated_url_post = null;
$current_url = amp_get_current_url();
$non_amp_url = amp_remove_endpoint( $current_url );
$amp_url = remove_query_arg(
array_merge(
wp_removable_query_args(),
array(
self::VALIDATE_QUERY_VAR,
'amp_preserve_source_comments',
)
),
$current_url
);
if ( ! amp_is_canonical() ) {
$amp_url = add_query_arg( amp_get_slug(), '', $amp_url );
}
$error_count = -1;
/*
* If not an AMP response, then obtain the count of validation errors from either the query param supplied after redirecting from AMP
* to non-AMP due to validation errors (see AMP_Theme_Support::prepare_response()), or if there is an amp_validated_url post that already
* is populated with the last-known validation errors. Otherwise, if it *is* an AMP response then the error count is obtained after
* when the response is being prepared by AMP_Validation_Manager::finalize_validation().
*/
if ( ! is_amp_endpoint() ) {
if ( isset( $_GET[ self::VALIDATION_ERRORS_QUERY_VAR ] ) && is_numeric( $_GET[ self::VALIDATION_ERRORS_QUERY_VAR ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
$error_count = (int) $_GET[ self::VALIDATION_ERRORS_QUERY_VAR ]; // phpcs:ignore WordPress.Security.NonceVerification.Recommended
}
if ( $error_count < 0 ) {
$amp_validated_url_post = AMP_Validated_URL_Post_Type::get_invalid_url_post( $amp_url );
if ( $amp_validated_url_post ) {
$error_count = 0;
foreach ( AMP_Validated_URL_Post_Type::get_invalid_url_validation_errors( $amp_validated_url_post ) as $error ) {
if ( AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_REJECTED_STATUS === $error['term_status'] || AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_NEW_REJECTED_STATUS === $error['term_status'] ) {
$error_count++;
}
}
}
}
}
$user_can_revalidate = $amp_validated_url_post ? current_user_can( 'edit_post', $amp_validated_url_post->ID ) : current_user_can( 'manage_options' );
if ( ! $user_can_revalidate ) {
return;
}
// @todo The amp_validated_url post should probably only be accessible to users who can manage_options, or limit access to a post if the user has the cap to edit the queried object?
$validate_url = AMP_Validated_URL_Post_Type::get_recheck_url( $amp_validated_url_post ? $amp_validated_url_post : $amp_url );
if ( is_amp_endpoint() ) {
$icon = '&#x2705;'; // WHITE HEAVY CHECK MARK. This will get overridden in AMP_Validation_Manager::finalize_validation() if there are unaccepted errors.
} elseif ( $error_count > 0 ) {
$icon = '&#x274C;'; // CROSS MARK.
} else {
$icon = '&#x1F517;'; // LINK SYMBOL.
}
$validate_item = array(
'parent' => 'amp',
'id' => 'amp-validity',
'href' => esc_url( $validate_url ),
);
if ( $error_count <= 0 ) {
$validate_item['item'] = esc_html__( 'Re-validate', 'amp' );
} else {
$validate_item['item'] = esc_html(
sprintf(
/* translators: %s is count of validation errors */
_n(
'Re-validate (%s validation error)',
'Re-validate (%s validation errors)',
$error_count,
'amp'
),
number_format_i18n( $error_count )
)
);
}
$parent = array(
'id' => 'amp',
'title' => sprintf(
'<span id="amp-admin-bar-item-status-icon">%s</span> %s',
$icon,
esc_html( is_amp_endpoint() ? __( 'AMP', 'amp' ) : __( 'View AMP', 'amp' ) )
),
);
$first_item_is_validate = (
amp_is_canonical()
||
( ! is_amp_endpoint() && $error_count > 0 )
);
if ( $first_item_is_validate ) {
$title = __( 'Validate AMP', 'amp' );
$parent['href'] = esc_url( $validate_url );
} elseif ( is_amp_endpoint() ) {
$title = __( 'AMP', 'amp' );
$parent['href'] = esc_url( $non_amp_url );
} else {
$title = __( 'View AMP', 'amp' );
$parent['href'] = esc_url( $amp_url );
}
$parent['title'] = sprintf(
'<span id="amp-admin-bar-item-status-icon">%s</span> %s',
$icon,
esc_html( $title )
);
$wp_admin_bar->add_menu( $parent );
// Add admin bar item to switch between AMP and non-AMP if parent node is also an AMP link.
if ( ! $first_item_is_validate ) {
$wp_admin_bar->add_menu(
array(
'parent' => 'amp',
'id' => 'amp-view',
'title' => esc_html( is_amp_endpoint() ? __( 'View non-AMP version', 'amp' ) : __( 'View AMP version', 'amp' ) ),
'href' => esc_url( is_amp_endpoint() ? $non_amp_url : $amp_url ),
)
);
}
// Validate admin bar item.
if ( $error_count <= 0 ) {
$title = esc_html__( 'Re-validate', 'amp' );
} else {
$title = esc_html(
sprintf(
/* translators: %s is count of validation errors */
_n(
'Re-validate (%s validation error)',
'Re-validate (%s validation errors)',
$error_count,
'amp'
),
number_format_i18n( $error_count )
)
);
}
$wp_admin_bar->add_menu(
array(
'parent' => 'amp',
'id' => 'amp-validity',
'title' => esc_html( $title ),
'href' => esc_url( $validate_url ),
)
);
// Scrub the query var from the URL.
if ( ! is_amp_endpoint() && isset( $_GET[ self::VALIDATION_ERRORS_QUERY_VAR ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
add_action(
'wp_before_admin_bar_render',
function() {
?>
<script>
(function( queryVar ) {
var urlParser = document.createElement( 'a' );
urlParser.href = location.href;
urlParser.search = urlParser.search.substr( 1 ).split( /&/ ).filter( function( part ) {
return 0 !== part.indexOf( queryVar + '=' );
} );
if ( urlParser.href !== location.href ) {
history.replaceState( {}, '', urlParser.href );
}
})( <?php echo wp_json_encode( AMP_Validation_Manager::VALIDATION_ERRORS_QUERY_VAR ); ?> );
</script>
<?php
}
);
}
self::$amp_admin_bar_item_added = true;
}
/**
* Add hooks for doing determining sources for validation errors during preprocessing/sanitizing.
*/
public static function add_validation_error_sourcing() {
// Capture overrides validation error status overrides from query var.
$can_override_validation_error_statuses = (
isset( $_REQUEST[ self::VALIDATE_QUERY_VAR ] ) // phpcs:ignore WordPress.Security.NonceVerification.Recommended
&&
self::get_amp_validate_nonce() === $_REQUEST[ self::VALIDATE_QUERY_VAR ] // phpcs:ignore WordPress.Security.NonceVerification.Recommended
&&
isset( $_REQUEST[ self::VALIDATION_ERROR_TERM_STATUS_QUERY_VAR ] ) // phpcs:ignore WordPress.Security.NonceVerification.Recommended
&&
is_array( $_REQUEST[ self::VALIDATION_ERROR_TERM_STATUS_QUERY_VAR ] ) // phpcs:ignore WordPress.Security.NonceVerification.Recommended
);
if ( $can_override_validation_error_statuses ) {
/*
* This can't just easily add an amp_validation_error_sanitized filter because the the filter_sanitizer_args() method
* currently needs to obtain the list of overrides to create a parsed_cache_variant.
*/
foreach ( $_REQUEST[ self::VALIDATION_ERROR_TERM_STATUS_QUERY_VAR ] as $slug => $status ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
$slug = sanitize_key( $slug );
$status = intval( $status );
self::$validation_error_status_overrides[ $slug ] = $status;
ksort( self::$validation_error_status_overrides );
}
}
add_action( 'wp', array( __CLASS__, 'wrap_widget_callbacks' ) );
add_action( 'all', array( __CLASS__, 'wrap_hook_callbacks' ) );
$wrapped_filters = array( 'the_content', 'the_excerpt' );
foreach ( $wrapped_filters as $wrapped_filter ) {
add_filter( $wrapped_filter, array( __CLASS__, 'decorate_filter_source' ), PHP_INT_MAX );
}
add_filter( 'do_shortcode_tag', array( __CLASS__, 'decorate_shortcode_source' ), PHP_INT_MAX, 2 );
add_filter( 'embed_oembed_html', array( __CLASS__, 'decorate_embed_source' ), PHP_INT_MAX, 3 );
$do_blocks_priority = has_filter( 'the_content', 'do_blocks' );
$is_gutenberg_active = (
false !== $do_blocks_priority
&&
class_exists( 'WP_Block_Type_Registry' )
);
if ( $is_gutenberg_active ) {
add_filter( 'the_content', array( __CLASS__, 'add_block_source_comments' ), $do_blocks_priority - 1 );
}
}
/**
* Handle save_post action to queue re-validation of the post on the frontend.
*
* This is intended to only apply to post edits made in the classic editor.
*
* @see AMP_Validation_Manager::get_amp_validity_rest_field() The method responsible for validation post changes via Gutenberg.
* @see AMP_Validation_Manager::validate_queued_posts_on_frontend()
*
* @param int $post_id Post ID.
*/
public static function handle_save_post_prompting_validation( $post_id ) {
global $pagenow;
$post = get_post( $post_id );
$is_classic_editor_post_save = (
isset( $_SERVER['REQUEST_METHOD'] )
&&
'POST' === $_SERVER['REQUEST_METHOD']
&&
'post.php' === $pagenow
&&
isset( $_POST['post_ID'] ) // phpcs:ignore WordPress.Security.NonceVerification.Missing
&&
(int) $_POST['post_ID'] === (int) $post_id // phpcs:ignore WordPress.Security.NonceVerification.Missing
);
$should_validate_post = (
$is_classic_editor_post_save
&&
is_post_type_viewable( $post->post_type )
&&
! wp_is_post_autosave( $post )
&&
! wp_is_post_revision( $post )
&&
'auto-draft' !== $post->post_status
&&
! isset( self::$posts_pending_frontend_validation[ $post_id ] )
);
if ( $should_validate_post ) {
self::$posts_pending_frontend_validation[ $post_id ] = true;
// The reason for shutdown is to ensure that all postmeta changes have been saved, including whether AMP is enabled.
if ( ! has_action( 'shutdown', array( __CLASS__, 'validate_queued_posts_on_frontend' ) ) ) {
add_action( 'shutdown', array( __CLASS__, 'validate_queued_posts_on_frontend' ) );
}
}
}
/**
* Validate the posts pending frontend validation.
*
* @see AMP_Validation_Manager::handle_save_post_prompting_validation()
*
* @return array Mapping of post ID to the result of validating or storing the validation result.
*/
public static function validate_queued_posts_on_frontend() {
$posts = array_filter(
array_map( 'get_post', array_keys( array_filter( self::$posts_pending_frontend_validation ) ) ),
function( $post ) {
return $post && post_supports_amp( $post ) && 'trash' !== $post->post_status;
}
);
$validation_posts = array();
/*
* It is unlikely that there will be more than one post in the array.
* For the bulk recheck action, see AMP_Validated_URL_Post_Type::handle_bulk_action().
*/
foreach ( $posts as $post ) {
$url = amp_get_permalink( $post->ID );
if ( ! $url ) {
$validation_posts[ $post->ID ] = new WP_Error( 'no_amp_permalink' );
continue;
}
// Prevent re-validating.
self::$posts_pending_frontend_validation[ $post->ID ] = false;
$validity = self::validate_url( $url );
if ( is_wp_error( $validity ) ) {
$validation_posts[ $post->ID ] = $validity;
} else {
$invalid_url_post_id = intval( get_post_meta( $post->ID, '_amp_validated_url_post_id', true ) );
$validation_posts[ $post->ID ] = AMP_Validated_URL_Post_Type::store_validation_errors(
wp_list_pluck( $validity['results'], 'error' ),
$validity['url'],
array_merge(
array(
'invalid_url_post' => $invalid_url_post_id,
),
wp_array_slice_assoc( $validity, array( 'queried_object' ) )
)
);
// Remember the amp_validated_url post so that when the slug changes the old amp_validated_url post can be updated.
if ( ! is_wp_error( $validation_posts[ $post->ID ] ) && $invalid_url_post_id !== $validation_posts[ $post->ID ] ) {
update_post_meta( $post->ID, '_amp_validated_url_post_id', $validation_posts[ $post->ID ] );
}
}
}
return $validation_posts;
}
/**
* Adds fields to the REST API responses, in order to display validation errors.
*
* @return void
*/
public static function add_rest_api_fields() {
if ( amp_is_canonical() ) {
$object_types = get_post_types_by_support( 'editor' );
} else {
$object_types = array_intersect(
get_post_types_by_support( 'amp' ),
get_post_types(
array(
'show_in_rest' => true,
)
)
);
}
register_rest_field(
$object_types,
self::VALIDITY_REST_FIELD_NAME,
array(
'get_callback' => array( __CLASS__, 'get_amp_validity_rest_field' ),
'schema' => array(
'description' => __( 'AMP validity status', 'amp' ),
'type' => 'object',
),
)
);
}
/**
* Adds a field to the REST API responses to display the validation status.
*
* First, get existing errors for the post.
* If there are none, validate the post and return any errors.
*
* @param array $post_data Data for the post.
* @param string $field_name The name of the field to add.
* @param WP_REST_Request $request The name of the field to add.
* @return array|null $validation_data Validation data if it's available, or null.
*/
public static function get_amp_validity_rest_field( $post_data, $field_name, $request ) {
unset( $field_name );
if ( ! current_user_can( 'edit_post', $post_data['id'] ) ) {
return null;
}
$post = get_post( $post_data['id'] );
$validation_status_post = null;
if ( in_array( $request->get_method(), array( 'PUT', 'POST' ), true ) ) {
if ( ! isset( self::$posts_pending_frontend_validation[ $post->ID ] ) ) {
self::$posts_pending_frontend_validation[ $post->ID ] = true;
}
$results = self::validate_queued_posts_on_frontend();
if ( isset( $results[ $post->ID ] ) && is_int( $results[ $post->ID ] ) ) {
$validation_status_post = get_post( $results[ $post->ID ] );
}
}
if ( empty( $validation_status_post ) ) {
$validation_status_post = AMP_Validated_URL_Post_Type::get_invalid_url_post( amp_get_permalink( $post->ID ) );
}
$field = array(
'results' => array(),
'review_link' => null,
);
if ( $validation_status_post ) {
$field['review_link'] = get_edit_post_link( $validation_status_post->ID, 'raw' );
foreach ( AMP_Validated_URL_Post_Type::get_invalid_url_validation_errors( $validation_status_post ) as $result ) {
$field['results'][] = array(
'sanitized' => AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_ACCEPTED_STATUS === $result['status'],
'error' => $result['data'],
'status' => $result['status'],
'term_status' => $result['term_status'],
'forced' => $result['forced'],
);
}
}
return $field;
}
/**
* Whether the user has the required capability.
*
* Checks for permissions before validating.
*
* @return boolean $has_cap Whether the current user has the capability.
*/
public static function has_cap() {
return current_user_can( 'edit_posts' );
}
/**
* Add validation error.
*
* @param array $error Error info, especially code.
* @param array $data Additional data, including the node.
*
* @return bool Whether the validation error should result in sanitization.
*/
public static function add_validation_error( array $error, array $data = array() ) {
$node = null;
$matches = null;
$sources = null;
if ( isset( $data['node'] ) && $data['node'] instanceof DOMNode ) {
$node = $data['node'];
}
if ( self::$should_locate_sources ) {
if ( ! empty( $error['sources'] ) ) {
$sources = $error['sources'];
} elseif ( $node ) {
$sources = self::locate_sources( $node );
}
}
unset( $error['sources'] );
if ( ! isset( $error['code'] ) ) {
$error['code'] = 'unknown';
}
/**
* Filters the validation error array.
*
* This allows plugins to add amend additional properties which can help with
* more accurately identifying a validation error beyond the name of the parent
* node and the element's attributes. The $sources are also omitted because
* these are only available during an explicit validation request and so they
* are not suitable for plugins to vary sanitization by. If looking to force a
* validation error to be ignored, use the 'amp_validation_error_sanitized'
* filter instead of attempting to return an empty value with this filter (as
* that is not supported).
*
* @since 1.0
*
* @param array $error Validation error to be printed.
* @param array $context {
* Context data for validation error sanitization.
*
* @type DOMNode $node Node for which the validation error is being reported. May be null.
* }
*/
$error = apply_filters( 'amp_validation_error', $error, compact( 'node' ) );
$sanitization = AMP_Validation_Error_Taxonomy::get_validation_error_sanitization( $error );
$sanitized = (
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_NEW_ACCEPTED_STATUS === $sanitization['status']
||
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_ACCEPTED_STATUS === $sanitization['status']
);
/*
* Ignore validation errors which are forcibly sanitized by filter. This includes tree shaking error
* accepted by options and via AMP_Validation_Error_Taxonomy::accept_validation_errors()).
* This was introduced in <https://github.com/ampproject/amp-wp/pull/1413> to prevent forcibly-sanitized
* validation errors from being reported, to avoid noise and wasted storage. It was inadvertently
* reverted in de7b04b but then restored as part of <https://github.com/ampproject/amp-wp/pull/1413>.
*/
if ( $sanitized && 'with_filter' === $sanitization['forced'] ) {
return true;
}
// Add sources back into the $error for referencing later. @todo It may be cleaner to store sources separately to avoid having to re-remove later during storage.
$error = array_merge( $error, compact( 'sources' ) );
self::$validation_results[] = compact( 'error', 'sanitized' );
return $sanitized;
}
/**
* Reset the stored removed nodes and attributes.
*
* After testing if the markup is valid,
* these static values will remain.
* So reset them in case another test is needed.
*
* @return void
*/
public static function reset_validation_results() {
self::$validation_results = array();
self::$enqueued_style_sources = array();
self::$enqueued_script_sources = array();
}
/**
* Checks the AMP validity of the post content.
*
* If it's not valid AMP, it displays an error message above the 'Classic' editor.
*
* This is essentially a PHP implementation of ampBlockValidation.handleValidationErrorsStateChange() in JS.
*
* @param WP_Post $post The updated post.
* @return void
*/
public static function print_edit_form_validation_status( $post ) {
if ( ! post_supports_amp( $post ) || ! self::has_cap() ) {
return;
}
// Skip if the post type is not viewable on the frontend, since we need a permalink to validate.
if ( ! is_post_type_viewable( $post->post_type ) ) {
return;
}
$invalid_url_post = AMP_Validated_URL_Post_Type::get_invalid_url_post( get_permalink( $post->ID ) );
if ( ! $invalid_url_post ) {
return;
}
// Show all validation errors which have not been explicitly acknowledged as accepted.
$validation_errors = array();
$has_rejected_error = false;
foreach ( AMP_Validated_URL_Post_Type::get_invalid_url_validation_errors( $invalid_url_post ) as $error ) {
$needs_moderation = (
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_REJECTED_STATUS === $error['status'] || // @todo Show differently since moderated?
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_NEW_REJECTED_STATUS === $error['status'] ||
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_NEW_ACCEPTED_STATUS === $error['status']
);
if ( $needs_moderation ) {
$validation_errors[] = $error['data'];
}
if (
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_NEW_REJECTED_STATUS === $error['status']
||
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_REJECTED_STATUS === $error['status']
) {
$has_rejected_error = true;
}
}
// No validation errors so abort.
if ( empty( $validation_errors ) ) {
return;
}
echo '<div class="notice notice-warning">';
echo '<p>';
// @todo Check if the error actually occurs in the_content, and if not, consider omitting the warning if the user does not have privileges to manage_options.
esc_html_e( 'There is content which fails AMP validation.', 'amp' );
echo ' ';
// Auto-acceptance is from either checking 'Automatically accept sanitization...' or from being in Native mode.
if ( self::is_sanitization_auto_accepted() ) {
if ( ! $has_rejected_error ) {
esc_html_e( 'However, your site is configured to automatically accept sanitization of the offending markup. You should review the issues to confirm whether or not sanitization should be accepted or rejected.', 'amp' );
} else {
/*
* Even if the 'auto_accept_sanitization' option is true, if there are non-accepted errors in non-Native mode, it will redirect to a non-AMP page.
* For example, the errors could have been stored as 'New Rejected' when auto-accept was false, and now auto-accept is true.
* In that case, this will block serving AMP.
* This could also apply if this is in 'Native' mode and the user has rejected a validation error.
*/
esc_html_e( 'Though your site is configured to automatically accept sanitization errors, there are rejected error(s). This could be because auto-acceptance of errors was disabled earlier. You should review the issues to confirm whether or not sanitization should be accepted or rejected.', 'amp' );
}
} else {
esc_html_e( 'Non-accepted validation errors prevent AMP from being served, and the user will be redirected to the non-AMP version.', 'amp' );
}
echo sprintf(
' <a href="%s" target="_blank">%s</a>',
esc_url( get_edit_post_link( $invalid_url_post ) ),
esc_html__( 'Review issues', 'amp' )
);
echo '</p>';
$results = AMP_Validation_Error_Taxonomy::summarize_validation_errors( array_unique( $validation_errors, SORT_REGULAR ) );
$removed_sets = array();
if ( ! empty( $results[ AMP_Validation_Error_Taxonomy::REMOVED_ELEMENTS ] ) && is_array( $results[ AMP_Validation_Error_Taxonomy::REMOVED_ELEMENTS ] ) ) {
$removed_sets[] = array(
'label' => __( 'Invalid elements:', 'amp' ),
'names' => array_map( 'sanitize_key', $results[ AMP_Validation_Error_Taxonomy::REMOVED_ELEMENTS ] ),
);
}
if ( ! empty( $results[ AMP_Validation_Error_Taxonomy::REMOVED_ATTRIBUTES ] ) && is_array( $results[ AMP_Validation_Error_Taxonomy::REMOVED_ATTRIBUTES ] ) ) {
$removed_sets[] = array(
'label' => __( 'Invalid attributes:', 'amp' ),
'names' => array_map( 'sanitize_key', $results[ AMP_Validation_Error_Taxonomy::REMOVED_ATTRIBUTES ] ),
);
}
// @todo There are other kinds of errors other than REMOVED_ELEMENTS and REMOVED_ATTRIBUTES.
foreach ( $removed_sets as $removed_set ) {
printf( '<p>%s ', esc_html( $removed_set['label'] ) );
$items = array();
foreach ( $removed_set['names'] as $name => $count ) {
if ( 1 === intval( $count ) ) {
$items[] = sprintf( '<code>%s</code>', esc_html( $name ) );
} else {
$items[] = sprintf( '<code>%s</code> (%d)', esc_html( $name ), $count );
}
}
echo implode( ', ', $items ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
echo '</p>';
}
echo '</div>';
}
/**
* Get source start comment.
*
* @param array $source Source data.
* @param bool $is_start Whether the comment is the start or end.
* @return string HTML Comment.
*/
public static function get_source_comment( array $source, $is_start = true ) {
unset( $source['reflection'] );
return sprintf(
'<!--%samp-source-stack %s-->',
$is_start ? '' : '/',
str_replace( '--', '', wp_json_encode( $source ) )
);
}
/**
* Parse source comment.
*
* @param DOMComment $comment Comment.
* @return array|null Parsed source or null if not a source comment.
*/
public static function parse_source_comment( DOMComment $comment ) {
if ( ! preg_match( '#^\s*(?P<closing>/)?amp-source-stack\s+(?P<args>{.+})\s*$#s', $comment->nodeValue, $matches ) ) {
return null;
}
$source = json_decode( $matches['args'], true );
$closing = ! empty( $matches['closing'] );
return compact( 'source', 'closing' );
}
/**
* Walk back tree to find the open sources.
*
* @todo This method and others for sourcing could be moved to a separate class.
*
* @param DOMNode $node Node to look for.
* @return array[][] {
* The data of the removed sources (theme, plugin, or mu-plugin).
*
* @type string $name The name of the source.
* @type string $type The type of the source.
* }
*/
public static function locate_sources( DOMNode $node ) {
$xpath = new DOMXPath( $node->ownerDocument );
$comments = $xpath->query( 'preceding::comment()[ starts-with( ., "amp-source-stack" ) or starts-with( ., "/amp-source-stack" ) ]', $node );
$sources = array();
$matches = array();
foreach ( $comments as $comment ) {
$parsed_comment = self::parse_source_comment( $comment );
if ( ! $parsed_comment ) {
continue;
}
if ( $parsed_comment['closing'] ) {
array_pop( $sources );
} else {
$sources[] = $parsed_comment['source'];
}
}
$is_enqueued_link = (
$node instanceof DOMElement
&&
'link' === $node->nodeName
&&
preg_match( '/(?P<handle>.+)-css$/', (string) $node->getAttribute( 'id' ), $matches )
&&
isset( self::$enqueued_style_sources[ $matches['handle'] ] )
);
if ( $is_enqueued_link ) {
$sources = array_merge(
self::$enqueued_style_sources[ $matches['handle'] ],
$sources
);
}
/**
* Script dependency.
*
* @var _WP_Dependency $script_dependency
*/
if ( $node instanceof DOMElement && 'script' === $node->nodeName ) {
$enqueued_script_handles = array_intersect( wp_scripts()->done, array_keys( self::$enqueued_script_sources ) );
if ( $node->hasAttribute( 'src' ) ) {
// External script.
$src = $node->getAttribute( 'src' );
foreach ( $enqueued_script_handles as $enqueued_script_handle ) {
$script_dependency = wp_scripts()->registered[ $enqueued_script_handle ];
$is_matching_script = (
$script_dependency
&&
$script_dependency->src
&&
// Script attribute is haystack because includes protocol and may include query args (like ver).
false !== strpos( $src, preg_replace( '#^https?:(?=//)#', '', $script_dependency->src ) )
);
if ( $is_matching_script ) {
$sources = array_merge(
self::$enqueued_script_sources[ $enqueued_script_handle ],
$sources
);
break;
}
}
} elseif ( $node->firstChild ) {
// Inline script.
$text = $node->textContent;
foreach ( $enqueued_script_handles as $enqueued_script_handle ) {
$inline_scripts = array_filter(
array_merge(
(array) wp_scripts()->get_data( $enqueued_script_handle, 'data' ),
(array) wp_scripts()->get_data( $enqueued_script_handle, 'before' ),
(array) wp_scripts()->get_data( $enqueued_script_handle, 'after' )
)
);
foreach ( $inline_scripts as $inline_script ) {
/*
* Check to see if the inline script is inside (or the same) as the script in the document.
* Note that WordPress takes the registered inline script and will output it with newlines
* padding it, and sometimes with the script wrapped by CDATA blocks.
*/
if ( false !== strpos( $text, trim( $inline_script ) ) ) {
$sources = array_merge(
self::$enqueued_script_sources[ $enqueued_script_handle ],
$sources
);
break;
}
}
}
}
}
return $sources;
}
/**
* Remove source comments.
*
* @param DOMDocument $dom Document.
*/
public static function remove_source_comments( $dom ) {
$xpath = new DOMXPath( $dom );
$comments = array();
foreach ( $xpath->query( '//comment()[ starts-with( ., "amp-source-stack" ) or starts-with( ., "/amp-source-stack" ) ]' ) as $comment ) {
if ( self::parse_source_comment( $comment ) ) {
$comments[] = $comment;
}
}
foreach ( $comments as $comment ) {
$comment->parentNode->removeChild( $comment );
}
}
/**
* Add block source comments.
*
* @param string $content Content prior to blocks being processed.
* @return string Content with source comments added.
*/
public static function add_block_source_comments( $content ) {
self::$block_content_index = 0;
$start_block_pattern = implode(
'',
array(
'#<!--\s+',
'(?P<closing>/)?',
'wp:(?P<name>\S+)',
'(?:\s+(?P<attributes>\{.*?\}))?',
'\s+(?P<self_closing>\/)?',
'-->#s',
)
);
return preg_replace_callback(
$start_block_pattern,
array( __CLASS__, 'handle_block_source_comment_replacement' ),
$content
);
}
/**
* Handle block source comment replacement.
*
* @see \AMP_Validation_Manager::add_block_source_comments()
*
* @param array $matches Matches.
*
* @return string Replaced.
*/
protected static function handle_block_source_comment_replacement( $matches ) {
$replaced = $matches[0];
// Obtain source information for block.
$source = array(
'block_name' => $matches['name'],
'post_id' => get_the_ID(),
);
if ( empty( $matches['closing'] ) ) {
$source['block_content_index'] = self::$block_content_index;
self::$block_content_index++;
}
// Make implicit core namespace explicit.
$is_implicit_core_namespace = ( false === strpos( $source['block_name'], '/' ) );
$source['block_name'] = $is_implicit_core_namespace ? 'core/' . $source['block_name'] : $source['block_name'];
if ( ! empty( $matches['attributes'] ) ) {
$source['block_attrs'] = json_decode( $matches['attributes'] );
}
$block_type = WP_Block_Type_Registry::get_instance()->get_registered( $source['block_name'] );
if ( $block_type && $block_type->is_dynamic() ) {
$callback_source = self::get_source( $block_type->render_callback );
if ( $callback_source ) {
$source = array_merge(
$source,
$callback_source
);
}
}
if ( ! empty( $matches['closing'] ) ) {
$replaced .= self::get_source_comment( $source, false );
} else {
$replaced = self::get_source_comment( $source, true ) . $replaced;
if ( ! empty( $matches['self_closing'] ) ) {
unset( $source['block_content_index'] );
$replaced .= self::get_source_comment( $source, false );
}
}
return $replaced;
}
/**
* Wrap callbacks for registered widgets to keep track of queued assets and the source for anything printed for validation.
*
* @global array $wp_filter
* @return void
*/
public static function wrap_widget_callbacks() {
global $wp_registered_widgets;
foreach ( $wp_registered_widgets as $widget_id => &$registered_widget ) {
$source = self::get_source( $registered_widget['callback'] );
if ( ! $source ) {
continue;
}
$source['widget_id'] = $widget_id;
unset( $source['reflection'] ); // Omit from stored source.
$function = $registered_widget['callback'];
$accepted_args = 2; // For the $instance and $args arguments.
$callback = compact( 'function', 'accepted_args', 'source' );
$registered_widget['callback'] = self::wrapped_callback( $callback );
}
}
/**
* Wrap filter/action callback functions for a given hook.
*
* Wrapped callback functions are reset to their original functions after invocation.
* This runs at the 'all' action. The shutdown hook is excluded.
*
* @global WP_Hook[] $wp_filter
* @param string $hook Hook name for action or filter.
* @return void
*/
public static function wrap_hook_callbacks( $hook ) {
global $wp_filter;
if ( ! isset( $wp_filter[ $hook ] ) || 'shutdown' === $hook ) {
return;
}
self::$current_hook_source_stack[ $hook ] = array();
foreach ( $wp_filter[ $hook ]->callbacks as $priority => &$callbacks ) {
foreach ( $callbacks as &$callback ) {
$source = self::get_source( $callback['function'] );
if ( ! $source ) {
continue;
}
$reflection = $source['reflection'];
unset( $source['reflection'] ); // Omit from stored source.
// Add hook to stack for decorate_filter_source to read from.
self::$current_hook_source_stack[ $hook ][] = $source;
/*
* A current limitation with wrapping callbacks is that the wrapped function cannot have
* any parameters passed by reference. Without this the result is:
*
* > PHP Warning: Parameter 1 to wp_default_styles() expected to be a reference, value given.
*/
if ( self::has_parameters_passed_by_reference( $reflection ) ) {
continue;
}
$source['hook'] = $hook;
$original_function = $callback['function'];
$wrapped_callback = self::wrapped_callback(
array_merge(
$callback,
compact( 'priority', 'source' )
)
);
$callback['function'] = function() use ( &$callback, $wrapped_callback, $original_function ) {
$callback['function'] = $original_function; // Restore original.
return call_user_func_array( $wrapped_callback, func_get_args() );
};
}
}
}
/**
* Determine whether the given reflection method/function has params passed by reference.
*
* @since 0.7
* @param ReflectionFunction|ReflectionMethod $reflection Reflection.
* @return bool Whether there are parameters passed by reference.
*/
protected static function has_parameters_passed_by_reference( $reflection ) {
foreach ( $reflection->getParameters() as $parameter ) {
if ( $parameter->isPassedByReference() ) {
return true;
}
}
return false;
}
/**
* Filters the output created by a shortcode callback.
*
* @since 0.7
*
* @param string $output Shortcode output.
* @param string $tag Shortcode name.
* @return string Output.
* @global array $shortcode_tags
*/
public static function decorate_shortcode_source( $output, $tag ) {
global $shortcode_tags;
if ( ! isset( $shortcode_tags[ $tag ] ) ) {
return $output;
}
$source = self::get_source( $shortcode_tags[ $tag ] );
if ( empty( $source ) ) {
return $output;
}
$source['shortcode'] = $tag;
$output = implode(
'',
array(
self::get_source_comment( $source, true ),
$output,
self::get_source_comment( $source, false ),
)
);
return $output;
}
/**
* Filters the output created by embeds.
*
* @since 1.0
*
* @param string $output Embed output.
* @param string $url URL.
* @param array $attr Attributes.
* @return string Output.
*/
public static function decorate_embed_source( $output, $url, $attr ) {
$source = array(
'embed' => $url,
'attr' => $attr,
);
return implode(
'',
array(
self::get_source_comment( $source, true ),
trim( $output ),
self::get_source_comment( $source, false ),
)
);
}
/**
* Wraps output of a filter to add source stack comments.
*
* @todo Duplicate with AMP_Validation_Manager::wrap_buffer_with_source_comments()?
* @param string $value Value.
* @return string Value wrapped in source comments.
*/
public static function decorate_filter_source( $value ) {
// Abort if the output is not a string and it doesn't contain any HTML tags.
if ( ! is_string( $value ) || ! preg_match( '/<.+?>/s', $value ) ) {
return $value;
}
$post = get_post();
$source = array(
'hook' => current_filter(),
'filter' => true,
);
if ( $post ) {
$source['post_id'] = $post->ID;
$source['post_type'] = $post->post_type;
}
if ( isset( self::$current_hook_source_stack[ current_filter() ] ) ) {
$sources = self::$current_hook_source_stack[ current_filter() ];
array_pop( $sources ); // Remove self.
$source['sources'] = $sources;
}
return implode(
'',
array(
self::get_source_comment( $source, true ),
$value,
self::get_source_comment( $source, false ),
)
);
}
/**
* Gets the plugin or theme of the callback, if one exists.
*
* @param string|array $callback The callback for which to get the plugin.
* @return array|null {
* The source data.
*
* @type string $type Source type (core, plugin, mu-plugin, or theme).
* @type string $name Source name.
* @type string $function Normalized function name.
* @type ReflectionMethod|ReflectionFunction $reflection
* }
*/
public static function get_source( $callback ) {
$reflection = null;
$class_name = null; // Because ReflectionMethod::getDeclaringClass() can return a parent class.
$file = null;
try {
if ( is_string( $callback ) && is_callable( $callback ) ) {
// The $callback is a function or static method.
$exploded_callback = explode( '::', $callback, 2 );
if ( 2 === count( $exploded_callback ) ) {
$class_name = $exploded_callback[0];
$reflection = new ReflectionMethod( $exploded_callback[0], $exploded_callback[1] );
} else {
$reflection = new ReflectionFunction( $callback );
}
} elseif ( is_array( $callback ) && isset( $callback[0], $callback[1] ) && method_exists( $callback[0], $callback[1] ) ) {
// The $callback is a method.
if ( is_string( $callback[0] ) ) {
$class_name = $callback[0];
} elseif ( is_object( $callback[0] ) ) {
$class_name = get_class( $callback[0] );
}
/*
* Obtain file from ReflectionClass because if the method is not on base class then
* file returned by ReflectionMethod will be for the base class not the subclass.
*/
$reflection = new ReflectionClass( $callback[0] );
$file = $reflection->getFileName();
// This is needed later for AMP_Validation_Manager::has_parameters_passed_by_reference().
$reflection = new ReflectionMethod( $callback[0], $callback[1] );
} elseif ( is_object( $callback ) && ( 'Closure' === get_class( $callback ) ) ) {
$reflection = new ReflectionFunction( $callback );
}
if ( $reflection && ! $file ) {
$file = $reflection->getFileName();
}
} catch ( Exception $e ) {
return null;
}
if ( ! $reflection ) {
return null;
}
$source = compact( 'reflection' );
if ( $file ) {
$file = wp_normalize_path( $file );
$slug_pattern = '([^/]+)';
if ( preg_match( ':' . preg_quote( trailingslashit( wp_normalize_path( WP_PLUGIN_DIR ) ), ':' ) . $slug_pattern . ':s', $file, $matches ) ) {
$source['type'] = 'plugin';
$source['name'] = $matches[1];
} elseif ( preg_match( ':' . preg_quote( trailingslashit( wp_normalize_path( get_theme_root() ) ), ':' ) . $slug_pattern . ':s', $file, $matches ) ) {
$source['type'] = 'theme';
$source['name'] = $matches[1];
} elseif ( preg_match( ':' . preg_quote( trailingslashit( wp_normalize_path( WPMU_PLUGIN_DIR ) ), ':' ) . $slug_pattern . ':s', $file, $matches ) ) {
$source['type'] = 'mu-plugin';
$source['name'] = $matches[1];
} elseif ( preg_match( ':' . preg_quote( trailingslashit( wp_normalize_path( ABSPATH ) ), ':' ) . '(wp-admin|wp-includes)/:s', $file, $matches ) ) {
$source['type'] = 'core';
$source['name'] = $matches[1];
}
}
if ( $class_name ) {
$source['function'] = $class_name . '::' . $reflection->getName();
} else {
$source['function'] = $reflection->getName();
}
return $source;
}
/**
* Check whether or not output buffering is currently possible.
*
* This is to guard against a fatal error: "ob_start(): Cannot use output buffering in output buffering display handlers".
*
* @return bool Whether output buffering is allowed.
*/
public static function can_output_buffer() {
// Output buffering for validation can only be done while overall output buffering is being done for the response.
if ( ! AMP_Theme_Support::is_output_buffering() ) {
return false;
}
// Abort when in shutdown since output has finished, when we're likely in the overall output buffering display handler.
if ( did_action( 'shutdown' ) ) {
return false;
}
// Check if any functions in call stack are output buffering display handlers.
$called_functions = array();
if ( defined( 'DEBUG_BACKTRACE_IGNORE_ARGS' ) ) {
$arg = DEBUG_BACKTRACE_IGNORE_ARGS; // phpcs:ignore PHPCompatibility.Constants.NewConstants.debug_backtrace_ignore_argsFound
} else {
$arg = false;
}
$backtrace = debug_backtrace( $arg ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_debug_backtrace -- Only way to find out if we are in a buffering display handler.
foreach ( $backtrace as $call_stack ) {
if ( '{closure}' === $call_stack['function'] ) {
$called_functions[] = 'Closure::__invoke';
} elseif ( isset( $call_stack['class'] ) ) {
$called_functions[] = sprintf( '%s::%s', $call_stack['class'], $call_stack['function'] );
} else {
$called_functions[] = $call_stack['function'];
}
}
return 0 === count( array_intersect( ob_list_handlers(), $called_functions ) );
}
/**
* Wraps a callback in comments if it outputs markup.
*
* If the sanitizer removes markup,
* this indicates which plugin it was from.
* The call_user_func_array() logic is mainly copied from WP_Hook:apply_filters().
*
* @param array $callback {
* The callback data.
*
* @type callable $function
* @type int $accepted_args
* @type array $source
* }
* @return closure $wrapped_callback The callback, wrapped in comments.
*/
public static function wrapped_callback( $callback ) {
return function() use ( $callback ) {
global $wp_styles, $wp_scripts;
$function = $callback['function'];
$accepted_args = $callback['accepted_args'];
$args = func_get_args();
$before_styles_enqueued = array();
if ( isset( $wp_styles ) && isset( $wp_styles->queue ) ) {
$before_styles_enqueued = $wp_styles->queue;
}
$before_scripts_enqueued = array();
if ( isset( $wp_scripts ) && isset( $wp_scripts->queue ) ) {
$before_scripts_enqueued = $wp_scripts->queue;
}
$is_filter = isset( $callback['source']['hook'] ) && ! did_action( $callback['source']['hook'] );
// Wrap the markup output of (action) hooks in source comments.
AMP_Validation_Manager::$hook_source_stack[] = $callback['source'];
$has_buffer_started = false;
if ( ! $is_filter && AMP_Validation_Manager::can_output_buffer() ) {
$has_buffer_started = ob_start( array( __CLASS__, 'wrap_buffer_with_source_comments' ) );
}
$result = call_user_func_array( $function, array_slice( $args, 0, intval( $accepted_args ) ) );
if ( $has_buffer_started ) {
ob_end_flush();
}
array_pop( AMP_Validation_Manager::$hook_source_stack );
// Keep track of which source enqueued the styles.
if ( isset( $wp_styles ) && isset( $wp_styles->queue ) ) {
foreach ( array_diff( $wp_styles->queue, $before_styles_enqueued ) as $handle ) {
AMP_Validation_Manager::$enqueued_style_sources[ $handle ][] = array_merge( $callback['source'], compact( 'handle' ) );
}
}
// Keep track of which source enqueued the scripts, and immediately report validity.
if ( isset( $wp_scripts ) && isset( $wp_scripts->queue ) ) {
foreach ( array_diff( $wp_scripts->queue, $before_scripts_enqueued ) as $queued_handle ) {
$handles = array( $queued_handle );
// Account for case where registered script is a placeholder for a set of scripts (e.g. jquery).
if ( isset( $wp_scripts->registered[ $queued_handle ] ) && false === $wp_scripts->registered[ $queued_handle ]->src ) {
$handles = array_merge( $handles, $wp_scripts->registered[ $queued_handle ]->deps );
}
foreach ( $handles as $handle ) {
AMP_Validation_Manager::$enqueued_script_sources[ $handle ][] = array_merge( $callback['source'], compact( 'handle' ) );
}
}
}
return $result;
};
}
/**
* Wrap output buffer with source comments.
*
* A key reason for why this is a method and not a closure is so that
* the can_output_buffer method will be able to identify it by name.
*
* @since 0.7
* @todo Is duplicate of \AMP_Validation_Manager::decorate_filter_source()?
*
* @param string $output Output buffer.
* @return string Output buffer conditionally wrapped with source comments.
*/
public static function wrap_buffer_with_source_comments( $output ) {
if ( empty( self::$hook_source_stack ) ) {
return $output;
}
$source = self::$hook_source_stack[ count( self::$hook_source_stack ) - 1 ];
// Wrap output that contains HTML tags (as opposed to actions that trigger in HTML attributes).
if ( ! empty( $output ) && preg_match( '/<.+?>/s', $output ) ) {
$output = implode(
'',
array(
self::get_source_comment( $source, true ),
$output,
self::get_source_comment( $source, false ),
)
);
}
return $output;
}
/**
* Get nonce for performing amp_validate request.
*
* The returned nonce is irrespective of the authenticated user.
*
* @return string Nonce.
*/
public static function get_amp_validate_nonce() {
return substr( wp_hash( self::VALIDATE_QUERY_VAR . (string) wp_nonce_tick(), 'nonce' ), -12, 10 );
}
/**
* Whether to validate the front end response.
*
* @return boolean Whether to validate.
*/
public static function should_validate_response() {
if ( ! isset( $_GET[ self::VALIDATE_QUERY_VAR ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
return false;
}
if ( self::has_cap() ) {
return true;
}
$validate_key = wp_unslash( $_GET[ self::VALIDATE_QUERY_VAR ] ); // phpcs:ignore WordPress.Security.NonceVerification.Recommended
return self::get_amp_validate_nonce() === $validate_key;
}
/**
* Finalize validation.
*
* @see AMP_Validation_Manager::add_admin_bar_menu_items()
*
* @param DOMDocument $dom Document.
* @param array $args {
* Args.
*
* @type bool $remove_source_comments Whether source comments should be removed. Defaults to true.
* @type bool $append_validation_status_comment Whether the validation errors should be appended as an HTML comment. Defaults to true.
* }
*/
public static function finalize_validation( DOMDocument $dom, $args = array() ) {
$args = array_merge(
array(
'remove_source_comments' => true,
'append_validation_status_comment' => true,
),
$args
);
/*
* Override AMP status in admin bar set in \AMP_Validation_Manager::add_admin_bar_menu_items()
* when there are validation errors which have not been explicitly accepted.
*/
if ( is_admin_bar_showing() && self::$amp_admin_bar_item_added ) {
$error_count = 0;
foreach ( self::$validation_results as $validation_result ) {
$validation_status = AMP_Validation_Error_Taxonomy::get_validation_error_sanitization( $validation_result['error'] );
$is_unaccepted = 'with_preview' === $validation_status['forced'] ?
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_ACCEPTED_STATUS !== $validation_status['status']
:
AMP_Validation_Error_Taxonomy::VALIDATION_ERROR_ACK_ACCEPTED_STATUS !== $validation_status['term_status'];
if ( $is_unaccepted ) {
$error_count++;
}
}
if ( $error_count > 0 ) {
$validate_item = $dom->getElementById( 'wp-admin-bar-amp-validity' );
if ( $validate_item ) {
$link = $validate_item->getElementsByTagName( 'a' )->item( 0 );
if ( $link ) {
$link->textContent = sprintf(
/* translators: %s is count of validation errors */
_n(
'Re-validate (%s validation error)',
'Re-validate (%s validation errors)',
$error_count,
'amp'
),
number_format_i18n( $error_count )
);
}
}
$admin_bar_icon = $dom->getElementById( 'amp-admin-bar-item-status-icon' );
if ( $admin_bar_icon ) {
$admin_bar_icon->textContent = "\xE2\x9A\xA0\xEF\xB8\x8F"; // WARNING SIGN: U+26A0, U+FE0F.
}
}
}
if ( self::should_validate_response() ) {
if ( $args['remove_source_comments'] ) {
self::remove_source_comments( $dom );
}
if ( $args['append_validation_status_comment'] ) {
$data = array(
'results' => self::$validation_results,
);
if ( get_queried_object() ) {
$data['queried_object'] = array();
if ( get_queried_object_id() ) {
$data['queried_object']['id'] = get_queried_object_id();
}
if ( get_queried_object() instanceof WP_Post ) {
$data['queried_object']['type'] = 'post';
} elseif ( get_queried_object() instanceof WP_Term ) {
$data['queried_object']['type'] = 'term';
} elseif ( get_queried_object() instanceof WP_User ) {
$data['queried_object']['type'] = 'user';
} elseif ( get_queried_object() instanceof WP_Post_Type ) {
$data['queried_object']['type'] = 'post_type';
}
}
$encoded = wp_json_encode( $data, 128 /* JSON_PRETTY_PRINT */ );
$encoded = str_replace( '--', '\u002d\u002d', $encoded ); // Prevent "--" in strings from breaking out of HTML comments.
$comment = $dom->createComment( 'AMP_VALIDATION:' . $encoded . "\n" );
$dom->documentElement->appendChild( $comment );
}
}
}
/**
* Adds the validation callback if front-end validation is needed.
*
* @param array $sanitizers The AMP sanitizers.
* @return array $sanitizers The filtered AMP sanitizers.
*/
public static function filter_sanitizer_args( $sanitizers ) {
foreach ( $sanitizers as $sanitizer => &$args ) {
$args['validation_error_callback'] = __CLASS__ . '::add_validation_error';
}
if ( isset( $sanitizers['AMP_Style_Sanitizer'] ) ) {
$sanitizers['AMP_Style_Sanitizer']['should_locate_sources'] = self::$should_locate_sources;
$css_validation_errors = array();
foreach ( self::$validation_error_status_overrides as $slug => $status ) {
$term = AMP_Validation_Error_Taxonomy::get_term( $slug );
if ( ! $term ) {
continue;
}
$validation_error = json_decode( $term->description, true );
$is_css_validation_error = (
is_array( $validation_error )
&&
isset( $validation_error['code'] )
&&
in_array( $validation_error['code'], AMP_Style_Sanitizer::get_css_parser_validation_error_codes(), true )
);
if ( $is_css_validation_error ) {
$css_validation_errors[ $slug ] = $status;
}
}
if ( ! empty( $css_validation_errors ) ) {
$sanitizers['AMP_Style_Sanitizer']['parsed_cache_variant'] = md5( wp_json_encode( $css_validation_errors ) );
}
$sanitizers['AMP_Style_Sanitizer']['accept_tree_shaking'] = AMP_Options_Manager::get_option( 'accept_tree_shaking' );
}
return $sanitizers;
}
/**
* Validates the latest published post.
*
* @return array|WP_Error The validation errors, or WP_Error.
*/
public static function validate_after_plugin_activation() {
$url = amp_admin_get_preview_permalink();
if ( ! $url ) {
return new WP_Error( 'no_published_post_url_available' );
}
$validity = self::validate_url( $url );
if ( is_wp_error( $validity ) ) {
return $validity;
}
$validation_errors = wp_list_pluck( $validity['results'], 'error' );
if ( is_array( $validity ) && count( $validation_errors ) > 0 ) { // @todo This should only warn when there are unaccepted validation errors.
AMP_Validated_URL_Post_Type::store_validation_errors(
$validation_errors,
$validity['url'],
wp_array_slice_assoc( $validity, array( 'queried_object_id', 'queried_object_type' ) )
);
set_transient( self::PLUGIN_ACTIVATION_VALIDATION_ERRORS_TRANSIENT_KEY, $validation_errors, 60 );
} else {
delete_transient( self::PLUGIN_ACTIVATION_VALIDATION_ERRORS_TRANSIENT_KEY );
}
return $validation_errors;
}
/**
* Validates a given URL.
*
* The validation errors will be stored in the validation status custom post type,
* as well as in a transient.
*
* @param string $url The URL to validate. This need not include the amp query var.
* @return WP_Error|array {
* Response.
*
* @type array $results Validation results, where each nested array contains an error key and sanitized key.
* @type string $url Final URL that was checked or redirected to.
* @type int $queried_object_id Queried object ID.
* @type string $queried_object_type Queried object type.
* }
*/
public static function validate_url( $url ) {
if ( amp_is_canonical() ) {
$url = remove_query_arg( amp_get_slug(), $url );
} else {
$url = add_query_arg( amp_get_slug(), '', $url );
}
$added_query_vars = array(
self::VALIDATE_QUERY_VAR => self::get_amp_validate_nonce(),
self::CACHE_BUST_QUERY_VAR => wp_rand(),
);
$validation_url = add_query_arg( $added_query_vars, $url );
$r = null;
/** This filter is documented in wp-includes/class-http.php */
$allowed_redirects = apply_filters( 'http_request_redirection_count', 5 );
for ( $redirect_count = 0; $redirect_count < $allowed_redirects; $redirect_count++ ) {
$r = wp_remote_get(
$validation_url,
array(
'cookies' => wp_unslash( $_COOKIE ), // Pass along cookies so private pages and drafts can be accessed.
'timeout' => 15, // Increase from default of 5 to give extra time for the plugin to identify the sources for any given validation errors; also, response caching is disabled when validating.
'sslverify' => false,
'redirection' => 0, // Because we're in a loop for redirection.
'headers' => array(
'Cache-Control' => 'no-cache',
),
)
);
// If the response is not a redirect, then break since $r is all we need.
$response_code = wp_remote_retrieve_response_code( $r );
$location_header = wp_remote_retrieve_header( $r, 'Location' );
$is_redirect = (
$response_code
&&
$response_code > 300 && $response_code < 400
&&
$location_header
);
if ( ! $is_redirect ) {
break;
}
// Ensure absolute URL.
if ( '/' === substr( $location_header, 0, 1 ) ) {
$location_header = preg_replace( '#(^https?://[^/]+)/.*#', '$1', home_url( '/' ) ) . $location_header;
}
// Block redirecting to a different host.
$location_header = wp_validate_redirect( $location_header );
if ( ! $location_header ) {
break;
}
// Ensure the redirect URL is formatted for the AMP.
if ( amp_is_canonical() ) {
$location_header = remove_query_arg( amp_get_slug(), $location_header );
} else {
$location_header = add_query_arg( amp_get_slug(), '', $location_header );
}
$validation_url = add_query_arg( $added_query_vars, $location_header );
}
if ( is_wp_error( $r ) ) {
return $r;
}
if ( wp_remote_retrieve_response_code( $r ) >= 300 ) {
return new WP_Error(
wp_remote_retrieve_response_code( $r ),
wp_remote_retrieve_response_message( $r )
);
}
$url = remove_query_arg(
array_keys( $added_query_vars ),
$validation_url
);
$response = wp_remote_retrieve_body( $r );
if ( strlen( trim( $response ) ) === 0 ) {
$error_code = 'white_screen_of_death';
return new WP_Error( $error_code, self::get_validate_url_error_message( $error_code ) );
}
if ( ! preg_match( '#</body>.*?<!--\s*AMP_VALIDATION\s*:\s*(\{.*?\})\s*-->#s', $response, $matches ) ) {
$error_code = 'response_comment_absent';
return new WP_Error( $error_code, self::get_validate_url_error_message( $error_code ) );
}
$validation = json_decode( $matches[1], true );
if ( json_last_error() || ! isset( $validation['results'] ) || ! is_array( $validation['results'] ) ) {
$error_code = 'malformed_json_validation_errors';
return new WP_Error( $error_code, self::get_validate_url_error_message( $error_code ) );
}
return array_merge(
$validation,
compact( 'url' )
);
}
/**
* Get error message for a validate URL failure.
*
* @param string $error_code Error code.
* @return string Error message.
*/
public static function get_validate_url_error_message( $error_code ) {
switch ( $error_code ) {
case 'http_request_failed':
return __( 'Failed to fetch URL(s) to validate. This may be due to a request timeout.', 'amp' );
case 'white_screen_of_death':
return __( 'Unable to validate URL. Encountered a white screen of death likely due to a fatal error. Please check your server\'s PHP error logs.', 'amp' );
case '404':
return __( 'The fetched URL was not found. It may have been deleted. If so, you can trash this.', 'amp' );
case '500':
return __( 'An internal server error occurred when fetching the URL for validation.', 'amp' );
case 'response_comment_absent':
return sprintf(
/* translators: %s: AMP_VALIDATION */
__( 'URL validation failed to due to the absence of the expected JSON-containing %s comment after the body.', 'amp' ),
'AMP_VALIDATION'
);
case 'malformed_json_validation_errors':
return sprintf(
/* translators: %s: AMP_VALIDATION */
__( 'URL validation failed to due to unexpected JSON in the %s comment after the body.', 'amp' ),
'AMP_VALIDATION'
);
default:
/* translators: %s is error code */
return sprintf( __( 'URL validation failed. Error code: %s.', 'amp' ), $error_code ); // Note that $error_code has been sanitized with sanitize_key(); will be escaped below as well.
};
}
/**
* On activating a plugin, display a notice if a plugin causes an AMP validation error.
*
* @return void
*/
public static function print_plugin_notice() {
global $pagenow;
if ( ( 'plugins.php' === $pagenow ) && ( ! empty( $_GET['activate'] ) || ! empty( $_GET['activate-multi'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
$validation_errors = get_transient( self::PLUGIN_ACTIVATION_VALIDATION_ERRORS_TRANSIENT_KEY );
if ( empty( $validation_errors ) || ! is_array( $validation_errors ) ) {
return;
}
delete_transient( self::PLUGIN_ACTIVATION_VALIDATION_ERRORS_TRANSIENT_KEY );
$errors = AMP_Validation_Error_Taxonomy::summarize_validation_errors( $validation_errors );
$invalid_plugins = isset( $errors[ AMP_Validation_Error_Taxonomy::SOURCES_INVALID_OUTPUT ]['plugin'] ) ? array_unique( $errors[ AMP_Validation_Error_Taxonomy::SOURCES_INVALID_OUTPUT ]['plugin'] ) : null;
if ( isset( $invalid_plugins ) ) {
$reported_plugins = array();
foreach ( $invalid_plugins as $plugin ) {
$reported_plugins[] = sprintf( '<code>%s</code>', esc_html( $plugin ) );
}
$more_details_link = sprintf(
'<a href="%s">%s</a>',
esc_url(
add_query_arg(
'post_type',
AMP_Validated_URL_Post_Type::POST_TYPE_SLUG,
admin_url( 'edit.php' )
)
),
__( 'More details', 'amp' )
);
printf(
'<div class="notice notice-warning is-dismissible"><p>%s %s %s</p><button type="button" class="notice-dismiss"><span class="screen-reader-text">%s</span></button></div>',
esc_html( _n( 'Warning: The following plugin may be incompatible with AMP:', 'Warning: The following plugins may be incompatible with AMP:', count( $invalid_plugins ), 'amp' ) ),
implode( ', ', $reported_plugins ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
$more_details_link, // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
esc_html__( 'Dismiss this notice.', 'amp' )
);
}
}
}
/**
* Enqueues the block validation script.
*
* @return void
*/
public static function enqueue_block_validation() {
$slug = 'amp-block-validation';
wp_enqueue_script(
$slug,
amp_get_asset_url( "js/{$slug}.js" ),
array( 'underscore', AMP_Post_Meta_Box::BLOCK_ASSET_HANDLE ),
AMP__VERSION,
true
);
$data = array(
'ampValidityRestField' => self::VALIDITY_REST_FIELD_NAME,
'isSanitizationAutoAccepted' => self::is_sanitization_auto_accepted(),
);
if ( function_exists( 'wp_set_script_translations' ) ) {
wp_set_script_translations( $slug, 'amp' );
} elseif ( function_exists( 'wp_get_jed_locale_data' ) || function_exists( 'gutenberg_get_jed_locale_data' ) ) {
$data['i18n'] = function_exists( 'wp_get_jed_locale_data' ) ? wp_get_jed_locale_data( 'amp' ) : gutenberg_get_jed_locale_data( 'amp' );
}
wp_add_inline_script( $slug, sprintf( 'ampBlockValidation.boot( %s );', wp_json_encode( $data ) ) );
}
}