The default will remain PHPUnit 4.x due to PHP 5.5 support. But, we should allow developers to run tests with newer PHPUnit versions which are noticably faster (especially for code coverage reports). * <https://github.com/sebastianbergmann/phpunit/wiki/Release-Announcement-for-PHPUnit-5.4.0> PHPUnit 5 deprecates the getMock() shortcut for getMockBuilder()->getMock(). It instead introduces the shortcut createMock() which has better defaults than getMockBuilder(). For example, it sets 'disableArgumentCloning' and other things by default. Going forward, code should either use getMockBuilder directly and configure it using the setter methods (instead of the confusing variadic arguments of getMock) or simply use the new minimalistic createMock method. This patch backports the createMock method to MediaWikiTestCase so that we can start using it. Change-Id: I091c0289b21d2b1c876adba89529dc3e72b99af2
373 lines
12 KiB
PHP
373 lines
12 KiB
PHP
<?php
|
|
|
|
namespace MediaWiki\Auth;
|
|
|
|
use MediaWiki\MediaWikiServices;
|
|
|
|
/**
|
|
* @group AuthManager
|
|
* @group Database
|
|
* @covers MediaWiki\Auth\LegacyHookPreAuthenticationProvider
|
|
*/
|
|
class LegacyHookPreAuthenticationProviderTest extends \MediaWikiTestCase {
|
|
/**
|
|
* Get an instance of the provider
|
|
* @return LegacyHookPreAuthenticationProvider
|
|
*/
|
|
protected function getProvider() {
|
|
$request = $this->getMockBuilder( 'FauxRequest' )
|
|
->setMethods( [ 'getIP' ] )->getMock();
|
|
$request->expects( $this->any() )->method( 'getIP' )->will( $this->returnValue( '127.0.0.42' ) );
|
|
|
|
$manager = new AuthManager(
|
|
$request,
|
|
MediaWikiServices::getInstance()->getMainConfig()
|
|
);
|
|
|
|
$provider = new LegacyHookPreAuthenticationProvider();
|
|
$provider->setManager( $manager );
|
|
$provider->setLogger( new \Psr\Log\NullLogger() );
|
|
$provider->setConfig( new \HashConfig( [
|
|
'PasswordAttemptThrottle' => [ 'count' => 23, 'seconds' => 42 ],
|
|
] ) );
|
|
return $provider;
|
|
}
|
|
|
|
/**
|
|
* Sets a mock on a hook
|
|
* @param string $hook
|
|
* @param object $expect From $this->once(), $this->never(), etc.
|
|
* @return object $mock->expects( $expect )->method( ... ).
|
|
*/
|
|
protected function hook( $hook, $expect ) {
|
|
$mock = $this->getMockBuilder( __CLASS__ )->setMethods( [ "on$hook" ] )->getMock();
|
|
$this->mergeMwGlobalArrayValue( 'wgHooks', [
|
|
$hook => [ $mock ],
|
|
] );
|
|
return $mock->expects( $expect )->method( "on$hook" );
|
|
}
|
|
|
|
/**
|
|
* Unsets a hook
|
|
* @param string $hook
|
|
*/
|
|
protected function unhook( $hook ) {
|
|
$this->mergeMwGlobalArrayValue( 'wgHooks', [
|
|
$hook => [],
|
|
] );
|
|
}
|
|
|
|
// Stubs for hooks taking reference parameters
|
|
public function onLoginUserMigrated( $user, &$msg ) {
|
|
}
|
|
public function onAbortLogin( $user, $password, &$abort, &$msg ) {
|
|
}
|
|
public function onAbortNewAccount( $user, &$abortError, &$abortStatus ) {
|
|
}
|
|
public function onAbortAutoAccount( $user, &$abortError ) {
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestForAuthentication
|
|
* @param string|null $username
|
|
* @param string|null $password
|
|
* @param string|null $msgForLoginUserMigrated
|
|
* @param int|null $abortForAbortLogin
|
|
* @param string|null $msgForAbortLogin
|
|
* @param string|null $failMsg
|
|
* @param array $failParams
|
|
*/
|
|
public function testTestForAuthentication(
|
|
$username, $password,
|
|
$msgForLoginUserMigrated, $abortForAbortLogin, $msgForAbortLogin,
|
|
$failMsg, $failParams = []
|
|
) {
|
|
$reqs = [];
|
|
if ( $username === null ) {
|
|
$this->hook( 'LoginUserMigrated', $this->never() );
|
|
$this->hook( 'AbortLogin', $this->never() );
|
|
} else {
|
|
if ( $password === null ) {
|
|
$req = $this->getMockForAbstractClass( AuthenticationRequest::class );
|
|
} else {
|
|
$req = new PasswordAuthenticationRequest();
|
|
$req->action = AuthManager::ACTION_LOGIN;
|
|
$req->password = $password;
|
|
}
|
|
$req->username = $username;
|
|
$reqs[get_class( $req )] = $req;
|
|
|
|
$h = $this->hook( 'LoginUserMigrated', $this->once() );
|
|
if ( $msgForLoginUserMigrated !== null ) {
|
|
$h->will( $this->returnCallback(
|
|
function ( $user, &$msg ) use ( $username, $msgForLoginUserMigrated ) {
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( $username, $user->getName() );
|
|
$msg = $msgForLoginUserMigrated;
|
|
return false;
|
|
}
|
|
) );
|
|
$this->hook( 'AbortLogin', $this->never() );
|
|
} else {
|
|
$h->will( $this->returnCallback(
|
|
function ( $user, &$msg ) use ( $username ) {
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( $username, $user->getName() );
|
|
return true;
|
|
}
|
|
) );
|
|
$h2 = $this->hook( 'AbortLogin', $this->once() );
|
|
if ( $abortForAbortLogin !== null ) {
|
|
$h2->will( $this->returnCallback(
|
|
function ( $user, $pass, &$abort, &$msg )
|
|
use ( $username, $password, $abortForAbortLogin, $msgForAbortLogin )
|
|
{
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( $username, $user->getName() );
|
|
if ( $password !== null ) {
|
|
$this->assertSame( $password, $pass );
|
|
} else {
|
|
$this->assertInternalType( 'string', $pass );
|
|
}
|
|
$abort = $abortForAbortLogin;
|
|
$msg = $msgForAbortLogin;
|
|
return false;
|
|
}
|
|
) );
|
|
} else {
|
|
$h2->will( $this->returnCallback(
|
|
function ( $user, $pass, &$abort, &$msg ) use ( $username, $password ) {
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( $username, $user->getName() );
|
|
if ( $password !== null ) {
|
|
$this->assertSame( $password, $pass );
|
|
} else {
|
|
$this->assertInternalType( 'string', $pass );
|
|
}
|
|
return true;
|
|
}
|
|
) );
|
|
}
|
|
}
|
|
}
|
|
unset( $h, $h2 );
|
|
|
|
$status = $this->getProvider()->testForAuthentication( $reqs );
|
|
|
|
$this->unhook( 'LoginUserMigrated' );
|
|
$this->unhook( 'AbortLogin' );
|
|
|
|
if ( $failMsg === null ) {
|
|
$this->assertEquals( \StatusValue::newGood(), $status, 'should succeed' );
|
|
} else {
|
|
$this->assertInstanceOf( 'StatusValue', $status, 'should fail (type)' );
|
|
$this->assertFalse( $status->isOk(), 'should fail (ok)' );
|
|
$errors = $status->getErrors();
|
|
$this->assertEquals( $failMsg, $errors[0]['message'], 'should fail (message)' );
|
|
$this->assertEquals( $failParams, $errors[0]['params'], 'should fail (params)' );
|
|
}
|
|
}
|
|
|
|
public static function provideTestForAuthentication() {
|
|
return [
|
|
'No valid requests' => [
|
|
null, null, null, null, null, null
|
|
],
|
|
'No hook errors' => [
|
|
'User', 'PaSsWoRd', null, null, null, null
|
|
],
|
|
'No hook errors, no password' => [
|
|
'User', null, null, null, null, null
|
|
],
|
|
'LoginUserMigrated no message' => [
|
|
'User', 'PaSsWoRd', false, null, null, 'login-migrated-generic'
|
|
],
|
|
'LoginUserMigrated with message' => [
|
|
'User', 'PaSsWoRd', 'LUM-abort', null, null, 'LUM-abort'
|
|
],
|
|
'LoginUserMigrated with message and params' => [
|
|
'User', 'PaSsWoRd', [ 'LUM-abort', 'foo' ], null, null, 'LUM-abort', [ 'foo' ]
|
|
],
|
|
'AbortLogin, SUCCESS' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::SUCCESS, null, null
|
|
],
|
|
'AbortLogin, NEED_TOKEN, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::NEED_TOKEN, null, 'nocookiesforlogin'
|
|
],
|
|
'AbortLogin, NEED_TOKEN, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::NEED_TOKEN, 'needtoken', 'needtoken'
|
|
],
|
|
'AbortLogin, WRONG_TOKEN, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_TOKEN, null, 'sessionfailure'
|
|
],
|
|
'AbortLogin, WRONG_TOKEN, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_TOKEN, 'wrongtoken', 'wrongtoken'
|
|
],
|
|
'AbortLogin, ILLEGAL, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::ILLEGAL, null, 'noname'
|
|
],
|
|
'AbortLogin, ILLEGAL, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::ILLEGAL, 'badname', 'badname'
|
|
],
|
|
'AbortLogin, NO_NAME, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::NO_NAME, null, 'noname'
|
|
],
|
|
'AbortLogin, NO_NAME, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::NO_NAME, 'badname', 'badname'
|
|
],
|
|
'AbortLogin, WRONG_PASS, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_PASS, null, 'wrongpassword'
|
|
],
|
|
'AbortLogin, WRONG_PASS, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_PASS, 'badpass', 'badpass'
|
|
],
|
|
'AbortLogin, WRONG_PLUGIN_PASS, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_PLUGIN_PASS, null, 'wrongpassword'
|
|
],
|
|
'AbortLogin, WRONG_PLUGIN_PASS, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::WRONG_PLUGIN_PASS, 'badpass', 'badpass'
|
|
],
|
|
'AbortLogin, NOT_EXISTS, no message' => [
|
|
"User'", 'A', null, \LoginForm::NOT_EXISTS, null, 'nosuchusershort', [ 'User'' ]
|
|
],
|
|
'AbortLogin, NOT_EXISTS, with message' => [
|
|
"User'", 'A', null, \LoginForm::NOT_EXISTS, 'badname', 'badname', [ 'User'' ]
|
|
],
|
|
'AbortLogin, EMPTY_PASS, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::EMPTY_PASS, null, 'wrongpasswordempty'
|
|
],
|
|
'AbortLogin, EMPTY_PASS, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::EMPTY_PASS, 'badpass', 'badpass'
|
|
],
|
|
'AbortLogin, RESET_PASS, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::RESET_PASS, null, 'resetpass_announce'
|
|
],
|
|
'AbortLogin, RESET_PASS, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::RESET_PASS, 'resetpass', 'resetpass'
|
|
],
|
|
'AbortLogin, THROTTLED, no message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::THROTTLED, null, 'login-throttled',
|
|
[ \Message::durationParam( 42 ) ]
|
|
],
|
|
'AbortLogin, THROTTLED, with message' => [
|
|
'User', 'PaSsWoRd', null, \LoginForm::THROTTLED, 't', 't',
|
|
[ \Message::durationParam( 42 ) ]
|
|
],
|
|
'AbortLogin, USER_BLOCKED, no message' => [
|
|
"User'", 'P', null, \LoginForm::USER_BLOCKED, null, 'login-userblocked', [ 'User'' ]
|
|
],
|
|
'AbortLogin, USER_BLOCKED, with message' => [
|
|
"User'", 'P', null, \LoginForm::USER_BLOCKED, 'blocked', 'blocked', [ 'User'' ]
|
|
],
|
|
'AbortLogin, ABORTED, no message' => [
|
|
"User'", 'P', null, \LoginForm::ABORTED, null, 'login-abort-generic', [ 'User'' ]
|
|
],
|
|
'AbortLogin, ABORTED, with message' => [
|
|
"User'", 'P', null, \LoginForm::ABORTED, 'aborted', 'aborted', [ 'User'' ]
|
|
],
|
|
'AbortLogin, USER_MIGRATED, no message' => [
|
|
'User', 'P', null, \LoginForm::USER_MIGRATED, null, 'login-migrated-generic'
|
|
],
|
|
'AbortLogin, USER_MIGRATED, with message' => [
|
|
'User', 'P', null, \LoginForm::USER_MIGRATED, 'migrated', 'migrated'
|
|
],
|
|
'AbortLogin, USER_MIGRATED, with message and params' => [
|
|
'User', 'P', null, \LoginForm::USER_MIGRATED, [ 'migrated', 'foo' ],
|
|
'migrated', [ 'foo' ]
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestForAccountCreation
|
|
* @param string $msg
|
|
* @param Status|null $status
|
|
* @param StatusValue Result
|
|
*/
|
|
public function testTestForAccountCreation( $msg, $status, $result ) {
|
|
$this->hook( 'AbortNewAccount', $this->once() )
|
|
->will( $this->returnCallback( function ( $user, &$error, &$abortStatus )
|
|
use ( $msg, $status )
|
|
{
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( 'User', $user->getName() );
|
|
$error = $msg;
|
|
$abortStatus = $status;
|
|
return $error === null && $status === null;
|
|
} ) );
|
|
|
|
$user = \User::newFromName( 'User' );
|
|
$creator = \User::newFromName( 'UTSysop' );
|
|
$ret = $this->getProvider()->testForAccountCreation( $user, $creator, [] );
|
|
|
|
$this->unhook( 'AbortNewAccount' );
|
|
|
|
$this->assertEquals( $result, $ret );
|
|
}
|
|
|
|
public static function provideTestForAccountCreation() {
|
|
return [
|
|
'No hook errors' => [
|
|
null, null, \StatusValue::newGood()
|
|
],
|
|
'AbortNewAccount, old style' => [
|
|
'foobar', null, \StatusValue::newFatal(
|
|
\Message::newFromKey( 'createaccount-hook-aborted' )->rawParams( 'foobar' )
|
|
)
|
|
],
|
|
'AbortNewAccount, new style' => [
|
|
'foobar',
|
|
\Status::newFatal( 'aborted!', 'param' ),
|
|
\StatusValue::newFatal( 'aborted!', 'param' )
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestUserForCreation
|
|
* @param string|null $error
|
|
* @param string|null $failMsg
|
|
*/
|
|
public function testTestUserForCreation( $error, $failMsg ) {
|
|
$testUser = self::getTestUser()->getUser();
|
|
$provider = $this->getProvider();
|
|
$options = [ 'flags' => \User::READ_LOCKING, 'creating' => true ];
|
|
|
|
$this->hook( 'AbortNewAccount', $this->never() );
|
|
$this->hook( 'AbortAutoAccount', $this->once() )
|
|
->will( $this->returnCallback( function ( $user, &$abortError ) use ( $testUser, $error ) {
|
|
$this->assertInstanceOf( 'User', $user );
|
|
$this->assertSame( $testUser->getName(), $user->getName() );
|
|
$abortError = $error;
|
|
return $error === null;
|
|
} ) );
|
|
$status = $provider->testUserForCreation(
|
|
$testUser, AuthManager::AUTOCREATE_SOURCE_SESSION, $options
|
|
);
|
|
$this->unhook( 'AbortNewAccount' );
|
|
$this->unhook( 'AbortAutoAccount' );
|
|
if ( $failMsg === null ) {
|
|
$this->assertEquals( \StatusValue::newGood(), $status, 'should succeed' );
|
|
} else {
|
|
$this->assertInstanceOf( 'StatusValue', $status, 'should fail (type)' );
|
|
$this->assertFalse( $status->isOk(), 'should fail (ok)' );
|
|
$errors = $status->getErrors();
|
|
$this->assertEquals( $failMsg, $errors[0]['message'], 'should fail (message)' );
|
|
}
|
|
|
|
$this->hook( 'AbortAutoAccount', $this->never() );
|
|
$this->hook( 'AbortNewAccount', $this->never() );
|
|
$status = $provider->testUserForCreation( $testUser, false, $options );
|
|
$this->unhook( 'AbortNewAccount' );
|
|
$this->unhook( 'AbortAutoAccount' );
|
|
$this->assertEquals( \StatusValue::newGood(), $status, 'should succeed' );
|
|
}
|
|
|
|
public static function provideTestUserForCreation() {
|
|
return [
|
|
'Success' => [ null, null ],
|
|
'Fail, no message' => [ false, 'login-abort-generic' ],
|
|
'Fail, with message' => [ 'fail', 'fail' ],
|
|
];
|
|
}
|
|
}
|