summaryrefslogtreecommitdiffstats
path: root/lib/private/Files/Node/Folder.php
diff options
context:
space:
mode:
Diffstat (limited to 'lib/private/Files/Node/Folder.php')
-rw-r--r--lib/private/Files/Node/Folder.php41
1 files changed, 16 insertions, 25 deletions
diff --git a/lib/private/Files/Node/Folder.php b/lib/private/Files/Node/Folder.php
index 82515feba03..eb4841c5719 100644
--- a/lib/private/Files/Node/Folder.php
+++ b/lib/private/Files/Node/Folder.php
@@ -234,18 +234,8 @@ class Folder extends Node implements \OCP\Files\Folder {
$query = $this->queryFromOperator(new SearchComparison(ISearchComparison::COMPARE_LIKE, 'name', '%' . $query . '%'));
}
- // Limit+offset for queries with ordering
- //
- // Because we currently can't do ordering between the results from different storages in sql
- // The only way to do ordering is requesting the $limit number of entries from all storages
- // sorting them and returning the first $limit entries.
- //
- // For offset we have the same problem, we don't know how many entries from each storage should be skipped
- // by a given $offset, so instead we query $offset + $limit from each storage and return entries $offset..($offset+$limit)
- // after merging and sorting them.
- //
- // This is suboptimal but because limit and offset tend to be fairly small in real world use cases it should
- // still be significantly better than disabling paging altogether
+ // search is handled by a single query covering all caches that this folder contains
+ // this is done by collect
$limitToHome = $query->limitToHome();
if ($limitToHome && count(explode('/', $this->path)) !== 3) {
@@ -257,11 +247,11 @@ class Folder extends Node implements \OCP\Files\Folder {
$storage = $mount->getStorage();
$internalPath = $mount->getInternalPath($this->path);
- $caches = ['' => new CacheJail($storage->getCache(''), $internalPath)];
- /** @var array{IMountPoint, string}[] $infoParams */
- $infoParams = [
- '' => [$mount, '']
- ];
+ // collect all caches for this folder, indexed by their mountpoint relative to this folder
+ // and save the mount which is needed later to construct the FileInfo objects
+
+ $caches = ['' => new CacheJail($storage->getCache(''), $internalPath)]; // a temporary CacheJail is used to handle filtering down the results to within this folder
+ $mountByMountPoint = ['' => $mount];
if (!$limitToHome) {
$mounts = $this->root->getMountsIn($this->path);
@@ -270,7 +260,7 @@ class Folder extends Node implements \OCP\Files\Folder {
if ($storage) {
$relativeMountPoint = ltrim(substr($mount->getMountPoint(), $rootLength), '/');
$caches[$relativeMountPoint] = $storage->getCache('');
- $infoParams[$relativeMountPoint] = [$mount, ''];
+ $mountByMountPoint[$relativeMountPoint] = $mount;
}
}
}
@@ -278,10 +268,12 @@ class Folder extends Node implements \OCP\Files\Folder {
/** @var QuerySearchHelper $searchHelper */
$searchHelper = \OC::$server->get(QuerySearchHelper::class);
$resultsPerCache = $searchHelper->searchInCaches($query, $caches);
- $files = array_merge(...array_map(function(array $results, $relativeMountPoint) use ($infoParams) {
- $params = $infoParams[$relativeMountPoint];
- return array_map(function(ICacheEntry $result) use ($relativeMountPoint, $params) {
- return $this->cacheEntryToFileInfo($params[0], $relativeMountPoint, $params[1], $result);
+
+ // loop trough all results per-cache, constructing the FileInfo object from the CacheEntry and merge them all
+ $files = array_merge(...array_map(function(array $results, $relativeMountPoint) use ($mountByMountPoint) {
+ $mount = $mountByMountPoint[$relativeMountPoint];
+ return array_map(function(ICacheEntry $result) use ($relativeMountPoint, $mount) {
+ return $this->cacheEntryToFileInfo($mount, $relativeMountPoint, $result);
}, $results);
}, array_values($resultsPerCache), array_keys($resultsPerCache)));
@@ -290,10 +282,9 @@ class Folder extends Node implements \OCP\Files\Folder {
}, $files);
}
- private function cacheEntryToFileInfo(IMountPoint $mount, string $appendRoot, string $trimRoot, ICacheEntry $cacheEntry): FileInfo {
- $trimLength = strlen($trimRoot);
+ private function cacheEntryToFileInfo(IMountPoint $mount, string $appendRoot, ICacheEntry $cacheEntry): FileInfo {
$cacheEntry['internalPath'] = $cacheEntry['path'];
- $cacheEntry['path'] = $appendRoot . substr($cacheEntry['path'], $trimLength);
+ $cacheEntry['path'] = $appendRoot . $cacheEntry->getPath();
return new \OC\Files\FileInfo($this->path . '/' . $cacheEntry['path'], $mount->getStorage(), $cacheEntry['internalPath'], $cacheEntry, $mount);
}