mirror of
https://github.com/ezyang/htmlpurifier.git
synced 2025-01-05 06:01:52 +00:00
Migrate AttrValidator to nested error format; modify generator logic in ErrorCollector.
AttrValidator's changes are fairly self-explanatory, but ErrorCollector's changes are worth a little discussion. ErrorCollector can use generators at various points during its flow control; there are two distinct generators that it should use: 1. The one used for the output, and 2. The one used for the error output. These will usually be the same, but in the odd case where they need to be different, getHTMLFormatted() will accept an alterate configuration object with an appropriate doctype. Signed-off-by: Edward Z. Yang <edwardzyang@thewritingpot.com>
This commit is contained in:
parent
c807ed5fe2
commit
d9e60350d3
@ -49,15 +49,23 @@ class HTMLPurifier_AttrValidator
|
|||||||
// do global transformations (pre)
|
// do global transformations (pre)
|
||||||
// nothing currently utilizes this
|
// nothing currently utilizes this
|
||||||
foreach ($definition->info_attr_transform_pre as $transform) {
|
foreach ($definition->info_attr_transform_pre as $transform) {
|
||||||
|
if ($e) $e->start();
|
||||||
$attr = $transform->transform($o = $attr, $config, $context);
|
$attr = $transform->transform($o = $attr, $config, $context);
|
||||||
if ($e && ($attr != $o)) $e->send(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
if ($e) {
|
||||||
|
if ($attr != $o) $e->end(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
||||||
|
else $e->end();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// do local transformations only applicable to this element (pre)
|
// do local transformations only applicable to this element (pre)
|
||||||
// ex. <p align="right"> to <p style="text-align:right;">
|
// ex. <p align="right"> to <p style="text-align:right;">
|
||||||
foreach ($definition->info[$token->name]->attr_transform_pre as $transform) {
|
foreach ($definition->info[$token->name]->attr_transform_pre as $transform) {
|
||||||
|
if ($e) $e->start();
|
||||||
$attr = $transform->transform($o = $attr, $config, $context);
|
$attr = $transform->transform($o = $attr, $config, $context);
|
||||||
if ($e && ($attr != $o)) $e->send(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
if ($e) {
|
||||||
|
if ($attr != $o) $e->end(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
||||||
|
else $e->end();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// create alias to this element's attribute definition array, see
|
// create alias to this element's attribute definition array, see
|
||||||
@ -72,6 +80,7 @@ class HTMLPurifier_AttrValidator
|
|||||||
// Watch out for name collisions: $key has previously been used
|
// Watch out for name collisions: $key has previously been used
|
||||||
foreach ($attr as $attr_key => $value) {
|
foreach ($attr as $attr_key => $value) {
|
||||||
|
|
||||||
|
if ($e) $e->start();
|
||||||
// call the definition
|
// call the definition
|
||||||
if ( isset($defs[$attr_key]) ) {
|
if ( isset($defs[$attr_key]) ) {
|
||||||
// there is a local definition defined
|
// there is a local definition defined
|
||||||
@ -103,7 +112,7 @@ class HTMLPurifier_AttrValidator
|
|||||||
if ($result === false || $result === null) {
|
if ($result === false || $result === null) {
|
||||||
// this is a generic error message that should replaced
|
// this is a generic error message that should replaced
|
||||||
// with more specific ones when possible
|
// with more specific ones when possible
|
||||||
if ($e) $e->send(E_ERROR, 'AttrValidator: Attribute removed');
|
if ($e) $e->end(E_ERROR, 'AttrValidator: Attribute removed');
|
||||||
|
|
||||||
// remove the attribute
|
// remove the attribute
|
||||||
unset($attr[$attr_key]);
|
unset($attr[$attr_key]);
|
||||||
@ -114,6 +123,9 @@ class HTMLPurifier_AttrValidator
|
|||||||
|
|
||||||
// simple substitution
|
// simple substitution
|
||||||
$attr[$attr_key] = $result;
|
$attr[$attr_key] = $result;
|
||||||
|
if ($e) $e->end();
|
||||||
|
} else {
|
||||||
|
if ($e) $e->end();
|
||||||
}
|
}
|
||||||
|
|
||||||
// we'd also want slightly more complicated substitution
|
// we'd also want slightly more complicated substitution
|
||||||
@ -129,14 +141,22 @@ class HTMLPurifier_AttrValidator
|
|||||||
|
|
||||||
// global (error reporting untested)
|
// global (error reporting untested)
|
||||||
foreach ($definition->info_attr_transform_post as $transform) {
|
foreach ($definition->info_attr_transform_post as $transform) {
|
||||||
|
if ($e) $e->start();
|
||||||
$attr = $transform->transform($o = $attr, $config, $context);
|
$attr = $transform->transform($o = $attr, $config, $context);
|
||||||
if ($e && ($attr != $o)) $e->send(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
if ($e) {
|
||||||
|
if ($attr != $o) $e->end(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
||||||
|
else $e->end();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// local (error reporting untested)
|
// local (error reporting untested)
|
||||||
foreach ($definition->info[$token->name]->attr_transform_post as $transform) {
|
foreach ($definition->info[$token->name]->attr_transform_post as $transform) {
|
||||||
|
if ($e) $e->start();
|
||||||
$attr = $transform->transform($o = $attr, $config, $context);
|
$attr = $transform->transform($o = $attr, $config, $context);
|
||||||
if ($e && ($attr != $o)) $e->send(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
if ($e) {
|
||||||
|
if ($attr != $o) $e->end(E_NOTICE, 'AttrValidator: Attributes transformed', $o, $attr);
|
||||||
|
else $e->end();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$token->attr = $attr;
|
$token->attr = $attr;
|
||||||
|
@ -25,7 +25,6 @@ class HTMLPurifier_ErrorCollector
|
|||||||
|
|
||||||
public function __construct($context) {
|
public function __construct($context) {
|
||||||
$this->locale =& $context->get('Locale');
|
$this->locale =& $context->get('Locale');
|
||||||
$this->generator =& $context->get('Generator');
|
|
||||||
$this->context = $context;
|
$this->context = $context;
|
||||||
$this->_current =& $this->_stacks[0];
|
$this->_current =& $this->_stacks[0];
|
||||||
$this->errors =& $this->_stacks[0];
|
$this->errors =& $this->_stacks[0];
|
||||||
@ -141,6 +140,7 @@ class HTMLPurifier_ErrorCollector
|
|||||||
public function getHTMLFormatted($config, $errors = null) {
|
public function getHTMLFormatted($config, $errors = null) {
|
||||||
$ret = array();
|
$ret = array();
|
||||||
|
|
||||||
|
$generator = new HTMLPurifier_Generator($config, $this->context);
|
||||||
if ($errors === null) $errors = $this->errors;
|
if ($errors === null) $errors = $this->errors;
|
||||||
|
|
||||||
// sort error array by line
|
// sort error array by line
|
||||||
@ -161,7 +161,7 @@ class HTMLPurifier_ErrorCollector
|
|||||||
list($line, $severity, $msg, $children) = $error;
|
list($line, $severity, $msg, $children) = $error;
|
||||||
$string = '';
|
$string = '';
|
||||||
$string .= '<strong>' . $this->locale->getErrorName($severity) . '</strong>: ';
|
$string .= '<strong>' . $this->locale->getErrorName($severity) . '</strong>: ';
|
||||||
$string .= $this->generator->escape($msg);
|
$string .= $generator->escape($msg);
|
||||||
if ($line) {
|
if ($line) {
|
||||||
// have javascript link generation that causes
|
// have javascript link generation that causes
|
||||||
// textarea to skip to the specified line
|
// textarea to skip to the specified line
|
||||||
|
@ -3,6 +3,15 @@
|
|||||||
class HTMLPurifier_AttrValidator_ErrorsTest extends HTMLPurifier_ErrorsHarness
|
class HTMLPurifier_AttrValidator_ErrorsTest extends HTMLPurifier_ErrorsHarness
|
||||||
{
|
{
|
||||||
|
|
||||||
|
public function setup() {
|
||||||
|
parent::setup();
|
||||||
|
$config = HTMLPurifier_Config::createDefault();
|
||||||
|
$this->language = HTMLPurifier_LanguageFactory::instance()->create($config, $this->context);
|
||||||
|
$this->context->register('Locale', $this->language);
|
||||||
|
$this->collector = new HTMLPurifier_ErrorCollector($this->context);
|
||||||
|
$this->context->register('Generator', new HTMLPurifier_Generator($config, $this->context));
|
||||||
|
}
|
||||||
|
|
||||||
protected function invoke($input) {
|
protected function invoke($input) {
|
||||||
$validator = new HTMLPurifier_AttrValidator();
|
$validator = new HTMLPurifier_AttrValidator();
|
||||||
$validator->validateToken($input, $this->config, $this->context);
|
$validator->validateToken($input, $this->config, $this->context);
|
||||||
@ -18,28 +27,40 @@ class HTMLPurifier_AttrValidator_ErrorsTest extends HTMLPurifier_ErrorsHarness
|
|||||||
$output = array('class' => 'value'); // must be valid
|
$output = array('class' => 'value'); // must be valid
|
||||||
$transform->setReturnValue('transform', $output, array($input, new AnythingExpectation(), new AnythingExpectation()));
|
$transform->setReturnValue('transform', $output, array($input, new AnythingExpectation(), new AnythingExpectation()));
|
||||||
$def->info_attr_transform_pre[] = $transform;
|
$def->info_attr_transform_pre[] = $transform;
|
||||||
$this->expectErrorCollection(E_NOTICE, 'AttrValidator: Attributes transformed', $input, $output);
|
|
||||||
$token = new HTMLPurifier_Token_Start('span', $input, 1);
|
$token = new HTMLPurifier_Token_Start('span', $input, 1);
|
||||||
$this->invoke($token);
|
$this->invoke($token);
|
||||||
|
|
||||||
|
$result = $this->collector->getRaw();
|
||||||
|
$expect = array(
|
||||||
|
array(1, E_NOTICE, 'Attributes on <span> transformed from original to class', array()),
|
||||||
|
);
|
||||||
|
$this->assertIdentical($result, $expect);
|
||||||
}
|
}
|
||||||
|
|
||||||
function testAttributesTransformedLocalPre() {
|
function testAttributesTransformedLocalPre() {
|
||||||
$this->config->set('HTML', 'TidyLevel', 'heavy');
|
$this->config->set('HTML', 'TidyLevel', 'heavy');
|
||||||
$input = array('align' => 'right');
|
$input = array('align' => 'right');
|
||||||
$output = array('style' => 'text-align:right;');
|
$output = array('style' => 'text-align:right;');
|
||||||
$this->expectErrorCollection(E_NOTICE, 'AttrValidator: Attributes transformed', $input, $output);
|
|
||||||
$token = new HTMLPurifier_Token_Start('p', $input, 1);
|
$token = new HTMLPurifier_Token_Start('p', $input, 1);
|
||||||
$this->invoke($token);
|
$this->invoke($token);
|
||||||
|
$result = $this->collector->getRaw();
|
||||||
|
$expect = array(
|
||||||
|
array(1, E_NOTICE, 'Attributes on <p> transformed from align to style', array()),
|
||||||
|
);
|
||||||
|
$this->assertIdentical($result, $expect);
|
||||||
}
|
}
|
||||||
|
|
||||||
// too lazy to check for global post and global pre
|
// too lazy to check for global post and global pre
|
||||||
|
|
||||||
function testAttributeRemoved() {
|
function testAttributeRemoved() {
|
||||||
$this->expectErrorCollection(E_ERROR, 'AttrValidator: Attribute removed');
|
|
||||||
$this->expectContext('CurrentAttr', 'foobar');
|
|
||||||
$token = new HTMLPurifier_Token_Start('p', array('foobar' => 'right'), 1);
|
$token = new HTMLPurifier_Token_Start('p', array('foobar' => 'right'), 1);
|
||||||
$this->expectContext('CurrentToken', $token);
|
|
||||||
$this->invoke($token);
|
$this->invoke($token);
|
||||||
|
$result = $this->collector->getRaw();
|
||||||
|
$expect = array(
|
||||||
|
array(1, E_ERROR, 'foobar attribute on <p> removed', array()),
|
||||||
|
);
|
||||||
|
$this->assertIdentical($result, $expect);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user