2014-02-20 17:05:45 +04:00
|
|
|
<?php
|
|
|
|
/**
|
2015-03-26 13:44:34 +03:00
|
|
|
* @author Andreas Fischer <bantu@owncloud.com>
|
|
|
|
* @author Arthur Schiwon <blizzz@owncloud.com>
|
|
|
|
* @author Joas Schilling <nickvergessen@owncloud.com>
|
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
2016-01-12 17:02:16 +03:00
|
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
2015-03-26 13:44:34 +03:00
|
|
|
*
|
2016-01-12 17:02:16 +03:00
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
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,
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*
|
|
|
|
*/
|
2015-02-26 13:37:37 +03:00
|
|
|
|
2014-02-20 17:05:45 +04:00
|
|
|
namespace OCA\user_ldap\tests;
|
|
|
|
|
|
|
|
use \OCA\user_ldap\lib\Access;
|
|
|
|
use \OCA\user_ldap\lib\Connection;
|
|
|
|
use \OCA\user_ldap\lib\ILDAPWrapper;
|
|
|
|
|
2015-11-25 18:58:54 +03:00
|
|
|
/**
|
|
|
|
* Class Test_Access
|
|
|
|
*
|
|
|
|
* @group DB
|
|
|
|
*
|
|
|
|
* @package OCA\user_ldap\tests
|
|
|
|
*/
|
2014-11-11 00:28:12 +03:00
|
|
|
class Test_Access extends \Test\TestCase {
|
2015-08-21 01:55:42 +03:00
|
|
|
private function getConnectorAndLdapMock() {
|
2014-02-20 17:05:45 +04:00
|
|
|
static $conMethods;
|
|
|
|
static $accMethods;
|
2014-03-31 12:48:38 +04:00
|
|
|
static $umMethods;
|
2014-02-20 17:05:45 +04:00
|
|
|
|
|
|
|
if(is_null($conMethods) || is_null($accMethods)) {
|
|
|
|
$conMethods = get_class_methods('\OCA\user_ldap\lib\Connection');
|
|
|
|
$accMethods = get_class_methods('\OCA\user_ldap\lib\Access');
|
2014-03-31 12:48:38 +04:00
|
|
|
$umMethods = get_class_methods('\OCA\user_ldap\lib\user\Manager');
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
|
|
|
$lw = $this->getMock('\OCA\user_ldap\lib\ILDAPWrapper');
|
|
|
|
$connector = $this->getMock('\OCA\user_ldap\lib\Connection',
|
|
|
|
$conMethods,
|
|
|
|
array($lw, null, null));
|
2014-03-31 12:48:38 +04:00
|
|
|
$um = $this->getMock('\OCA\user_ldap\lib\user\Manager',
|
|
|
|
$umMethods, array(
|
|
|
|
$this->getMock('\OCP\IConfig'),
|
|
|
|
$this->getMock('\OCA\user_ldap\lib\FilesystemHelper'),
|
|
|
|
$this->getMock('\OCA\user_ldap\lib\LogWrapper'),
|
|
|
|
$this->getMock('\OCP\IAvatarManager'),
|
2015-01-07 02:52:18 +03:00
|
|
|
$this->getMock('\OCP\Image'),
|
2016-01-19 17:48:38 +03:00
|
|
|
$this->getMock('\OCP\IDBConnection'),
|
|
|
|
$this->getMock('\OCP\IUserManager')));
|
2014-02-20 17:05:45 +04:00
|
|
|
|
2014-03-31 12:48:38 +04:00
|
|
|
return array($lw, $connector, $um);
|
2014-02-20 17:05:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartValidChars() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-03-31 12:48:38 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
2014-02-20 17:05:45 +04:00
|
|
|
|
|
|
|
$input = 'okay';
|
|
|
|
$this->assertTrue($input === $access->escapeFilterPart($input));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartEscapeWildcard() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-03-31 12:48:38 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
2014-02-20 17:05:45 +04:00
|
|
|
|
|
|
|
$input = '*';
|
|
|
|
$expected = '\\\\*';
|
|
|
|
$this->assertTrue($expected === $access->escapeFilterPart($input));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testEscapeFilterPartEscapeWildcard2() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-03-31 12:48:38 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
2014-02-20 17:05:45 +04:00
|
|
|
|
|
|
|
$input = 'foo*bar';
|
|
|
|
$expected = 'foo\\\\*bar';
|
|
|
|
$this->assertTrue($expected === $access->escapeFilterPart($input));
|
|
|
|
}
|
2014-07-02 00:02:41 +04:00
|
|
|
|
2014-09-19 02:01:57 +04:00
|
|
|
/** @dataProvider convertSID2StrSuccessData */
|
|
|
|
public function testConvertSID2StrSuccess(array $sidArray, $sidExpected) {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-07-02 00:02:41 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
2014-09-19 02:01:57 +04:00
|
|
|
$sidBinary = implode('', $sidArray);
|
2014-07-02 00:02:41 +04:00
|
|
|
$this->assertSame($sidExpected, $access->convertSID2Str($sidBinary));
|
|
|
|
}
|
|
|
|
|
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() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-07-02 00:02:41 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
|
|
|
$sidIllegal = 'foobar';
|
|
|
|
$sidExpected = '';
|
|
|
|
|
|
|
|
$this->assertSame($sidExpected, $access->convertSID2Str($sidIllegal));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetDomainDNFromDNSuccess() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-07-02 00:02:41 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
|
|
|
$inputDN = 'uid=zaphod,cn=foobar,dc=my,dc=server,dc=com';
|
|
|
|
$domainDN = 'dc=my,dc=server,dc=com';
|
|
|
|
|
|
|
|
$lw->expects($this->once())
|
|
|
|
->method('explodeDN')
|
|
|
|
->with($inputDN, 0)
|
|
|
|
->will($this->returnValue(explode(',', $inputDN)));
|
|
|
|
|
|
|
|
$this->assertSame($domainDN, $access->getDomainDNFromDN($inputDN));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGetDomainDNFromDNError() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-07-02 00:02:41 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
|
|
|
$inputDN = 'foobar';
|
|
|
|
$expected = '';
|
|
|
|
|
|
|
|
$lw->expects($this->once())
|
|
|
|
->method('explodeDN')
|
|
|
|
->with($inputDN, 0)
|
|
|
|
->will($this->returnValue(false));
|
|
|
|
|
|
|
|
$this->assertSame($expected, $access->getDomainDNFromDN($inputDN));
|
|
|
|
}
|
2014-08-11 18:40:41 +04:00
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
private function getResemblesDNInputData() {
|
|
|
|
return $cases = array(
|
|
|
|
array(
|
|
|
|
'input' => 'foo=bar,bar=foo,dc=foobar',
|
|
|
|
'interResult' => array(
|
|
|
|
'count' => 3,
|
|
|
|
0 => 'foo=bar',
|
|
|
|
1 => 'bar=foo',
|
|
|
|
2 => 'dc=foobar'
|
|
|
|
),
|
|
|
|
'expectedResult' => true
|
|
|
|
),
|
|
|
|
array(
|
|
|
|
'input' => 'foobarbarfoodcfoobar',
|
|
|
|
'interResult' => false,
|
|
|
|
'expectedResult' => false
|
|
|
|
)
|
2014-08-11 18:40:41 +04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
public function testStringResemblesDN() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-08-11 18:40:41 +04:00
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
$cases = $this->getResemblesDNInputData();
|
2014-08-11 18:40:41 +04:00
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
$lw->expects($this->exactly(2))
|
2014-08-11 18:40:41 +04:00
|
|
|
->method('explodeDN')
|
2014-08-18 18:55:08 +04:00
|
|
|
->will($this->returnCallback(function ($dn) use ($cases) {
|
|
|
|
foreach($cases as $case) {
|
|
|
|
if($dn === $case['input']) {
|
|
|
|
return $case['interResult'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
foreach($cases as $case) {
|
|
|
|
$this->assertSame($case['expectedResult'], $access->stringResemblesDN($case['input']));
|
|
|
|
}
|
2014-08-11 18:40:41 +04:00
|
|
|
}
|
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
public function testStringResemblesDNLDAPmod() {
|
2015-08-21 01:55:42 +03:00
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
2014-08-11 18:40:41 +04:00
|
|
|
$lw = new \OCA\user_ldap\lib\LDAP();
|
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
|
|
|
if(!function_exists('ldap_explode_dn')) {
|
|
|
|
$this->markTestSkipped('LDAP Module not available');
|
|
|
|
}
|
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
$cases = $this->getResemblesDNInputData();
|
2014-08-11 18:40:41 +04:00
|
|
|
|
2014-08-18 18:55:08 +04:00
|
|
|
foreach($cases as $case) {
|
|
|
|
$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() {
|
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
|
|
|
|
$con->expects($this->once())
|
|
|
|
->method('writeToCache');
|
|
|
|
|
|
|
|
$access->cacheUserHome('foobar', '/foobars/path');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBatchApplyUserAttributes() {
|
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
$mapperMock = $this->getMockBuilder('\OCA\User_LDAP\Mapping\UserMapping')
|
|
|
|
->disableOriginalConstructor()
|
|
|
|
->getMock();
|
2015-10-27 21:03:40 +03:00
|
|
|
|
|
|
|
$mapperMock->expects($this->any())
|
|
|
|
->method('getNameByDN')
|
|
|
|
->will($this->returnValue('a_username'));
|
|
|
|
|
2015-08-21 01:55:42 +03:00
|
|
|
$userMock = $this->getMockBuilder('\OCA\user_ldap\lib\user\User')
|
|
|
|
->disableOriginalConstructor()
|
|
|
|
->getMock();
|
|
|
|
|
2015-10-27 21:03:40 +03:00
|
|
|
$access->connection->expects($this->any())
|
|
|
|
->method('__get')
|
|
|
|
->will($this->returnValue('displayName'));
|
|
|
|
|
2015-08-21 01:55:42 +03:00
|
|
|
$access->setUserMapper($mapperMock);
|
|
|
|
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute = strtolower($access->connection->ldapUserDisplayName);
|
2015-08-21 01:55:42 +03:00
|
|
|
$data = array(
|
|
|
|
array(
|
|
|
|
'dn' => 'foobar',
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'barfoo'
|
2015-08-21 01:55:42 +03:00
|
|
|
),
|
|
|
|
array(
|
|
|
|
'dn' => 'foo',
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'bar'
|
2015-08-21 01:55:42 +03:00
|
|
|
),
|
|
|
|
array(
|
|
|
|
'dn' => 'raboof',
|
2015-10-27 21:03:40 +03:00
|
|
|
$displayNameAttribute => 'oofrab'
|
2015-08-21 01:55:42 +03:00
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$userMock->expects($this->exactly(count($data)))
|
|
|
|
->method('processAttributes');
|
|
|
|
|
|
|
|
$um->expects($this->exactly(count($data)))
|
|
|
|
->method('get')
|
|
|
|
->will($this->returnValue($userMock));
|
|
|
|
|
|
|
|
$access->batchApplyUserAttributes($data);
|
|
|
|
}
|
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
|
|
|
|
*/
|
|
|
|
public function testSanitizeDN($attribute) {
|
|
|
|
list($lw, $con, $um) = $this->getConnectorAndLdapMock();
|
|
|
|
|
|
|
|
|
|
|
|
$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)
|
|
|
|
)));
|
|
|
|
|
|
|
|
$access = new Access($con, $lw, $um);
|
|
|
|
$values = $access->readAttribute('uid=whoever,dc=example,dc=org', $attribute);
|
|
|
|
$this->assertSame($values[0], strtolower($dnFromServer));
|
|
|
|
}
|
2014-07-02 00:02:41 +04:00
|
|
|
}
|