mirror of
				https://github.com/ezyang/htmlpurifier.git
				synced 2025-10-24 18:16:19 +02:00 
			
		
		
		
	git-svn-id: http://htmlpurifier.org/svnroot/htmlpurifier/trunk@1332 48356398-32a2-884e-a903-53898d9a118a
		
			
				
	
	
		
			346 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			346 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| require_once 'HTMLPurifier/ConfigSchema.php';
 | |
| 
 | |
| if (!class_exists('CS')) {
 | |
|     class CS extends HTMLPurifier_ConfigSchema {}
 | |
| }
 | |
| 
 | |
| class HTMLPurifier_ConfigSchemaTest extends HTMLPurifier_Harness
 | |
| {
 | |
|     
 | |
|     /**
 | |
|      * Munged name of current file.
 | |
|      */
 | |
|     var $file;
 | |
|     
 | |
|     /**
 | |
|      * Copy of the real ConfigSchema to revert to.
 | |
|      */
 | |
|     var $old_copy;
 | |
|     
 | |
|     /**
 | |
|      * Copy of dummy ConfigSchema for testing purposes.
 | |
|      */
 | |
|     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);
 | |
|         
 | |
|         $this->file = $this->our_copy->mungeFilename(__FILE__);
 | |
|     }
 | |
|     
 | |
|     function tearDown() {
 | |
|         // testing is done, restore the old copy
 | |
|         HTMLPurifier_ConfigSchema::instance($this->old_copy);
 | |
|         tally_errors($this);
 | |
|     }
 | |
|     
 | |
|     function test_defineNamespace() {
 | |
|         CS::defineNamespace('http', $d = 'This is an internet protocol.');
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->info_namespace, array(
 | |
|             'http' => new HTMLPurifier_ConfigDef_Namespace($d)
 | |
|         ));
 | |
|         
 | |
|         $this->expectError('Cannot redefine namespace');
 | |
|         CS::defineNamespace('http', 'It is used to serve webpages.');
 | |
|         
 | |
|         $this->expectError('Namespace name must be alphanumeric');
 | |
|         CS::defineNamespace('ssh+http', 'This http is tunneled through SSH.');
 | |
|         
 | |
|         $this->expectError('Description must be non-empty');
 | |
|         CS::defineNamespace('ftp', null);
 | |
|     }
 | |
|     
 | |
|     function test_define() {
 | |
|         CS::defineNamespace('Car', 'Automobiles, those gas-guzzlers!');
 | |
|         
 | |
|         CS::define('Car', 'Seats', 5, 'int', $d = 'Standard issue.'); $l = __LINE__;
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->defaults['Car']['Seats'], 5);
 | |
|         $this->assertIdentical($this->our_copy->info['Car']['Seats'],
 | |
|             new HTMLPurifier_ConfigDef_Directive('int',
 | |
|                 array($this->file => array($l => $d))
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         CS::define('Car', 'Age', null, 'int/null', $d = 'Not always known.'); $l = __LINE__;
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->defaults['Car']['Age'], null);
 | |
|         $this->assertIdentical($this->our_copy->info['Car']['Age'], 
 | |
|             new HTMLPurifier_ConfigDef_Directive('int',
 | |
|                 array($this->file => array($l => $d)), true
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define directive for undefined namespace');
 | |
|         CS::define('Train', 'Cars', 10, 'int', 'Including the caboose.');
 | |
|         
 | |
|         $this->expectError('Directive name must be alphanumeric');
 | |
|         CS::define('Car', 'Is it shiny?', true, 'bool', 'Indicates regular waxing.');
 | |
|         
 | |
|         $this->expectError('Invalid type for configuration directive');
 | |
|         CS::define('Car', 'Efficiency', 50, 'mpg', 'The higher the better.');
 | |
|         
 | |
|         $this->expectError('Default value does not match directive type');
 | |
|         CS::define('Car', 'Producer', 'Ford', 'int', 'ID of the company that made the car.');
 | |
|         
 | |
|         $this->expectError('Description must be non-empty');
 | |
|         CS::define('Car', 'ComplexAttribute', 'lawyers', 'istring', null);
 | |
|     }
 | |
|     
 | |
|     function testRedefinition_define() {
 | |
|         CS::defineNamespace('Cat', 'Belongs to Schrodinger.');
 | |
|         
 | |
|         CS::define('Cat', 'Dead', false, 'bool', $d1 = 'Well, is it?'); $l1 = __LINE__;
 | |
|         CS::define('Cat', 'Dead', false, 'bool', $d2 = 'It is difficult to say.'); $l2 = __LINE__;
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->defaults['Cat']['Dead'], false);
 | |
|         $this->assertIdentical($this->our_copy->info['Cat']['Dead'], 
 | |
|             new HTMLPurifier_ConfigDef_Directive('bool',
 | |
|                 array($this->file => array($l1 => $d1, $l2 => $d2))
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Inconsistent default or type, cannot redefine');
 | |
|         CS::define('Cat', 'Dead', true, 'bool', 'Quantum mechanics does not know.');
 | |
|         
 | |
|         $this->expectError('Inconsistent default or type, cannot redefine');
 | |
|         CS::define('Cat', 'Dead', 'maybe', 'string', 'Perhaps if we look we will know.');
 | |
|     }
 | |
|     
 | |
|     function test_defineAllowedValues() {
 | |
|         CS::defineNamespace('QuantumNumber', 'D');
 | |
|         CS::define('QuantumNumber', 'Spin', 0.5, 'float',
 | |
|             'Spin of particle. Fourth quantum number, represented by s.');
 | |
|         CS::define('QuantumNumber', 'Current', 's', 'string',
 | |
|             'Currently selected quantum number.');
 | |
|         CS::define('QuantumNumber', 'Difficulty', null, 'string/null', $d = 'How hard are the problems?'); $l = __LINE__;
 | |
|         
 | |
|         CS::defineAllowedValues( // okay, since default is null
 | |
|             'QuantumNumber', 'Difficulty', array('easy', 'medium', 'hard')
 | |
|         );
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->defaults['QuantumNumber']['Difficulty'], null);
 | |
|         $this->assertIdentical($this->our_copy->info['QuantumNumber']['Difficulty'], 
 | |
|             new HTMLPurifier_ConfigDef_Directive(
 | |
|                 'string',
 | |
|                 array($this->file => array($l => $d)),
 | |
|                 true,
 | |
|                 array(
 | |
|                     'easy' => true,
 | |
|                     'medium' => true,
 | |
|                     'hard' => true
 | |
|                 )
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define allowed values for undefined directive');
 | |
|         CS::defineAllowedValues(
 | |
|             'SpaceTime', 'Symmetry', array('time', 'spatial', 'projective')
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define allowed values for directive whose type is not string');
 | |
|         CS::defineAllowedValues(
 | |
|             'QuantumNumber', 'Spin', array(0.5, -0.5)
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Default value must be in allowed range of variables');
 | |
|         CS::defineAllowedValues(
 | |
|             'QuantumNumber', 'Current', array('n', 'l', 'm') // forgot s!
 | |
|         );
 | |
|     }
 | |
|     
 | |
|     function test_defineValueAliases() {
 | |
|         CS::defineNamespace('Abbrev', 'Stuff on abbreviations.');
 | |
|         CS::define('Abbrev', 'HTH', 'Happy to Help', 'string', $d = 'Three-letters'); $l = __LINE__;
 | |
|         CS::defineAllowedValues(
 | |
|             'Abbrev', 'HTH', array(
 | |
|                 'Happy to Help',
 | |
|                 'Hope that Helps',
 | |
|                 'HAIL THE HAND!'
 | |
|             )
 | |
|         );
 | |
|         CS::defineValueAliases(
 | |
|             'Abbrev', 'HTH', array(
 | |
|                 'happy' => 'Happy to Help',
 | |
|                 'hope' => 'Hope that Helps'
 | |
|             )
 | |
|         );
 | |
|         CS::defineValueAliases( // delayed addition
 | |
|             'Abbrev', 'HTH', array(
 | |
|                 'hail' => 'HAIL THE HAND!'
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->assertIdentical($this->our_copy->defaults['Abbrev']['HTH'], 'Happy to Help');
 | |
|         $this->assertIdentical($this->our_copy->info['Abbrev']['HTH'], 
 | |
|             new HTMLPurifier_ConfigDef_Directive(
 | |
|                 'string',
 | |
|                 array($this->file => array($l => $d)),
 | |
|                 false,
 | |
|                 array(
 | |
|                     'Happy to Help' => true,
 | |
|                     'Hope that Helps' => true,
 | |
|                     'HAIL THE HAND!' => true
 | |
|                 ),
 | |
|                 array(
 | |
|                     'happy' => 'Happy to Help',
 | |
|                     'hope' => 'Hope that Helps',
 | |
|                     'hail' => 'HAIL THE HAND!'
 | |
|                 )
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define alias to value that is not allowed');
 | |
|         CS::defineValueAliases(
 | |
|             'Abbrev', 'HTH', array(
 | |
|                 'head' => 'Head to Head'
 | |
|             )
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define alias over allowed value');
 | |
|         CS::defineValueAliases(
 | |
|             'Abbrev', 'HTH', array(
 | |
|                 'Hope that Helps' => 'Happy to Help'
 | |
|             )
 | |
|         );
 | |
|         
 | |
|     }
 | |
|    
 | |
|     function testAlias() {
 | |
|         CS::defineNamespace('Home', 'Sweet home.');
 | |
|         CS::define('Home', 'Rug', 3, 'int', 'ID.');
 | |
|         CS::defineAlias('Home', 'Carpet', 'Home', 'Rug');
 | |
|         
 | |
|         $this->assertTrue(!isset($this->our_copy->defaults['Home']['Carpet']));
 | |
|         $this->assertIdentical($this->our_copy->info['Home']['Carpet'], 
 | |
|             new HTMLPurifier_ConfigDef_DirectiveAlias('Home', 'Rug')
 | |
|         );
 | |
|         
 | |
|         $this->expectError('Cannot define directive alias in undefined namespace');
 | |
|         CS::defineAlias('Store', 'Rug', 'Home', 'Rug');
 | |
|         
 | |
|         $this->expectError('Directive name must be alphanumeric');
 | |
|         CS::defineAlias('Home', 'R.g', 'Home', 'Rug');
 | |
|         
 | |
|         CS::define('Home', 'Rugger', 'Bob Max', 'string', 'Name of.');
 | |
|         $this->expectError('Cannot define alias over directive');
 | |
|         CS::defineAlias('Home', 'Rugger', 'Home', 'Rug');
 | |
|         
 | |
|         $this->expectError('Cannot define alias to undefined directive');
 | |
|         CS::defineAlias('Home', 'Rug2', 'Home', 'Rugavan');
 | |
|         
 | |
|         $this->expectError('Cannot define alias to alias');
 | |
|         CS::defineAlias('Home', 'Rug2', 'Home', 'Carpet');
 | |
|     }
 | |
|     
 | |
|     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', 'text'); // aliases, lstring = long string
 | |
|         $this->assertValid('FOOBAR', 'istring', 'foobar');
 | |
|         $this->assertValid('FOOBAR', 'itext', '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->assertValid('true', 'bool', true);
 | |
|         $this->assertValid('false', 'bool', false);
 | |
|         $this->assertValid('1', 'bool', true);
 | |
|         $this->assertInvalid(34, 'bool');
 | |
|         $this->assertInvalid(null, 'bool');
 | |
|         
 | |
|         $this->assertValid(array('1', '2', '3'), 'list');
 | |
|         $this->assertValid('foo,bar, cow', 'list', array('foo', 'bar', 'cow'));
 | |
|         $this->assertValid('', 'list', array());
 | |
|         $this->assertValid("foo\nbar", 'list', array('foo', 'bar'));
 | |
|         $this->assertValid("foo\nbar,baz", 'list', array('foo', 'bar', 'baz'));
 | |
|         
 | |
|         $this->assertValid(array('1' => true, '2' => true), 'lookup');
 | |
|         $this->assertValid(array('1', '2'), 'lookup', array('1' => true, '2' => true));
 | |
|         $this->assertValid('foo,bar', 'lookup', array('foo' => true, 'bar' => true));
 | |
|         $this->assertValid("foo\nbar", 'lookup', array('foo' => true, 'bar' => true));
 | |
|         $this->assertValid("foo\nbar,baz", 'lookup', array('foo' => true, 'bar' => true, 'baz' => true));
 | |
|         $this->assertValid('', 'lookup', array());
 | |
|         
 | |
|         $this->assertValid(array('foo' => 'bar'), 'hash');
 | |
|         $this->assertValid(array(1 => 'moo'), 'hash');
 | |
|         $this->assertInvalid(array(0 => 'moo'), 'hash');
 | |
|         $this->assertValid('', 'hash', array());
 | |
|         $this->assertValid('foo:bar,too:two', 'hash', array('foo' => 'bar', 'too' => 'two'));
 | |
|         $this->assertValid("foo:bar\ntoo:two,three:free", 'hash', array('foo' => 'bar', 'too' => 'two', 'three' => 'free'));
 | |
|         $this->assertValid('foo:bar,too', 'hash', array('foo' => 'bar'));
 | |
|         $this->assertValid('foo:bar,', 'hash', array('foo' => 'bar'));
 | |
|         $this->assertValid('foo:bar:baz', 'hash', array('foo' => 'bar:baz'));
 | |
|         
 | |
|         $this->assertValid(23, 'mixed');
 | |
|         
 | |
|     }
 | |
|     
 | |
|     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\\My Libraries\\htmlpurifier\\library\\HTMLPurifier\\AttrDef.php',
 | |
|             'HTMLPurifier/AttrDef.php'
 | |
|         );
 | |
|         
 | |
|         $this->assertMungeFilename(
 | |
|             'C:\\php\\My Libraries\\htmlpurifier\\library\\HTMLPurifier.php',
 | |
|             'HTMLPurifier.php'
 | |
|         );
 | |
|         
 | |
|     }
 | |
|     
 | |
| }
 | |
| 
 |