mirror of
https://github.com/WordPress/WordPress.git
synced 2025-01-18 14:21:25 +01:00
9fc546b2ce
Provides a custom decoder for strings coming from HTML attributes and markup. This custom decoder is necessary because of deficiencies in PHP's `html_entity_decode()` function: - It isn't aware of 720 of the possible named character references in HTML, leaving many out that should be translated. - It isn't aware of the ambiguous ampersand rule, which allows conversion of character references in certain contexts when they are missing their closing `;`. - It doesn't draw a distinction for the ambiguous ampersand rule when decoding attribute values instead of markup values. - Use of `html_entity_decode()` requires manually passing non-default paramter values to ensure it decodes properly. This decoder also provides some conveniences, such as making a single-pass and interruptable decode operation possible. This will provide a number of opportunities to optimize detection and decoding of things like value prefixes, and whether a value contains a given substring. Developed in https://github.com/WordPress/wordpress-develop/pull/6387 Discussed in https://core.trac.wordpress.org/ticket/61072 Props dmsnell, gziolo, jonsurrell, jorbin, westonruter, zieladam. Fixes #61072. Built from https://develop.svn.wordpress.org/trunk@58281 git-svn-id: http://core.svn.wordpress.org/trunk@57741 1a063a9b-81f0-0310-95a4-ce76da25c4cd
462 lines
16 KiB
PHP
462 lines
16 KiB
PHP
<?php
|
||
|
||
/**
|
||
* HTML API: WP_HTML_Decoder class
|
||
*
|
||
* Decodes spans of raw text found inside HTML content.
|
||
*
|
||
* @package WordPress
|
||
* @subpackage HTML-API
|
||
* @since 6.6.0
|
||
*/
|
||
class WP_HTML_Decoder {
|
||
/**
|
||
* Indicates if an attribute value starts with a given raw string value.
|
||
*
|
||
* Use this method to determine if an attribute value starts with a given string, regardless
|
||
* of how it might be encoded in HTML. For instance, `http:` could be represented as `http:`
|
||
* or as `http:` or as `http:` or as `http:`, or in many other ways.
|
||
*
|
||
* Example:
|
||
*
|
||
* $value = 'http://wordpress.org/';
|
||
* true === WP_HTML_Decoder::attribute_starts_with( $value, 'http:', 'ascii-case-insensitive' );
|
||
* false === WP_HTML_Decoder::attribute_starts_with( $value, 'https:', 'ascii-case-insensitive' );
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @param string $haystack String containing the raw non-decoded attribute value.
|
||
* @param string $search_text Does the attribute value start with this plain string.
|
||
* @param string $case_sensitivity Optional. Pass 'ascii-case-insensitive' to ignore ASCII case when matching.
|
||
* Default 'case-sensitive'.
|
||
* @return bool Whether the attribute value starts with the given string.
|
||
*/
|
||
public static function attribute_starts_with( $haystack, $search_text, $case_sensitivity = 'case-sensitive' ) {
|
||
$search_length = strlen( $search_text );
|
||
$loose_case = 'ascii-case-insensitive' === $case_sensitivity;
|
||
$haystack_end = strlen( $haystack );
|
||
$search_at = 0;
|
||
$haystack_at = 0;
|
||
|
||
while ( $search_at < $search_length && $haystack_at < $haystack_end ) {
|
||
$chars_match = $loose_case
|
||
? strtolower( $haystack[ $haystack_at ] ) === strtolower( $search_text[ $search_at ] )
|
||
: $haystack[ $haystack_at ] === $search_text[ $search_at ];
|
||
|
||
$is_introducer = '&' === $haystack[ $haystack_at ];
|
||
$next_chunk = $is_introducer
|
||
? self::read_character_reference( 'attribute', $haystack, $haystack_at, $token_length )
|
||
: null;
|
||
|
||
// If there's no character reference and the characters don't match, the match fails.
|
||
if ( null === $next_chunk && ! $chars_match ) {
|
||
return false;
|
||
}
|
||
|
||
// If there's no character reference but the character do match, then it could still match.
|
||
if ( null === $next_chunk && $chars_match ) {
|
||
++$haystack_at;
|
||
++$search_at;
|
||
continue;
|
||
}
|
||
|
||
// If there is a character reference, then the decoded value must exactly match what follows in the search string.
|
||
if ( 0 !== substr_compare( $search_text, $next_chunk, $search_at, strlen( $next_chunk ), $loose_case ) ) {
|
||
return false;
|
||
}
|
||
|
||
// The character reference matched, so continue checking.
|
||
$haystack_at += $token_length;
|
||
$search_at += strlen( $next_chunk );
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* Returns a string containing the decoded value of a given HTML text node.
|
||
*
|
||
* Text nodes appear in HTML DATA sections, which are the text segments inside
|
||
* and around tags, excepting SCRIPT and STYLE elements (and some others),
|
||
* whose inner text is not decoded. Use this function to read the decoded
|
||
* value of such a text span in an HTML document.
|
||
*
|
||
* Example:
|
||
*
|
||
* '“😄”' === WP_HTML_Decode::decode_text_node( '“😄”' );
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @param string $text Text containing raw and non-decoded text node to decode.
|
||
* @return string Decoded UTF-8 value of given text node.
|
||
*/
|
||
public static function decode_text_node( $text ) {
|
||
return static::decode( 'data', $text );
|
||
}
|
||
|
||
/**
|
||
* Returns a string containing the decoded value of a given HTML attribute.
|
||
*
|
||
* Text found inside an HTML attribute has different parsing rules than for
|
||
* text found inside other markup, or DATA segments. Use this function to
|
||
* read the decoded value of an HTML string inside a quoted attribute.
|
||
*
|
||
* Example:
|
||
*
|
||
* '“😄”' === WP_HTML_Decode::decode_attribute( '“😄”' );
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @param string $text Text containing raw and non-decoded attribute value to decode.
|
||
* @return string Decoded UTF-8 value of given attribute value.
|
||
*/
|
||
public static function decode_attribute( $text ) {
|
||
return static::decode( 'attribute', $text );
|
||
}
|
||
|
||
/**
|
||
* Decodes a span of HTML text, depending on the context in which it's found.
|
||
*
|
||
* This is a low-level method; prefer calling WP_HTML_Decoder::decode_attribute() or
|
||
* WP_HTML_Decoder::decode_text_node() instead. It's provided for cases where this
|
||
* may be difficult to do from calling code.
|
||
*
|
||
* Example:
|
||
*
|
||
* '©' = WP_HTML_Decoder::decode( 'data', '©' );
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @access private
|
||
*
|
||
* @param string $context `attribute` for decoding attribute values, `data` otherwise.
|
||
* @param string $text Text document containing span of text to decode.
|
||
* @return string Decoded UTF-8 string.
|
||
*/
|
||
public static function decode( $context, $text ) {
|
||
$decoded = '';
|
||
$end = strlen( $text );
|
||
$at = 0;
|
||
$was_at = 0;
|
||
|
||
while ( $at < $end ) {
|
||
$next_character_reference_at = strpos( $text, '&', $at );
|
||
if ( false === $next_character_reference_at || $next_character_reference_at >= $end ) {
|
||
break;
|
||
}
|
||
|
||
$character_reference = self::read_character_reference( $context, $text, $next_character_reference_at, $token_length );
|
||
if ( isset( $character_reference ) ) {
|
||
$at = $next_character_reference_at;
|
||
$decoded .= substr( $text, $was_at, $at - $was_at );
|
||
$decoded .= $character_reference;
|
||
$at += $token_length;
|
||
$was_at = $at;
|
||
continue;
|
||
}
|
||
|
||
++$at;
|
||
}
|
||
|
||
if ( 0 === $was_at ) {
|
||
return $text;
|
||
}
|
||
|
||
if ( $was_at < $end ) {
|
||
$decoded .= substr( $text, $was_at, $end - $was_at );
|
||
}
|
||
|
||
return $decoded;
|
||
}
|
||
|
||
/**
|
||
* Attempt to read a character reference at the given location in a given string,
|
||
* depending on the context in which it's found.
|
||
*
|
||
* If a character reference is found, this function will return the translated value
|
||
* that the reference maps to. It will then set `$match_byte_length` the
|
||
* number of bytes of input it read while consuming the character reference. This
|
||
* gives calling code the opportunity to advance its cursor when traversing a string
|
||
* and decoding.
|
||
*
|
||
* Example:
|
||
*
|
||
* null === WP_HTML_Decoder::read_character_reference( 'attribute', 'Ships…', 0 );
|
||
* '…' === WP_HTML_Decoder::read_character_reference( 'attribute', 'Ships…', 5, $token_length );
|
||
* 8 === $token_length; // `…`
|
||
*
|
||
* null === WP_HTML_Decoder::read_character_reference( 'attribute', '¬in', 0 );
|
||
* '∉' === WP_HTML_Decoder::read_character_reference( 'attribute', '∉', 0, $token_length );
|
||
* 7 === $token_length; // `∉`
|
||
*
|
||
* '¬' === WP_HTML_Decoder::read_character_reference( 'data', '¬in', 0, $token_length );
|
||
* 4 === $token_length; // `¬`
|
||
* '∉' === WP_HTML_Decoder::read_character_reference( 'data', '∉', 0, $token_length );
|
||
* 7 === $token_length; // `∉`
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @param string $context `attribute` for decoding attribute values, `data` otherwise.
|
||
* @param string $text Text document containing span of text to decode.
|
||
* @param int $at Optional. Byte offset into text where span begins, defaults to the beginning (0).
|
||
* @param int &$match_byte_length Optional. Set to byte-length of character reference if provided and if a match
|
||
* is found, otherwise not set. Default null.
|
||
* @return string|false Decoded character reference in UTF-8 if found, otherwise `false`.
|
||
*/
|
||
public static function read_character_reference( $context, $text, $at = 0, &$match_byte_length = null ) {
|
||
/**
|
||
* Mappings for HTML5 named character references.
|
||
*
|
||
* @var WP_Token_Map $html5_named_character_references
|
||
*/
|
||
global $html5_named_character_references;
|
||
|
||
$length = strlen( $text );
|
||
if ( $at + 1 >= $length ) {
|
||
return null;
|
||
}
|
||
|
||
if ( '&' !== $text[ $at ] ) {
|
||
return null;
|
||
}
|
||
|
||
/*
|
||
* Numeric character references.
|
||
*
|
||
* When truncated, these will encode the code point found by parsing the
|
||
* digits that are available. For example, when `🅰` is truncated
|
||
* to `DZ` it will encode `DZ`. It does not:
|
||
* - know how to parse the original `🅰`.
|
||
* - fail to parse and return plaintext `DZ`.
|
||
* - fail to parse and return the replacement character `<60>`
|
||
*/
|
||
if ( '#' === $text[ $at + 1 ] ) {
|
||
if ( $at + 2 >= $length ) {
|
||
return null;
|
||
}
|
||
|
||
/** Tracks inner parsing within the numeric character reference. */
|
||
$digits_at = $at + 2;
|
||
|
||
if ( 'x' === $text[ $digits_at ] || 'X' === $text[ $digits_at ] ) {
|
||
$numeric_base = 16;
|
||
$numeric_digits = '0123456789abcdefABCDEF';
|
||
$max_digits = 6; // 
|
||
++$digits_at;
|
||
} else {
|
||
$numeric_base = 10;
|
||
$numeric_digits = '0123456789';
|
||
$max_digits = 7; // 
|
||
}
|
||
|
||
// Cannot encode invalid Unicode code points. Max is to U+10FFFF.
|
||
$zero_count = strspn( $text, '0', $digits_at );
|
||
$digit_count = strspn( $text, $numeric_digits, $digits_at + $zero_count );
|
||
$after_digits = $digits_at + $zero_count + $digit_count;
|
||
$has_semicolon = $after_digits < $length && ';' === $text[ $after_digits ];
|
||
$end_of_span = $has_semicolon ? $after_digits + 1 : $after_digits;
|
||
|
||
// `&#` or `&#x` without digits returns into plaintext.
|
||
if ( 0 === $digit_count && 0 === $zero_count ) {
|
||
return null;
|
||
}
|
||
|
||
// Whereas `&#` and only zeros is invalid.
|
||
if ( 0 === $digit_count ) {
|
||
$match_byte_length = $end_of_span - $at;
|
||
return '<27>';
|
||
}
|
||
|
||
// If there are too many digits then it's not worth parsing. It's invalid.
|
||
if ( $digit_count > $max_digits ) {
|
||
$match_byte_length = $end_of_span - $at;
|
||
return '<27>';
|
||
}
|
||
|
||
$digits = substr( $text, $digits_at + $zero_count, $digit_count );
|
||
$code_point = intval( $digits, $numeric_base );
|
||
|
||
/*
|
||
* Noncharacters, 0x0D, and non-ASCII-whitespace control characters.
|
||
*
|
||
* > A noncharacter is a code point that is in the range U+FDD0 to U+FDEF,
|
||
* > inclusive, or U+FFFE, U+FFFF, U+1FFFE, U+1FFFF, U+2FFFE, U+2FFFF,
|
||
* > U+3FFFE, U+3FFFF, U+4FFFE, U+4FFFF, U+5FFFE, U+5FFFF, U+6FFFE,
|
||
* > U+6FFFF, U+7FFFE, U+7FFFF, U+8FFFE, U+8FFFF, U+9FFFE, U+9FFFF,
|
||
* > U+AFFFE, U+AFFFF, U+BFFFE, U+BFFFF, U+CFFFE, U+CFFFF, U+DFFFE,
|
||
* > U+DFFFF, U+EFFFE, U+EFFFF, U+FFFFE, U+FFFFF, U+10FFFE, or U+10FFFF.
|
||
*
|
||
* A C0 control is a code point that is in the range of U+00 to U+1F,
|
||
* but ASCII whitespace includes U+09, U+0A, U+0C, and U+0D.
|
||
*
|
||
* These characters are invalid but still decode as any valid character.
|
||
* This comment is here to note and explain why there's no check to
|
||
* remove these characters or replace them.
|
||
*
|
||
* @see https://infra.spec.whatwg.org/#noncharacter
|
||
*/
|
||
|
||
/*
|
||
* Code points in the C1 controls area need to be remapped as if they
|
||
* were stored in Windows-1252. Note! This transformation only happens
|
||
* for numeric character references. The raw code points in the byte
|
||
* stream are not translated.
|
||
*
|
||
* > If the number is one of the numbers in the first column of
|
||
* > the following table, then find the row with that number in
|
||
* > the first column, and set the character reference code to
|
||
* > the number in the second column of that row.
|
||
*/
|
||
if ( $code_point >= 0x80 && $code_point <= 0x9F ) {
|
||
$windows_1252_mapping = array(
|
||
0x20AC, // 0x80 -> EURO SIGN (€).
|
||
0x81, // 0x81 -> (no change).
|
||
0x201A, // 0x82 -> SINGLE LOW-9 QUOTATION MARK (‚).
|
||
0x0192, // 0x83 -> LATIN SMALL LETTER F WITH HOOK (ƒ).
|
||
0x201E, // 0x84 -> DOUBLE LOW-9 QUOTATION MARK („).
|
||
0x2026, // 0x85 -> HORIZONTAL ELLIPSIS (…).
|
||
0x2020, // 0x86 -> DAGGER (†).
|
||
0x2021, // 0x87 -> DOUBLE DAGGER (‡).
|
||
0x02C6, // 0x88 -> MODIFIER LETTER CIRCUMFLEX ACCENT (ˆ).
|
||
0x2030, // 0x89 -> PER MILLE SIGN (‰).
|
||
0x0160, // 0x8A -> LATIN CAPITAL LETTER S WITH CARON (Š).
|
||
0x2039, // 0x8B -> SINGLE LEFT-POINTING ANGLE QUOTATION MARK (‹).
|
||
0x0152, // 0x8C -> LATIN CAPITAL LIGATURE OE (Œ).
|
||
0x8D, // 0x8D -> (no change).
|
||
0x017D, // 0x8E -> LATIN CAPITAL LETTER Z WITH CARON (Ž).
|
||
0x8F, // 0x8F -> (no change).
|
||
0x90, // 0x90 -> (no change).
|
||
0x2018, // 0x91 -> LEFT SINGLE QUOTATION MARK (‘).
|
||
0x2019, // 0x92 -> RIGHT SINGLE QUOTATION MARK (’).
|
||
0x201C, // 0x93 -> LEFT DOUBLE QUOTATION MARK (“).
|
||
0x201D, // 0x94 -> RIGHT DOUBLE QUOTATION MARK (”).
|
||
0x2022, // 0x95 -> BULLET (•).
|
||
0x2013, // 0x96 -> EN DASH (–).
|
||
0x2014, // 0x97 -> EM DASH (—).
|
||
0x02DC, // 0x98 -> SMALL TILDE (˜).
|
||
0x2122, // 0x99 -> TRADE MARK SIGN (™).
|
||
0x0161, // 0x9A -> LATIN SMALL LETTER S WITH CARON (š).
|
||
0x203A, // 0x9B -> SINGLE RIGHT-POINTING ANGLE QUOTATION MARK (›).
|
||
0x0153, // 0x9C -> LATIN SMALL LIGATURE OE (œ).
|
||
0x9D, // 0x9D -> (no change).
|
||
0x017E, // 0x9E -> LATIN SMALL LETTER Z WITH CARON (ž).
|
||
0x0178, // 0x9F -> LATIN CAPITAL LETTER Y WITH DIAERESIS (Ÿ).
|
||
);
|
||
|
||
$code_point = $windows_1252_mapping[ $code_point - 0x80 ];
|
||
}
|
||
|
||
$match_byte_length = $end_of_span - $at;
|
||
return self::code_point_to_utf8_bytes( $code_point );
|
||
}
|
||
|
||
/** Tracks inner parsing within the named character reference. */
|
||
$name_at = $at + 1;
|
||
// Minimum named character reference is two characters. E.g. `GT`.
|
||
if ( $name_at + 2 > $length ) {
|
||
return null;
|
||
}
|
||
|
||
$name_length = 0;
|
||
$replacement = $html5_named_character_references->read_token( $text, $name_at, $name_length );
|
||
if ( false === $replacement ) {
|
||
return null;
|
||
}
|
||
|
||
$after_name = $name_at + $name_length;
|
||
|
||
// If the match ended with a semicolon then it should always be decoded.
|
||
if ( ';' === $text[ $name_at + $name_length - 1 ] ) {
|
||
$match_byte_length = $after_name - $at;
|
||
return $replacement;
|
||
}
|
||
|
||
/*
|
||
* At this point though there's a match for an entry in the named
|
||
* character reference table but the match doesn't end in `;`.
|
||
* It may be allowed if it's followed by something unambiguous.
|
||
*/
|
||
$ambiguous_follower = (
|
||
$after_name < $length &&
|
||
$name_at < $length &&
|
||
(
|
||
ctype_alnum( $text[ $after_name ] ) ||
|
||
'=' === $text[ $after_name ]
|
||
)
|
||
);
|
||
|
||
// It's non-ambiguous, safe to leave it in.
|
||
if ( ! $ambiguous_follower ) {
|
||
$match_byte_length = $after_name - $at;
|
||
return $replacement;
|
||
}
|
||
|
||
// It's ambiguous, which isn't allowed inside attributes.
|
||
if ( 'attribute' === $context ) {
|
||
return null;
|
||
}
|
||
|
||
$match_byte_length = $after_name - $at;
|
||
return $replacement;
|
||
}
|
||
|
||
/**
|
||
* Encode a code point number into the UTF-8 encoding.
|
||
*
|
||
* This encoder implements the UTF-8 encoding algorithm for converting
|
||
* a code point into a byte sequence. If it receives an invalid code
|
||
* point it will return the Unicode Replacement Character U+FFFD `<60>`.
|
||
*
|
||
* Example:
|
||
*
|
||
* '🅰' === WP_HTML_Decoder::code_point_to_utf8_bytes( 0x1f170 );
|
||
*
|
||
* // Half of a surrogate pair is an invalid code point.
|
||
* '<27>' === WP_HTML_Decoder::code_point_to_utf8_bytes( 0xd83c );
|
||
*
|
||
* @since 6.6.0
|
||
*
|
||
* @see https://www.rfc-editor.org/rfc/rfc3629 For the UTF-8 standard.
|
||
*
|
||
* @param int $code_point Which code point to convert.
|
||
* @return string Converted code point, or `<60>` if invalid.
|
||
*/
|
||
public static function code_point_to_utf8_bytes( $code_point ) {
|
||
// Pre-check to ensure a valid code point.
|
||
if (
|
||
$code_point <= 0 ||
|
||
( $code_point >= 0xD800 && $code_point <= 0xDFFF ) ||
|
||
$code_point > 0x10FFFF
|
||
) {
|
||
return '<27>';
|
||
}
|
||
|
||
if ( $code_point <= 0x7F ) {
|
||
return chr( $code_point );
|
||
}
|
||
|
||
if ( $code_point <= 0x7FF ) {
|
||
$byte1 = ( $code_point >> 6 ) | 0xC0;
|
||
$byte2 = $code_point & 0x3F | 0x80;
|
||
|
||
return pack( 'CC', $byte1, $byte2 );
|
||
}
|
||
|
||
if ( $code_point <= 0xFFFF ) {
|
||
$byte1 = ( $code_point >> 12 ) | 0xE0;
|
||
$byte2 = ( $code_point >> 6 ) & 0x3F | 0x80;
|
||
$byte3 = $code_point & 0x3F | 0x80;
|
||
|
||
return pack( 'CCC', $byte1, $byte2, $byte3 );
|
||
}
|
||
|
||
// Any values above U+10FFFF are eliminated above in the pre-check.
|
||
$byte1 = ( $code_point >> 18 ) | 0xF0;
|
||
$byte2 = ( $code_point >> 12 ) & 0x3F | 0x80;
|
||
$byte3 = ( $code_point >> 6 ) & 0x3F | 0x80;
|
||
$byte4 = $code_point & 0x3F | 0x80;
|
||
|
||
return pack( 'CCCC', $byte1, $byte2, $byte3, $byte4 );
|
||
}
|
||
}
|