aboutsummaryrefslogtreecommitdiffstats
path: root/apps/dav/lib/DAV/CustomPropertiesBackend.php
diff options
context:
space:
mode:
Diffstat (limited to 'apps/dav/lib/DAV/CustomPropertiesBackend.php')
-rw-r--r--apps/dav/lib/DAV/CustomPropertiesBackend.php100
1 files changed, 95 insertions, 5 deletions
diff --git a/apps/dav/lib/DAV/CustomPropertiesBackend.php b/apps/dav/lib/DAV/CustomPropertiesBackend.php
index f9a4f8ee986..be7345f25df 100644
--- a/apps/dav/lib/DAV/CustomPropertiesBackend.php
+++ b/apps/dav/lib/DAV/CustomPropertiesBackend.php
@@ -9,13 +9,20 @@
namespace OCA\DAV\DAV;
use Exception;
+use OCA\DAV\CalDAV\CalDavBackend;
use OCA\DAV\CalDAV\Calendar;
+use OCA\DAV\CalDAV\CalendarHome;
use OCA\DAV\CalDAV\CalendarObject;
use OCA\DAV\CalDAV\DefaultCalendarValidator;
+use OCA\DAV\CalDAV\Integration\ExternalCalendar;
+use OCA\DAV\CalDAV\Outbox;
+use OCA\DAV\CalDAV\Trashbin\TrashbinHome;
use OCA\DAV\Connector\Sabre\Directory;
+use OCA\DAV\Db\PropertyMapper;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IDBConnection;
use OCP\IUser;
+use Sabre\CalDAV\Schedule\Inbox;
use Sabre\DAV\Exception as DavException;
use Sabre\DAV\PropertyStorage\Backend\BackendInterface;
use Sabre\DAV\PropFind;
@@ -97,11 +104,17 @@ class CustomPropertiesBackend implements BackendInterface {
];
/**
+ * Map of well-known property names to default values
+ */
+ private const PROPERTY_DEFAULT_VALUES = [
+ '{http://owncloud.org/ns}calendar-enabled' => '1',
+ ];
+
+ /**
* Properties cache
- *
- * @var array
*/
- private $userCache = [];
+ private array $userCache = [];
+ private array $publishedCache = [];
private XmlService $xmlService;
/**
@@ -114,6 +127,7 @@ class CustomPropertiesBackend implements BackendInterface {
private Tree $tree,
private IDBConnection $connection,
private IUser $user,
+ private PropertyMapper $propertyMapper,
private DefaultCalendarValidator $defaultCalendarValidator,
) {
$this->xmlService = new XmlService();
@@ -197,6 +211,13 @@ class CustomPropertiesBackend implements BackendInterface {
$this->cacheDirectory($path, $node);
}
+ if ($node instanceof CalendarHome && $propFind->getDepth() !== 0) {
+ $backend = $node->getCalDAVBackend();
+ if ($backend instanceof CalDavBackend) {
+ $this->cacheCalendars($node, $requestedProps);
+ }
+ }
+
if ($node instanceof CalendarObject) {
// No custom properties supported on individual events
return;
@@ -316,6 +337,10 @@ class CustomPropertiesBackend implements BackendInterface {
return [];
}
+ if (isset($this->publishedCache[$path])) {
+ return $this->publishedCache[$path];
+ }
+
$qb = $this->connection->getQueryBuilder();
$qb->select('*')
->from(self::TABLE_NAME)
@@ -326,6 +351,7 @@ class CustomPropertiesBackend implements BackendInterface {
$props[$row['propertyname']] = $this->decodeValueFromDatabase($row['propertyvalue'], $row['valuetype']);
}
$result->closeCursor();
+ $this->publishedCache[$path] = $props;
return $props;
}
@@ -364,6 +390,62 @@ class CustomPropertiesBackend implements BackendInterface {
$this->userCache = array_merge($this->userCache, $propsByPath);
}
+ private function cacheCalendars(CalendarHome $node, array $requestedProperties): void {
+ $calendars = $node->getChildren();
+
+ $users = [];
+ foreach ($calendars as $calendar) {
+ if ($calendar instanceof Calendar) {
+ $user = str_replace('principals/users/', '', $calendar->getPrincipalURI());
+ if (!isset($users[$user])) {
+ $users[$user] = ['calendars/' . $user];
+ }
+ $users[$user][] = 'calendars/' . $user . '/' . $calendar->getUri();
+ } elseif ($calendar instanceof Inbox || $calendar instanceof Outbox || $calendar instanceof TrashbinHome || $calendar instanceof ExternalCalendar) {
+ if ($calendar->getOwner()) {
+ $user = str_replace('principals/users/', '', $calendar->getOwner());
+ if (!isset($users[$user])) {
+ $users[$user] = ['calendars/' . $user];
+ }
+ $users[$user][] = 'calendars/' . $user . '/' . $calendar->getName();
+ }
+ }
+ }
+
+ // user properties
+ $properties = $this->propertyMapper->findPropertiesByPathsAndUsers($users);
+
+ $propsByPath = [];
+ foreach ($users as $paths) {
+ foreach ($paths as $path) {
+ $propsByPath[$path] = [];
+ }
+ }
+
+ foreach ($properties as $property) {
+ $propsByPath[$property->getPropertypath()][$property->getPropertyname()] = $this->decodeValueFromDatabase($property->getPropertyvalue(), $property->getValuetype());
+ }
+ $this->userCache = array_merge($this->userCache, $propsByPath);
+
+ // published properties
+ $allowedProps = array_intersect(self::PUBLISHED_READ_ONLY_PROPERTIES, $requestedProperties);
+ if (empty($allowedProps)) {
+ return;
+ }
+ $paths = [];
+ foreach ($users as $nestedPaths) {
+ $paths = array_merge($paths, $nestedPaths);
+ }
+ $paths = array_unique($paths);
+
+ $propsByPath = array_fill_keys(array_values($paths), []);
+ $properties = $this->propertyMapper->findPropertiesByPaths($paths, $allowedProps);
+ foreach ($properties as $property) {
+ $propsByPath[$property->getPropertypath()][$property->getPropertyname()] = $this->decodeValueFromDatabase($property->getPropertyvalue(), $property->getValuetype());
+ }
+ $this->publishedCache = array_merge($this->publishedCache, $propsByPath);
+ }
+
/**
* Returns a list of properties for the given path and current user
*
@@ -410,6 +492,14 @@ class CustomPropertiesBackend implements BackendInterface {
return $props;
}
+ private function isPropertyDefaultValue(string $name, mixed $value): bool {
+ if (!isset(self::PROPERTY_DEFAULT_VALUES[$name])) {
+ return false;
+ }
+
+ return self::PROPERTY_DEFAULT_VALUES[$name] === $value;
+ }
+
/**
* @throws Exception
*/
@@ -426,8 +516,8 @@ class CustomPropertiesBackend implements BackendInterface {
'propertyName' => $propertyName,
];
- // If it was null, we need to delete the property
- if (is_null($propertyValue)) {
+ // If it was null or set to the default value, we need to delete the property
+ if (is_null($propertyValue) || $this->isPropertyDefaultValue($propertyName, $propertyValue)) {
if (array_key_exists($propertyName, $existing)) {
$deleteQuery = $deleteQuery ?? $this->createDeleteQuery();
$deleteQuery