wiki.techinc.nl/tests/phpunit/includes/HooksTest.php
Lucas Werkmeister 5f2bfc44ce Simplify HookContainer (v2)
This reverts change I50c3d1c5df (commit b0317287bc), thus reinstating
change I7d690a1172 (commit d139eb07fe). The only change from the
original is in getHookMethodName(), additionally replacing '-' with '_'
(not just ':' and '\'). The original commit message follows:

This converts all hook handlers to the same internal representation.
This is done lazily, when the hook is run for the first time.

The logic for temporarily disabling handlers by calling scopedRegister()
with the $replace parameter set has been greatly simplified.

There are some minor changes to the class's interface and behavior,
none of which should be breaking changes:
* run() will emit deprecation warnings if and only if it was called
  with the deprecationVersion option set, for all kinds of handlers.
  The idea is that deprecated hooks should emit a warning either from
  run(), or from emitDeprecationWarnings(). The latter happens if the
  hook is listed in DeprecatedHooks.
* register() now also accepts hook handlers declared in the way that
  extensions register hooks.
* Attempts to call register() with an invalid hook definition now
  result in an invalidArgumentException.
* Attempts to call register() for a deprecated hook will consistently
  result in a deprecation warning.
* The internal getRegisteredHooks() method has been removed in favor
  of the identical getHookNames() method.
* The internal getLegacyHandlers method has been removed in favor
  of getHandlerDescriptions() and getHandlerCallbacks().
* The call order changed so that dynamically registered handlers
  are called last, instead of getting called before handler objects
  from extensions.

Bug: T338213
Change-Id: I6efb09e314ad2b124a33a757fdda2a07ae0d8f7c
2023-06-06 12:06:23 +02:00

356 lines
9.7 KiB
PHP

<?php
use Wikimedia\ScopedCallback;
class HooksTest extends MediaWikiIntegrationTestCase {
private const MOCK_HOOK_NAME = 'MediaWikiHooksTest001';
protected function setUp(): void {
parent::setUp();
}
public static function provideHooks() {
$obj = new HookTestDummyHookHandlerClass();
return [
[
'Object and method',
[ $obj, 'someNonStatic' ],
'changed-nonstatic',
'changed-nonstatic'
],
[ 'Object and no method', [ $obj ], 'changed-onevent', 'original' ],
[
'Object and method with data',
[ $obj, 'someNonStaticWithData', 'data' ],
'data',
'original'
],
[ 'Object and static method', [ $obj, 'someStatic' ], 'changed-static', 'original' ],
[
'Class::method static call',
[ 'HookTestDummyHookHandlerClass::someStatic' ],
'changed-static',
'original'
],
[
'Class::method static call as array',
[ [ 'HookTestDummyHookHandlerClass::someStatic' ] ],
'changed-static',
'original'
],
[ 'Global function', [ 'wfNothingFunction' ], 'changed-func', 'original' ],
[ 'Global function with data', [ 'wfNothingFunctionData', 'data' ], 'data', 'original' ],
[ 'Closure', [ static function ( &$foo, $bar ) {
$foo = 'changed-closure';
return true;
} ], 'changed-closure', 'original' ],
[ 'Closure with data', [ static function ( $data, &$foo, $bar ) {
$foo = $data;
return true;
}, 'data' ], 'data', 'original' ]
];
}
/**
* @dataProvider provideHooks
* @covers Hooks::register
* @covers Hooks::run
*/
public function testRunningNewStyleHooks( $msg, $hook, $expectedFoo, $expectedBar ) {
$this->hideDeprecated( 'Hooks::run' );
$foo = $bar = 'original';
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, $hook );
Hooks::run( self::MOCK_HOOK_NAME, [ &$foo, &$bar ] );
$this->assertSame( $expectedFoo, $foo, $msg );
$this->assertSame( $expectedBar, $bar, $msg );
}
/**
* @covers Hooks::getHandlers
*/
public function testGetHandlers() {
global $wgHooks;
$hookContainer = $this->getServiceContainer()->getHookContainer();
$this->filterDeprecated( '/\$wgHooks/' );
$this->hideDeprecated( 'Hooks::getHandlers' );
$this->assertSame(
[],
Hooks::getHandlers( 'MediaWikiHooksTest001' ),
'No hooks registered'
);
$a = [ new HookTestDummyHookHandlerClass(), 'someStatic' ];
$b = [ new HookTestDummyHookHandlerClass(), 'onMediaWikiHooksTest001' ];
$wgHooks['MediaWikiHooksTest001'][] = $a;
$this->assertSame(
[ $a ],
array_values( Hooks::getHandlers( 'MediaWikiHooksTest001' ) ),
'Hook registered by $wgHooks'
);
$reset = $hookContainer->scopedRegister( 'MediaWikiHooksTest001', $b );
$this->assertSame(
[ $a, $b ],
array_values( Hooks::getHandlers( 'MediaWikiHooksTest001' ) ),
'Hooks::getHandlers() should return hooks registered via wgHooks as well as Hooks::register'
);
ScopedCallback::consume( $reset );
unset( $wgHooks['MediaWikiHooksTest001'] );
$hookContainer->register( 'MediaWikiHooksTest001', $b );
$this->assertSame(
[ $b ],
array_values( Hooks::getHandlers( 'MediaWikiHooksTest001' ) ),
'Hook registered by Hook::register'
);
}
/**
* @covers Hooks::isRegistered
* @covers Hooks::getHandlers
* @covers Hooks::register
* @covers Hooks::run
*/
public function testRegistration() {
$this->hideDeprecated( 'Hooks::isRegistered' );
$this->hideDeprecated( 'Hooks::getHandlers' );
$this->hideDeprecated( 'Hooks::run' );
global $wgHooks;
$hookContainer = $this->getServiceContainer()->getHookContainer();
$this->expectDeprecationAndContinue( '/\$wgHooks .* deprecated/' );
$this->expectDeprecationAndContinue( '/Use of Hooks::register was deprecated/' );
$a = new HookTestDummyHookHandlerClass();
$b = new HookTestDummyHookHandlerClass();
$c = new HookTestDummyHookHandlerClass();
$wgHooks[ self::MOCK_HOOK_NAME ][] = $a;
Hooks::register( self::MOCK_HOOK_NAME, $b );
$hookContainer->register( self::MOCK_HOOK_NAME, $c );
$this->assertTrue( Hooks::isRegistered( self::MOCK_HOOK_NAME ) );
$this->assertCount( 3, Hooks::getHandlers( self::MOCK_HOOK_NAME ) );
$foo = 'quux';
$bar = 'qaax';
Hooks::run( self::MOCK_HOOK_NAME, [ &$foo, &$bar ] );
$this->assertSame(
1,
$a->calls,
'Hooks::run() should run hooks registered via $wgHooks'
);
$this->assertSame(
1,
$b->calls,
'Hooks::run() should run hooks registered via Hooks::register'
);
$this->assertSame(
1,
$c->calls,
'Hooks::run() should run hooks registered via HookContainer::register'
);
}
/**
* @covers Hooks::run
*/
public function testUncallableFunction() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$this->expectException( InvalidArgumentException::class );
$hookContainer->register( self::MOCK_HOOK_NAME, 'ThisFunctionDoesntExist' );
Hooks::run( self::MOCK_HOOK_NAME, [] );
}
/**
* @covers Hooks::run
*/
public function testFalseReturn() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
return false;
} );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
$foo = 'test';
return true;
} );
$foo = 'original';
Hooks::run( self::MOCK_HOOK_NAME, [ &$foo ] );
$this->assertSame( 'original', $foo, 'Hooks abort after a false return.' );
}
/**
* @covers Hooks::run
*/
public function testNullReturn() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
return;
} );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
$foo = 'test';
return true;
} );
$foo = 'original';
Hooks::run( self::MOCK_HOOK_NAME, [ &$foo ] );
$this->assertSame( 'test', $foo, 'Hooks continue after a null return.' );
}
/**
* @covers Hooks::run
*/
public function testCallHook_FalseHook() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, false );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
$foo = 'test';
return true;
} );
$foo = 'original';
Hooks::run( self::MOCK_HOOK_NAME, [ &$foo ] );
$this->assertSame( 'test', $foo, 'Hooks that are falsey are skipped.' );
}
/**
* @covers Hooks::run
*/
public function testCallHook_UnknownDatatype() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$this->expectException( InvalidArgumentException::class );
$hookContainer->register( self::MOCK_HOOK_NAME, 12345 );
Hooks::run( self::MOCK_HOOK_NAME );
}
/**
* @covers Hooks::run
*/
public function testCallHook_Deprecated() {
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, 'HookTestDummyHookHandlerClass::someStatic' );
$this->expectDeprecationAndContinue( '/Use of MediaWikiHooksTest001 hook/' );
$a = $b = 0;
$this->hideDeprecated( 'Hooks::run' );
Hooks::run( self::MOCK_HOOK_NAME, [ $a, $b ], '1.31' );
}
/**
* @covers Hooks::runWithoutAbort
*/
public function testRunWithoutAbort() {
$this->hideDeprecated( 'Hooks::runWithoutAbort' );
$list = [];
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$list ) {
$list[] = 1;
return true; // Explicit true
} );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$list ) {
$list[] = 2;
return; // Implicit null
} );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$list ) {
$list[] = 3;
// No return
} );
Hooks::runWithoutAbort( self::MOCK_HOOK_NAME, [ &$list ] );
$this->assertSame( [ 1, 2, 3 ], $list, 'All hooks ran.' );
}
/**
* @covers Hooks::runWithoutAbort
*/
public function testRunWithoutAbortWarning() {
$this->hideDeprecated( 'Hooks::runWithoutAbort' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
return false;
} );
$hookContainer->register( self::MOCK_HOOK_NAME, static function ( &$foo ) {
$foo = 'test';
return true;
} );
$foo = 'original';
$this->expectException( UnexpectedValueException::class );
$this->expectExceptionMessage( 'unabortable MediaWikiHooksTest001' );
Hooks::runWithoutAbort( self::MOCK_HOOK_NAME, [ &$foo ] );
}
/**
* @covers Hooks::run
*/
public function testBadHookFunction() {
$this->hideDeprecated( 'Hooks::run' );
$hookContainer = $this->getServiceContainer()->getHookContainer();
$hookContainer->register( self::MOCK_HOOK_NAME, static function () {
return 'test';
} );
$this->expectException( UnexpectedValueException::class );
Hooks::run( self::MOCK_HOOK_NAME, [] );
}
}
function wfNothingFunction( &$foo, &$bar ) {
$foo = 'changed-func';
return true;
}
function wfNothingFunctionData( $data, &$foo, &$bar ) {
$foo = $data;
return true;
}
class HookTestDummyHookHandlerClass {
public $calls = 0;
public static function someStatic( &$foo, &$bar ) {
$foo = 'changed-static';
return true;
}
public function someNonStatic( &$foo, &$bar ) {
$this->calls++;
$foo = 'changed-nonstatic';
$bar = 'changed-nonstatic';
return true;
}
public function onMediaWikiHooksTest001( &$foo, &$bar ) {
$this->calls++;
$foo = 'changed-onevent';
return true;
}
public function someNonStaticWithData( $data, &$foo, &$bar ) {
$this->calls++;
$foo = $data;
return true;
}
}