<?php require_once 'HTMLPurifier/ConfigSchema.php'; class HTMLPurifier_ConfigSchemaTest extends UnitTestCase { var $old_copy; var $our_copy; function setUp() { // yes, I know this is slightly convoluted, but that's the price // you pay for using Singletons. Good thing we can overload it. // first, let's get a clean copy to do tests $our_copy = new HTMLPurifier_ConfigSchema(); // get the old copy $this->old_copy = HTMLPurifier_ConfigSchema::instance(); // put in our copy, and reassign to the REAL reference $this->our_copy =& HTMLPurifier_ConfigSchema::instance($our_copy); } function tearDown() { // testing is done, restore the old copy HTMLPurifier_ConfigSchema::instance($this->old_copy); } function testNormal() { $file = $this->our_copy->mungeFilename(__FILE__); // define a namespace $description = 'Configuration that is always available.'; HTMLPurifier_ConfigSchema::defineNamespace( 'Core', $description ); $this->assertIdentical($this->our_copy->defaults, array( 'Core' => array() )); $this->assertIdentical($this->our_copy->info, array( 'Core' => array() )); $namespace = new HTMLPurifier_ConfigEntity_Namespace(); $namespace->description = $description; $this->assertIdentical($this->our_copy->info_namespace, array( 'Core' => $namespace )); // define a directive $description = 'This is a description of the directive.'; HTMLPurifier_ConfigSchema::define( 'Core', 'Name', 'default value', 'string', $description ); $line = __LINE__; $this->assertIdentical($this->our_copy->defaults, array( 'Core' => array( 'Name' => 'default value' ) )); $directive = new HTMLPurifier_ConfigEntity_Directive(); $directive->type = 'string'; $directive->addDescription($file, $line, $description); $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // define a directive in an undefined namespace HTMLPurifier_ConfigSchema::define( 'Extension', 'Name', false, 'bool', 'This is for an extension, but we have not defined its namespace!' ); $this->assertError('Cannot define directive for undefined namespace'); $this->assertNoErrors(); // redefine a value in a valid manner $description = 'Alternative configuration definition'; HTMLPurifier_ConfigSchema::define( 'Core', 'Name', 'default value', 'string', $description ); $line = __LINE__; $this->assertNoErrors(); $directive->addDescription($file, $line, $description); $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // redefine a directive in an invalid manner HTMLPurifier_ConfigSchema::define( 'Core', 'Name', 'different default', 'string', 'Inconsistent default or type, cannot redefine' ); $this->assertError('Inconsistent default or type, cannot redefine'); $this->assertNoErrors(); // make an enumeration HTMLPurifier_ConfigSchema::defineAllowedValues( 'Core', 'Name', array( 'Real Value', 'Real Value 2' ) ); $directive->allowed = array( 'Real Value' => true, 'Real Value 2' => true ); $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // redefinition of enumeration is cumulative HTMLPurifier_ConfigSchema::defineAllowedValues( 'Core', 'Name', array( 'Real Value 3', ) ); $directive->allowed['Real Value 3'] = true; $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // cannot define enumeration for undefined directive HTMLPurifier_ConfigSchema::defineAllowedValues( 'Core', 'Foobar', array( 'Real Value 9', ) ); $this->assertError('Cannot define allowed values for undefined directive'); $this->assertNoErrors(); // test defining value aliases for an enumerated value HTMLPurifier_ConfigSchema::defineValueAliases( 'Core', 'Name', array( 'Aliased Value' => 'Real Value' ) ); $directive->aliases['Aliased Value'] = 'Real Value'; $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // redefine should be cumulative HTMLPurifier_ConfigSchema::defineValueAliases( 'Core', 'Name', array( 'Aliased Value 2' => 'Real Value 2' ) ); $directive->aliases['Aliased Value 2'] = 'Real Value 2'; $this->assertIdentical($this->our_copy->info, array( 'Core' => array( 'Name' => $directive ) )); // cannot create alias to not-allowed value HTMLPurifier_ConfigSchema::defineValueAliases( 'Core', 'Name', array( 'Aliased Value 3' => 'Invalid Value' ) ); $this->assertError('Cannot define alias to value that is not allowed'); $this->assertNoErrors(); // cannot create alias for already allowed value HTMLPurifier_ConfigSchema::defineValueAliases( 'Core', 'Name', array( 'Real Value' => 'Real Value 2' ) ); $this->assertError('Cannot define alias over allowed value'); $this->assertNoErrors(); // define a directive with an invalid type HTMLPurifier_ConfigSchema::define( 'Core', 'Foobar', false, 'omen', 'Omen is not a valid type, so we reject this.' ); $this->assertError('Invalid type for configuration directive'); $this->assertNoErrors(); // define a directive with inconsistent type HTMLPurifier_ConfigSchema::define( 'Core', 'Foobaz', 10, 'string', 'If we say string, we should mean it, not integer 10.' ); $this->assertError('Default value does not match directive type'); $this->assertNoErrors(); // define a directive that allows null HTMLPurifier_ConfigSchema::define( 'Core', 'Foobaz', null, 'string/null', 'Nulls are allowed if you add on /null, cool huh?' ); $this->assertNoErrors(); // define a directive with bad characters HTMLPurifier_ConfigSchema::define( 'Core', 'Core.Attr', 10, 'int', 'No periods! >:-(' ); $this->assertError('Directive name must be alphanumeric'); $this->assertNoErrors(); // define a namespace with bad characters HTMLPurifier_ConfigSchema::defineNamespace( 'Foobar&Gromit', $description ); $this->assertError('Namespace name must be alphanumeric'); $this->assertNoErrors(); } function assertValid($var, $type, $ret = null) { $ret = ($ret === null) ? $var : $ret; $this->assertIdentical($this->our_copy->validate($var, $type), $ret); } function assertInvalid($var, $type) { $this->assertTrue( $this->our_copy->isError( $this->our_copy->validate($var, $type) ) ); } function testValidate() { $this->assertValid('foobar', 'string'); $this->assertValid('FOOBAR', 'istring', 'foobar'); $this->assertValid(34, 'int'); $this->assertValid(3.34, 'float'); $this->assertValid(false, 'bool'); $this->assertValid(0, 'bool', false); $this->assertValid(1, 'bool', true); $this->assertInvalid(34, 'bool'); $this->assertInvalid(null, 'bool'); $this->assertValid(array('1', '2', '3'), 'list'); $this->assertValid(array('1' => true, '2' => true), 'lookup'); $this->assertValid(array('1', '2'), 'lookup', array('1' => true, '2' => true)); $this->assertValid(array('foo' => 'bar'), 'hash'); $this->assertInvalid(array(0 => 'moo'), 'hash'); $this->assertValid(array(1 => 'moo'), 'hash'); $this->assertValid(23, 'mixed'); $this->assertValid('foo,bar, cow', 'list', array('foo', 'bar', 'cow')); $this->assertValid('foo,bar', 'lookup', array('foo' => true, 'bar' => true)); $this->assertValid('true', 'bool', true); $this->assertValid('false', 'bool', false); $this->assertValid('1', 'bool', true); } function testValidate_null() { $this->assertTrue( $this->our_copy->isError( $this->our_copy->validate(null, 'string', false) ) ); $this->assertFalse( $this->our_copy->isError( $this->our_copy->validate(null, 'string', true) ) ); } function assertMungeFilename($oldname, $newname) { $this->assertIdentical( $this->our_copy->mungeFilename($oldname), $newname ); } function testMungeFilename() { $this->assertMungeFilename( 'C:\\php\\libs\\htmlpurifier\\library\\HTMLPurifier\\AttrDef.php', 'HTMLPurifier/AttrDef.php' ); $this->assertMungeFilename( 'C:\\php\\libs\\htmlpurifier\\library\\HTMLPurifier.php', 'HTMLPurifier.php' ); } } ?>