2015-10-21 16:06:48 +03:00
|
|
|
<?php
|
|
|
|
/**
|
2016-07-21 17:49:16 +03:00
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
2017-02-25 16:26:02 +03:00
|
|
|
* @copyright Copyright (c) 2017, Georg Ehrke <oc.list@georgehrke.com>
|
2016-07-21 17:49:16 +03:00
|
|
|
*
|
2020-03-31 11:49:10 +03:00
|
|
|
* @author Georg Ehrke <oc.list@georgehrke.com>
|
|
|
|
* @author Robin Appelman <robin@icewind.nl>
|
2015-10-21 16:06:48 +03:00
|
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
|
|
|
*
|
|
|
|
* @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,
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
2015-10-21 16:06:48 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-03-27 20:15:51 +03:00
|
|
|
namespace OCA\DAV\DAV;
|
2015-10-21 16:06:48 +03:00
|
|
|
|
2020-01-31 17:06:26 +03:00
|
|
|
use OCA\DAV\Connector\Sabre\Node;
|
2015-10-21 16:06:48 +03:00
|
|
|
use OCP\IDBConnection;
|
|
|
|
use OCP\IUser;
|
|
|
|
use Sabre\DAV\PropertyStorage\Backend\BackendInterface;
|
|
|
|
use Sabre\DAV\PropFind;
|
|
|
|
use Sabre\DAV\PropPatch;
|
|
|
|
use Sabre\DAV\Tree;
|
|
|
|
|
|
|
|
class CustomPropertiesBackend implements BackendInterface {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Ignored properties
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
2020-01-31 18:10:19 +03:00
|
|
|
private $ignoredProperties = [
|
2015-10-21 16:06:48 +03:00
|
|
|
'{DAV:}getcontentlength',
|
|
|
|
'{DAV:}getcontenttype',
|
|
|
|
'{DAV:}getetag',
|
|
|
|
'{DAV:}quota-used-bytes',
|
|
|
|
'{DAV:}quota-available-bytes',
|
|
|
|
'{http://owncloud.org/ns}permissions',
|
|
|
|
'{http://owncloud.org/ns}downloadURL',
|
|
|
|
'{http://owncloud.org/ns}dDC',
|
|
|
|
'{http://owncloud.org/ns}size',
|
2018-08-28 19:27:53 +03:00
|
|
|
'{http://nextcloud.org/ns}is-encrypted',
|
2020-01-31 18:10:19 +03:00
|
|
|
];
|
2015-10-21 16:06:48 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var Tree
|
|
|
|
*/
|
|
|
|
private $tree;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var IDBConnection
|
|
|
|
*/
|
|
|
|
private $connection;
|
|
|
|
|
|
|
|
/**
|
2020-01-31 17:06:26 +03:00
|
|
|
* @var IUser
|
2015-10-21 16:06:48 +03:00
|
|
|
*/
|
|
|
|
private $user;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Properties cache
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $cache = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Tree $tree node tree
|
|
|
|
* @param IDBConnection $connection database connection
|
|
|
|
* @param IUser $user owner of the tree and properties
|
|
|
|
*/
|
|
|
|
public function __construct(
|
|
|
|
Tree $tree,
|
|
|
|
IDBConnection $connection,
|
|
|
|
IUser $user) {
|
|
|
|
$this->tree = $tree;
|
|
|
|
$this->connection = $connection;
|
2020-01-31 17:06:26 +03:00
|
|
|
$this->user = $user;
|
2015-10-21 16:06:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetches properties for a path.
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
* @param PropFind $propFind
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function propFind($path, PropFind $propFind) {
|
|
|
|
$requestedProps = $propFind->get404Properties();
|
|
|
|
|
|
|
|
// these might appear
|
|
|
|
$requestedProps = array_diff(
|
|
|
|
$requestedProps,
|
|
|
|
$this->ignoredProperties
|
|
|
|
);
|
|
|
|
|
2017-02-25 16:26:02 +03:00
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-21 16:06:48 +03:00
|
|
|
if (empty($requestedProps)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-03 15:37:30 +03:00
|
|
|
$props = $this->getProperties($path, $requestedProps);
|
2015-10-21 16:06:48 +03:00
|
|
|
foreach ($props as $propName => $propValue) {
|
|
|
|
$propFind->set($propName, $propValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates properties for a path
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
* @param PropPatch $propPatch
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function propPatch($path, PropPatch $propPatch) {
|
2020-02-03 15:37:30 +03:00
|
|
|
$propPatch->handleRemaining(function ($changedProps) use ($path) {
|
|
|
|
return $this->updateProperties($path, $changedProps);
|
2015-10-21 16:06:48 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method is called after a node is deleted.
|
|
|
|
*
|
|
|
|
* @param string $path path of node for which to delete properties
|
|
|
|
*/
|
|
|
|
public function delete($path) {
|
|
|
|
$statement = $this->connection->prepare(
|
|
|
|
'DELETE FROM `*PREFIX*properties` WHERE `userid` = ? AND `propertypath` = ?'
|
|
|
|
);
|
2020-01-31 18:10:19 +03:00
|
|
|
$statement->execute([$this->user->getUID(), $this->formatPath($path)]);
|
2015-10-21 16:06:48 +03:00
|
|
|
$statement->closeCursor();
|
|
|
|
|
|
|
|
unset($this->cache[$path]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method is called after a successful MOVE
|
|
|
|
*
|
|
|
|
* @param string $source
|
|
|
|
* @param string $destination
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function move($source, $destination) {
|
|
|
|
$statement = $this->connection->prepare(
|
|
|
|
'UPDATE `*PREFIX*properties` SET `propertypath` = ?' .
|
|
|
|
' WHERE `userid` = ? AND `propertypath` = ?'
|
|
|
|
);
|
2020-01-31 18:10:19 +03:00
|
|
|
$statement->execute([$this->formatPath($destination), $this->user->getUID(), $this->formatPath($source)]);
|
2015-10-21 16:06:48 +03:00
|
|
|
$statement->closeCursor();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of properties for this nodes.;
|
2020-01-31 18:10:19 +03:00
|
|
|
*
|
2020-02-03 15:37:30 +03:00
|
|
|
* @param string $path
|
2015-10-21 16:06:48 +03:00
|
|
|
* @param array $requestedProperties requested properties or empty array for "all"
|
|
|
|
* @return array
|
|
|
|
* @note The properties list is a list of propertynames the client
|
|
|
|
* requested, encoded as xmlnamespace#tagName, for example:
|
|
|
|
* http://www.example.org/namespace#author If the array is empty, all
|
|
|
|
* properties should be returned
|
|
|
|
*/
|
2020-02-03 15:37:30 +03:00
|
|
|
private function getProperties(string $path, array $requestedProperties) {
|
2015-10-21 16:06:48 +03:00
|
|
|
if (isset($this->cache[$path])) {
|
|
|
|
return $this->cache[$path];
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: chunking if more than 1000 properties
|
|
|
|
$sql = 'SELECT * FROM `*PREFIX*properties` WHERE `userid` = ? AND `propertypath` = ?';
|
|
|
|
|
2020-01-31 18:10:19 +03:00
|
|
|
$whereValues = [$this->user->getUID(), $this->formatPath($path)];
|
|
|
|
$whereTypes = [null, null];
|
2015-10-21 16:06:48 +03:00
|
|
|
|
|
|
|
if (!empty($requestedProperties)) {
|
|
|
|
// request only a subset
|
|
|
|
$sql .= ' AND `propertyname` in (?)';
|
|
|
|
$whereValues[] = $requestedProperties;
|
|
|
|
$whereTypes[] = \Doctrine\DBAL\Connection::PARAM_STR_ARRAY;
|
|
|
|
}
|
|
|
|
|
|
|
|
$result = $this->connection->executeQuery(
|
|
|
|
$sql,
|
|
|
|
$whereValues,
|
|
|
|
$whereTypes
|
|
|
|
);
|
|
|
|
|
|
|
|
$props = [];
|
|
|
|
while ($row = $result->fetch()) {
|
|
|
|
$props[$row['propertyname']] = $row['propertyvalue'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$result->closeCursor();
|
|
|
|
|
|
|
|
$this->cache[$path] = $props;
|
|
|
|
return $props;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update properties
|
|
|
|
*
|
2020-02-03 15:37:30 +03:00
|
|
|
* @param string $path path for which to update properties
|
2015-10-21 16:06:48 +03:00
|
|
|
* @param array $properties array of properties to update
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2020-02-03 15:37:30 +03:00
|
|
|
private function updateProperties(string $path, array $properties) {
|
2015-10-21 16:06:48 +03:00
|
|
|
|
|
|
|
$deleteStatement = 'DELETE FROM `*PREFIX*properties`' .
|
|
|
|
' WHERE `userid` = ? AND `propertypath` = ? AND `propertyname` = ?';
|
|
|
|
|
|
|
|
$insertStatement = 'INSERT INTO `*PREFIX*properties`' .
|
|
|
|
' (`userid`,`propertypath`,`propertyname`,`propertyvalue`) VALUES(?,?,?,?)';
|
|
|
|
|
|
|
|
$updateStatement = 'UPDATE `*PREFIX*properties` SET `propertyvalue` = ?' .
|
|
|
|
' WHERE `userid` = ? AND `propertypath` = ? AND `propertyname` = ?';
|
|
|
|
|
|
|
|
// TODO: use "insert or update" strategy ?
|
2020-02-03 15:37:30 +03:00
|
|
|
$existing = $this->getProperties($path, []);
|
2015-10-21 16:06:48 +03:00
|
|
|
$this->connection->beginTransaction();
|
|
|
|
foreach ($properties as $propertyName => $propertyValue) {
|
|
|
|
// If it was null, we need to delete the property
|
|
|
|
if (is_null($propertyValue)) {
|
|
|
|
if (array_key_exists($propertyName, $existing)) {
|
|
|
|
$this->connection->executeUpdate($deleteStatement,
|
2020-01-31 18:10:19 +03:00
|
|
|
[
|
2020-01-31 17:06:26 +03:00
|
|
|
$this->user->getUID(),
|
2020-01-31 18:10:19 +03:00
|
|
|
$this->formatPath($path),
|
|
|
|
$propertyName,
|
|
|
|
]
|
2015-10-21 16:06:48 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!array_key_exists($propertyName, $existing)) {
|
|
|
|
$this->connection->executeUpdate($insertStatement,
|
2020-01-31 18:10:19 +03:00
|
|
|
[
|
2020-01-31 17:06:26 +03:00
|
|
|
$this->user->getUID(),
|
2020-01-31 18:10:19 +03:00
|
|
|
$this->formatPath($path),
|
2015-10-21 16:06:48 +03:00
|
|
|
$propertyName,
|
2020-01-31 18:10:19 +03:00
|
|
|
$propertyValue,
|
|
|
|
]
|
2015-10-21 16:06:48 +03:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$this->connection->executeUpdate($updateStatement,
|
2020-01-31 18:10:19 +03:00
|
|
|
[
|
2015-10-21 16:06:48 +03:00
|
|
|
$propertyValue,
|
2020-01-31 17:06:26 +03:00
|
|
|
$this->user->getUID(),
|
2020-01-31 18:10:19 +03:00
|
|
|
$this->formatPath($path),
|
|
|
|
$propertyName,
|
|
|
|
]
|
2015-10-21 16:06:48 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->connection->commit();
|
|
|
|
unset($this->cache[$path]);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2020-01-31 18:10:19 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* long paths are hashed to ensure they fit in the database
|
|
|
|
*
|
|
|
|
* @param string $path
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function formatPath(string $path): string {
|
|
|
|
if (strlen($path) > 250) {
|
|
|
|
return sha1($path);
|
|
|
|
} else {
|
|
|
|
return $path;
|
|
|
|
}
|
|
|
|
}
|
2015-10-21 16:06:48 +03:00
|
|
|
}
|