This patch adds two more commonly used methods for params/rawParams to getMockMessage and relocates the functionality to the MediaWikiTestCaseTrait. Follows-Up: Ia94521b786 Change-Id: If9a6ccf1885ba11fbf3fb1b586006118462c1410
166 lines
5 KiB
PHP
166 lines
5 KiB
PHP
<?php
|
|
/**
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
* http://www.gnu.org/copyleft/gpl.html
|
|
*
|
|
* @file
|
|
* @ingroup Testing
|
|
*/
|
|
|
|
use MediaWiki\Logger\LoggerFactory;
|
|
use MediaWiki\MediaWikiServices;
|
|
use PHPUnit\Framework\Exception;
|
|
use PHPUnit\Framework\TestCase;
|
|
use Wikimedia\ObjectFactory;
|
|
|
|
/**
|
|
* Base class for unit tests.
|
|
*
|
|
* Extend this class if you are testing classes which use dependency injection and do not access
|
|
* global functions, variables, services or a storage backend.
|
|
*
|
|
* @since 1.34
|
|
*/
|
|
abstract class MediaWikiUnitTestCase extends TestCase {
|
|
use MediaWikiCoversValidator;
|
|
use MediaWikiTestCaseTrait;
|
|
|
|
private static $originalGlobals;
|
|
private static $unitGlobals;
|
|
private static $temporaryHooks;
|
|
|
|
/**
|
|
* Whitelist of globals to allow in MediaWikiUnitTestCase.
|
|
*
|
|
* Please, keep this list to the bare minimum.
|
|
*
|
|
* @return string[]
|
|
*/
|
|
private static function getGlobalsWhitelist() {
|
|
return [
|
|
// The autoloader may change between bootstrap and the first test,
|
|
// so (lazily) capture these here instead.
|
|
'wgAutoloadClasses',
|
|
'wgAutoloadLocalClasses',
|
|
// Need for LoggerFactory. Default is NullSpi.
|
|
'wgMWLoggerDefaultSpi',
|
|
'wgAutoloadAttemptLowercase',
|
|
'wgLegalTitleChars',
|
|
'wgDevelopmentWarnings',
|
|
];
|
|
}
|
|
|
|
public static function setUpBeforeClass() : void {
|
|
parent::setUpBeforeClass();
|
|
|
|
$reflection = new ReflectionClass( static::class );
|
|
$dirSeparator = DIRECTORY_SEPARATOR;
|
|
if ( stripos( $reflection->getFilename(), "${dirSeparator}unit${dirSeparator}" ) === false ) {
|
|
self::fail( 'This unit test needs to be in "tests/phpunit/unit"!' );
|
|
}
|
|
|
|
self::$unitGlobals =& TestSetup::$bootstrapGlobals;
|
|
|
|
foreach ( self::getGlobalsWhitelist() as $global ) {
|
|
self::$unitGlobals[ $global ] =& $GLOBALS[ $global ];
|
|
}
|
|
self::$temporaryHooks = [];
|
|
|
|
// Would be nice if we could simply replace $GLOBALS as a whole,
|
|
// but un-setting or re-assigning that breaks the reference of this magic
|
|
// variable. Thus we have to modify it in place.
|
|
self::$originalGlobals = [];
|
|
foreach ( $GLOBALS as $key => $_ ) {
|
|
// Stash current values
|
|
self::$originalGlobals[$key] =& $GLOBALS[$key];
|
|
|
|
// Remove globals not part of the snapshot (see bootstrap.php, phpunit.php).
|
|
if ( $key !== 'GLOBALS' && !array_key_exists( $key, self::$unitGlobals ) ) {
|
|
unset( $GLOBALS[$key] );
|
|
}
|
|
}
|
|
// Restore values from the early snapshot
|
|
// Not by ref because tests must not be able to modify the snapshot.
|
|
foreach ( self::$unitGlobals as $key => $value ) {
|
|
$GLOBALS[ $key ] = $value;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
protected function runTest() {
|
|
try {
|
|
return parent::runTest();
|
|
} catch ( ConfigException $exception ) {
|
|
throw new Exception(
|
|
'Config variables must be mocked, they cannot be accessed directly in tests which extend '
|
|
. self::class,
|
|
$exception->getCode(),
|
|
$exception
|
|
);
|
|
}
|
|
|
|
// Don't let LoggerFactory::getProvider() access globals or other things we don't want.
|
|
LoggerFactory::registerProvider( ObjectFactory::getObjectFromSpec( [
|
|
'class' => \MediaWiki\Logger\NullSpi::class
|
|
] ) );
|
|
}
|
|
|
|
protected function tearDown() : void {
|
|
// Quick reset between tests
|
|
foreach ( $GLOBALS as $key => $_ ) {
|
|
if ( $key !== 'GLOBALS' && !array_key_exists( $key, self::$unitGlobals ) ) {
|
|
unset( $GLOBALS[$key] );
|
|
}
|
|
}
|
|
foreach ( self::$unitGlobals as $key => $value ) {
|
|
$GLOBALS[ $key ] = $value;
|
|
}
|
|
self::$temporaryHooks = [];
|
|
|
|
parent::tearDown();
|
|
}
|
|
|
|
public static function tearDownAfterClass() : void {
|
|
// Remove globals created by the test
|
|
foreach ( $GLOBALS as $key => $_ ) {
|
|
if ( $key !== 'GLOBALS' && !array_key_exists( $key, self::$originalGlobals ) ) {
|
|
unset( $GLOBALS[$key] );
|
|
}
|
|
}
|
|
// Restore values (including reference!)
|
|
foreach ( self::$originalGlobals as $key => &$value ) {
|
|
$GLOBALS[ $key ] =& $value;
|
|
}
|
|
|
|
parent::tearDownAfterClass();
|
|
}
|
|
|
|
/**
|
|
* Create a temporary hook handler which will be reset by tearDown.
|
|
* @param string $hookName Hook name
|
|
* @param mixed $handler Value suitable for a hook handler
|
|
* @since 1.34
|
|
*/
|
|
protected function setTemporaryHook( $hookName, $handler ) {
|
|
// Adds handler to list of hook handlers
|
|
$hookContainer = MediaWikiServices::getInstance()->getHookContainer();
|
|
$hookToRemove = $hookContainer->scopedRegister( $hookName, $handler, true );
|
|
// Keep reference to the ScopedCallback
|
|
self::$temporaryHooks[] = $hookToRemove;
|
|
}
|
|
|
|
}
|