summaryrefslogtreecommitdiffstats
path: root/apps
diff options
context:
space:
mode:
authorRobin Appelman <robin@icewind.nl>2023-07-09 02:41:10 +0200
committerbackportbot-nextcloud[bot] <backportbot-nextcloud[bot]@users.noreply.github.com>2023-07-10 20:19:22 +0000
commiteb44900d350f3b604ddc1b95afd6a80489b1c05d (patch)
tree579fb588b5013dffb09a7dc4f487be4b02dfff7b /apps
parent70e2243f0f8287121454eeb67389654ac8e11b04 (diff)
downloadnextcloud-server-eb44900d350f3b604ddc1b95afd6a80489b1c05d.tar.gz
nextcloud-server-eb44900d350f3b604ddc1b95afd6a80489b1c05d.zip
preload custom properties when propfinding folders
Signed-off-by: Robin Appelman <robin@icewind.nl>
Diffstat (limited to 'apps')
-rw-r--r--apps/dav/lib/DAV/CustomPropertiesBackend.php44
1 files changed, 42 insertions, 2 deletions
diff --git a/apps/dav/lib/DAV/CustomPropertiesBackend.php b/apps/dav/lib/DAV/CustomPropertiesBackend.php
index 3bc3ba33173..bc822aee974 100644
--- a/apps/dav/lib/DAV/CustomPropertiesBackend.php
+++ b/apps/dav/lib/DAV/CustomPropertiesBackend.php
@@ -22,9 +22,11 @@
* along with this program. If not, see <http://www.gnu.org/licenses/>
*
*/
+
namespace OCA\DAV\DAV;
use Exception;
+use OCA\DAV\Connector\Sabre\Directory;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IDBConnection;
use OCP\IUser;
@@ -134,7 +136,8 @@ class CustomPropertiesBackend implements BackendInterface {
public function __construct(
Tree $tree,
IDBConnection $connection,
- IUser $user) {
+ IUser $user
+ ) {
$this->tree = $tree;
$this->connection = $connection;
$this->user = $user;
@@ -180,6 +183,11 @@ class CustomPropertiesBackend implements BackendInterface {
return;
}
+ $node = $this->tree->getNodeForPath($path);
+ if ($node instanceof Directory && $propFind->getDepth() !== 0) {
+ $this->cacheDirectory($path, $node);
+ }
+
// First fetch the published properties (set by another user), then get the ones set by
// the current user. If both are set then the latter as priority.
foreach ($this->getPublishedProperties($path, $requestedProps) as $propName => $propValue) {
@@ -263,6 +271,38 @@ class CustomPropertiesBackend implements BackendInterface {
}
/**
+ * prefetch all user properties in a directory
+ */
+ private function cacheDirectory(string $path, Directory $node): void {
+ $prefix = ltrim($path . '/', '/');
+ $query = $this->connection->getQueryBuilder();
+ $query->select('name', 'propertypath', 'propertyname', 'propertyvalue', 'valuetype')
+ ->from('filecache', 'f')
+ ->leftJoin('f', 'properties', 'p', $query->expr()->andX(
+ $query->expr()->eq('propertypath', $query->func()->concat(
+ $query->createNamedParameter($prefix),
+ 'name'
+ )),
+ $query->expr()->eq('userid', $query->createNamedParameter($this->user->getUID()))
+ ))
+ ->where($query->expr()->eq('parent', $query->createNamedParameter($node->getInternalFileId(), IQueryBuilder::PARAM_INT)));
+ $result = $query->executeQuery();
+
+ $propsByPath = [];
+
+ while ($row = $result->fetch()) {
+ $childPath = $prefix . $row['name'];
+ if (!isset($propsByPath[$childPath])) {
+ $propsByPath[$childPath] = [];
+ }
+ if (isset($row['propertyname'])) {
+ $propsByPath[$childPath][$row['propertyname']] = $this->decodeValueFromDatabase($row['propertyvalue'], $row['valuetype']);
+ }
+ }
+ $this->userCache = array_merge($this->userCache, $propsByPath);
+ }
+
+ /**
* Returns a list of properties for the given path and current user
*
* @param string $path
@@ -321,7 +361,7 @@ class CustomPropertiesBackend implements BackendInterface {
$dbParameters = [
'userid' => $this->user->getUID(),
'propertyPath' => $this->formatPath($path),
- 'propertyName' => $propertyName
+ 'propertyName' => $propertyName,
];
// If it was null, we need to delete the property