allow sharees to edit certain calendar properties for themselves
Signed-off-by: Georg Ehrke <developer@georgehrke.com>
This commit is contained in:
parent
dc5ba95469
commit
b887adf386
|
@ -175,13 +175,12 @@ class Calendar extends \Sabre\CalDAV\Calendar implements IShareable {
|
||||||
}
|
}
|
||||||
|
|
||||||
function propPatch(PropPatch $propPatch) {
|
function propPatch(PropPatch $propPatch) {
|
||||||
$mutations = $propPatch->getMutations();
|
// parent::propPatch will only update calendars table
|
||||||
// If this is a shared calendar, the user can only change the enabled property, to hide it.
|
// if calendar is shared, changes have to be made to the properties table
|
||||||
if ($this->isShared() && (sizeof($mutations) !== 1 || !isset($mutations['{http://owncloud.org/ns}calendar-enabled']))) {
|
if (!$this->isShared()) {
|
||||||
throw new Forbidden();
|
|
||||||
}
|
|
||||||
parent::propPatch($propPatch);
|
parent::propPatch($propPatch);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function getChild($name) {
|
function getChild($name) {
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
<?php
|
<?php
|
||||||
/**
|
/**
|
||||||
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
||||||
|
* @copyright Copyright (c) 2017, Georg Ehrke <oc.list@georgehrke.com>
|
||||||
*
|
*
|
||||||
* @author Robin Appelman <robin@icewind.nl>
|
* @author Robin Appelman <robin@icewind.nl>
|
||||||
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
||||||
|
* @author Georg Ehrke <oc.list@georgehrke.com>
|
||||||
*
|
*
|
||||||
* @license AGPL-3.0
|
* @license AGPL-3.0
|
||||||
*
|
*
|
||||||
|
@ -102,6 +104,26 @@ class CustomPropertiesBackend implements BackendInterface {
|
||||||
$this->ignoredProperties
|
$this->ignoredProperties
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// substr of calendars/ => path is inside the CalDAV component
|
||||||
|
// two '/' => this a calendar (no calendar-home nor calendar object)
|
||||||
|
if (substr($path, 0, 10) === 'calendars/' && substr_count($path, '/') === 2) {
|
||||||
|
$allRequestedProps = $propFind->getRequestedProperties();
|
||||||
|
$customPropertiesForShares = [
|
||||||
|
'{DAV:}displayname',
|
||||||
|
'{urn:ietf:params:xml:ns:caldav}calendar-description',
|
||||||
|
'{urn:ietf:params:xml:ns:caldav}calendar-timezone',
|
||||||
|
'{http://apple.com/ns/ical/}calendar-order',
|
||||||
|
'{http://apple.com/ns/ical/}calendar-color',
|
||||||
|
'{urn:ietf:params:xml:ns:caldav}schedule-calendar-transp',
|
||||||
|
];
|
||||||
|
|
||||||
|
foreach ($customPropertiesForShares as $customPropertyForShares) {
|
||||||
|
if (in_array($customPropertyForShares, $allRequestedProps)) {
|
||||||
|
$requestedProps[] = $customPropertyForShares;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (empty($requestedProps)) {
|
if (empty($requestedProps)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,7 +109,7 @@ class CalendarTest extends TestCase {
|
||||||
['user1', 'user2', [], true],
|
['user1', 'user2', [], true],
|
||||||
['user1', 'user2', [
|
['user1', 'user2', [
|
||||||
'{http://owncloud.org/ns}calendar-enabled' => true,
|
'{http://owncloud.org/ns}calendar-enabled' => true,
|
||||||
], false],
|
], true],
|
||||||
['user1', 'user2', [
|
['user1', 'user2', [
|
||||||
'{DAV:}displayname' => true,
|
'{DAV:}displayname' => true,
|
||||||
], true],
|
], true],
|
||||||
|
@ -134,7 +134,7 @@ class CalendarTest extends TestCase {
|
||||||
/**
|
/**
|
||||||
* @dataProvider dataPropPatch
|
* @dataProvider dataPropPatch
|
||||||
*/
|
*/
|
||||||
public function testPropPatch($ownerPrincipal, $principalUri, $mutations, $throws) {
|
public function testPropPatch($ownerPrincipal, $principalUri, $mutations, $shared) {
|
||||||
/** @var \PHPUnit_Framework_MockObject_MockObject | CalDavBackend $backend */
|
/** @var \PHPUnit_Framework_MockObject_MockObject | CalDavBackend $backend */
|
||||||
$backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
|
$backend = $this->getMockBuilder(CalDavBackend::class)->disableOriginalConstructor()->getMock();
|
||||||
$calendarInfo = [
|
$calendarInfo = [
|
||||||
|
@ -144,15 +144,16 @@ class CalendarTest extends TestCase {
|
||||||
'uri' => 'default'
|
'uri' => 'default'
|
||||||
];
|
];
|
||||||
$c = new Calendar($backend, $calendarInfo, $this->l10n);
|
$c = new Calendar($backend, $calendarInfo, $this->l10n);
|
||||||
|
$propPatch = new PropPatch($mutations);
|
||||||
|
|
||||||
if ($throws) {
|
if (!$shared) {
|
||||||
$this->setExpectedException('\Sabre\DAV\Exception\Forbidden');
|
$backend->expects($this->once())
|
||||||
|
->method('updateCalendar')
|
||||||
|
->with(666, $propPatch);
|
||||||
}
|
}
|
||||||
$c->propPatch(new PropPatch($mutations));
|
$c->propPatch($propPatch);
|
||||||
if (!$throws) {
|
|
||||||
$this->assertTrue(true);
|
$this->assertTrue(true);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @dataProvider providesReadOnlyInfo
|
* @dataProvider providesReadOnlyInfo
|
||||||
|
|
|
@ -0,0 +1,170 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* @copyright Copyright (c) 2017, Georg Ehrke <oc.list@georgehrke.com>
|
||||||
|
*
|
||||||
|
* @author Georg Ehrke <oc.list@georgehrke.com>
|
||||||
|
*
|
||||||
|
* @license GNU AGPL version 3 or any later version
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as
|
||||||
|
* published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
namespace OCA\DAV\Tests\Files;
|
||||||
|
|
||||||
|
use OCA\DAV\Files\CustomPropertiesBackend;
|
||||||
|
use OCP\IDBConnection;
|
||||||
|
use OCP\IUser;
|
||||||
|
use Sabre\DAV\PropFind;
|
||||||
|
use Sabre\DAV\PropPatch;
|
||||||
|
use Sabre\DAV\Tree;
|
||||||
|
use Test\TestCase;
|
||||||
|
|
||||||
|
class CustomPropertiesBackendTest extends TestCase {
|
||||||
|
|
||||||
|
/** @var Tree | \PHPUnit_Framework_MockObject_MockObject */
|
||||||
|
private $tree;
|
||||||
|
|
||||||
|
/** @var IDBConnection | \PHPUnit_Framework_MockObject_MockObject */
|
||||||
|
private $dbConnection;
|
||||||
|
|
||||||
|
/** @var IUser | \PHPUnit_Framework_MockObject_MockObject */
|
||||||
|
private $user;
|
||||||
|
|
||||||
|
/** @var CustomPropertiesBackend | \PHPUnit_Framework_MockObject_MockObject */
|
||||||
|
private $backend;
|
||||||
|
|
||||||
|
public function setUp() {
|
||||||
|
parent::setUp();
|
||||||
|
|
||||||
|
$this->tree = $this->createMock(Tree::class);
|
||||||
|
$this->dbConnection = $this->createMock(IDBConnection::class);
|
||||||
|
$this->user = $this->createMock(IUser::class);
|
||||||
|
$this->user->expects($this->once())
|
||||||
|
->method('getUID')
|
||||||
|
->with()
|
||||||
|
->will($this->returnValue('dummy_user_42'));
|
||||||
|
|
||||||
|
$this->backend = new CustomPropertiesBackend($this->tree,
|
||||||
|
$this->dbConnection, $this->user);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testPropFindNoDbCalls() {
|
||||||
|
$propFind = $this->createMock(PropFind::class);
|
||||||
|
$propFind->expects($this->at(0))
|
||||||
|
->method('get404Properties')
|
||||||
|
->with()
|
||||||
|
->will($this->returnValue([
|
||||||
|
'{http://owncloud.org/ns}permissions',
|
||||||
|
'{http://owncloud.org/ns}downloadURL',
|
||||||
|
'{http://owncloud.org/ns}dDC',
|
||||||
|
'{http://owncloud.org/ns}size',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$this->dbConnection->expects($this->never())
|
||||||
|
->method($this->anything());
|
||||||
|
|
||||||
|
$this->backend->propFind('foo_bar_path_1337_0', $propFind);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testPropFindCalendarCall() {
|
||||||
|
$propFind = $this->createMock(PropFind::class);
|
||||||
|
$propFind->expects($this->at(0))
|
||||||
|
->method('get404Properties')
|
||||||
|
->with()
|
||||||
|
->will($this->returnValue([
|
||||||
|
'{DAV:}getcontentlength',
|
||||||
|
'{DAV:}getcontenttype',
|
||||||
|
'{DAV:}getetag',
|
||||||
|
'{abc}def'
|
||||||
|
]));
|
||||||
|
|
||||||
|
$propFind->expects($this->at(1))
|
||||||
|
->method('getRequestedProperties')
|
||||||
|
->with()
|
||||||
|
->will($this->returnValue([
|
||||||
|
'{DAV:}getcontentlength',
|
||||||
|
'{DAV:}getcontenttype',
|
||||||
|
'{DAV:}getetag',
|
||||||
|
'{DAV:}displayname',
|
||||||
|
'{urn:ietf:params:xml:ns:caldav}calendar-description',
|
||||||
|
'{urn:ietf:params:xml:ns:caldav}calendar-timezone',
|
||||||
|
'{abc}def'
|
||||||
|
]));
|
||||||
|
|
||||||
|
$statement = $this->createMock('\Doctrine\DBAL\Driver\Statement');
|
||||||
|
$this->dbConnection->expects($this->once())
|
||||||
|
->method('executeQuery')
|
||||||
|
->with('SELECT * FROM `*PREFIX*properties` WHERE `userid` = ? AND `propertypath` = ? AND `propertyname` in (?)',
|
||||||
|
['dummy_user_42', 'calendars/foo/bar_path_1337_0', [
|
||||||
|
3 => '{abc}def',
|
||||||
|
4 => '{DAV:}displayname',
|
||||||
|
5 => '{urn:ietf:params:xml:ns:caldav}calendar-description',
|
||||||
|
6 => '{urn:ietf:params:xml:ns:caldav}calendar-timezone']],
|
||||||
|
[null, null, 102])
|
||||||
|
->will($this->returnValue($statement));
|
||||||
|
|
||||||
|
$this->backend->propFind('calendars/foo/bar_path_1337_0', $propFind);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider propPatchProvider
|
||||||
|
*/
|
||||||
|
public function testPropPatch($path, $propPatch) {
|
||||||
|
$propPatch->expects($this->once())
|
||||||
|
->method('handleRemaining');
|
||||||
|
|
||||||
|
$this->backend->propPatch($path, $propPatch);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function propPatchProvider() {
|
||||||
|
$propPatchMock = $this->createMock(PropPatch::class);
|
||||||
|
return [
|
||||||
|
['foo_bar_path_1337', $propPatchMock],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testDelete() {
|
||||||
|
$statement = $this->createMock('\Doctrine\DBAL\Driver\Statement');
|
||||||
|
$statement->expects($this->at(0))
|
||||||
|
->method('execute')
|
||||||
|
->with(['dummy_user_42', 'foo_bar_path_1337']);
|
||||||
|
$statement->expects($this->at(1))
|
||||||
|
->method('closeCursor')
|
||||||
|
->with();
|
||||||
|
|
||||||
|
$this->dbConnection->expects($this->at(0))
|
||||||
|
->method('prepare')
|
||||||
|
->with('DELETE FROM `*PREFIX*properties` WHERE `userid` = ? AND `propertypath` = ?')
|
||||||
|
->will($this->returnValue($statement));
|
||||||
|
|
||||||
|
$this->backend->delete('foo_bar_path_1337');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testMove() {
|
||||||
|
$statement = $this->createMock('\Doctrine\DBAL\Driver\Statement');
|
||||||
|
$statement->expects($this->at(0))
|
||||||
|
->method('execute')
|
||||||
|
->with(['bar_foo_path_7331', 'dummy_user_42', 'foo_bar_path_1337']);
|
||||||
|
$statement->expects($this->at(1))
|
||||||
|
->method('closeCursor')
|
||||||
|
->with();
|
||||||
|
|
||||||
|
$this->dbConnection->expects($this->at(0))
|
||||||
|
->method('prepare')
|
||||||
|
->with('UPDATE `*PREFIX*properties` SET `propertypath` = ? WHERE `userid` = ? AND `propertypath` = ?')
|
||||||
|
->will($this->returnValue($statement));
|
||||||
|
|
||||||
|
$this->backend->move('foo_bar_path_1337', 'bar_foo_path_7331');
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue