2008-05-15 09:15:33 +02:00
|
|
|
<?php
|
|
|
|
|
2016-08-19 00:51:35 +02:00
|
|
|
namespace SilverStripe\Dev;
|
|
|
|
|
2017-05-17 07:40:13 +02:00
|
|
|
use SilverStripe\Core\Injector\Injectable;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SimpleXMLElement;
|
|
|
|
use tidy;
|
|
|
|
use Exception;
|
|
|
|
|
2008-05-15 09:15:33 +02:00
|
|
|
/**
|
|
|
|
* CSSContentParser enables parsing & assertion running of HTML content via CSS selectors.
|
|
|
|
* It works by converting the content to XHTML using tidy, rewriting the CSS selectors as XPath queries, and executing
|
|
|
|
* those using SimpeXML.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2008-05-15 09:15:33 +02:00
|
|
|
* It was built to facilitate testing using PHPUnit and contains a number of assert methods that will throw PHPUnit
|
|
|
|
* assertion exception when applicable.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2014-08-21 03:56:05 +02:00
|
|
|
* Tries to use the PHP tidy extension (http://php.net/tidy),
|
2008-10-12 18:15:24 +02:00
|
|
|
* and falls back to the "tidy" CLI tool. If none of those exists,
|
|
|
|
* the string is parsed directly without sanitization.
|
2014-08-15 08:53:05 +02:00
|
|
|
*
|
2012-06-05 11:38:27 +02:00
|
|
|
* Caution: Doesn't fully support HTML elements like <header>
|
|
|
|
* due to them being declared illegal by the "tidy" preprocessing step.
|
2008-05-15 09:15:33 +02:00
|
|
|
*/
|
2017-05-17 07:40:13 +02:00
|
|
|
class CSSContentParser
|
2016-11-29 00:31:16 +01:00
|
|
|
{
|
2017-05-17 07:40:13 +02:00
|
|
|
use Injectable;
|
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
protected $simpleXML = null;
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
public function __construct($content)
|
|
|
|
{
|
|
|
|
if (extension_loaded('tidy')) {
|
|
|
|
// using the tidy php extension
|
|
|
|
$tidy = new tidy();
|
|
|
|
$tidy->parseString(
|
|
|
|
$content,
|
|
|
|
array(
|
|
|
|
'output-xhtml' => true,
|
|
|
|
'numeric-entities' => true,
|
|
|
|
'wrap' => 0, // We need this to be consistent for functional test string comparisons
|
|
|
|
),
|
|
|
|
'utf8'
|
|
|
|
);
|
|
|
|
$tidy->cleanRepair();
|
|
|
|
$tidy = str_replace('xmlns="http://www.w3.org/1999/xhtml"', '', $tidy);
|
|
|
|
$tidy = str_replace(' ', '', $tidy);
|
|
|
|
} elseif (@shell_exec('which tidy')) {
|
|
|
|
// using tiny through cli
|
|
|
|
$CLI_content = escapeshellarg($content);
|
|
|
|
$tidy = `echo $CLI_content | tidy --force-output 1 -n -q -utf8 -asxhtml -w 0 2> /dev/null`;
|
|
|
|
$tidy = str_replace('xmlns="http://www.w3.org/1999/xhtml"', '', $tidy);
|
|
|
|
$tidy = str_replace(' ', '', $tidy);
|
|
|
|
} else {
|
|
|
|
// no tidy library found, hence no sanitizing
|
|
|
|
$tidy = $content;
|
|
|
|
}
|
2010-10-13 05:57:07 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
$this->simpleXML = @simplexml_load_string($tidy, 'SimpleXMLElement', LIBXML_NOWARNING);
|
|
|
|
if (!$this->simpleXML) {
|
|
|
|
throw new Exception('CSSContentParser::__construct(): Could not parse content.'
|
|
|
|
. ' Please check the PHP extension tidy is installed.');
|
|
|
|
}
|
|
|
|
}
|
2014-08-15 08:53:05 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Returns a number of SimpleXML elements that match the given CSS selector.
|
|
|
|
* Currently the selector engine only supports querying by tag, id, and class.
|
|
|
|
* See {@link getByXpath()} for a more direct selector syntax.
|
|
|
|
*
|
|
|
|
* @param String $selector
|
2017-06-22 12:50:45 +02:00
|
|
|
* @return SimpleXMLElement[]
|
2016-11-29 00:31:16 +01:00
|
|
|
*/
|
|
|
|
public function getBySelector($selector)
|
|
|
|
{
|
|
|
|
$xpath = $this->selector2xpath($selector);
|
|
|
|
return $this->getByXpath($xpath);
|
|
|
|
}
|
2008-05-15 09:15:33 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Allows querying the content through XPATH selectors.
|
|
|
|
*
|
|
|
|
* @param String $xpath SimpleXML compatible XPATH statement
|
2017-06-22 12:50:45 +02:00
|
|
|
* @return SimpleXMLElement[]
|
2016-11-29 00:31:16 +01:00
|
|
|
*/
|
|
|
|
public function getByXpath($xpath)
|
|
|
|
{
|
|
|
|
return $this->simpleXML->xpath($xpath);
|
|
|
|
}
|
2010-10-15 03:26:24 +02:00
|
|
|
|
2016-11-29 00:31:16 +01:00
|
|
|
/**
|
|
|
|
* Converts a CSS selector into an equivalent xpath expression.
|
|
|
|
* Currently the selector engine only supports querying by tag, id, and class.
|
|
|
|
*
|
|
|
|
* @param String $selector See {@link getBySelector()}
|
|
|
|
* @return String XPath expression
|
|
|
|
*/
|
|
|
|
public function selector2xpath($selector)
|
|
|
|
{
|
|
|
|
$parts = preg_split('/\\s+/', $selector);
|
|
|
|
$xpath = "";
|
|
|
|
foreach ($parts as $part) {
|
|
|
|
if (preg_match('/^([A-Za-z][A-Za-z0-9]*)/', $part, $matches)) {
|
|
|
|
$xpath .= "//$matches[1]";
|
|
|
|
} else {
|
|
|
|
$xpath .= "//*";
|
|
|
|
}
|
|
|
|
$xfilters = array();
|
|
|
|
if (preg_match('/#([^#.\[]+)/', $part, $matches)) {
|
|
|
|
$xfilters[] = "@id='$matches[1]'";
|
|
|
|
}
|
|
|
|
if (preg_match('/\.([^#.\[]+)/', $part, $matches)) {
|
|
|
|
$xfilters[] = "contains(@class,'$matches[1]')";
|
|
|
|
}
|
|
|
|
if ($xfilters) {
|
|
|
|
$xpath .= '[' . implode(',', $xfilters) . ']';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $xpath;
|
|
|
|
}
|
2012-03-24 04:04:52 +01:00
|
|
|
}
|