mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
FIX ShortcodeParser producing bad output after escaped tag
Also tightens up matching of shortcodes so we dont match on invalid shortcodes
This commit is contained in:
parent
8a70019e78
commit
1ee01c39d4
@ -145,14 +145,7 @@ class ShortcodeParser {
|
|||||||
'pre', 'section', 'table', 'ul'
|
'pre', 'section', 'table', 'ul'
|
||||||
);
|
);
|
||||||
|
|
||||||
private static $tagrx = '/
|
private static $attrrx = '
|
||||||
<(?<element>(?:"[^"]*"[\'"]*|\'[^\']*\'[\'"]*|[^\'">])+)> | # HTML Tag - skip attribute scoped tags
|
|
||||||
\[ (?<escaped>\[.*?\]) \] | # Escaped block
|
|
||||||
\[ (?<open>\w+) (?<attrs>.*?) (?<selfclosed>\/?) \] | # Opening tag
|
|
||||||
\[\/ (?<close>\w+) \] # Closing tag
|
|
||||||
/x';
|
|
||||||
|
|
||||||
private static $attrrx = '/
|
|
||||||
([^\s\/\'"=,]+) # Name
|
([^\s\/\'"=,]+) # Name
|
||||||
\s* = \s*
|
\s* = \s*
|
||||||
(?:
|
(?:
|
||||||
@ -160,8 +153,35 @@ class ShortcodeParser {
|
|||||||
(?:"([^"]+)") | # Value surrounded by "
|
(?:"([^"]+)") | # Value surrounded by "
|
||||||
(\w+) # Bare value
|
(\w+) # Bare value
|
||||||
)
|
)
|
||||||
/x';
|
';
|
||||||
|
|
||||||
|
private static function attrrx() {
|
||||||
|
return '/'.self::$attrrx.'/xS';
|
||||||
|
}
|
||||||
|
|
||||||
|
private static $tagrx = '
|
||||||
|
# HTML Tag
|
||||||
|
<(?<element>(?:"[^"]*"[\'"]*|\'[^\']*\'[\'"]*|[^\'">])+)>
|
||||||
|
|
||||||
|
| # Opening tag
|
||||||
|
(?<oesc>\[?)
|
||||||
|
\[
|
||||||
|
(?<open>\w+)
|
||||||
|
[\s,]*
|
||||||
|
(?<attrs> (?: %s [\s,]*)* )
|
||||||
|
\/?\]
|
||||||
|
(?<cesc1>\]?)
|
||||||
|
|
||||||
|
| # Closing tag
|
||||||
|
\[\/
|
||||||
|
(?<close>\w+)
|
||||||
|
\]
|
||||||
|
(?<cesc2>\]?)
|
||||||
|
';
|
||||||
|
|
||||||
|
private static function tagrx() {
|
||||||
|
return '/'.sprintf(self::$tagrx, self::$attrrx).'/xS';
|
||||||
|
}
|
||||||
|
|
||||||
const WARN = 'warn';
|
const WARN = 'warn';
|
||||||
const STRIP = 'strip';
|
const STRIP = 'strip';
|
||||||
@ -181,21 +201,11 @@ class ShortcodeParser {
|
|||||||
* @return array - The list of tags found. When using an open/close pair, only one item will be in the array,
|
* @return array - The list of tags found. When using an open/close pair, only one item will be in the array,
|
||||||
* with "content" set to the text between the tags
|
* with "content" set to the text between the tags
|
||||||
*/
|
*/
|
||||||
protected function extractTags(&$content) {
|
protected function extractTags($content) {
|
||||||
$tags = array();
|
$tags = array();
|
||||||
$escapes = array();
|
|
||||||
|
|
||||||
if(preg_match_all(self::$tagrx, $content, $matches, PREG_SET_ORDER | PREG_OFFSET_CAPTURE)) {
|
if(preg_match_all(self::tagrx(), $content, $matches, PREG_SET_ORDER | PREG_OFFSET_CAPTURE)) {
|
||||||
foreach($matches as $match) {
|
foreach($matches as $match) {
|
||||||
// Record escaped tags
|
|
||||||
if (!empty($match['escaped'][0])) {
|
|
||||||
$escapes[] = array(
|
|
||||||
's' => $match[0][1],
|
|
||||||
'e' => $match[0][1] + strlen($match[0][0]),
|
|
||||||
'content' => $match['escaped']
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ignore any elements
|
// Ignore any elements
|
||||||
if (empty($match['open'][0]) && empty($match['close'][0])) continue;
|
if (empty($match['open'][0]) && empty($match['close'][0])) continue;
|
||||||
|
|
||||||
@ -203,7 +213,7 @@ class ShortcodeParser {
|
|||||||
$attrs = array();
|
$attrs = array();
|
||||||
|
|
||||||
if (!empty($match['attrs'][0])) {
|
if (!empty($match['attrs'][0])) {
|
||||||
preg_match_all(self::$attrrx, $match['attrs'][0], $attrmatches, PREG_SET_ORDER);
|
preg_match_all(self::attrrx(), $match['attrs'][0], $attrmatches, PREG_SET_ORDER);
|
||||||
|
|
||||||
foreach ($attrmatches as $attr) {
|
foreach ($attrmatches as $attr) {
|
||||||
list($whole, $name, $value) = array_values(array_filter($attr));
|
list($whole, $name, $value) = array_values(array_filter($attr));
|
||||||
@ -219,14 +229,15 @@ class ShortcodeParser {
|
|||||||
'open' => @$match['open'][0],
|
'open' => @$match['open'][0],
|
||||||
'close' => @$match['close'][0],
|
'close' => @$match['close'][0],
|
||||||
'attrs' => $attrs,
|
'attrs' => $attrs,
|
||||||
'content' => ''
|
'content' => '',
|
||||||
|
'escaped' => !empty($match['oesc'][0]) || !empty($match['cesc1'][0]) || !empty($match['cesc2'][0])
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$i = count($tags);
|
$i = count($tags);
|
||||||
while($i--) {
|
while($i--) {
|
||||||
if($tags[$i]['close']) {
|
if(!empty($tags[$i]['close'])) {
|
||||||
// If the tag just before this one isn't the related opening tag, throw an error
|
// If the tag just before this one isn't the related opening tag, throw an error
|
||||||
$err = null;
|
$err = null;
|
||||||
|
|
||||||
@ -244,21 +255,29 @@ class ShortcodeParser {
|
|||||||
if(self::$error_behavior == self::ERROR) user_error($err, E_USER_ERRROR);
|
if(self::$error_behavior == self::ERROR) user_error($err, E_USER_ERRROR);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
if ($tags[$i]['escaped']) {
|
||||||
|
if (!$tags[$i-1]['escaped']) {
|
||||||
|
$tags[$i]['e'] -= 1;
|
||||||
|
$tags[$i]['escaped'] = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if ($tags[$i-1]['escaped']) {
|
||||||
|
$tags[$i-1]['s'] += 1;
|
||||||
|
$tags[$i-1]['escaped'] = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Otherwise, grab content between tags, save in opening tag & delete the closing one
|
// Otherwise, grab content between tags, save in opening tag & delete the closing one
|
||||||
$tags[$i-1]['text'] = substr($content, $tags[$i-1]['s'], $tags[$i]['e'] - $tags[$i-1]['s']);
|
$tags[$i-1]['text'] = substr($content, $tags[$i-1]['s'], $tags[$i]['e'] - $tags[$i-1]['s']);
|
||||||
$tags[$i-1]['content'] = substr($content, $tags[$i-1]['e'], $tags[$i]['s'] - $tags[$i-1]['e']);
|
$tags[$i-1]['content'] = substr($content, $tags[$i-1]['e'], $tags[$i]['s'] - $tags[$i-1]['e']);
|
||||||
$tags[$i-1]['e'] = $tags[$i]['e'];
|
$tags[$i-1]['e'] = $tags[$i]['e'];
|
||||||
|
|
||||||
unset($tags[$i]);
|
unset($tags[$i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$i = count($escapes);
|
|
||||||
while($i--) {
|
|
||||||
$escape = $escapes[$i];
|
|
||||||
$content = substr_replace($content, $escape['content'], $escape['s'], $escape['e'] - $escape['s']);
|
|
||||||
}
|
|
||||||
|
|
||||||
return array_values($tags);
|
return array_values($tags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -281,7 +300,13 @@ class ShortcodeParser {
|
|||||||
if ($li === null) $tail = substr($content, $tags[$i]['e']);
|
if ($li === null) $tail = substr($content, $tags[$i]['e']);
|
||||||
else $tail = substr($content, $tags[$i]['e'], $li - $tags[$i]['e']);
|
else $tail = substr($content, $tags[$i]['e'], $li - $tags[$i]['e']);
|
||||||
|
|
||||||
$str = $generator($i, $tags[$i]). $tail . $str;
|
if ($tags[$i]['escaped']) {
|
||||||
|
$str = substr($content, $tags[$i]['s']+1, $tags[$i]['e'] - $tags[$i]['s'] - 2) . $tail . $str;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$str = $generator($i, $tags[$i]) . $tail . $str;
|
||||||
|
}
|
||||||
|
|
||||||
$li = $tags[$i]['s'];
|
$li = $tags[$i]['s'];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,9 +100,40 @@ class ShortcodeParserTest extends SapphireTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public function testShortcodeEscaping() {
|
public function testShortcodeEscaping() {
|
||||||
$this->assertEquals('[test_shortcode]', $this->parser->parse('[[test_shortcode]]'));
|
$this->assertEquals(
|
||||||
$this->assertEquals('[test_shortcode]content[/test_shortcode]',
|
'[test_shortcode]',
|
||||||
$this->parser->parse('[[test_shortcode]content[/test_shortcode]]'));
|
$this->parser->parse('[[test_shortcode]]')
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[test_shortcode /]',
|
||||||
|
$this->parser->parse('[[test_shortcode /]]')
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[test_shortcode]content[/test_shortcode]',
|
||||||
|
$this->parser->parse('[[test_shortcode]content[/test_shortcode]]'
|
||||||
|
));
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[test_shortcode]content',
|
||||||
|
$this->parser->parse('[[test_shortcode]][test_shortcode]content[/test_shortcode]')
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[test_shortcode]content[/test_shortcode]content2',
|
||||||
|
$this->parser->parse('[[test_shortcode]content[/test_shortcode]][test_shortcode]content2[/test_shortcode]'
|
||||||
|
));
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[[Doesnt strip double [ character if not a shortcode',
|
||||||
|
$this->parser->parse('[[Doesnt strip double [ character if not a [test_shortcode]shortcode[/test_shortcode]'
|
||||||
|
));
|
||||||
|
|
||||||
|
$this->assertEquals(
|
||||||
|
'[[Doesnt shortcode get confused by double ]] characters',
|
||||||
|
$this->parser->parse('[[Doesnt [test_shortcode]shortcode[/test_shortcode] get confused by double ]] characters'
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testUnquotedArguments() {
|
public function testUnquotedArguments() {
|
||||||
|
Loading…
Reference in New Issue
Block a user