2021-01-05 22:08:41 +00:00
|
|
|
<?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
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
namespace MediaWiki\Tests\Unit\Permissions;
|
|
|
|
|
|
|
|
|
|
use InvalidArgumentException;
|
|
|
|
|
use MediaWiki\Page\PageIdentity;
|
|
|
|
|
use MediaWiki\Page\PageIdentityValue;
|
|
|
|
|
use MediaWiki\Permissions\Authority;
|
|
|
|
|
use MediaWiki\Permissions\PermissionManager;
|
|
|
|
|
use MediaWiki\Permissions\PermissionStatus;
|
2021-01-07 20:57:19 +00:00
|
|
|
use MediaWiki\Permissions\UserAuthority;
|
2021-01-05 22:08:41 +00:00
|
|
|
use MediaWikiUnitTestCase;
|
|
|
|
|
use PHPUnit\Framework\MockObject\MockObject;
|
|
|
|
|
use User;
|
|
|
|
|
|
|
|
|
|
/**
|
2021-01-07 20:57:19 +00:00
|
|
|
* @covers \MediaWiki\Permissions\UserAuthority
|
2021-01-05 22:08:41 +00:00
|
|
|
*/
|
2021-01-07 20:57:19 +00:00
|
|
|
class UserAuthorityTest extends MediaWikiUnitTestCase {
|
2021-01-05 22:08:41 +00:00
|
|
|
|
|
|
|
|
private function newPermissionsManager( array $permissions ): PermissionManager {
|
|
|
|
|
/** @var PermissionManager|MockObject $manager */
|
|
|
|
|
$manager = $this->createNoOpMock(
|
|
|
|
|
PermissionManager::class,
|
|
|
|
|
[ 'userHasRight', 'userCan', 'getPermissionErrors' ]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$manager->method( 'userHasRight' )->willReturnCallback(
|
2021-02-07 13:10:36 +00:00
|
|
|
static function ( $user, $permission ) use ( $permissions ) {
|
2021-01-05 22:08:41 +00:00
|
|
|
return in_array( $permission, $permissions );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$manager->method( 'userCan' )->willReturnCallback(
|
2021-02-07 13:10:36 +00:00
|
|
|
static function ( $permission, $user ) use ( $manager ) {
|
2021-01-05 22:08:41 +00:00
|
|
|
return $manager->userHasRight( $user, $permission );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$manager->method( 'getPermissionErrors' )->willReturnCallback(
|
2021-02-07 13:10:36 +00:00
|
|
|
static function ( $permission, $user, $target ) use ( $manager ) {
|
2021-01-05 22:08:41 +00:00
|
|
|
$errors = [];
|
|
|
|
|
if ( !$manager->userCan( $permission, $user, $target ) ) {
|
|
|
|
|
$errors[] = [ 'permissionserrors' ];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $errors;
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return $manager;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private function newUser(): User {
|
|
|
|
|
return $this->createNoOpMock( User::class );
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-07 20:57:19 +00:00
|
|
|
private function newAuthority( User $actor, array $permissions ): Authority {
|
2021-01-05 22:08:41 +00:00
|
|
|
$permissionManager = $this->newPermissionsManager( $permissions );
|
2021-01-07 20:57:19 +00:00
|
|
|
return new UserAuthority(
|
|
|
|
|
$actor,
|
2021-01-05 22:08:41 +00:00
|
|
|
$permissionManager
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testGetAuthor() {
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [] );
|
|
|
|
|
|
2021-03-04 19:45:28 +00:00
|
|
|
$this->assertSame( $actor, $authority->getUser() );
|
2021-01-05 22:08:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testPermissions() {
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->isAllowed( 'foo' ) );
|
|
|
|
|
$this->assertTrue( $authority->isAllowed( 'bar' ) );
|
|
|
|
|
$this->assertFalse( $authority->isAllowed( 'quux' ) );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->isAllowedAll( 'foo', 'bar' ) );
|
|
|
|
|
$this->assertTrue( $authority->isAllowedAny( 'bar', 'quux' ) );
|
|
|
|
|
|
|
|
|
|
$this->assertFalse( $authority->isAllowedAll( 'foo', 'quux' ) );
|
|
|
|
|
$this->assertFalse( $authority->isAllowedAny( 'xyzzy', 'quux' ) );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testProbablyCan() {
|
|
|
|
|
$target = new PageIdentityValue( 321, NS_MAIN, __METHOD__, PageIdentity::LOCAL );
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->probablyCan( 'foo', $target ) );
|
|
|
|
|
$this->assertTrue( $authority->probablyCan( 'bar', $target ) );
|
|
|
|
|
$this->assertFalse( $authority->probablyCan( 'quux', $target ) );
|
|
|
|
|
|
|
|
|
|
$status = new PermissionStatus();
|
|
|
|
|
$authority->probablyCan( 'foo', $target, $status );
|
|
|
|
|
$this->assertTrue( $status->isOK() );
|
|
|
|
|
|
|
|
|
|
$authority->probablyCan( 'quux', $target, $status );
|
|
|
|
|
$this->assertFalse( $status->isOK() );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testDefinitlyCan() {
|
|
|
|
|
$target = new PageIdentityValue( 321, NS_MAIN, __METHOD__, PageIdentity::LOCAL );
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->definitelyCan( 'foo', $target ) );
|
|
|
|
|
$this->assertTrue( $authority->definitelyCan( 'bar', $target ) );
|
|
|
|
|
$this->assertFalse( $authority->definitelyCan( 'quux', $target ) );
|
|
|
|
|
|
|
|
|
|
$status = new PermissionStatus();
|
|
|
|
|
$authority->definitelyCan( 'foo', $target, $status );
|
|
|
|
|
$this->assertTrue( $status->isOK() );
|
|
|
|
|
|
|
|
|
|
$authority->definitelyCan( 'quux', $target, $status );
|
|
|
|
|
$this->assertFalse( $status->isOK() );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testAuthorizeRead() {
|
|
|
|
|
$target = new PageIdentityValue( 321, NS_MAIN, __METHOD__, PageIdentity::LOCAL );
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->authorizeRead( 'foo', $target ) );
|
|
|
|
|
$this->assertTrue( $authority->authorizeRead( 'bar', $target ) );
|
|
|
|
|
$this->assertFalse( $authority->authorizeRead( 'quux', $target ) );
|
|
|
|
|
|
|
|
|
|
$status = new PermissionStatus();
|
|
|
|
|
$authority->authorizeRead( 'foo', $target, $status );
|
|
|
|
|
$this->assertTrue( $status->isOK() );
|
|
|
|
|
|
|
|
|
|
$authority->authorizeRead( 'quux', $target, $status );
|
|
|
|
|
$this->assertFalse( $status->isOK() );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testAuthorizeWrite() {
|
|
|
|
|
$target = new PageIdentityValue( 321, NS_MAIN, __METHOD__, PageIdentity::LOCAL );
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->assertTrue( $authority->authorizeWrite( 'foo', $target ) );
|
|
|
|
|
$this->assertTrue( $authority->authorizeWrite( 'bar', $target ) );
|
|
|
|
|
$this->assertFalse( $authority->authorizeWrite( 'quux', $target ) );
|
|
|
|
|
|
|
|
|
|
$status = new PermissionStatus();
|
|
|
|
|
$authority->authorizeWrite( 'foo', $target, $status );
|
|
|
|
|
$this->assertTrue( $status->isOK() );
|
|
|
|
|
|
|
|
|
|
$authority->authorizeWrite( 'quux', $target, $status );
|
|
|
|
|
$this->assertFalse( $status->isOK() );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testIsAllowedAnyThrowsOnEmptySet() {
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->expectException( InvalidArgumentException::class );
|
|
|
|
|
$authority->isAllowedAny();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function testIsAllowedAllThrowsOnEmptySet() {
|
|
|
|
|
$actor = $this->newUser();
|
|
|
|
|
$authority = $this->newAuthority( $actor, [ 'foo', 'bar' ] );
|
|
|
|
|
|
|
|
|
|
$this->expectException( InvalidArgumentException::class );
|
|
|
|
|
$authority->isAllowedAll();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|