211 lines
5.4 KiB
PHP
211 lines
5.4 KiB
PHP
<?php
|
|
|
|
/**
|
|
* SPDX-FileCopyrightText: 2020 Nextcloud GmbH and Nextcloud contributors
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
*/
|
|
namespace OCA\Deck\DAV;
|
|
|
|
use OCA\DAV\CalDAV\Integration\ExternalCalendar;
|
|
use OCA\DAV\CalDAV\Plugin;
|
|
use OCA\Deck\Db\Acl;
|
|
use OCA\Deck\Db\Board;
|
|
use Sabre\CalDAV\CalendarQueryValidator;
|
|
use Sabre\CalDAV\Xml\Property\SupportedCalendarComponentSet;
|
|
use Sabre\DAV\Exception\Forbidden;
|
|
use Sabre\DAV\Exception\NotFound;
|
|
use Sabre\DAV\PropPatch;
|
|
use Sabre\VObject\InvalidDataException;
|
|
use Sabre\VObject\Reader;
|
|
|
|
class Calendar extends ExternalCalendar {
|
|
|
|
/** @var string */
|
|
private $principalUri;
|
|
/** @var string[] */
|
|
private $children;
|
|
/** @var DeckCalendarBackend */
|
|
private $backend;
|
|
/** @var Board */
|
|
private $board;
|
|
|
|
public function __construct(string $principalUri, string $calendarUri, Board $board, DeckCalendarBackend $backend) {
|
|
parent::__construct('deck', $calendarUri);
|
|
|
|
$this->backend = $backend;
|
|
$this->board = $board;
|
|
|
|
$this->principalUri = $principalUri;
|
|
}
|
|
|
|
public function getOwner() {
|
|
return $this->principalUri;
|
|
}
|
|
|
|
public function getACL() {
|
|
// the calendar should always have the read and the write-properties permissions
|
|
// write-properties is needed to allow the user to toggle the visibility of shared deck calendars
|
|
$acl = [
|
|
[
|
|
'privilege' => '{DAV:}read',
|
|
'principal' => $this->getOwner(),
|
|
'protected' => true,
|
|
],
|
|
[
|
|
'privilege' => '{DAV:}write-properties',
|
|
'principal' => $this->getOwner(),
|
|
'protected' => true,
|
|
]
|
|
];
|
|
|
|
return $acl;
|
|
}
|
|
|
|
public function setACL(array $acl) {
|
|
throw new Forbidden('Setting ACL is not supported on this node');
|
|
}
|
|
|
|
public function getSupportedPrivilegeSet() {
|
|
return null;
|
|
}
|
|
|
|
public function calendarQuery(array $filters) {
|
|
$result = [];
|
|
$objects = $this->getChildren();
|
|
|
|
foreach ($objects as $object) {
|
|
if ($this->validateFilterForObject($object, $filters)) {
|
|
$result[] = $object->getName();
|
|
}
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
protected function validateFilterForObject($object, array $filters) {
|
|
$vObject = Reader::read($object->get());
|
|
|
|
$validator = new CalendarQueryValidator();
|
|
$result = $validator->validate($vObject, $filters);
|
|
|
|
// Destroy circular references so PHP will GC the object.
|
|
$vObject->destroy();
|
|
|
|
return $result;
|
|
}
|
|
|
|
public function createFile($name, $data = null) {
|
|
throw new Forbidden('Creating a new entry is not implemented');
|
|
}
|
|
|
|
public function getChild($name) {
|
|
if ($this->childExists($name)) {
|
|
$card = array_values(array_filter(
|
|
$this->getBackendChildren(),
|
|
function ($card) use (&$name) {
|
|
return $card->getCalendarPrefix() . '-' . $card->getId() . '.ics' === $name;
|
|
}
|
|
));
|
|
if (count($card) > 0) {
|
|
return new CalendarObject($this, $name, $this->backend, $card[0]);
|
|
}
|
|
}
|
|
throw new NotFound('Node not found');
|
|
}
|
|
|
|
public function getChildren() {
|
|
$childNames = array_map(function ($card) {
|
|
return $card->getCalendarPrefix() . '-' . $card->getId() . '.ics';
|
|
}, $this->getBackendChildren());
|
|
|
|
$children = [];
|
|
|
|
foreach ($childNames as $name) {
|
|
$children[] = $this->getChild($name);
|
|
}
|
|
|
|
return $children;
|
|
}
|
|
|
|
private function getBackendChildren() {
|
|
if ($this->children) {
|
|
return $this->children;
|
|
}
|
|
|
|
if ($this->board) {
|
|
$this->children = $this->backend->getChildren($this->board->getId());
|
|
} else {
|
|
$this->children = [];
|
|
}
|
|
|
|
return $this->children;
|
|
}
|
|
|
|
public function childExists($name) {
|
|
return count(array_filter(
|
|
$this->getBackendChildren(),
|
|
function ($card) use (&$name) {
|
|
return $card->getCalendarPrefix() . '-' . $card->getId() . '.ics' === $name;
|
|
}
|
|
)) > 0;
|
|
}
|
|
|
|
|
|
public function delete() {
|
|
throw new Forbidden('Deleting an entry is not implemented');
|
|
}
|
|
|
|
public function getLastModified() {
|
|
return $this->board->getLastModified();
|
|
}
|
|
|
|
public function getGroup() {
|
|
return [];
|
|
}
|
|
|
|
public function propPatch(PropPatch $propPatch) {
|
|
$properties = [
|
|
'{DAV:}displayname',
|
|
'{http://apple.com/ns/ical/}calendar-color'
|
|
];
|
|
$propPatch->handle($properties, function ($properties) {
|
|
foreach ($properties as $key => $value) {
|
|
switch ($key) {
|
|
case '{DAV:}displayname':
|
|
if (!$this->backend->checkBoardPermission($this->board->getId(), Acl::PERMISSION_MANAGE)) {
|
|
throw new Forbidden('no permission to change the displayname');
|
|
}
|
|
if (mb_strpos($value, 'Deck: ') === 0) {
|
|
$value = mb_substr($value, strlen('Deck: '));
|
|
}
|
|
$this->board->setTitle($value);
|
|
break;
|
|
case '{http://apple.com/ns/ical/}calendar-color':
|
|
if (!$this->backend->checkBoardPermission($this->board->getId(), Acl::PERMISSION_MANAGE)) {
|
|
throw new Forbidden('no permission to change the calendar color');
|
|
}
|
|
$color = substr($value, 1, 6);
|
|
if (!preg_match('/[a-f0-9]{6}/i', $color)) {
|
|
throw new InvalidDataException('No valid color provided');
|
|
}
|
|
$this->board->setColor($color);
|
|
break;
|
|
}
|
|
}
|
|
return $this->backend->updateBoard($this->board);
|
|
});
|
|
// We can just return here and let oc_properties handle everything
|
|
}
|
|
|
|
/**
|
|
* @inheritDoc
|
|
*/
|
|
public function getProperties($properties) {
|
|
return [
|
|
'{DAV:}displayname' => 'Deck: ' . ($this->board ? $this->board->getTitle() : 'no board object provided'),
|
|
'{http://apple.com/ns/ical/}calendar-color' => '#' . $this->board->getColor(),
|
|
'{' . Plugin::NS_CALDAV . '}supported-calendar-component-set' => new SupportedCalendarComponentSet(['VTODO']),
|
|
];
|
|
}
|
|
}
|