250 lines
13 KiB
PHP
250 lines
13 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Symfony\Component\CssSelector\Tests\Parser;
|
||
|
|
||
|
use PHPUnit\Framework\TestCase;
|
||
|
use Symfony\Component\CssSelector\Exception\SyntaxErrorException;
|
||
|
use Symfony\Component\CssSelector\Node\FunctionNode;
|
||
|
use Symfony\Component\CssSelector\Node\SelectorNode;
|
||
|
use Symfony\Component\CssSelector\Parser\Parser;
|
||
|
use Symfony\Component\CssSelector\Parser\Token;
|
||
|
|
||
|
class ParserTest extends TestCase
|
||
|
{
|
||
|
/** @dataProvider getParserTestData */
|
||
|
public function testParser($source, $representation)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
|
||
|
$this->assertEquals($representation, array_map(function (SelectorNode $node) {
|
||
|
return (string) $node->getTree();
|
||
|
}, $parser->parse($source)));
|
||
|
}
|
||
|
|
||
|
/** @dataProvider getParserExceptionTestData */
|
||
|
public function testParserException($source, $message)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
|
||
|
try {
|
||
|
$parser->parse($source);
|
||
|
$this->fail('Parser should throw a SyntaxErrorException.');
|
||
|
} catch (SyntaxErrorException $e) {
|
||
|
$this->assertEquals($message, $e->getMessage());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/** @dataProvider getPseudoElementsTestData */
|
||
|
public function testPseudoElements($source, $element, $pseudo)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
$selectors = $parser->parse($source);
|
||
|
$this->assertCount(1, $selectors);
|
||
|
|
||
|
/** @var SelectorNode $selector */
|
||
|
$selector = $selectors[0];
|
||
|
$this->assertEquals($element, (string) $selector->getTree());
|
||
|
$this->assertEquals($pseudo, (string) $selector->getPseudoElement());
|
||
|
}
|
||
|
|
||
|
/** @dataProvider getSpecificityTestData */
|
||
|
public function testSpecificity($source, $value)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
$selectors = $parser->parse($source);
|
||
|
$this->assertCount(1, $selectors);
|
||
|
|
||
|
/** @var SelectorNode $selector */
|
||
|
$selector = $selectors[0];
|
||
|
$this->assertEquals($value, $selector->getSpecificity()->getValue());
|
||
|
}
|
||
|
|
||
|
/** @dataProvider getParseSeriesTestData */
|
||
|
public function testParseSeries($series, $a, $b)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
$selectors = $parser->parse(sprintf(':nth-child(%s)', $series));
|
||
|
$this->assertCount(1, $selectors);
|
||
|
|
||
|
/** @var FunctionNode $function */
|
||
|
$function = $selectors[0]->getTree();
|
||
|
$this->assertEquals(array($a, $b), Parser::parseSeries($function->getArguments()));
|
||
|
}
|
||
|
|
||
|
/** @dataProvider getParseSeriesExceptionTestData */
|
||
|
public function testParseSeriesException($series)
|
||
|
{
|
||
|
$parser = new Parser();
|
||
|
$selectors = $parser->parse(sprintf(':nth-child(%s)', $series));
|
||
|
$this->assertCount(1, $selectors);
|
||
|
|
||
|
/** @var FunctionNode $function */
|
||
|
$function = $selectors[0]->getTree();
|
||
|
$this->{method_exists($this, $_ = 'expectException') ? $_ : 'setExpectedException'}('Symfony\Component\CssSelector\Exception\SyntaxErrorException');
|
||
|
Parser::parseSeries($function->getArguments());
|
||
|
}
|
||
|
|
||
|
public function getParserTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('*', array('Element[*]')),
|
||
|
array('*|*', array('Element[*]')),
|
||
|
array('*|foo', array('Element[foo]')),
|
||
|
array('foo|*', array('Element[foo|*]')),
|
||
|
array('foo|bar', array('Element[foo|bar]')),
|
||
|
array('#foo#bar', array('Hash[Hash[Element[*]#foo]#bar]')),
|
||
|
array('div>.foo', array('CombinedSelector[Element[div] > Class[Element[*].foo]]')),
|
||
|
array('div> .foo', array('CombinedSelector[Element[div] > Class[Element[*].foo]]')),
|
||
|
array('div >.foo', array('CombinedSelector[Element[div] > Class[Element[*].foo]]')),
|
||
|
array('div > .foo', array('CombinedSelector[Element[div] > Class[Element[*].foo]]')),
|
||
|
array("div \n> \t \t .foo", array('CombinedSelector[Element[div] > Class[Element[*].foo]]')),
|
||
|
array('td.foo,.bar', array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array('td.foo, .bar', array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array("td.foo\t\r\n\f ,\t\r\n\f .bar", array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array('td.foo,.bar', array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array('td.foo, .bar', array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array("td.foo\t\r\n\f ,\t\r\n\f .bar", array('Class[Element[td].foo]', 'Class[Element[*].bar]')),
|
||
|
array('div, td.foo, div.bar span', array('Element[div]', 'Class[Element[td].foo]', 'CombinedSelector[Class[Element[div].bar] <followed> Element[span]]')),
|
||
|
array('div > p', array('CombinedSelector[Element[div] > Element[p]]')),
|
||
|
array('td:first', array('Pseudo[Element[td]:first]')),
|
||
|
array('td :first', array('CombinedSelector[Element[td] <followed> Pseudo[Element[*]:first]]')),
|
||
|
array('a[name]', array('Attribute[Element[a][name]]')),
|
||
|
array("a[ name\t]", array('Attribute[Element[a][name]]')),
|
||
|
array('a [name]', array('CombinedSelector[Element[a] <followed> Attribute[Element[*][name]]]')),
|
||
|
array('a[rel="include"]', array("Attribute[Element[a][rel = 'include']]")),
|
||
|
array('a[rel = include]', array("Attribute[Element[a][rel = 'include']]")),
|
||
|
array("a[hreflang |= 'en']", array("Attribute[Element[a][hreflang |= 'en']]")),
|
||
|
array('a[hreflang|=en]', array("Attribute[Element[a][hreflang |= 'en']]")),
|
||
|
array('div:nth-child(10)', array("Function[Element[div]:nth-child(['10'])]")),
|
||
|
array(':nth-child(2n+2)', array("Function[Element[*]:nth-child(['2', 'n', '+2'])]")),
|
||
|
array('div:nth-of-type(10)', array("Function[Element[div]:nth-of-type(['10'])]")),
|
||
|
array('div div:nth-of-type(10) .aclass', array("CombinedSelector[CombinedSelector[Element[div] <followed> Function[Element[div]:nth-of-type(['10'])]] <followed> Class[Element[*].aclass]]")),
|
||
|
array('label:only', array('Pseudo[Element[label]:only]')),
|
||
|
array('a:lang(fr)', array("Function[Element[a]:lang(['fr'])]")),
|
||
|
array('div:contains("foo")', array("Function[Element[div]:contains(['foo'])]")),
|
||
|
array('div#foobar', array('Hash[Element[div]#foobar]')),
|
||
|
array('div:not(div.foo)', array('Negation[Element[div]:not(Class[Element[div].foo])]')),
|
||
|
array('td ~ th', array('CombinedSelector[Element[td] ~ Element[th]]')),
|
||
|
array('.foo[data-bar][data-baz=0]', array("Attribute[Attribute[Class[Element[*].foo][data-bar]][data-baz = '0']]")),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
public function getParserExceptionTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('attributes(href)/html/body/a', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '(', 10))->getMessage()),
|
||
|
array('attributes(href)', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '(', 10))->getMessage()),
|
||
|
array('html/body/a', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '/', 4))->getMessage()),
|
||
|
array(' ', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_FILE_END, '', 1))->getMessage()),
|
||
|
array('div, ', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_FILE_END, '', 5))->getMessage()),
|
||
|
array(' , div', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, ',', 1))->getMessage()),
|
||
|
array('p, , div', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, ',', 3))->getMessage()),
|
||
|
array('div > ', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_FILE_END, '', 6))->getMessage()),
|
||
|
array(' > div', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '>', 2))->getMessage()),
|
||
|
array('foo|#bar', SyntaxErrorException::unexpectedToken('identifier or "*"', new Token(Token::TYPE_HASH, 'bar', 4))->getMessage()),
|
||
|
array('#.foo', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '#', 0))->getMessage()),
|
||
|
array('.#foo', SyntaxErrorException::unexpectedToken('identifier', new Token(Token::TYPE_HASH, 'foo', 1))->getMessage()),
|
||
|
array(':#foo', SyntaxErrorException::unexpectedToken('identifier', new Token(Token::TYPE_HASH, 'foo', 1))->getMessage()),
|
||
|
array('[*]', SyntaxErrorException::unexpectedToken('"|"', new Token(Token::TYPE_DELIMITER, ']', 2))->getMessage()),
|
||
|
array('[foo|]', SyntaxErrorException::unexpectedToken('identifier', new Token(Token::TYPE_DELIMITER, ']', 5))->getMessage()),
|
||
|
array('[#]', SyntaxErrorException::unexpectedToken('identifier or "*"', new Token(Token::TYPE_DELIMITER, '#', 1))->getMessage()),
|
||
|
array('[foo=#]', SyntaxErrorException::unexpectedToken('string or identifier', new Token(Token::TYPE_DELIMITER, '#', 5))->getMessage()),
|
||
|
array(':nth-child()', SyntaxErrorException::unexpectedToken('at least one argument', new Token(Token::TYPE_DELIMITER, ')', 11))->getMessage()),
|
||
|
array('[href]a', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_IDENTIFIER, 'a', 6))->getMessage()),
|
||
|
array('[rel:stylesheet]', SyntaxErrorException::unexpectedToken('operator', new Token(Token::TYPE_DELIMITER, ':', 4))->getMessage()),
|
||
|
array('[rel=stylesheet', SyntaxErrorException::unexpectedToken('"]"', new Token(Token::TYPE_FILE_END, '', 15))->getMessage()),
|
||
|
array(':lang(fr', SyntaxErrorException::unexpectedToken('an argument', new Token(Token::TYPE_FILE_END, '', 8))->getMessage()),
|
||
|
array(':contains("foo', SyntaxErrorException::unclosedString(10)->getMessage()),
|
||
|
array('foo!', SyntaxErrorException::unexpectedToken('selector', new Token(Token::TYPE_DELIMITER, '!', 3))->getMessage()),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
public function getPseudoElementsTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('foo', 'Element[foo]', ''),
|
||
|
array('*', 'Element[*]', ''),
|
||
|
array(':empty', 'Pseudo[Element[*]:empty]', ''),
|
||
|
array(':BEfore', 'Element[*]', 'before'),
|
||
|
array(':aftER', 'Element[*]', 'after'),
|
||
|
array(':First-Line', 'Element[*]', 'first-line'),
|
||
|
array(':First-Letter', 'Element[*]', 'first-letter'),
|
||
|
array('::befoRE', 'Element[*]', 'before'),
|
||
|
array('::AFter', 'Element[*]', 'after'),
|
||
|
array('::firsT-linE', 'Element[*]', 'first-line'),
|
||
|
array('::firsT-letteR', 'Element[*]', 'first-letter'),
|
||
|
array('::Selection', 'Element[*]', 'selection'),
|
||
|
array('foo:after', 'Element[foo]', 'after'),
|
||
|
array('foo::selection', 'Element[foo]', 'selection'),
|
||
|
array('lorem#ipsum ~ a#b.c[href]:empty::selection', 'CombinedSelector[Hash[Element[lorem]#ipsum] ~ Pseudo[Attribute[Class[Hash[Element[a]#b].c][href]]:empty]]', 'selection'),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
public function getSpecificityTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('*', 0),
|
||
|
array(' foo', 1),
|
||
|
array(':empty ', 10),
|
||
|
array(':before', 1),
|
||
|
array('*:before', 1),
|
||
|
array(':nth-child(2)', 10),
|
||
|
array('.bar', 10),
|
||
|
array('[baz]', 10),
|
||
|
array('[baz="4"]', 10),
|
||
|
array('[baz^="4"]', 10),
|
||
|
array('#lipsum', 100),
|
||
|
array(':not(*)', 0),
|
||
|
array(':not(foo)', 1),
|
||
|
array(':not(.foo)', 10),
|
||
|
array(':not([foo])', 10),
|
||
|
array(':not(:empty)', 10),
|
||
|
array(':not(#foo)', 100),
|
||
|
array('foo:empty', 11),
|
||
|
array('foo:before', 2),
|
||
|
array('foo::before', 2),
|
||
|
array('foo:empty::before', 12),
|
||
|
array('#lorem + foo#ipsum:first-child > bar:first-line', 213),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
public function getParseSeriesTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('1n+3', 1, 3),
|
||
|
array('1n +3', 1, 3),
|
||
|
array('1n + 3', 1, 3),
|
||
|
array('1n+ 3', 1, 3),
|
||
|
array('1n-3', 1, -3),
|
||
|
array('1n -3', 1, -3),
|
||
|
array('1n - 3', 1, -3),
|
||
|
array('1n- 3', 1, -3),
|
||
|
array('n-5', 1, -5),
|
||
|
array('odd', 2, 1),
|
||
|
array('even', 2, 0),
|
||
|
array('3n', 3, 0),
|
||
|
array('n', 1, 0),
|
||
|
array('+n', 1, 0),
|
||
|
array('-n', -1, 0),
|
||
|
array('5', 0, 5),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
public function getParseSeriesExceptionTestData()
|
||
|
{
|
||
|
return array(
|
||
|
array('foo'),
|
||
|
array('n+'),
|
||
|
);
|
||
|
}
|
||
|
}
|