2016-11-04 14:05:15 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @copyright Copyright (c) 2016, Roeland Jago Douma <roeland@famdouma.nl>
|
2016-11-15 16:01:40 +00:00
|
|
|
* @copyright Copyright (c) 2016, Joas Schilling <coding@schilljs.com>
|
2016-11-04 14:05:15 +00:00
|
|
|
*
|
2016-11-15 16:01:40 +00:00
|
|
|
* @author Joas Schilling <coding@schilljs.com>
|
2016-11-04 14:05:15 +00:00
|
|
|
* @author Roeland Jago Douma <roeland@famdouma.nl>
|
|
|
|
*
|
|
|
|
* @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\CalDAV\Schedule;
|
|
|
|
|
2016-11-15 16:01:40 +00:00
|
|
|
use OCA\DAV\CalDAV\CalDavBackend;
|
|
|
|
use OCA\DAV\CalDAV\CalendarHome;
|
|
|
|
use Sabre\DAV\INode;
|
|
|
|
use Sabre\DAV\PropFind;
|
|
|
|
use Sabre\DAV\Server;
|
|
|
|
use Sabre\DAV\Xml\Property\LocalHref;
|
|
|
|
use Sabre\DAVACL\IPrincipal;
|
2018-05-25 18:26:36 +00:00
|
|
|
use Sabre\HTTP\RequestInterface;
|
|
|
|
use Sabre\HTTP\ResponseInterface;
|
|
|
|
use Sabre\VObject\Component\VCalendar;
|
|
|
|
use Sabre\VObject\Reader;
|
2016-11-15 16:01:40 +00:00
|
|
|
|
2016-11-04 14:05:15 +00:00
|
|
|
class Plugin extends \Sabre\CalDAV\Schedule\Plugin {
|
|
|
|
|
2016-11-15 16:01:40 +00:00
|
|
|
/**
|
|
|
|
* Initializes the plugin
|
|
|
|
*
|
|
|
|
* @param Server $server
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function initialize(Server $server) {
|
|
|
|
parent::initialize($server);
|
|
|
|
$server->on('propFind', [$this, 'propFindDefaultCalendarUrl'], 90);
|
|
|
|
}
|
|
|
|
|
2016-11-04 14:05:15 +00:00
|
|
|
/**
|
|
|
|
* Returns a list of addresses that are associated with a principal.
|
|
|
|
*
|
|
|
|
* @param string $principal
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getAddressesForPrincipal($principal) {
|
|
|
|
$result = parent::getAddressesForPrincipal($principal);
|
|
|
|
|
|
|
|
if ($result === null) {
|
|
|
|
$result = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
2016-11-15 16:01:40 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Always use the personal calendar as target for scheduled events
|
|
|
|
*
|
|
|
|
* @param PropFind $propFind
|
|
|
|
* @param INode $node
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function propFindDefaultCalendarUrl(PropFind $propFind, INode $node) {
|
|
|
|
if ($node instanceof IPrincipal) {
|
|
|
|
$propFind->handle('{' . self::NS_CALDAV . '}schedule-default-calendar-URL', function() use ($node) {
|
|
|
|
/** @var \OCA\DAV\CalDAV\Plugin $caldavPlugin */
|
|
|
|
$caldavPlugin = $this->server->getPlugin('caldav');
|
|
|
|
$principalUrl = $node->getPrincipalUrl();
|
|
|
|
|
|
|
|
$calendarHomePath = $caldavPlugin->getCalendarHomeForPrincipal($principalUrl);
|
|
|
|
if (!$calendarHomePath) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-05-28 18:12:13 +00:00
|
|
|
if (strpos($principalUrl, 'principals/users') === 0) {
|
|
|
|
$uri = CalDavBackend::PERSONAL_CALENDAR_URI;
|
|
|
|
$displayname = CalDavBackend::PERSONAL_CALENDAR_NAME;
|
|
|
|
} elseif (strpos($principalUrl, 'principals/calendar-resources') === 0 ||
|
|
|
|
strpos($principalUrl, 'principals/calendar-rooms') === 0) {
|
|
|
|
$uri = CalDavBackend::RESOURCE_BOOKING_CALENDAR_URI;
|
|
|
|
$displayname = CalDavBackend::RESOURCE_BOOKING_CALENDAR_NAME;
|
|
|
|
} else {
|
|
|
|
// How did we end up here?
|
|
|
|
// TODO - throw exception or just ignore?
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-11-15 16:01:40 +00:00
|
|
|
/** @var CalendarHome $calendarHome */
|
|
|
|
$calendarHome = $this->server->tree->getNodeForPath($calendarHomePath);
|
2018-05-28 18:12:13 +00:00
|
|
|
if (!$calendarHome->childExists($uri)) {
|
|
|
|
$calendarHome->getCalDAVBackend()->createCalendar($principalUrl, $uri, [
|
|
|
|
'{DAV:}displayname' => $displayname,
|
2016-11-15 16:01:40 +00:00
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2018-05-28 18:12:13 +00:00
|
|
|
$result = $this->server->getPropertiesForPath($calendarHomePath . '/' . $uri, [], 1);
|
2016-11-15 16:01:40 +00:00
|
|
|
if (empty($result)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new LocalHref($result[0]['href']);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2018-05-25 18:26:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This method is triggered whenever there was a calendar object gets
|
|
|
|
* created or updated.
|
|
|
|
*
|
|
|
|
* Basically just a copy of parent::calendarObjectChange, with the change
|
|
|
|
* from:
|
|
|
|
* $addresses = $this->getAddressesForPrincipal($calendarNode->getOwner());
|
|
|
|
* to:
|
|
|
|
* $addresses = $this->getAddressesForPrincipal($calendarNode->getPrincipalURI());
|
|
|
|
*
|
|
|
|
* @param RequestInterface $request HTTP request
|
|
|
|
* @param ResponseInterface $response HTTP Response
|
|
|
|
* @param VCalendar $vCal Parsed iCalendar object
|
|
|
|
* @param mixed $calendarPath Path to calendar collection
|
|
|
|
* @param mixed $modified The iCalendar object has been touched.
|
|
|
|
* @param mixed $isNew Whether this was a new item or we're updating one
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function calendarObjectChange(RequestInterface $request, ResponseInterface $response, VCalendar $vCal, $calendarPath, &$modified, $isNew) {
|
|
|
|
|
|
|
|
if (!$this->scheduleReply($this->server->httpRequest)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$calendarNode = $this->server->tree->getNodeForPath($calendarPath);
|
|
|
|
|
|
|
|
$addresses = $this->getAddressesForPrincipal(
|
|
|
|
$calendarNode->getPrincipalURI()
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!$isNew) {
|
|
|
|
$node = $this->server->tree->getNodeForPath($request->getPath());
|
|
|
|
$oldObj = Reader::read($node->get());
|
|
|
|
} else {
|
|
|
|
$oldObj = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->processICalendarChange($oldObj, $vCal, $addresses, [], $modified);
|
|
|
|
|
|
|
|
if ($oldObj) {
|
|
|
|
// Destroy circular references so PHP will GC the object.
|
|
|
|
$oldObj->destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method checks the 'Schedule-Reply' header
|
|
|
|
* and returns false if it's 'F', otherwise true.
|
|
|
|
*
|
|
|
|
* Copied from Sabre/DAV's Schedule plugin, because it's
|
|
|
|
* private for whatever reason
|
|
|
|
*
|
|
|
|
* @param RequestInterface $request
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
private function scheduleReply(RequestInterface $request) {
|
|
|
|
|
|
|
|
$scheduleReply = $request->getHeader('Schedule-Reply');
|
|
|
|
return $scheduleReply !== 'F';
|
|
|
|
|
|
|
|
}
|
2016-11-04 14:05:15 +00:00
|
|
|
}
|