-
-
Notifications
You must be signed in to change notification settings - Fork 4.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(files_versions): Add listener and interfaces to allow versions m…
…igration across storages Signed-off-by: Louis Chemineau <[email protected]>
- Loading branch information
Showing
16 changed files
with
484 additions
and
36 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
139 changes: 139 additions & 0 deletions
139
apps/files_versions/lib/Listener/VersionStorageMoveListener.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
/** | ||
* @copyright Copyright (c) 2024 Louis Chmn <[email protected]> | ||
* | ||
* @author Louis Chmn <[email protected]> | ||
* | ||
* @license GNU AGPL-3.0-or-later | ||
* | ||
* 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, | ||
* along with this program. If not, see <http://www.gnu.org/licenses/> | ||
* | ||
*/ | ||
namespace OCA\Files_Versions\Listener; | ||
|
||
use Exception; | ||
use OC\Files\Node\NonExistingFile; | ||
use OCA\Files_Versions\Versions\IVersionBackend; | ||
use OCA\Files_Versions\Versions\IVersionManager; | ||
use OCA\Files_Versions\Versions\IVersionsImporterBackend; | ||
use OCP\EventDispatcher\Event; | ||
use OCP\EventDispatcher\IEventListener; | ||
use OCP\Files\Events\Node\AbstractNodesEvent; | ||
use OCP\Files\Events\Node\BeforeNodeCopiedEvent; | ||
use OCP\Files\Events\Node\BeforeNodeRenamedEvent; | ||
use OCP\Files\Events\Node\NodeCopiedEvent; | ||
use OCP\Files\Events\Node\NodeRenamedEvent; | ||
use OCP\Files\File; | ||
use OCP\Files\Folder; | ||
use OCP\Files\Node; | ||
use OCP\Files\Storage\IStorage; | ||
use OCP\IUser; | ||
use OCP\IUserSession; | ||
|
||
/** @template-implements IEventListener<Event> */ | ||
class VersionStorageMoveListener implements IEventListener { | ||
/** @var File[] */ | ||
private array $movedNodes = []; | ||
|
||
public function __construct( | ||
private IVersionManager $versionManager, | ||
private IUserSession $userSession, | ||
) { | ||
} | ||
|
||
/** | ||
* @abstract Moves version across storages if necessary. | ||
* @throws Exception No user in session | ||
*/ | ||
public function handle(Event $event): void { | ||
if (!($event instanceof AbstractNodesEvent)) { | ||
return; | ||
} | ||
|
||
$source = $event->getSource(); | ||
$target = $event->getTarget(); | ||
|
||
$sourceStorage = $this->getNodeStorage($source); | ||
$targetStorage = $this->getNodeStorage($target); | ||
|
||
$sourceBackend = $this->versionManager->getBackendForStorage($sourceStorage); | ||
$targetBackend = $this->versionManager->getBackendForStorage($targetStorage); | ||
|
||
// If same backend, nothing to do. | ||
if ($sourceBackend === $targetBackend) { | ||
return; | ||
} | ||
|
||
$user = $this->userSession->getUser() ?? $source->getOwner(); | ||
|
||
if ($user === null) { | ||
throw new Exception("Cannot move versions across storages without a user."); | ||
} | ||
|
||
if ($event instanceof BeforeNodeRenamedEvent || $event instanceof BeforeNodeCopiedEvent) { | ||
$this->recursivelyPrepareMoveOrCopy($source); | ||
} elseif ($event instanceof NodeRenamedEvent || $event instanceof NodeCopiedEvent) { | ||
$this->recursivelyHandleMoveOrCopy($event, $user, $target, $sourceBackend, $targetBackend); | ||
} | ||
} | ||
|
||
/** Store all sub files in this->movedNodes so their info can be used after the operation. */ | ||
private function recursivelyPrepareMoveOrCopy(Node $source): void { | ||
if ($source instanceof File) { | ||
$this->movedNodes[$source->getId()] = $source; | ||
} elseif ($source instanceof Folder) { | ||
foreach ($source->getDirectoryListing() as $child) { | ||
$this->recursivelyPrepareMoveOrCopy($child); | ||
} | ||
} | ||
} | ||
|
||
/** Call handleMoveOrCopy on each sub files */ | ||
private function recursivelyHandleMoveOrCopy(Event $event, IUser $user, Node $target, IVersionBackend $sourceBackend, IVersionBackend $targetBackend): void { | ||
if ($target instanceof File) { | ||
/** @var File $originalSource */ | ||
$originalSource = $this->movedNodes[$target->getId()]; | ||
$this->handleMoveOrCopy($event, $user, $originalSource, $target, $sourceBackend, $targetBackend); | ||
} elseif ($target instanceof Folder) { | ||
foreach ($target->getDirectoryListing() as $child) { | ||
$this->recursivelyHandleMoveOrCopy($event, $user, $child, $sourceBackend, $targetBackend); | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* Called only during NodeRenamedEvent or NodeCopiedEvent | ||
* Will send the source node versions to the new backend, and then delete them from the old backend. | ||
*/ | ||
private function handleMoveOrCopy(Event $event, IUser $user, File $source, File $target, IVersionBackend $sourceBackend, IVersionBackend $targetBackend): void { | ||
if ($targetBackend instanceof IVersionsImporterBackend) { | ||
$versions = $sourceBackend->getVersionsForFile($user, $source); | ||
$targetBackend->importVersionsForFile($user, $source, $target, $versions); | ||
} | ||
|
||
if ($event instanceof NodeRenamedEvent && $sourceBackend instanceof IVersionsImporterBackend) { | ||
$sourceBackend->clearVersionsForFile($user, $source, $target); | ||
} | ||
} | ||
|
||
private function getNodeStorage(Node $node): IStorage { | ||
if ($node instanceof NonExistingFile) { | ||
return $node->getParent()->getStorage(); | ||
} else { | ||
return $node->getStorage(); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
50 changes: 50 additions & 0 deletions
50
apps/files_versions/lib/Versions/IVersionsImporterBackend.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
/** | ||
* @copyright Copyright (c) 2024 Louis Chmn <[email protected]> | ||
* | ||
* @author Louis Chmn <[email protected]> | ||
* | ||
* @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\Versions; | ||
|
||
use OCP\Files\Node; | ||
use OCP\IUser; | ||
|
||
/** | ||
* @since 29.0.0 | ||
*/ | ||
interface IVersionsImporterBackend { | ||
/** | ||
* Import the given versions for the target file. | ||
* | ||
* @param Node $target - The target is not yet created. | ||
* @param IVersion[] $versions | ||
* @since 29.0.0 | ||
*/ | ||
public function importVersionsForFile(IUser $user, Node $source, Node $target, array $versions): void; | ||
|
||
/** | ||
* Clear all versions for a file | ||
* | ||
* @since 29.0.0 | ||
*/ | ||
public function clearVersionsForFile(IUser $user, Node $source, Node $target): void; | ||
} |
Oops, something went wrong.