wiki.techinc.nl/tests/phpunit/includes/auth/AbstractPrimaryAuthenticationProviderTest.php
Brad Jorsch d245bd25ae Add AuthManager
This implements the AuthManager class and its needed interfaces and
subclasses, and integrates them into the backend portion of MediaWiki.
Integration with frontend portions of MediaWiki (e.g. ApiLogin,
Special:Login) is left for a followup.

Bug: T91699
Bug: T71589
Bug: T111299
Co-Authored-By: Gergő Tisza <gtisza@wikimedia.org>
Change-Id: If89d24838e326fe25fe867d02181eebcfbb0e196
2016-05-16 15:11:02 +00:00

183 lines
5.6 KiB
PHP

<?php
namespace MediaWiki\Auth;
/**
* @group AuthManager
* @covers MediaWiki\Auth\AbstractPrimaryAuthenticationProvider
*/
class AbstractPrimaryAuthenticationProviderTest extends \MediaWikiTestCase {
protected function setUp() {
global $wgDisableAuthManager;
parent::setUp();
if ( $wgDisableAuthManager ) {
$this->markTestSkipped( '$wgDisableAuthManager is set' );
}
}
public function testAbstractPrimaryAuthenticationProvider() {
$user = \User::newFromName( 'UTSysop' );
$provider = $this->getMockForAbstractClass( AbstractPrimaryAuthenticationProvider::class );
try {
$provider->continuePrimaryAuthentication( [] );
$this->fail( 'Expected exception not thrown' );
} catch ( \BadMethodCallException $ex ) {
}
try {
$provider->continuePrimaryAccountCreation( $user, $user, [] );
$this->fail( 'Expected exception not thrown' );
} catch ( \BadMethodCallException $ex ) {
}
$req = $this->getMockForAbstractClass( AuthenticationRequest::class );
$this->assertTrue( $provider->providerAllowsPropertyChange( 'foo' ) );
$this->assertEquals(
\StatusValue::newGood(),
$provider->testForAccountCreation( $user, $user, [] )
);
$this->assertEquals(
\StatusValue::newGood(),
$provider->testUserForCreation( $user, AuthManager::AUTOCREATE_SOURCE_SESSION )
);
$this->assertEquals(
\StatusValue::newGood(),
$provider->testUserForCreation( $user, false )
);
$this->assertNull(
$provider->finishAccountCreation( $user, $user, AuthenticationResponse::newPass() )
);
$provider->autoCreatedAccount( $user, AuthManager::AUTOCREATE_SOURCE_SESSION );
$res = AuthenticationResponse::newPass();
$provider->postAuthentication( $user, $res );
$provider->postAccountCreation( $user, $user, $res );
$provider->postAccountLink( $user, $res );
$provider->expects( $this->once() )
->method( 'testUserExists' )
->with( $this->equalTo( 'foo' ) )
->will( $this->returnValue( true ) );
$this->assertTrue( $provider->testUserCanAuthenticate( 'foo' ) );
}
public function testProviderRevokeAccessForUser() {
$reqs = [];
for ( $i = 0; $i < 3; $i++ ) {
$reqs[$i] = $this->getMock( AuthenticationRequest::class );
$reqs[$i]->done = false;
}
$provider = $this->getMockForAbstractClass( AbstractPrimaryAuthenticationProvider::class );
$provider->expects( $this->once() )->method( 'getAuthenticationRequests' )
->with(
$this->identicalTo( AuthManager::ACTION_REMOVE ),
$this->identicalTo( [ 'username' => 'UTSysop' ] )
)
->will( $this->returnValue( $reqs ) );
$provider->expects( $this->exactly( 3 ) )->method( 'providerChangeAuthenticationData' )
->will( $this->returnCallback( function ( $req ) {
$this->assertSame( 'UTSysop', $req->username );
$this->assertFalse( $req->done );
$req->done = true;
} ) );
$provider->providerRevokeAccessForUser( 'UTSysop' );
foreach ( $reqs as $i => $req ) {
$this->assertTrue( $req->done, "#$i" );
}
}
/**
* @dataProvider providePrimaryAccountLink
* @param string $type PrimaryAuthenticationProvider::TYPE_* constant
* @param string $msg Error message from beginPrimaryAccountLink
*/
public function testPrimaryAccountLink( $type, $msg ) {
$provider = $this->getMockForAbstractClass( AbstractPrimaryAuthenticationProvider::class );
$provider->expects( $this->any() )->method( 'accountCreationType' )
->will( $this->returnValue( $type ) );
$class = AbstractPrimaryAuthenticationProvider::class;
$msg1 = "{$class}::beginPrimaryAccountLink $msg";
$msg2 = "{$class}::continuePrimaryAccountLink is not implemented.";
$user = \User::newFromName( 'Whatever' );
try {
$provider->beginPrimaryAccountLink( $user, [] );
$this->fail( 'Expected exception not thrown' );
} catch ( \BadMethodCallException $ex ) {
$this->assertSame( $msg1, $ex->getMessage() );
}
try {
$provider->continuePrimaryAccountLink( $user, [] );
$this->fail( 'Expected exception not thrown' );
} catch ( \BadMethodCallException $ex ) {
$this->assertSame( $msg2, $ex->getMessage() );
}
}
public static function providePrimaryAccountLink() {
return [
[
PrimaryAuthenticationProvider::TYPE_NONE,
'should not be called on a non-link provider.',
],
[
PrimaryAuthenticationProvider::TYPE_CREATE,
'should not be called on a non-link provider.',
],
[
PrimaryAuthenticationProvider::TYPE_LINK,
'is not implemented.',
],
];
}
/**
* @dataProvider provideProviderNormalizeUsername
*/
public function testProviderNormalizeUsername( $name, $expect ) {
// fake interwiki map for the 'Interwiki prefix' testcase
$this->mergeMwGlobalArrayValue( 'wgHooks', [
'InterwikiLoadPrefix' => [
function ( $prefix, &$iwdata ) {
if ( $prefix === 'interwiki' ) {
$iwdata = [
'iw_url' => 'http://example.com/',
'iw_local' => 0,
'iw_trans' => 0,
];
return false;
}
},
],
] );
$provider = $this->getMockForAbstractClass( AbstractPrimaryAuthenticationProvider::class );
$this->assertSame( $expect, $provider->providerNormalizeUsername( $name ) );
}
public static function provideProviderNormalizeUsername() {
return [
'Leading space' => [ ' Leading space', 'Leading space' ],
'Trailing space ' => [ 'Trailing space ', 'Trailing space' ],
'Namespace prefix' => [ 'Talk:Username', null ],
'Interwiki prefix' => [ 'interwiki:Username', null ],
'With hash' => [ 'name with # hash', null ],
'Multi spaces' => [ 'Multi spaces', 'Multi spaces' ],
'Lowercase' => [ 'lowercase', 'Lowercase' ],
'Invalid character' => [ 'in[]valid', null ],
'With slash' => [ 'with / slash', null ],
'Underscores' => [ '___under__scores___', 'Under scores' ],
];
}
}