157 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace PhpOffice\PhpSpreadsheetTests\Reader\Security;
 | 
						|
 | 
						|
use PhpOffice\PhpSpreadsheet\Reader\Security\XmlScanner;
 | 
						|
use PhpOffice\PhpSpreadsheet\Reader\Xls;
 | 
						|
use PhpOffice\PhpSpreadsheet\Reader\Xlsx;
 | 
						|
use PHPUnit\Framework\TestCase;
 | 
						|
use XMLReader;
 | 
						|
 | 
						|
class XmlScannerTest extends TestCase
 | 
						|
{
 | 
						|
    protected function setUp(): void
 | 
						|
    {
 | 
						|
        // php 8.+ deprecated libxml_disable_entity_loader() - It's on by default
 | 
						|
        if (\PHP_VERSION_ID < 80000) {
 | 
						|
            libxml_disable_entity_loader(false);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider providerValidXML
 | 
						|
     *
 | 
						|
     * @param mixed $filename
 | 
						|
     * @param mixed $expectedResult
 | 
						|
     * @param $libxmlDisableEntityLoader
 | 
						|
     */
 | 
						|
    public function testValidXML($filename, $expectedResult, $libxmlDisableEntityLoader): void
 | 
						|
    {
 | 
						|
        // php 8.+ deprecated libxml_disable_entity_loader() - It's on by default
 | 
						|
        if (\PHP_VERSION_ID < 80000) {
 | 
						|
            $oldDisableEntityLoaderState = libxml_disable_entity_loader($libxmlDisableEntityLoader);
 | 
						|
        }
 | 
						|
 | 
						|
        $reader = XmlScanner::getInstance(new \PhpOffice\PhpSpreadsheet\Reader\Xml());
 | 
						|
        $result = $reader->scanFile($filename);
 | 
						|
        self::assertEquals($expectedResult, $result);
 | 
						|
 | 
						|
        // php 8.+ deprecated libxml_disable_entity_loader() - It's on by default
 | 
						|
        if (\PHP_VERSION_ID < 80000) {
 | 
						|
            libxml_disable_entity_loader($oldDisableEntityLoaderState);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function providerValidXML()
 | 
						|
    {
 | 
						|
        $tests = [];
 | 
						|
        foreach (glob('tests/data/Reader/Xml/XEETestValid*.xml') as $file) {
 | 
						|
            $filename = realpath($file);
 | 
						|
            $expectedResult = file_get_contents($file);
 | 
						|
            $tests[basename($file) . '_libxml_entity_loader_disabled'] = [$filename, $expectedResult, true];
 | 
						|
            $tests[basename($file) . '_libxml_entity_loader_enabled'] = [$filename, $expectedResult, false];
 | 
						|
        }
 | 
						|
 | 
						|
        return $tests;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider providerInvalidXML
 | 
						|
     *
 | 
						|
     * @param mixed $filename
 | 
						|
     * @param $libxmlDisableEntityLoader
 | 
						|
     */
 | 
						|
    public function testInvalidXML($filename, $libxmlDisableEntityLoader): void
 | 
						|
    {
 | 
						|
        $this->expectException(\PhpOffice\PhpSpreadsheet\Reader\Exception::class);
 | 
						|
 | 
						|
        // php 8.+ deprecated libxml_disable_entity_loader() - It's on by default
 | 
						|
        if (\PHP_VERSION_ID < 80000) {
 | 
						|
            libxml_disable_entity_loader($libxmlDisableEntityLoader);
 | 
						|
        }
 | 
						|
 | 
						|
        $reader = XmlScanner::getInstance(new \PhpOffice\PhpSpreadsheet\Reader\Xml());
 | 
						|
        $expectedResult = 'FAILURE: Should throw an Exception rather than return a value';
 | 
						|
        $result = $reader->scanFile($filename);
 | 
						|
        self::assertEquals($expectedResult, $result);
 | 
						|
        // php 8.+ deprecated libxml_disable_entity_loader() - It's on by default
 | 
						|
        if (\PHP_VERSION_ID < 80000) {
 | 
						|
            self::assertEquals($libxmlDisableEntityLoader, libxml_disable_entity_loader());
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function providerInvalidXML()
 | 
						|
    {
 | 
						|
        $tests = [];
 | 
						|
        foreach (glob('tests/data/Reader/Xml/XEETestInvalidUTF*.xml') as $file) {
 | 
						|
            $filename = realpath($file);
 | 
						|
            $tests[basename($file) . '_libxml_entity_loader_disabled'] = [$filename, true];
 | 
						|
            $tests[basename($file) . '_libxml_entity_loader_enabled'] = [$filename, false];
 | 
						|
        }
 | 
						|
 | 
						|
        return $tests;
 | 
						|
    }
 | 
						|
 | 
						|
    public function testGetSecurityScannerForXmlBasedReader(): void
 | 
						|
    {
 | 
						|
        $fileReader = new Xlsx();
 | 
						|
        $scanner = $fileReader->getSecurityScanner();
 | 
						|
 | 
						|
        //    Must return an object...
 | 
						|
        self::assertIsObject($scanner);
 | 
						|
        //    ... of the correct type
 | 
						|
        self::assertInstanceOf(XmlScanner::class, $scanner);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testGetSecurityScannerForNonXmlBasedReader(): void
 | 
						|
    {
 | 
						|
        $fileReader = new Xls();
 | 
						|
        $scanner = $fileReader->getSecurityScanner();
 | 
						|
        //    Must return a null...
 | 
						|
        self::assertNull($scanner);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider providerValidXMLForCallback
 | 
						|
     *
 | 
						|
     * @param mixed $filename
 | 
						|
     * @param mixed $expectedResult
 | 
						|
     */
 | 
						|
    public function testSecurityScanWithCallback($filename, $expectedResult): void
 | 
						|
    {
 | 
						|
        $fileReader = new Xlsx();
 | 
						|
        $scanner = $fileReader->getSecurityScanner();
 | 
						|
        $scanner->setAdditionalCallback('strrev');
 | 
						|
        $xml = $scanner->scanFile($filename);
 | 
						|
 | 
						|
        self::assertEquals(strrev($expectedResult), $xml);
 | 
						|
    }
 | 
						|
 | 
						|
    public function providerValidXMLForCallback()
 | 
						|
    {
 | 
						|
        $tests = [];
 | 
						|
        foreach (glob('tests/data/Reader/Xml/SecurityScannerWithCallback*.xml') as $file) {
 | 
						|
            $tests[basename($file)] = [realpath($file), file_get_contents($file)];
 | 
						|
        }
 | 
						|
 | 
						|
        return $tests;
 | 
						|
    }
 | 
						|
 | 
						|
    public function testLibxmlDisableEntityLoaderIsRestoredWithoutShutdown(): void
 | 
						|
    {
 | 
						|
        $reader = new Xlsx();
 | 
						|
        unset($reader);
 | 
						|
 | 
						|
        $reader = new XMLReader();
 | 
						|
        $opened = $reader->open('tests/data/Reader/Xml/SecurityScannerWithCallbackExample.xml');
 | 
						|
        self::assertTrue($opened);
 | 
						|
    }
 | 
						|
 | 
						|
    public function testEncodingAllowsMixedCase(): void
 | 
						|
    {
 | 
						|
        $scanner = new XmlScanner();
 | 
						|
        $output = $scanner->scan($input = '<?xml version="1.0" encoding="utf-8"?><foo>bar</foo>');
 | 
						|
        self::assertSame($input, $output);
 | 
						|
    }
 | 
						|
}
 |