mirror of
https://github.com/ezyang/htmlpurifier.git
synced 2025-01-10 07:51:52 +00:00
eaabccdd9b
- Removed HTMLPurifier_Error - Documentation updates - Removed more copy() methods in favor of clone - HTMLPurifier::getInstance() to HTMLPurifier::instance() - Fix InterchangeBuilder to use HTMLPURIFIER_PREFIX git-svn-id: http://htmlpurifier.org/svnroot/htmlpurifier/trunk@1689 48356398-32a2-884e-a903-53898d9a118a
445 lines
19 KiB
PHP
445 lines
19 KiB
PHP
<?php
|
|
|
|
class HTMLPurifier_ConfigTest extends HTMLPurifier_Harness
|
|
{
|
|
|
|
protected $schema;
|
|
|
|
public function setUp() {
|
|
// set up a dummy schema object for testing
|
|
$this->schema = new HTMLPurifier_ConfigSchema();
|
|
}
|
|
|
|
// test functionality based on ConfigSchema
|
|
|
|
function testNormal() {
|
|
$this->schema->addNamespace('Element');
|
|
|
|
$this->schema->add('Element', 'Abbr', 'H', 'string', false);
|
|
$this->schema->add('Element', 'Name', 'hydrogen', 'istring', false);
|
|
$this->schema->add('Element', 'Number', 1, 'int', false);
|
|
$this->schema->add('Element', 'Mass', 1.00794, 'float', false);
|
|
$this->schema->add('Element', 'Radioactive', false, 'bool', false);
|
|
$this->schema->add('Element', 'Isotopes', array(1 => true, 2 => true, 3 => true), 'lookup', false);
|
|
$this->schema->add('Element', 'Traits', array('nonmetallic', 'odorless', 'flammable'), 'list', false);
|
|
$this->schema->add('Element', 'IsotopeNames', array(1 => 'protium', 2 => 'deuterium', 3 => 'tritium'), 'hash', false);
|
|
$this->schema->add('Element', 'Object', new stdClass(), 'mixed', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
// test default value retrieval
|
|
$this->assertIdentical($config->get('Element', 'Abbr'), 'H');
|
|
$this->assertIdentical($config->get('Element', 'Name'), 'hydrogen');
|
|
$this->assertIdentical($config->get('Element', 'Number'), 1);
|
|
$this->assertIdentical($config->get('Element', 'Mass'), 1.00794);
|
|
$this->assertIdentical($config->get('Element', 'Radioactive'), false);
|
|
$this->assertIdentical($config->get('Element', 'Isotopes'), array(1 => true, 2 => true, 3 => true));
|
|
$this->assertIdentical($config->get('Element', 'Traits'), array('nonmetallic', 'odorless', 'flammable'));
|
|
$this->assertIdentical($config->get('Element', 'IsotopeNames'), array(1 => 'protium', 2 => 'deuterium', 3 => 'tritium'));
|
|
$this->assertIdentical($config->get('Element', 'Object'), new stdClass());
|
|
|
|
// test setting values
|
|
$config->set('Element', 'Abbr', 'Pu');
|
|
$config->set('Element', 'Name', 'PLUTONIUM'); // test decaps
|
|
$config->set('Element', 'Number', '94'); // test parsing
|
|
$config->set('Element', 'Mass', '244.'); // test parsing
|
|
$config->set('Element', 'Radioactive', true);
|
|
$config->set('Element', 'Isotopes', array(238, 239)); // test inversion
|
|
$config->set('Element', 'Traits', 'nuclear, heavy, actinide'); // test parsing
|
|
$config->set('Element', 'IsotopeNames', array(238 => 'Plutonium-238', 239 => 'Plutonium-239'));
|
|
$config->set('Element', 'Object', false); // unmodeled
|
|
|
|
$this->expectError('Cannot set undefined directive Element.Metal to value');
|
|
$config->set('Element', 'Metal', true);
|
|
|
|
$this->expectError('Value for Element.Radioactive is of invalid type, should be bool');
|
|
$config->set('Element', 'Radioactive', 'very');
|
|
|
|
// test value retrieval
|
|
$this->assertIdentical($config->get('Element', 'Abbr'), 'Pu');
|
|
$this->assertIdentical($config->get('Element', 'Name'), 'plutonium');
|
|
$this->assertIdentical($config->get('Element', 'Number'), 94);
|
|
$this->assertIdentical($config->get('Element', 'Mass'), 244.);
|
|
$this->assertIdentical($config->get('Element', 'Radioactive'), true);
|
|
$this->assertIdentical($config->get('Element', 'Isotopes'), array(238 => true, 239 => true));
|
|
$this->assertIdentical($config->get('Element', 'Traits'), array('nuclear', 'heavy', 'actinide'));
|
|
$this->assertIdentical($config->get('Element', 'IsotopeNames'), array(238 => 'Plutonium-238', 239 => 'Plutonium-239'));
|
|
$this->assertIdentical($config->get('Element', 'Object'), false);
|
|
|
|
$this->expectError('Cannot retrieve value of undefined directive Element.Metal');
|
|
$config->get('Element', 'Metal');
|
|
|
|
}
|
|
|
|
function testEnumerated() {
|
|
|
|
$this->schema->addNamespace('Instrument', 'Of the musical type.');
|
|
|
|
// case sensitive
|
|
$this->schema->add('Instrument', 'Manufacturer', 'Yamaha', 'string', false);
|
|
$this->schema->addAllowedValues('Instrument', 'Manufacturer', array(
|
|
'Yamaha' => true, 'Conn-Selmer' => true, 'Vandoren' => true,
|
|
'Laubin' => true, 'Buffet' => true, 'other' => true));
|
|
$this->schema->addValueAliases('Instrument', 'Manufacturer', array(
|
|
'Selmer' => 'Conn-Selmer'));
|
|
|
|
// case insensitive
|
|
$this->schema->add('Instrument', 'Family', 'woodwind', 'istring', false);
|
|
$this->schema->addAllowedValues('Instrument', 'Family', array(
|
|
'brass' => true, 'woodwind' => true, 'percussion' => true,
|
|
'string' => true, 'keyboard' => true, 'electronic' => true));
|
|
$this->schema->addValueAliases('Instrument', 'Family', array(
|
|
'synth' => 'electronic'));
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
// case sensitive
|
|
|
|
$config->set('Instrument', 'Manufacturer', 'Vandoren');
|
|
$this->assertIdentical($config->get('Instrument', 'Manufacturer'), 'Vandoren');
|
|
|
|
$config->set('Instrument', 'Manufacturer', 'Selmer');
|
|
$this->assertIdentical($config->get('Instrument', 'Manufacturer'), 'Conn-Selmer');
|
|
|
|
$this->expectError('Value not supported, valid values are: Yamaha, Conn-Selmer, Vandoren, Laubin, Buffet, other');
|
|
$config->set('Instrument', 'Manufacturer', 'buffet');
|
|
|
|
// case insensitive
|
|
|
|
$config->set('Instrument', 'Family', 'brass');
|
|
$this->assertIdentical($config->get('Instrument', 'Family'), 'brass');
|
|
|
|
$config->set('Instrument', 'Family', 'PERCUSSION');
|
|
$this->assertIdentical($config->get('Instrument', 'Family'), 'percussion');
|
|
|
|
$config->set('Instrument', 'Family', 'synth');
|
|
$this->assertIdentical($config->get('Instrument', 'Family'), 'electronic');
|
|
|
|
$config->set('Instrument', 'Family', 'Synth');
|
|
$this->assertIdentical($config->get('Instrument', 'Family'), 'electronic');
|
|
|
|
}
|
|
|
|
function testNull() {
|
|
|
|
$this->schema->addNamespace('ReportCard');
|
|
$this->schema->add('ReportCard', 'English', null, 'string', true);
|
|
$this->schema->add('ReportCard', 'Absences', 0, 'int', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
$config->set('ReportCard', 'English', 'B-');
|
|
$this->assertIdentical($config->get('ReportCard', 'English'), 'B-');
|
|
|
|
$config->set('ReportCard', 'English', null); // not yet graded
|
|
$this->assertIdentical($config->get('ReportCard', 'English'), null);
|
|
|
|
// error
|
|
$this->expectError('Value for ReportCard.Absences is of invalid type, should be int');
|
|
$config->set('ReportCard', 'Absences', null);
|
|
|
|
}
|
|
|
|
function testAliases() {
|
|
|
|
$this->schema->addNamespace('Home');
|
|
$this->schema->add('Home', 'Rug', 3, 'int', false);
|
|
$this->schema->addAlias('Home', 'Carpet', 'Home', 'Rug');
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
$this->assertIdentical($config->get('Home', 'Rug'), 3);
|
|
|
|
$this->expectError('Cannot get value from aliased directive, use real name Home.Rug');
|
|
$config->get('Home', 'Carpet');
|
|
|
|
$this->expectError('Home.Carpet is an alias, preferred directive name is Home.Rug');
|
|
$config->set('Home', 'Carpet', 999);
|
|
$this->assertIdentical($config->get('Home', 'Rug'), 999);
|
|
|
|
}
|
|
|
|
// test functionality based on method
|
|
|
|
function test_getBatch() {
|
|
|
|
$this->schema->addNamespace('Variables');
|
|
$this->schema->add('Variables', 'TangentialAcceleration', 'a_tan', 'string', false);
|
|
$this->schema->add('Variables', 'AngularAcceleration', 'alpha', 'string', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
// grab a namespace
|
|
$this->assertIdentical(
|
|
$config->getBatch('Variables'),
|
|
array(
|
|
'TangentialAcceleration' => 'a_tan',
|
|
'AngularAcceleration' => 'alpha'
|
|
)
|
|
);
|
|
|
|
// grab a non-existant namespace
|
|
$this->expectError('Cannot retrieve undefined namespace Constants');
|
|
$config->getBatch('Constants');
|
|
|
|
}
|
|
|
|
function test_loadIni() {
|
|
|
|
$this->schema->addNamespace('Shortcut', 'Keyboard shortcuts for commands');
|
|
$this->schema->add('Shortcut', 'Copy', 'c', 'istring', false);
|
|
$this->schema->add('Shortcut', 'Paste', 'v', 'istring', false);
|
|
$this->schema->add('Shortcut', 'Cut', 'x', 'istring', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
$config->loadIni(dirname(__FILE__) . '/ConfigTest-loadIni.ini');
|
|
|
|
$this->assertIdentical($config->get('Shortcut', 'Copy'), 'q');
|
|
$this->assertIdentical($config->get('Shortcut', 'Paste'), 'p');
|
|
$this->assertIdentical($config->get('Shortcut', 'Cut'), 't');
|
|
|
|
}
|
|
|
|
function test_getHTMLDefinition() {
|
|
|
|
// we actually want to use the old copy, because the definition
|
|
// generation routines have dependencies on configuration values
|
|
|
|
$config = HTMLPurifier_Config::createDefault();
|
|
$config->set('HTML', 'Doctype', 'XHTML 1.0 Strict');
|
|
$config->autoFinalize = false;
|
|
|
|
$def = $config->getCSSDefinition();
|
|
$this->assertIsA($def, 'HTMLPurifier_CSSDefinition');
|
|
|
|
$def = $config->getHTMLDefinition();
|
|
$def2 = $config->getHTMLDefinition();
|
|
$this->assertIsA($def, 'HTMLPurifier_HTMLDefinition');
|
|
$this->assertSame($def, $def2);
|
|
$this->assertTrue($def->setup);
|
|
|
|
$old_def = clone $def2;
|
|
|
|
$config->set('HTML', 'Doctype', 'HTML 4.01 Transitional');
|
|
$def = $config->getHTMLDefinition();
|
|
$this->assertIsA($def, 'HTMLPurifier_HTMLDefinition');
|
|
$this->assertNotEqual($def, $old_def);
|
|
$this->assertTrue($def->setup);
|
|
|
|
// test retrieval of raw definition
|
|
$config->set('HTML', 'DefinitionID', 'HTMLPurifier_ConfigTest->test_getHTMLDefinition()');
|
|
$config->set('HTML', 'DefinitionRev', 3);
|
|
$def = $config->getHTMLDefinition(true);
|
|
$this->assertNotEqual($def, $old_def);
|
|
$this->assertEqual(false, $def->setup);
|
|
|
|
// auto initialization
|
|
$config->getHTMLDefinition();
|
|
$this->assertTrue($def->setup);
|
|
|
|
}
|
|
|
|
function test_getHTMLDefinition_rawError() {
|
|
$config = HTMLPurifier_Config::createDefault();
|
|
$this->expectException(new HTMLPurifier_Exception('Cannot retrieve raw version without specifying %HTML.DefinitionID'));
|
|
$def = $config->getHTMLDefinition(true);
|
|
}
|
|
|
|
function test_getCSSDefinition() {
|
|
$config = HTMLPurifier_Config::createDefault();
|
|
$def = $config->getCSSDefinition();
|
|
$this->assertIsA($def, 'HTMLPurifier_CSSDefinition');
|
|
}
|
|
|
|
function test_getDefinition() {
|
|
$this->schema->addNamespace('Cache', 'Cache stuff');
|
|
$this->schema->add('Cache', 'DefinitionImpl', null, 'string', true);
|
|
$this->schema->addNamespace('Crust', 'Krusty Krabs');
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$this->expectException(new HTMLPurifier_Exception("Definition of Crust type not supported"));
|
|
$config->getDefinition('Crust');
|
|
}
|
|
|
|
function test_loadArray() {
|
|
// setup a few dummy namespaces/directives for our testing
|
|
$this->schema->addNamespace('Zoo');
|
|
$this->schema->add('Zoo', 'Aadvark', 0, 'int', false);
|
|
$this->schema->add('Zoo', 'Boar', 0, 'int', false);
|
|
$this->schema->add('Zoo', 'Camel', 0, 'int', false);
|
|
$this->schema->add('Zoo', 'Others', array(), 'list', false);
|
|
|
|
$config_manual = new HTMLPurifier_Config($this->schema);
|
|
$config_loadabbr = new HTMLPurifier_Config($this->schema);
|
|
$config_loadfull = new HTMLPurifier_Config($this->schema);
|
|
|
|
$config_manual->set('Zoo', 'Aadvark', 3);
|
|
$config_manual->set('Zoo', 'Boar', 5);
|
|
$config_manual->set('Zoo', 'Camel', 2000); // that's a lotta camels!
|
|
$config_manual->set('Zoo', 'Others', array('Peacock', 'Dodo')); // wtf!
|
|
|
|
// condensed form
|
|
$config_loadabbr->loadArray(array(
|
|
'Zoo.Aadvark' => 3,
|
|
'Zoo.Boar' => 5,
|
|
'Zoo.Camel' => 2000,
|
|
'Zoo.Others' => array('Peacock', 'Dodo')
|
|
));
|
|
|
|
// fully expanded form
|
|
$config_loadfull->loadArray(array(
|
|
'Zoo' => array(
|
|
'Aadvark' => 3,
|
|
'Boar' => 5,
|
|
'Camel' => 2000,
|
|
'Others' => array('Peacock', 'Dodo')
|
|
)
|
|
));
|
|
|
|
$this->assertIdentical($config_manual, $config_loadabbr);
|
|
$this->assertIdentical($config_manual, $config_loadfull);
|
|
|
|
}
|
|
|
|
function test_create() {
|
|
|
|
$this->schema->addNamespace('Cake');
|
|
$this->schema->add('Cake', 'Sprinkles', 666, 'int', false);
|
|
$this->schema->add('Cake', 'Flavor', 'vanilla', 'string', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->set('Cake', 'Sprinkles', 42);
|
|
|
|
// test flat pass-through
|
|
$created_config = HTMLPurifier_Config::create($config, $this->schema);
|
|
$this->assertIdentical($config, $created_config);
|
|
|
|
// test loadArray
|
|
$created_config = HTMLPurifier_Config::create(array('Cake.Sprinkles' => 42), $this->schema);
|
|
$this->assertIdentical($config, $created_config);
|
|
|
|
// test loadIni
|
|
$created_config = HTMLPurifier_Config::create(dirname(__FILE__) . '/ConfigTest-create.ini', $this->schema);
|
|
$this->assertIdentical($config, $created_config);
|
|
|
|
}
|
|
|
|
function test_finalize() {
|
|
|
|
// test finalization
|
|
|
|
$this->schema->addNamespace('Poem');
|
|
$this->schema->add('Poem', 'Meter', 'iambic', 'string', false);
|
|
|
|
$config = new HTMLPurifier_Config($this->schema);
|
|
$config->autoFinalize = false;
|
|
|
|
$config->set('Poem', 'Meter', 'irregular');
|
|
|
|
$config->finalize();
|
|
|
|
$this->expectError('Cannot set directive after finalization');
|
|
$config->set('Poem', 'Meter', 'vedic');
|
|
|
|
$this->expectError('Cannot load directives after finalization');
|
|
$config->loadArray(array('Poem.Meter' => 'octosyllable'));
|
|
|
|
$this->expectError('Cannot load directives after finalization');
|
|
$config->loadIni(dirname(__FILE__) . '/ConfigTest-finalize.ini');
|
|
|
|
}
|
|
|
|
function test_loadArrayFromForm() {
|
|
|
|
$this->schema->addNamespace('Pancake');
|
|
$this->schema->add('Pancake', 'Mix', 'buttermilk', 'string', false);
|
|
$this->schema->add('Pancake', 'Served', true, 'bool', false);
|
|
$this->schema->addNamespace('Toppings', false);
|
|
$this->schema->add('Toppings', 'Syrup', true, 'bool', false);
|
|
$this->schema->add('Toppings', 'Flavor', 'maple', 'string', false);
|
|
$this->schema->add('Toppings', 'Strawberries', 3, 'int', false);
|
|
$this->schema->add('Toppings', 'Calories', 2000, 'int', true);
|
|
$this->schema->add('Toppings', 'DefinitionID', null, 'string', true);
|
|
$this->schema->add('Toppings', 'DefinitionRev', 1, 'int', false);
|
|
$this->schema->add('Toppings', 'Protected', 1, 'int', false);
|
|
|
|
$get = array(
|
|
'breakfast' => array(
|
|
'Pancake.Mix' => 'nasty',
|
|
'Pancake.Served' => '0',
|
|
'Toppings.Syrup' => '0',
|
|
'Toppings.Flavor' => "juice",
|
|
'Toppings.Strawberries' => '999',
|
|
'Toppings.Calories' => '',
|
|
'Null_Toppings.Calories' => '1',
|
|
'Toppings.DefinitionID' => '<argh>',
|
|
'Toppings.DefinitionRev' => '65',
|
|
'Toppings.Protected' => '4',
|
|
)
|
|
);
|
|
|
|
$config_expect = HTMLPurifier_Config::create(array(
|
|
'Pancake.Served' => false,
|
|
'Toppings.Syrup' => false,
|
|
'Toppings.Flavor' => "juice",
|
|
'Toppings.Strawberries' => 999,
|
|
'Toppings.Calories' => null
|
|
), $this->schema);
|
|
|
|
$config_result = HTMLPurifier_Config::loadArrayFromForm(
|
|
$get, 'breakfast',
|
|
array('Pancake.Served', 'Toppings', '-Toppings.Protected'),
|
|
false, // mq fix
|
|
$this->schema
|
|
);
|
|
|
|
$this->assertEqual($config_expect, $config_result);
|
|
|
|
/*
|
|
MAGIC QUOTES NOT TESTED!!!
|
|
|
|
$get = array(
|
|
'breakfast' => array(
|
|
'Pancake.Mix' => 'n\\asty'
|
|
)
|
|
);
|
|
$config_expect = HTMLPurifier_Config::create(array(
|
|
'Pancake.Mix' => 'n\\asty'
|
|
));
|
|
$config_result = HTMLPurifier_Config::loadArrayFromForm($get, 'breakfast', true, false);
|
|
$this->assertEqual($config_expect, $config_result);
|
|
*/
|
|
}
|
|
|
|
function test_getAllowedDirectivesForForm() {
|
|
$this->schema->addNamespace('Unused');
|
|
$this->schema->add('Unused', 'Unused', 'Foobar', 'string', false);
|
|
$this->schema->addNamespace('Partial');
|
|
$this->schema->add('Partial', 'Allowed', true, 'bool', false);
|
|
$this->schema->add('Partial', 'Unused', 'Foobar', 'string', false);
|
|
$this->schema->addNamespace('All');
|
|
$this->schema->add('All', 'Allowed', true, 'bool', false);
|
|
$this->schema->add('All', 'Blacklisted', 'Foobar', 'string', false); // explicitly blacklisted
|
|
$this->schema->add('All', 'DefinitionID', 'Foobar', 'string', true); // auto-blacklisted
|
|
$this->schema->add('All', 'DefinitionRev', 2, 'int', false); // auto-blacklisted
|
|
|
|
$input = array('Partial.Allowed', 'All', '-All.Blacklisted');
|
|
$output = HTMLPurifier_Config::getAllowedDirectivesForForm($input, $this->schema);
|
|
$expect = array(
|
|
array('Partial', 'Allowed'),
|
|
array('All', 'Allowed')
|
|
);
|
|
|
|
$this->assertEqual($output, $expect);
|
|
|
|
}
|
|
|
|
}
|
|
|