2014-02-20 17:05:45 +04:00
|
|
|
<?php
|
|
|
|
/**
|
2016-07-21 17:49:16 +03:00
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
2016-11-23 21:58:43 +03:00
|
|
|
* @copyright Copyright (c) 2016, Lukas Reschke <lukas@statuscode.ch>
|
2016-07-21 17:49:16 +03:00
|
|
|
*
|
2015-03-26 13:44:34 +03:00
|
|
|
* @author Andreas Fischer <bantu@owncloud.com>
|
2016-05-26 20:56:05 +03:00
|
|
|
* @author Arthur Schiwon <blizzz@arthur-schiwon.de>
|
2016-07-21 17:49:16 +03:00
|
|
|
* @author Joas Schilling <coding@schilljs.com>
|
2017-11-06 17:56:42 +03:00
|
|
|
* @author Jörn Friedrich Dreyer <jfd@butonic.de>
|
|
|
|
* @author Lukas Reschke <lukas@statuscode.ch>
|
2015-03-26 13:44:34 +03:00
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
2017-11-06 17:56:42 +03:00
|
|
|
* @author Roeland Jago Douma <roeland@famdouma.nl>
|
|
|
|
* @author Roger Szabo <roger.szabo@web.de>
|
|
|
|
* @author root <root@localhost.localdomain>
|
2016-01-12 17:02:16 +03:00
|
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
2017-11-06 17:56:42 +03:00
|
|
|
* @author Victor Dubiniuk <dubiniuk@owncloud.com>
|
2015-03-26 13:44:34 +03:00
|
|
|
*
|
|
|
|
* @license AGPL-3.0
|
|
|
|
*
|
|
|
|
* This code is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
|
|
* as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* 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 Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License, version 3,
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
2015-03-26 13:44:34 +03:00
|
|
|
*
|
|
|
|
*/
|
2015-02-26 13:37:37 +03:00
|
|
|
|
2016-05-12 18:14:59 +03:00
|
|
|
namespace OCA\User_LDAP\Tests;
|
2014-02-20 17:05:45 +04:00
|
|
|
|
2016-05-12 17:42:57 +03:00
|
|
|
use OCA\User_LDAP\Access;
|
2016-10-13 16:34:40 +03:00
|
|
|
use OCA\User_LDAP\Connection;
|
2016-11-23 21:58:43 +03:00
|
|
|
use OCA\User_LDAP\Exceptions\ConstraintViolationException;
|
2016-09-02 12:02:12 +03:00
|
|
|
use OCA\User_LDAP\FilesystemHelper;
|
2016-10-13 16:34:40 +03:00
|
|
|
use OCA\User_LDAP\Helper;
|
2016-09-02 12:02:12 +03:00
|
|
|
use OCA\User_LDAP\ILDAPWrapper;
|
2016-11-23 21:58:43 +03:00
|
|
|
use OCA\User_LDAP\LDAP;
|
2016-09-02 12:02:12 +03:00
|
|
|
use OCA\User_LDAP\LogWrapper;
|
2017-10-31 14:20:44 +03:00
|
|
|
use OCA\User_LDAP\Mapping\UserMapping;
|
2016-10-13 16:34:40 +03:00
|
|
|
use OCA\User_LDAP\User\Manager;
|
2018-05-29 02:10:03 +03:00
|
|
|
use OCA\User_LDAP\User\OfflineUser;
|
2017-10-31 14:20:44 +03:00
|
|
|
use OCA\User_LDAP\User\User;
|
2016-09-02 12:02:12 +03:00
|
|
|
use OCP\IAvatarManager;
|
|
|
|
use OCP\IConfig;
|
|
|
|
use OCP\IDBConnection;
|
|
|
|
use OCP\Image;
|
|
|
|
use OCP\IUserManager;
|
2017-03-31 10:16:22 +03:00
|
|
|
use OCP\Notification\IManager as INotificationManager;
|
2017-10-31 14:20:44 +03:00
|
|
|
use Test\TestCase;
|
2014-02-20 17:05:45 +04:00
|
|
|
|
2015-11-25 18:58:54 +03:00
|
|
|
/**
|
2016-05-12 18:14:59 +03:00
|
|
|
* Class AccessTest
|
2015-11-25 18:58:54 +03:00
|
|
|
*
|
|
|
|
* @group DB
|
|
|
|
*
|
2016-05-12 18:14:59 +03:00
|
|
|
* @package OCA\User_LDAP\Tests
|
2015-11-25 18:58:54 +03:00
|
|
|
*/
|
2017-10-31 14:20:44 +03:00
|
|
|
class AccessTest extends TestCase {
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @var UserMapping|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
protected $userMapper;
|
2016-11-23 21:58:43 +03:00
|
|
|
/** @var Connection|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $connection;
|
|
|
|
/** @var LDAP|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $ldap;
|
|
|
|
/** @var Manager|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $userManager;
|
|
|
|
/** @var Helper|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $helper;
|
2017-11-08 03:51:14 +03:00
|
|
|
/** @var IConfig|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $config;
|
2018-03-19 17:53:30 +03:00
|
|
|
/** @var IUserManager|\PHPUnit_Framework_MockObject_MockObject */
|
|
|
|
private $ncUserManager;
|
2016-11-23 21:58:43 +03:00
|
|
|
/** @var Access */
|
|
|
|
private $access;
|
|
|
|
|
2019-11-27 17:27:18 +03:00
|
|
|
protected function setUp(): void {
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->connection = $this->createMock(Connection::class);
|
|
|
|
$this->ldap = $this->createMock(LDAP::class);
|
|
|
|
$this->userManager = $this->createMock(Manager::class);
|
|
|
|
$this->helper = $this->createMock(Helper::class);
|
2017-11-08 03:51:14 +03:00
|
|
|
$this->config = $this->createMock(IConfig::class);
|
2017-12-07 19:49:33 +03:00
|
|
|
$this->userMapper = $this->createMock(UserMapping::class);
|
2018-03-19 17:53:30 +03:00
|
|
|
$this->ncUserManager = $this->createMock(IUserManager::class);
|
2016-11-23 21:58:43 +03:00
|
|
|
|
|
|
|
$this->access = new Access(
|
|
|
|
$this->connection,
|
|
|
|
$this->ldap,
|
|
|
|
$this->userManager,
|
2017-10-31 20:13:48 +03:00
|
|
|
$this->helper,
|
2018-03-19 17:53:30 +03:00
|
|
|
$this->config,
|
|
|
|
$this->ncUserManager
|
2016-11-23 21:58:43 +03:00
|
|
|
);
|
2017-12-07 19:49:33 +03:00
|
|
|
$this->access->setUserMapper($this->userMapper);
|
2016-11-23 21:58:43 +03:00
|
|
|
}
|
|
|
|
|
2015-08-21 01:55:42 +03:00
|
|
|
private function getConnectorAndLdapMock() {
|
2016-09-02 12:02:12 +03:00
|
|
|
$lw = $this->createMock(ILDAPWrapper::class);
|
2016-10-13 16:34:40 +03:00
|
|
|
$connector = $this->getMockBuilder(Connection::class)
|
2016-09-27 23:54:37 +03:00
|
|
|
->setConstructorArgs([$lw, null, null])
|
|
|
|
->getMock();
|
2016-10-13 16:34:40 +03:00
|
|
|
$um = $this->getMockBuilder(Manager::class)
|
|
|
|
->setConstructorArgs([
|
2016-09-02 12:02:12 +03:00
|
|
|
$this->createMock(IConfig::class),
|
|
|
|
$this->createMock(FilesystemHelper::class),
|
|
|
|
$this->createMock(LogWrapper::class),
|
|
|
|
$this->createMock(IAvatarManager::class),
|
|
|
|
$this->createMock(Image::class),
|
|
|
|
$this->createMock(IDBConnection::class),
|
2017-03-31 10:16:22 +03:00
|
|
|
$this->createMock(IUserManager::class),
|
|
|
|
$this->createMock(INotificationManager::class)])
|
2016-09-27 23:54:37 +03:00
|
|
|
->getMock();
|
2016-10-13 16:34:40 +03:00
|
|
|
$helper = new Helper(\OC::$server->getConfig());
|
2014-02-20 17:05:45 +04:00
|
|
|
|
2016-07-22 11:46:29 +03:00
|
|
|
return array($lw, $connector, $um, $helper);
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartValidChars() {
|
|
|
|
$input = 'okay';
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertTrue($input === $this->access->escapeFilterPart($input));
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartEscapeWildcard() {
|
|
|
|
$input = '*';
|
|
|
|
$expected = '\\\\*';
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertTrue($expected === $this->access->escapeFilterPart($input));
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartEscapeWildcard2() {
|
|
|
|
$input = 'foo*bar';
|
|
|
|
$expected = 'foo\\\\*bar';
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertTrue($expected === $this->access->escapeFilterPart($input));
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
2014-07-02 00:02:41 +04:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
/**
|
|
|
|
* @dataProvider convertSID2StrSuccessData
|
|
|
|
* @param array $sidArray
|
|
|
|
* @param $sidExpected
|
|
|
|
*/
|
2014-09-19 02:01:57 +04:00
|
|
|
public function testConvertSID2StrSuccess(array $sidArray, $sidExpected) {
|
|
|
|
$sidBinary = implode('', $sidArray);
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($sidExpected, $this->access->convertSID2Str($sidBinary));
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|
|
|
|
|
2014-09-19 02:01:57 +04:00
|
|
|
public function convertSID2StrSuccessData() {
|
|
|
|
return array(
|
|
|
|
array(
|
|
|
|
array(
|
|
|
|
"\x01",
|
|
|
|
"\x04",
|
|
|
|
"\x00\x00\x00\x00\x00\x05",
|
|
|
|
"\x15\x00\x00\x00",
|
|
|
|
"\xa6\x81\xe5\x0e",
|
|
|
|
"\x4d\x6c\x6c\x2b",
|
|
|
|
"\xca\x32\x05\x5f",
|
|
|
|
),
|
|
|
|
'S-1-5-21-249921958-728525901-1594176202',
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
array(
|
|
|
|
"\x01",
|
|
|
|
"\x02",
|
|
|
|
"\xFF\xFF\xFF\xFF\xFF\xFF",
|
|
|
|
"\xFF\xFF\xFF\xFF",
|
|
|
|
"\xFF\xFF\xFF\xFF",
|
|
|
|
),
|
|
|
|
'S-1-281474976710655-4294967295-4294967295',
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-07-02 00:02:41 +04:00
|
|
|
public function testConvertSID2StrInputError() {
|
|
|
|
$sidIllegal = 'foobar';
|
|
|
|
$sidExpected = '';
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($sidExpected, $this->access->convertSID2Str($sidIllegal));
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetDomainDNFromDNSuccess() {
|
|
|
|
$inputDN = 'uid=zaphod,cn=foobar,dc=my,dc=server,dc=com';
|
|
|
|
$domainDN = 'dc=my,dc=server,dc=com';
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->ldap->expects($this->once())
|
2014-07-02 00:02:41 +04:00
|
|
|
->method('explodeDN')
|
|
|
|
->with($inputDN, 0)
|
|
|
|
->will($this->returnValue(explode(',', $inputDN)));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($domainDN, $this->access->getDomainDNFromDN($inputDN));
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetDomainDNFromDNError() {
|
|
|
|
$inputDN = 'foobar';
|
|
|
|
$expected = '';
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->ldap->expects($this->once())
|
2014-07-02 00:02:41 +04:00
|
|
|
->method('explodeDN')
|
|
|
|
->with($inputDN, 0)
|
|
|
|
->will($this->returnValue(false));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($expected, $this->access->getDomainDNFromDN($inputDN));
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|
2014-08-11 18:40:41 +04:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
public function dnInputDataProvider() {
|
|
|
|
return [[
|
|
|
|
[
|
2014-08-18 18:55:08 +04:00
|
|
|
'input' => 'foo=bar,bar=foo,dc=foobar',
|
|
|
|
'interResult' => array(
|
|
|
|
'count' => 3,
|
|
|
|
0 => 'foo=bar',
|
|
|
|
1 => 'bar=foo',
|
|
|
|
2 => 'dc=foobar'
|
|
|
|
),
|
|
|
|
'expectedResult' => true
|
2017-10-31 14:20:44 +03:00
|
|
|
],
|
|
|
|
[
|
2014-08-18 18:55:08 +04:00
|
|
|
'input' => 'foobarbarfoodcfoobar',
|
|
|
|
'interResult' => false,
|
|
|
|
'expectedResult' => false
|
2017-10-31 14:20:44 +03:00
|
|
|
]
|
|
|
|
]];
|
2014-08-11 18:40:41 +04:00
|
|
|
}
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
/**
|
|
|
|
* @dataProvider dnInputDataProvider
|
2017-12-07 19:49:33 +03:00
|
|
|
* @param array $case
|
2017-10-31 14:20:44 +03:00
|
|
|
*/
|
|
|
|
public function testStringResemblesDN($case) {
|
2016-07-22 11:46:29 +03:00
|
|
|
list($lw, $con, $um, $helper) = $this->getConnectorAndLdapMock();
|
2017-11-08 03:51:14 +03:00
|
|
|
/** @var IConfig|\PHPUnit_Framework_MockObject_MockObject $config */
|
|
|
|
$config = $this->createMock(IConfig::class);
|
2018-03-19 17:53:30 +03:00
|
|
|
$access = new Access($con, $lw, $um, $helper, $config, $this->ncUserManager);
|
2014-08-11 18:40:41 +04:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$lw->expects($this->exactly(1))
|
2014-08-11 18:40:41 +04:00
|
|
|
->method('explodeDN')
|
2017-10-31 14:20:44 +03:00
|
|
|
->will($this->returnCallback(function ($dn) use ($case) {
|
|
|
|
if($dn === $case['input']) {
|
|
|
|
return $case['interResult'];
|
2014-08-18 18:55:08 +04:00
|
|
|
}
|
2016-08-17 10:46:54 +03:00
|
|
|
return null;
|
2014-08-18 18:55:08 +04:00
|
|
|
}));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($case['expectedResult'], $access->stringResemblesDN($case['input']));
|
2014-08-11 18:40:41 +04:00
|
|
|
}
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
/**
|
|
|
|
* @dataProvider dnInputDataProvider
|
|
|
|
* @param $case
|
|
|
|
*/
|
|
|
|
public function testStringResemblesDNLDAPmod($case) {
|
|
|
|
list(, $con, $um, $helper) = $this->getConnectorAndLdapMock();
|
2017-11-08 03:51:14 +03:00
|
|
|
/** @var IConfig|\PHPUnit_Framework_MockObject_MockObject $config */
|
|
|
|
$config = $this->createMock(IConfig::class);
|
2017-10-31 14:20:44 +03:00
|
|
|
$lw = new LDAP();
|
2018-03-19 17:53:30 +03:00
|
|
|
$access = new Access($con, $lw, $um, $helper, $config, $this->ncUserManager);
|
2014-08-11 18:40:41 +04:00
|
|
|
|
|
|
|
if(!function_exists('ldap_explode_dn')) {
|
|
|
|
$this->markTestSkipped('LDAP Module not available');
|
|
|
|
}
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->assertSame($case['expectedResult'], $access->stringResemblesDN($case['input']));
|
2014-08-11 18:40:41 +04:00
|
|
|
}
|
2015-08-21 01:55:42 +03:00
|
|
|
|
|
|
|
public function testCacheUserHome() {
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->connection->expects($this->once())
|
2015-08-21 01:55:42 +03:00
|
|
|
->method('writeToCache');
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->access->cacheUserHome('foobar', '/foobars/path');
|
2015-08-21 01:55:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testBatchApplyUserAttributes() {
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->ldap->expects($this->any())
|
|
|
|
->method('isResource')
|
|
|
|
->willReturn(true);
|
|
|
|
|
|
|
|
$this->ldap->expects($this->any())
|
|
|
|
->method('getAttributes')
|
|
|
|
->willReturn(['displayname' => ['bar', 'count' => 1]]);
|
2015-10-27 21:03:40 +03:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
/** @var UserMapping|\PHPUnit_Framework_MockObject_MockObject $mapperMock */
|
|
|
|
$mapperMock = $this->createMock(UserMapping::class);
|
2015-10-27 21:03:40 +03:00
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('getNameByDN')
|
2017-10-31 14:20:44 +03:00
|
|
|
->willReturn(false);
|
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('map')
|
|
|
|
->willReturn(true);
|
2015-10-27 21:03:40 +03:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$userMock = $this->createMock(User::class);
|
2015-08-21 01:55:42 +03:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
// also returns for userUuidAttribute
|
|
|
|
$this->access->connection->expects($this->any())
|
2015-10-27 21:03:40 +03:00
|
|
|
->method('__get')
|
|
|
|
->will($this->returnValue('displayName'));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->access->setUserMapper($mapperMock);
|
2015-08-21 01:55:42 +03:00
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$displayNameAttribute = strtolower($this->access->connection->ldapUserDisplayName);
|
|
|
|
$data = [
|
|
|
|
[
|
|
|
|
'dn' => ['foobar'],
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'barfoo'
|
2017-10-31 14:20:44 +03:00
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['foo'],
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'bar'
|
2017-10-31 14:20:44 +03:00
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['raboof'],
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'oofrab'
|
2017-10-31 14:20:44 +03:00
|
|
|
]
|
|
|
|
];
|
2015-08-21 01:55:42 +03:00
|
|
|
|
|
|
|
$userMock->expects($this->exactly(count($data)))
|
|
|
|
->method('processAttributes');
|
|
|
|
|
2018-05-29 02:10:03 +03:00
|
|
|
$this->userManager->expects($this->exactly(count($data) * 2))
|
2015-08-21 01:55:42 +03:00
|
|
|
->method('get')
|
|
|
|
->will($this->returnValue($userMock));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->access->batchApplyUserAttributes($data);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBatchApplyUserAttributesSkipped() {
|
|
|
|
/** @var UserMapping|\PHPUnit_Framework_MockObject_MockObject $mapperMock */
|
|
|
|
$mapperMock = $this->createMock(UserMapping::class);
|
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('getNameByDN')
|
|
|
|
->will($this->returnValue('a_username'));
|
|
|
|
|
|
|
|
$userMock = $this->createMock(User::class);
|
|
|
|
|
|
|
|
$this->access->connection->expects($this->any())
|
|
|
|
->method('__get')
|
|
|
|
->will($this->returnValue('displayName'));
|
|
|
|
|
|
|
|
$this->access->setUserMapper($mapperMock);
|
|
|
|
|
|
|
|
$displayNameAttribute = strtolower($this->access->connection->ldapUserDisplayName);
|
|
|
|
$data = [
|
|
|
|
[
|
|
|
|
'dn' => ['foobar'],
|
|
|
|
$displayNameAttribute => 'barfoo'
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['foo'],
|
|
|
|
$displayNameAttribute => 'bar'
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['raboof'],
|
|
|
|
$displayNameAttribute => 'oofrab'
|
|
|
|
]
|
|
|
|
];
|
|
|
|
|
|
|
|
$userMock->expects($this->never())
|
|
|
|
->method('processAttributes');
|
|
|
|
|
2017-11-03 19:40:05 +03:00
|
|
|
$this->userManager->expects($this->any())
|
|
|
|
->method('get')
|
|
|
|
->willReturn($this->createMock(User::class));
|
2017-10-31 14:20:44 +03:00
|
|
|
|
2017-10-31 20:13:48 +03:00
|
|
|
$this->access->batchApplyUserAttributes($data);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBatchApplyUserAttributesDontSkip() {
|
|
|
|
/** @var UserMapping|\PHPUnit_Framework_MockObject_MockObject $mapperMock */
|
|
|
|
$mapperMock = $this->createMock(UserMapping::class);
|
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('getNameByDN')
|
|
|
|
->will($this->returnValue('a_username'));
|
|
|
|
|
|
|
|
$userMock = $this->createMock(User::class);
|
|
|
|
|
|
|
|
$this->access->connection->expects($this->any())
|
|
|
|
->method('__get')
|
|
|
|
->will($this->returnValue('displayName'));
|
|
|
|
|
|
|
|
$this->access->setUserMapper($mapperMock);
|
|
|
|
|
|
|
|
$displayNameAttribute = strtolower($this->access->connection->ldapUserDisplayName);
|
|
|
|
$data = [
|
|
|
|
[
|
|
|
|
'dn' => ['foobar'],
|
|
|
|
$displayNameAttribute => 'barfoo'
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['foo'],
|
|
|
|
$displayNameAttribute => 'bar'
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => ['raboof'],
|
|
|
|
$displayNameAttribute => 'oofrab'
|
|
|
|
]
|
|
|
|
];
|
|
|
|
|
|
|
|
$userMock->expects($this->exactly(count($data)))
|
|
|
|
->method('processAttributes');
|
|
|
|
|
2018-05-29 02:10:03 +03:00
|
|
|
$this->userManager->expects($this->exactly(count($data) * 2))
|
2017-10-31 20:13:48 +03:00
|
|
|
->method('get')
|
|
|
|
->will($this->returnValue($userMock));
|
|
|
|
|
2017-10-31 14:20:44 +03:00
|
|
|
$this->access->batchApplyUserAttributes($data);
|
2015-08-21 01:55:42 +03:00
|
|
|
}
|
2015-09-28 19:38:57 +03:00
|
|
|
|
|
|
|
public function dNAttributeProvider() {
|
|
|
|
// corresponds to Access::resemblesDN()
|
|
|
|
return array(
|
|
|
|
'dn' => array('dn'),
|
|
|
|
'uniqueMember' => array('uniquemember'),
|
|
|
|
'member' => array('member'),
|
|
|
|
'memberOf' => array('memberof')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider dNAttributeProvider
|
2017-10-31 14:20:44 +03:00
|
|
|
* @param $attribute
|
2015-09-28 19:38:57 +03:00
|
|
|
*/
|
|
|
|
public function testSanitizeDN($attribute) {
|
2016-07-22 11:46:29 +03:00
|
|
|
list($lw, $con, $um, $helper) = $this->getConnectorAndLdapMock();
|
2017-11-08 03:51:14 +03:00
|
|
|
/** @var IConfig|\PHPUnit_Framework_MockObject_MockObject $config */
|
|
|
|
$config = $this->createMock(IConfig::class);
|
2015-09-28 19:38:57 +03:00
|
|
|
|
|
|
|
$dnFromServer = 'cn=Mixed Cases,ou=Are Sufficient To,ou=Test,dc=example,dc=org';
|
|
|
|
|
|
|
|
$lw->expects($this->any())
|
|
|
|
->method('isResource')
|
|
|
|
->will($this->returnValue(true));
|
|
|
|
$lw->expects($this->any())
|
|
|
|
->method('getAttributes')
|
|
|
|
->will($this->returnValue(array(
|
|
|
|
$attribute => array('count' => 1, $dnFromServer)
|
|
|
|
)));
|
|
|
|
|
2018-03-19 17:53:30 +03:00
|
|
|
$access = new Access($con, $lw, $um, $helper, $config, $this->ncUserManager);
|
2015-09-28 19:38:57 +03:00
|
|
|
$values = $access->readAttribute('uid=whoever,dc=example,dc=org', $attribute);
|
|
|
|
$this->assertSame($values[0], strtolower($dnFromServer));
|
|
|
|
}
|
2016-11-23 21:58:43 +03:00
|
|
|
|
2019-11-27 17:27:18 +03:00
|
|
|
|
2016-11-23 21:58:43 +03:00
|
|
|
public function testSetPasswordWithDisabledChanges() {
|
2019-11-27 17:27:18 +03:00
|
|
|
$this->expectException(\Exception::class);
|
|
|
|
$this->expectExceptionMessage('LDAP password changes are disabled');
|
|
|
|
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->connection
|
|
|
|
->method('__get')
|
|
|
|
->willReturn(false);
|
|
|
|
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->access->setPassword('CN=foo', 'MyPassword');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetPasswordWithLdapNotAvailable() {
|
|
|
|
$this->connection
|
|
|
|
->method('__get')
|
|
|
|
->willReturn(true);
|
|
|
|
$connection = $this->createMock(LDAP::class);
|
|
|
|
$this->connection
|
|
|
|
->expects($this->once())
|
|
|
|
->method('getConnectionResource')
|
|
|
|
->willReturn($connection);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('isResource')
|
|
|
|
->with($connection)
|
|
|
|
->willReturn(false);
|
|
|
|
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->assertFalse($this->access->setPassword('CN=foo', 'MyPassword'));
|
|
|
|
}
|
|
|
|
|
2019-11-27 17:27:18 +03:00
|
|
|
|
2016-11-23 21:58:43 +03:00
|
|
|
public function testSetPasswordWithRejectedChange() {
|
2019-11-27 17:27:18 +03:00
|
|
|
$this->expectException(\OC\HintException::class);
|
|
|
|
$this->expectExceptionMessage('Password change rejected.');
|
|
|
|
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->connection
|
|
|
|
->method('__get')
|
|
|
|
->willReturn(true);
|
|
|
|
$connection = $this->createMock(LDAP::class);
|
|
|
|
$this->connection
|
|
|
|
->expects($this->once())
|
|
|
|
->method('getConnectionResource')
|
|
|
|
->willReturn($connection);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('isResource')
|
|
|
|
->with($connection)
|
|
|
|
->willReturn(true);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('modReplace')
|
|
|
|
->with($connection, 'CN=foo', 'MyPassword')
|
|
|
|
->willThrowException(new ConstraintViolationException());
|
|
|
|
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->access->setPassword('CN=foo', 'MyPassword');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSetPassword() {
|
|
|
|
$this->connection
|
|
|
|
->method('__get')
|
|
|
|
->willReturn(true);
|
|
|
|
$connection = $this->createMock(LDAP::class);
|
|
|
|
$this->connection
|
|
|
|
->expects($this->once())
|
|
|
|
->method('getConnectionResource')
|
|
|
|
->willReturn($connection);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('isResource')
|
|
|
|
->with($connection)
|
|
|
|
->willReturn(true);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('modReplace')
|
|
|
|
->with($connection, 'CN=foo', 'MyPassword')
|
|
|
|
->willReturn(true);
|
|
|
|
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
2016-11-23 21:58:43 +03:00
|
|
|
$this->assertTrue($this->access->setPassword('CN=foo', 'MyPassword'));
|
|
|
|
}
|
2017-12-07 19:49:33 +03:00
|
|
|
|
|
|
|
protected function prepareMocksForSearchTests(
|
|
|
|
$base,
|
|
|
|
$fakeConnection,
|
|
|
|
$fakeSearchResultResource,
|
|
|
|
$fakeLdapEntries
|
|
|
|
) {
|
|
|
|
$this->connection
|
|
|
|
->expects($this->any())
|
|
|
|
->method('getConnectionResource')
|
|
|
|
->willReturn($fakeConnection);
|
|
|
|
$this->connection->expects($this->any())
|
|
|
|
->method('__get')
|
|
|
|
->willReturnCallback(function($key) use ($base) {
|
|
|
|
if(stripos($key, 'base') !== false) {
|
|
|
|
return $base;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->any())
|
|
|
|
->method('isResource')
|
|
|
|
->willReturnCallback(function ($resource) use ($fakeConnection) {
|
|
|
|
return $resource === $fakeConnection;
|
|
|
|
});
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->any())
|
|
|
|
->method('errno')
|
|
|
|
->willReturn(0);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->once())
|
|
|
|
->method('search')
|
|
|
|
->willReturn([$fakeSearchResultResource]);
|
|
|
|
$this->ldap
|
|
|
|
->expects($this->exactly(count($base)))
|
|
|
|
->method('getEntries')
|
|
|
|
->willReturn($fakeLdapEntries);
|
|
|
|
|
|
|
|
$this->helper->expects($this->any())
|
|
|
|
->method('sanitizeDN')
|
|
|
|
->willReturnArgument(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSearchNoPagedSearch() {
|
|
|
|
// scenario: no pages search, 1 search base
|
|
|
|
$filter = 'objectClass=nextcloudUser';
|
|
|
|
$base = ['ou=zombies,dc=foobar,dc=nextcloud,dc=com'];
|
|
|
|
|
|
|
|
$fakeConnection = new \stdClass();
|
|
|
|
$fakeSearchResultResource = new \stdClass();
|
|
|
|
$fakeLdapEntries = [
|
|
|
|
'count' => 2,
|
|
|
|
[
|
|
|
|
'dn' => 'uid=sgarth,' . $base[0],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => 'uid=wwilson,' . $base[0],
|
|
|
|
]
|
|
|
|
];
|
|
|
|
|
|
|
|
$expected = $fakeLdapEntries;
|
|
|
|
unset($expected['count']);
|
|
|
|
|
|
|
|
$this->prepareMocksForSearchTests($base, $fakeConnection, $fakeSearchResultResource, $fakeLdapEntries);
|
|
|
|
|
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
|
|
|
$result = $this->access->search($filter, $base);
|
|
|
|
$this->assertSame($expected, $result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFetchListOfUsers() {
|
|
|
|
$filter = 'objectClass=nextcloudUser';
|
|
|
|
$base = ['ou=zombies,dc=foobar,dc=nextcloud,dc=com'];
|
|
|
|
$attrs = ['dn', 'uid'];
|
|
|
|
|
|
|
|
$fakeConnection = new \stdClass();
|
|
|
|
$fakeSearchResultResource = new \stdClass();
|
|
|
|
$fakeLdapEntries = [
|
|
|
|
'count' => 2,
|
|
|
|
[
|
|
|
|
'dn' => 'uid=sgarth,' . $base[0],
|
|
|
|
'uid' => [ 'sgarth' ],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'dn' => 'uid=wwilson,' . $base[0],
|
|
|
|
'uid' => [ 'wwilson' ],
|
|
|
|
]
|
|
|
|
];
|
|
|
|
$expected = $fakeLdapEntries;
|
|
|
|
unset($expected['count']);
|
|
|
|
array_walk($expected, function(&$v) {
|
|
|
|
$v['dn'] = [$v['dn']]; // dn is translated into an array internally for consistency
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->prepareMocksForSearchTests($base, $fakeConnection, $fakeSearchResultResource, $fakeLdapEntries);
|
|
|
|
|
|
|
|
$this->connection->expects($this->exactly($fakeLdapEntries['count']))
|
|
|
|
->method('writeToCache')
|
|
|
|
->with($this->stringStartsWith('userExists'), true);
|
|
|
|
|
2017-12-08 00:02:54 +03:00
|
|
|
$this->userMapper->expects($this->exactly($fakeLdapEntries['count']))
|
|
|
|
->method('getNameByDN')
|
|
|
|
->willReturnCallback(function($fdn) {
|
|
|
|
$parts = ldap_explode_dn($fdn, false);
|
|
|
|
return $parts[0];
|
|
|
|
});
|
|
|
|
|
2017-12-07 19:49:33 +03:00
|
|
|
/** @noinspection PhpUnhandledExceptionInspection */
|
|
|
|
$list = $this->access->fetchListOfUsers($filter, $attrs);
|
|
|
|
$this->assertSame($expected, $list);
|
|
|
|
}
|
|
|
|
|
2018-03-02 19:44:06 +03:00
|
|
|
public function intUsernameProvider() {
|
2018-03-05 16:03:08 +03:00
|
|
|
// system dependent :-/
|
|
|
|
$translitExpected = @iconv('UTF-8', 'ASCII//TRANSLIT', 'fränk') ? 'frank' : 'frnk';
|
|
|
|
|
2018-03-02 19:44:06 +03:00
|
|
|
return [
|
|
|
|
['alice', 'alice'],
|
|
|
|
['b/ob', 'bob'],
|
|
|
|
['charly🐬', 'charly'],
|
|
|
|
['debo rah', 'debo_rah'],
|
|
|
|
['epost@poste.test', 'epost@poste.test'],
|
2018-03-05 16:03:08 +03:00
|
|
|
['fränk', $translitExpected],
|
2018-03-02 19:44:06 +03:00
|
|
|
[' gerda ', 'gerda'],
|
|
|
|
['🕱🐵🐘🐑', null]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider intUsernameProvider
|
|
|
|
*
|
|
|
|
* @param $name
|
|
|
|
* @param $expected
|
|
|
|
*/
|
|
|
|
public function testSanitizeUsername($name, $expected) {
|
|
|
|
if($expected === null) {
|
|
|
|
$this->expectException(\InvalidArgumentException::class);
|
|
|
|
}
|
|
|
|
$sanitizedName = $this->access->sanitizeUsername($name);
|
|
|
|
$this->assertSame($expected, $sanitizedName);
|
|
|
|
}
|
|
|
|
|
2018-05-29 02:10:03 +03:00
|
|
|
public function testUserStateUpdate() {
|
|
|
|
$this->connection->expects($this->any())
|
|
|
|
->method('__get')
|
|
|
|
->willReturnMap([
|
|
|
|
[ 'ldapUserDisplayName', 'displayName' ],
|
|
|
|
[ 'ldapUserDisplayName2', null],
|
|
|
|
]);
|
|
|
|
|
|
|
|
$offlineUserMock = $this->createMock(OfflineUser::class);
|
|
|
|
$offlineUserMock->expects($this->once())
|
|
|
|
->method('unmark');
|
|
|
|
|
|
|
|
$regularUserMock = $this->createMock(User::class);
|
|
|
|
|
|
|
|
$this->userManager->expects($this->atLeastOnce())
|
|
|
|
->method('get')
|
|
|
|
->with('detta')
|
|
|
|
->willReturnOnConsecutiveCalls($offlineUserMock, $regularUserMock);
|
|
|
|
|
|
|
|
/** @var UserMapping|\PHPUnit_Framework_MockObject_MockObject $mapperMock */
|
|
|
|
$mapperMock = $this->createMock(UserMapping::class);
|
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('getNameByDN')
|
|
|
|
->with('uid=detta,ou=users,dc=hex,dc=ample')
|
|
|
|
->willReturn('detta');
|
|
|
|
$this->access->setUserMapper($mapperMock);
|
2018-03-02 19:44:06 +03:00
|
|
|
|
2018-05-29 02:10:03 +03:00
|
|
|
$records = [
|
|
|
|
[
|
|
|
|
'dn' => ['uid=detta,ou=users,dc=hex,dc=ample'],
|
|
|
|
'displayName' => ['Detta Detkova'],
|
|
|
|
]
|
|
|
|
];
|
|
|
|
$this->access->nextcloudUserNames($records);
|
|
|
|
}
|
2017-12-07 19:49:33 +03:00
|
|
|
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|