Skip to content

Commit

Permalink
HTML API: Refactor wp_targeted_link_rel
Browse files Browse the repository at this point in the history
See WordPress#5143

This patch refactors `wp_targeted_link_rel` to be more reliable and clear by
using the HTML API to replace existing PCRE code. It provides backwards
compatability for the `rel` filter by creating a quick HTML string representing
the opening A tag for the matched element and providing the `href`, `rel`, and
`target` attributes on that tag.
  • Loading branch information
dmsnell committed Oct 30, 2023
1 parent ffcf5fb commit df24c55
Show file tree
Hide file tree
Showing 2 changed files with 67 additions and 65 deletions.
21 changes: 21 additions & 0 deletions src/dumb.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php

$r = function ( $o ) {
return new class( $o ) {
private $o;
public function __construct( $o ) {
$this->o = $o;
}

public function __get( $n ) {
$f = get_class_vars( $this->o )[ $n ];

return $f();
}
};
};

$o = new stdClass();
$h = 'f';
$o->$h = function() { echo "Hello from the inside.\n"; };
$r($o)->$h();
111 changes: 46 additions & 65 deletions src/wp-includes/formatting.php
Original file line number Diff line number Diff line change
Expand Up @@ -3311,84 +3311,65 @@ static function ( $matches ) {
* @return string Converted content.
*/
function wp_targeted_link_rel( $text ) {
// Don't run (more expensive) regex if no links with targets.
// Don't run more expensive code if it's not possible that A links with targets set are in the text.
if ( stripos( $text, 'target' ) === false || stripos( $text, '<a ' ) === false || is_serialized( $text ) ) {
return $text;
}

$script_and_style_regex = '/<(script|style).*?<\/\\1>/si';
$processor = new WP_HTML_Tag_Processor( $text );
while ( $processor->next_tag( 'A' ) ) {
$rel = $processor->get_attribute( 'rel' );
$target = $processor->get_attribute( 'target' );

preg_match_all( $script_and_style_regex, $text, $matches );
$extra_parts = $matches[0];
$html_parts = preg_split( $script_and_style_regex, $text );
/**
* Generated to support backwards compatability for old filters
* which process the HTML tag string containing the link.
*/
$fake_link_html = sprintf(
'<a href="%s" rel="%s" target="%s">',
esc_url( $processor->get_attribute( 'href' ) ),
esc_attr( $rel ),
esc_attr( $target )
);

foreach ( $html_parts as &$part ) {
$part = preg_replace_callback( '|<a\s([^>]*target\s*=[^>]*)>|i', 'wp_targeted_link_rel_callback', $part );
}
/**
* Filters the rel values that are added to links with `target` attribute.
*
* This filter provides an HTML snippet containing the A element's opening tag
* and it should return a new value for the "rel" attribute.
*
* @since 5.1.0
* @since 6.5.0 Only sends href, rel, and target attributes in the HTML tag string.
*
* @param string $rel The rel values.
* @param string $link_html HTML snippet containing an opening A tag with href, rel, and target attributes.
*/
$filtered_rel = apply_filters( 'wp_targeted_link_rel', 'noopener', $fake_link_html );

$text = '';
for ( $i = 0; $i < count( $html_parts ); $i++ ) {
$text .= $html_parts[ $i ];
if ( isset( $extra_parts[ $i ] ) ) {
$text .= $extra_parts[ $i ];
// If someone wanted to remove the `rel` attribute then do that and move on.
if ( ! $filtered_rel ) {
$processor->remove_attribute( 'rel' );
continue;
}
}

return $text;
}

/**
* Callback to add `rel="noopener"` string to HTML A element.
*
* Will not duplicate an existing 'noopener' value to avoid invalidating the HTML.
*
* @since 5.1.0
* @since 5.6.0 Removed 'noreferrer' relationship.
*
* @param array $matches Single match.
* @return string HTML A Element with `rel="noopener"` in addition to any existing values.
*/
function wp_targeted_link_rel_callback( $matches ) {
$link_html = $matches[1];
$original_link_html = $link_html;

// Consider the HTML escaped if there are no unescaped quotes.
$is_escaped = ! preg_match( '/(^|[^\\\\])[\'"]/', $link_html );
if ( $is_escaped ) {
// Replace only the quotes so that they are parsable by wp_kses_hair(), leave the rest as is.
$link_html = preg_replace( '/\\\\([\'"])/', '$1', $link_html );
}

$atts = wp_kses_hair( $link_html, wp_allowed_protocols() );

/**
* Filters the rel values that are added to links with `target` attribute.
*
* @since 5.1.0
*
* @param string $rel The rel values.
* @param string $link_html The matched content of the link tag including all HTML attributes.
*/
$rel = apply_filters( 'wp_targeted_link_rel', 'noopener', $link_html );

// Return early if no rel values to be added or if no actual target attribute.
if ( ! $rel || ! isset( $atts['target'] ) ) {
return "<a $original_link_html>";
}

if ( isset( $atts['rel'] ) ) {
$all_parts = preg_split( '/\s/', "{$atts['rel']['value']} $rel", -1, PREG_SPLIT_NO_EMPTY );
$rel = implode( ' ', array_unique( $all_parts ) );
}
// If there is no `target` attribute then there's no need to add special `rel` values.
if ( ! is_string( $target ) ) {
continue;
}

$atts['rel']['whole'] = 'rel="' . esc_attr( $rel ) . '"';
$link_html = implode( ' ', array_column( $atts, 'whole' ) );
// Otherwise, append the filtered `rel` value to the original one and only retian the unique components.
$new_rel = ' ';
$rel_parts = preg_split( "~[ \t\f\n\r]+", "{$rel} {$filtered_rel}", -1, PREG_SPLIT_NO_EMPTY );
foreach ( $rel_parts as $part ) {
if ( ! str_contains( $new_rel, " {$part} " ) ) {
$new_rel .= "{$part} ";
}
}

if ( $is_escaped ) {
$link_html = preg_replace( '/[\'"]/', '\\\\$0', $link_html );
$processor->set_attribute( 'rel', trim( $new_rel ) );
}

return "<a $link_html>";
return $processor->get_updated_html();
}

/**
Expand Down

0 comments on commit df24c55

Please sign in to comment.