summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMorris Jobke <hey@morrisjobke.de>2019-05-21 21:00:15 +0200
committerGitHub <noreply@github.com>2019-05-21 21:00:15 +0200
commit553543e85c59b72b45f2a1b83fabea67d4681c64 (patch)
treefe0a76e1d21970a163cba58779b8efb0d9303792
parent9daee196a3ebe8373cdce9c6770c5b0641566839 (diff)
parentccfbee2af953924c6b290104cdac1d5c6139f15b (diff)
downloadnextcloud-server-553543e85c59b72b45f2a1b83fabea67d4681c64.tar.gz
nextcloud-server-553543e85c59b72b45f2a1b83fabea67d4681c64.zip
Merge pull request #15659 from nextcloud/version-backend-use
add way for version backends to programmatically specify if they should be used
-rw-r--r--apps/files_versions/lib/Versions/IVersionBackend.php12
-rw-r--r--apps/files_versions/lib/Versions/LegacyVersionsBackend.php5
-rw-r--r--apps/files_versions/lib/Versions/VersionManager.php40
-rw-r--r--apps/files_versions/tests/Versions/VersionManagerTest.php84
4 files changed, 129 insertions, 12 deletions
diff --git a/apps/files_versions/lib/Versions/IVersionBackend.php b/apps/files_versions/lib/Versions/IVersionBackend.php
index 616d535f7fd..913745c9850 100644
--- a/apps/files_versions/lib/Versions/IVersionBackend.php
+++ b/apps/files_versions/lib/Versions/IVersionBackend.php
@@ -25,6 +25,7 @@ use OCP\Files\File;
use OCP\Files\FileInfo;
use OCP\Files\NotFoundException;
use OCP\Files\SimpleFS\ISimpleFile;
+use OCP\Files\Storage\IStorage;
use OCP\IUser;
/**
@@ -32,6 +33,17 @@ use OCP\IUser;
*/
interface IVersionBackend {
/**
+ * Whether or not this version backend should be used for a storage
+ *
+ * If false is returned then the next applicable backend will be used
+ *
+ * @param IStorage $storage
+ * @return bool
+ * @since 17.0.0
+ */
+ public function useBackendForStorage(IStorage $storage): bool;
+
+ /**
* Get all versions for a file
*
* @param IUser $user
diff --git a/apps/files_versions/lib/Versions/LegacyVersionsBackend.php b/apps/files_versions/lib/Versions/LegacyVersionsBackend.php
index 99424f54564..aceddef29d2 100644
--- a/apps/files_versions/lib/Versions/LegacyVersionsBackend.php
+++ b/apps/files_versions/lib/Versions/LegacyVersionsBackend.php
@@ -29,6 +29,7 @@ use OCP\Files\FileInfo;
use OCP\Files\Folder;
use OCP\Files\IRootFolder;
use OCP\Files\NotFoundException;
+use OCP\Files\Storage\IStorage;
use OCP\IUser;
use OCP\IUserManager;
@@ -43,6 +44,10 @@ class LegacyVersionsBackend implements IVersionBackend {
$this->userManager = $userManager;
}
+ public function useBackendForStorage(IStorage $storage): bool {
+ return true;
+ }
+
public function getVersionsForFile(IUser $user, FileInfo $file): array {
$storage = $file->getStorage();
if ($storage->instanceOfStorage(SharedStorage::class)) {
diff --git a/apps/files_versions/lib/Versions/VersionManager.php b/apps/files_versions/lib/Versions/VersionManager.php
index 757b6002710..99fa9f62bc8 100644
--- a/apps/files_versions/lib/Versions/VersionManager.php
+++ b/apps/files_versions/lib/Versions/VersionManager.php
@@ -27,15 +27,18 @@ use OCP\Files\Storage\IStorage;
use OCP\IUser;
class VersionManager implements IVersionManager {
- /** @var IVersionBackend[] */
+ /** @var (IVersionBackend[])[] */
private $backends = [];
public function registerBackend(string $storageType, IVersionBackend $backend) {
- $this->backends[$storageType] = $backend;
+ if (!isset($this->backends[$storageType])) {
+ $this->backends[$storageType] = [];
+ }
+ $this->backends[$storageType][] = $backend;
}
/**
- * @return IVersionBackend[]
+ * @return (IVersionBackend[])[]
*/
private function getBackends(): array {
return $this->backends;
@@ -49,20 +52,29 @@ class VersionManager implements IVersionManager {
public function getBackendForStorage(IStorage $storage): IVersionBackend {
$fullType = get_class($storage);
$backends = $this->getBackends();
- $foundType = array_reduce(array_keys($backends), function ($type, $registeredType) use ($storage) {
+
+ $foundType = '';
+ $foundBackend = null;
+
+ foreach ($backends as $type => $backendsForType) {
if (
- $storage->instanceOfStorage($registeredType) &&
- ($type === '' || is_subclass_of($registeredType, $type))
+ $storage->instanceOfStorage($type) &&
+ ($foundType === '' || is_subclass_of($type, $foundType))
) {
- return $registeredType;
- } else {
- return $type;
+ foreach ($backendsForType as $backend) {
+ /** @var IVersionBackend $backend */
+ if ($backend->useBackendForStorage($storage)) {
+ $foundBackend = $backend;
+ $foundType = $type;
+ }
+ }
}
- }, '');
- if ($foundType === '') {
+ }
+
+ if ($foundType === '' || $foundBackend === null) {
throw new BackendNotFoundException("Version backend for $fullType not found");
} else {
- return $backends[$foundType];
+ return $foundBackend;
}
}
@@ -90,4 +102,8 @@ class VersionManager implements IVersionManager {
$backend = $this->getBackendForStorage($sourceFile->getStorage());
return $backend->getVersionFile($user, $sourceFile, $revision);
}
+
+ public function useBackendForStorage(IStorage $storage): bool {
+ return false;
+ }
}
diff --git a/apps/files_versions/tests/Versions/VersionManagerTest.php b/apps/files_versions/tests/Versions/VersionManagerTest.php
new file mode 100644
index 00000000000..25afa7f53e8
--- /dev/null
+++ b/apps/files_versions/tests/Versions/VersionManagerTest.php
@@ -0,0 +1,84 @@
+<?php declare(strict_types=1);
+/**
+ * @copyright Copyright (c) 2019 Robin Appelman <robin@icewind.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\files_versions\tests\Versions;
+
+use OC\Files\Storage\Local;
+use OCA\Files_Versions\Versions\IVersionBackend;
+use OCA\Files_Versions\Versions\VersionManager;
+use OCP\Files\Storage\IStorage;
+use Test\TestCase;
+
+class VersionManagerTest extends TestCase {
+ private function getBackend(bool $shouldUse = true): IVersionBackend {
+ $backend = $this->createMock(IVersionBackend::class);
+ $backend->method('useBackendForStorage')
+ ->willReturn($shouldUse);
+ return $backend;
+ }
+
+ private function getStorage(string $class): IStorage {
+ return $this->getMockBuilder($class)
+ ->disableOriginalConstructor()
+ ->setMethodsExcept(['instanceOfStorage'])
+ ->getMock();
+ }
+
+ public function testGetBackendSingle() {
+ $manager = new VersionManager();
+ $backend = $this->getBackend();
+ $manager->registerBackend(IStorage::class, $backend);
+
+ $this->assertEquals($backend, $manager->getBackendForStorage($this->getStorage(Local::class)));
+ }
+
+ public function testGetBackendMoreSpecific() {
+ $manager = new VersionManager();
+ $backend1 = $this->getBackend();
+ $backend2 = $this->getBackend();
+ $manager->registerBackend(IStorage::class, $backend1);
+ $manager->registerBackend(Local::class, $backend2);
+
+ $this->assertEquals($backend2, $manager->getBackendForStorage($this->getStorage(Local::class)));
+ }
+
+ public function testGetBackendNoUse() {
+ $manager = new VersionManager();
+ $backend1 = $this->getBackend();
+ $backend2 = $this->getBackend(false);
+ $manager->registerBackend(IStorage::class, $backend1);
+ $manager->registerBackend(Local::class, $backend2);
+
+ $this->assertEquals($backend1, $manager->getBackendForStorage($this->getStorage(Local::class)));
+ }
+
+ public function testGetBackendMultiple() {
+ $manager = new VersionManager();
+ $backend1 = $this->getBackend();
+ $backend2 = $this->getBackend(false);
+ $backend3 = $this->getBackend();
+ $manager->registerBackend(IStorage::class, $backend1);
+ $manager->registerBackend(Local::class, $backend2);
+ $manager->registerBackend(Local::class, $backend3);
+
+ $this->assertEquals($backend3, $manager->getBackendForStorage($this->getStorage(Local::class)));
+ }
+}