ptRecorder = new PhpunitTestRecorder; $this->ptRunner = new ParserTestRunner( $this->ptRecorder ); if ( is_string( $flags ) ) { $flags = self::CORE_ONLY; } global $IP; $mwTestDir = $IP . '/tests/'; # Human friendly helpers $wantsCore = ( $flags & self::CORE_ONLY ); $wantsRest = ( $flags & self::NO_CORE ); # Will hold the .txt parser test files we will include $filesToTest = []; # Filter out .txt files $files = ParserTestRunner::getParserTestFiles(); foreach ( $files as $extName => $parserTestFile ) { $isCore = ( strpos( $parserTestFile, $mwTestDir ) === 0 ); if ( $isCore && $wantsCore ) { self::debug( "included core parser tests: $parserTestFile" ); $filesToTest[$extName] = $parserTestFile; } elseif ( !$isCore && $wantsRest ) { self::debug( "included non core parser tests: $parserTestFile" ); $filesToTest[$extName] = $parserTestFile; } else { self::debug( "skipped parser tests: $parserTestFile" ); } } self::debug( 'parser tests files: ' . implode( ' ', $filesToTest ) ); $testList = []; $counter = 0; foreach ( $filesToTest as $extensionName => $fileName ) { if ( is_int( $extensionName ) ) { // If there's no extension name because this is coming // from the legacy global, then assume the next level directory // is the extension name (e.g. extensions/FooBar/parserTests.txt). $extensionName = basename( dirname( $fileName ) ); } $testsName = $extensionName . '__' . basename( $fileName, '.txt' ); $parserTestClassName = ucfirst( $testsName ); // Official spec for class names: https://www.php.net/manual/en/language.oop5.basic.php // Prepend 'ParserTest_' to be paranoid about it not starting with a number $parserTestClassName = 'ParserTest_' . preg_replace( '/[^a-zA-Z0-9_\x7f-\xff]/', '_', $parserTestClassName ); if ( isset( $testList[$parserTestClassName] ) ) { // If there is a conflict, append a number. $counter++; $parserTestClassName .= $counter; } $testList[$parserTestClassName] = true; // Previously we actually created a class here, with eval(). We now // just override the name. self::debug( "Adding test class $parserTestClassName" ); $this->addTest( new ParserTestFileSuite( $this->ptRunner, $parserTestClassName, $fileName ) ); } } protected function setUp() : void { wfDebug( __METHOD__ ); $lb = MediaWikiServices::getInstance()->getDBLoadBalancer(); $db = $lb->getConnection( DB_MASTER ); $type = $db->getType(); $prefix = MediaWikiIntegrationTestCase::DB_PREFIX; $this->oldTablePrefix = $db->tablePrefix(); MediaWikiIntegrationTestCase::setupTestDB( $db, $prefix ); CloneDatabase::changePrefix( $prefix ); $this->ptRunner->setDatabase( $db ); MediaWikiIntegrationTestCase::resetNonServiceCaches(); MediaWikiIntegrationTestCase::installMockMwServices(); $teardown = new ScopedCallback( function () { MediaWikiIntegrationTestCase::restoreMwServices(); } ); $teardown = $this->ptRunner->setupUploads( $teardown ); $this->ptTeardownScope = $teardown; } protected function tearDown() : void { wfDebug( __METHOD__ ); if ( $this->ptTeardownScope ) { ScopedCallback::consume( $this->ptTeardownScope ); } CloneDatabase::changePrefix( $this->oldTablePrefix ); } /** * Write $msg under log group 'tests-parser' * @param string $msg Message to log */ protected static function debug( $msg ) { wfDebugLog( 'tests-parser', wfGetCaller() . ' ' . $msg ); } }