mirror of
https://github.com/ezyang/htmlpurifier.git
synced 2025-01-05 06:01:52 +00:00
fac747bdbd
With minor corrections. Signed-off-by: Marcus Bointon <marcus@synchromedia.co.uk> Signed-off-by: Edward Z. Yang <ezyang@mit.edu>
320 lines
8.9 KiB
PHP
320 lines
8.9 KiB
PHP
<?php
|
|
|
|
class HTMLPurifier_GeneratorTest extends HTMLPurifier_Harness
|
|
{
|
|
|
|
/**
|
|
* Entity lookup table to help for a few tests.
|
|
*/
|
|
private $_entity_lookup;
|
|
|
|
public function __construct()
|
|
{
|
|
parent::__construct();
|
|
$this->_entity_lookup = HTMLPurifier_EntityLookup::instance();
|
|
}
|
|
|
|
public function setUp()
|
|
{
|
|
parent::setUp();
|
|
$this->config->set('Output.Newline', "\n");
|
|
}
|
|
|
|
/**
|
|
* Creates a generator based on config and context member variables.
|
|
*/
|
|
protected function createGenerator()
|
|
{
|
|
return new HTMLPurifier_Generator($this->config, $this->context);
|
|
}
|
|
|
|
protected function assertGenerateFromToken($token, $html)
|
|
{
|
|
$generator = $this->createGenerator();
|
|
$result = $generator->generateFromToken($token);
|
|
$this->assertIdentical($result, $html);
|
|
}
|
|
|
|
public function test_generateFromToken_text()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Text('Foobar.<>'),
|
|
'Foobar.<>'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_startWithAttr()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Start('a',
|
|
array('href' => 'dyn?a=foo&b=bar')
|
|
),
|
|
'<a href="dyn?a=foo&b=bar">'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_end()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_End('b'),
|
|
'</b>'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_emptyWithAttr()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Empty('br',
|
|
array('style' => 'font-family:"Courier New";')
|
|
),
|
|
'<br style="font-family:"Courier New";" />'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_startNoAttr()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Start('asdf'),
|
|
'<asdf>'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_emptyNoAttr()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Empty('br'),
|
|
'<br />'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_error()
|
|
{
|
|
$this->expectError('Cannot generate HTML from non-HTMLPurifier_Token object');
|
|
$this->assertGenerateFromToken( null, '' );
|
|
}
|
|
|
|
public function test_generateFromToken_unicode()
|
|
{
|
|
$theta_char = $this->_entity_lookup->table['theta'];
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Text($theta_char),
|
|
$theta_char
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_backtick()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Start('img', array('alt' => '`foo')),
|
|
'<img alt="`foo ">'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_backtickDisabled()
|
|
{
|
|
$this->config->set('Output.FixInnerHTML', false);
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Start('img', array('alt' => '`')),
|
|
'<img alt="`">'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromToken_backtickNoChange()
|
|
{
|
|
$this->assertGenerateFromToken(
|
|
new HTMLPurifier_Token_Start('img', array('alt' => '`foo` bar')),
|
|
'<img alt="`foo` bar">'
|
|
);
|
|
}
|
|
|
|
public function assertGenerateAttributes($attr, $expect, $element = false)
|
|
{
|
|
$generator = $this->createGenerator();
|
|
$result = $generator->generateAttributes($attr, $element);
|
|
$this->assertIdentical($result, $expect);
|
|
}
|
|
|
|
public function test_generateAttributes_blank()
|
|
{
|
|
$this->assertGenerateAttributes(array(), '');
|
|
}
|
|
|
|
public function test_generateAttributes_basic()
|
|
{
|
|
$this->assertGenerateAttributes(
|
|
array('href' => 'dyn?a=foo&b=bar'),
|
|
'href="dyn?a=foo&b=bar"'
|
|
);
|
|
}
|
|
|
|
public function test_generateAttributes_doubleQuote()
|
|
{
|
|
$this->assertGenerateAttributes(
|
|
array('style' => 'font-family:"Courier New";'),
|
|
'style="font-family:"Courier New";"'
|
|
);
|
|
}
|
|
|
|
public function test_generateAttributes_singleQuote()
|
|
{
|
|
$this->assertGenerateAttributes(
|
|
array('style' => 'font-family:\'Courier New\';'),
|
|
'style="font-family:\'Courier New\';"'
|
|
);
|
|
}
|
|
|
|
public function test_generateAttributes_multiple()
|
|
{
|
|
$this->assertGenerateAttributes(
|
|
array('src' => 'picture.jpg', 'alt' => 'Short & interesting'),
|
|
'src="picture.jpg" alt="Short & interesting"'
|
|
);
|
|
}
|
|
|
|
public function test_generateAttributes_specialChar()
|
|
{
|
|
$theta_char = $this->_entity_lookup->table['theta'];
|
|
$this->assertGenerateAttributes(
|
|
array('title' => 'Theta is ' . $theta_char),
|
|
'title="Theta is ' . $theta_char . '"'
|
|
);
|
|
}
|
|
|
|
|
|
public function test_generateAttributes_minimized()
|
|
{
|
|
$this->config->set('HTML.Doctype', 'HTML 4.01 Transitional');
|
|
$this->assertGenerateAttributes(
|
|
array('compact' => 'compact'), 'compact', 'menu'
|
|
);
|
|
}
|
|
|
|
public function test_generateFromTokens()
|
|
{
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('b'),
|
|
new HTMLPurifier_Token_Text('Foobar!'),
|
|
new HTMLPurifier_Token_End('b')
|
|
),
|
|
'<b>Foobar!</b>'
|
|
);
|
|
|
|
}
|
|
|
|
protected function assertGeneration($tokens, $expect)
|
|
{
|
|
$generator = new HTMLPurifier_Generator($this->config, $this->context);
|
|
$result = $generator->generateFromTokens($tokens);
|
|
$this->assertIdentical($expect, $result);
|
|
}
|
|
|
|
public function test_generateFromTokens_Scripting()
|
|
{
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('script'),
|
|
new HTMLPurifier_Token_Text('alert(3 < 5);'),
|
|
new HTMLPurifier_Token_End('script')
|
|
),
|
|
"<script><!--//--><![CDATA[//><!--\nalert(3 < 5);\n//--><!]]></script>"
|
|
);
|
|
}
|
|
|
|
public function test_generateFromTokens_Scripting_missingCloseTag()
|
|
{
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('script'),
|
|
new HTMLPurifier_Token_Text('alert(3 < 5);'),
|
|
),
|
|
"<script>alert(3 < 5);"
|
|
);
|
|
}
|
|
|
|
public function test_generateFromTokens_Scripting_doubleBlock()
|
|
{
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('script'),
|
|
new HTMLPurifier_Token_Text('alert(3 < 5);'),
|
|
new HTMLPurifier_Token_Text('foo();'),
|
|
new HTMLPurifier_Token_End('script')
|
|
),
|
|
"<script>alert(3 < 5);foo();</script>"
|
|
);
|
|
}
|
|
|
|
public function test_generateFromTokens_Scripting_disableWrapper()
|
|
{
|
|
$this->config->set('Output.CommentScriptContents', false);
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('script'),
|
|
new HTMLPurifier_Token_Text('alert(3 < 5);'),
|
|
new HTMLPurifier_Token_End('script')
|
|
),
|
|
"<script>alert(3 < 5);</script>"
|
|
);
|
|
}
|
|
|
|
public function test_generateFromTokens_XHTMLoff()
|
|
{
|
|
$this->config->set('HTML.XHTML', false);
|
|
|
|
// omit trailing slash
|
|
$this->assertGeneration(
|
|
array( new HTMLPurifier_Token_Empty('br') ),
|
|
'<br>'
|
|
);
|
|
|
|
// there should be a test for attribute minimization, but it is
|
|
// impossible for something like that to happen due to our current
|
|
// definitions! fix it later
|
|
|
|
// namespaced attributes must be dropped
|
|
$this->assertGeneration(
|
|
array( new HTMLPurifier_Token_Start('p', array('xml:lang'=>'fr')) ),
|
|
'<p>'
|
|
);
|
|
|
|
}
|
|
|
|
public function test_generateFromTokens_TidyFormat()
|
|
{
|
|
// abort test if tidy isn't loaded
|
|
if (!extension_loaded('tidy')) return;
|
|
|
|
// just don't test; Tidy is exploding on me.
|
|
return;
|
|
|
|
$this->config->set('Core.TidyFormat', true);
|
|
$this->config->set('Output.Newline', "\n");
|
|
|
|
// nice wrapping please
|
|
$this->assertGeneration(
|
|
array(
|
|
new HTMLPurifier_Token_Start('div'),
|
|
new HTMLPurifier_Token_Text('Text'),
|
|
new HTMLPurifier_Token_End('div')
|
|
),
|
|
"<div>\n Text\n</div>\n"
|
|
);
|
|
|
|
}
|
|
|
|
public function test_generateFromTokens_sortAttr()
|
|
{
|
|
$this->config->set('Output.SortAttr', true);
|
|
|
|
$this->assertGeneration(
|
|
array( new HTMLPurifier_Token_Start('p', array('b'=>'c', 'a'=>'d')) ),
|
|
'<p a="d" b="c">'
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// vim: et sw=4 sts=4
|