From baafbb65f7263778d82b675b96b6c10e80f3d74a Mon Sep 17 00:00:00 2001 From: Martin Lagler Date: Tue, 27 Feb 2024 14:01:37 +0100 Subject: [PATCH] Fix restore when parent page was deleted --- .github/workflows/test-application.yaml | 2 +- Document/Subscriber/WebspaceSubscriber.php | 13 ++- .../articles/default_with_page_tree_route.xml | 23 ++++ .../Trash/ArticleTrashItemHandlerTest.php | 104 ++++++++++++++++++ 4 files changed, 137 insertions(+), 5 deletions(-) create mode 100644 Tests/Application/config/templates/articles/default_with_page_tree_route.xml diff --git a/.github/workflows/test-application.yaml b/.github/workflows/test-application.yaml index 2290cb30..32033a84 100644 --- a/.github/workflows/test-application.yaml +++ b/.github/workflows/test-application.yaml @@ -92,7 +92,7 @@ jobs: options: --health-cmd="mysqladmin ping" --health-interval=10s --health-timeout=5s --health-retries=5 jackrabbit: - image: sulu/jackrabbit:2.20-standalone + image: sulu/jackrabbit:2.20-tomcat-filesystem env: DATABASE_HOST: mysql DATABASE_PORT: 3306 diff --git a/Document/Subscriber/WebspaceSubscriber.php b/Document/Subscriber/WebspaceSubscriber.php index 0231c8f2..5bb8d23b 100644 --- a/Document/Subscriber/WebspaceSubscriber.php +++ b/Document/Subscriber/WebspaceSubscriber.php @@ -20,6 +20,7 @@ use Sulu\Component\DocumentManager\DocumentManagerInterface; use Sulu\Component\DocumentManager\Event\AbstractMappingEvent; use Sulu\Component\DocumentManager\Events; +use Sulu\Component\DocumentManager\Exception\DocumentNotFoundException; use Symfony\Component\EventDispatcher\EventSubscriberInterface; /** @@ -103,10 +104,14 @@ public function saveProperties(AbstractMappingEvent $event): void if ($parentPageUuid) { // we know now it's a `page_tree_route` route // so load the parent and find out the webspace - $parentDocument = $this->documentManager->find($parentPageUuid, $event->getLocale()); - $mainWebspace = $this->documentInspector->getWebspace($parentDocument); - $document->setMainWebspace($mainWebspace); - $document->setAdditionalWebspaces([]); + try { + $parentDocument = $this->documentManager->find($parentPageUuid, $event->getLocale()); + $mainWebspace = $this->documentInspector->getWebspace($parentDocument); + $document->setMainWebspace($mainWebspace); + $document->setAdditionalWebspaces([]); + } catch (DocumentNotFoundException $exception) { + // Do nothing, because when the parent page was deleted it should still work while restoring. + } } $mainWebspace = $document->getMainWebspace(); diff --git a/Tests/Application/config/templates/articles/default_with_page_tree_route.xml b/Tests/Application/config/templates/articles/default_with_page_tree_route.xml new file mode 100644 index 00000000..7a8ccc51 --- /dev/null +++ b/Tests/Application/config/templates/articles/default_with_page_tree_route.xml @@ -0,0 +1,23 @@ + + diff --git a/Tests/Functional/Trash/ArticleTrashItemHandlerTest.php b/Tests/Functional/Trash/ArticleTrashItemHandlerTest.php index 7a757ade..fb304a03 100644 --- a/Tests/Functional/Trash/ArticleTrashItemHandlerTest.php +++ b/Tests/Functional/Trash/ArticleTrashItemHandlerTest.php @@ -14,6 +14,7 @@ use Sulu\Bundle\ArticleBundle\Controller\ArticleController; use Sulu\Bundle\ArticleBundle\Document\ArticleDocument; use Sulu\Bundle\ArticleBundle\Trash\ArticleTrashItemHandler; +use Sulu\Bundle\PageBundle\Document\PageDocument; use Sulu\Bundle\TestBundle\Testing\SuluTestCase; use Sulu\Bundle\TrashBundle\SuluTrashBundle; use Sulu\Component\DocumentManager\DocumentManagerInterface; @@ -210,4 +211,107 @@ public function testStoreAndRestoreShadowArticle(): void static::assertSame('en', $restoredArticleEn->getOriginalLocale()); static::assertSame('de', $restoredArticleEn->getShadowLocale()); } + + public function testStoreAndRestorePageTreeRoute(): void + { + /** @var PageDocument $page */ + $page = $this->documentManager->create('page'); + $page->setTitle('test parent page'); + $page->setStructureType('default'); + $page->setResourceSegment('/test-parent-page'); + + $this->documentManager->persist($page, 'de', ['parent_path' => '/cmf/sulu_io/contents',]); + $this->documentManager->publish($page, 'de'); + $this->documentManager->flush(); + + /** @var ArticleDocument $article1De */ + $article1De = $this->documentManager->create(ArticleController::DOCUMENT_TYPE); + $article1De->setParent($this->documentManager->find('/cmf/sulu_io/contents', 'de')); + $article1De->setTitle('test-title-de'); + $article1De->setLocale('de'); + $article1De->setStructureType('default_with_page_tree_route'); + $article1De->setRoutePath($page->getStructure()->getProperty('url')->getValue() . '/test-title-de'); + $article1De->getStructure()->bind([ + 'routePath' => [ + 'page' => [ + 'uuid' => $page->getUuid(), + 'path' => $page->getStructure()->getProperty('url')->getValue(), + ], + 'path' => $page->getStructure()->getProperty('url')->getValue() . '/test-title-de', + 'suffix' => '/test-title-de', + ], + ]); + $article1De->setMainWebspace('sulu_io'); + $this->documentManager->persist($article1De, 'de'); + + $this->documentManager->flush(); + $originalArticleUuid = $article1De->getUuid(); + + $trashItem = $this->articleTrashItemHandler->store($article1De); + $this->documentManager->remove($article1De); + $this->documentManager->flush(); + $this->documentManager->clear(); + + static::assertSame($originalArticleUuid, $trashItem->getResourceId()); + static::assertSame('test-title-de', $trashItem->getResourceTitle()); + + /** @var ArticleDocument $restoredArticle */ + $restoredArticle = $this->articleTrashItemHandler->restore($trashItem); + static::assertSame($originalArticleUuid, $restoredArticle->getUuid()); + + /** @var ArticleDocument $restoredArticleDe */ + $restoredArticleDe = $this->documentManager->find($originalArticleUuid, 'de'); + static::assertSame($originalArticleUuid, $restoredArticleDe->getUuid()); + static::assertSame('test-title-de', $restoredArticleDe->getTitle()); + static::assertSame('/test-parent-page/test-title-de', $restoredArticleDe->getRoutePath()); + static::assertSame('de', $restoredArticleDe->getLocale()); + static::assertSame('default_with_page_tree_route', $restoredArticleDe->getStructureType()); + static::assertSame('sulu_io', $restoredArticleDe->getMainWebspace()); + } + + public function testStoreAndRestorePageTreeRouteNoValidParent(): void + { + /** @var ArticleDocument $article1De */ + $article1De = $this->documentManager->create(ArticleController::DOCUMENT_TYPE); + $article1De->setParent($this->documentManager->find('/cmf/sulu_io/contents', 'de')); + $article1De->setTitle('test-title-de'); + $article1De->setLocale('de'); + $article1De->setStructureType('default_with_page_tree_route'); + $article1De->getStructure()->bind([ + 'routePath' => [ + 'page' => [ + 'uuid' => 'not-existing-uuid', + 'path' => '/not-existing-path', + ], + 'path' => '/not-existing-path/test-title-de', + 'suffix' => '/test-title-de', + ], + ]); + $article1De->setMainWebspace('sulu_io'); + $this->documentManager->persist($article1De, 'de'); + + $this->documentManager->flush(); + $originalArticleUuid = $article1De->getUuid(); + + $trashItem = $this->articleTrashItemHandler->store($article1De); + $this->documentManager->remove($article1De); + $this->documentManager->flush(); + $this->documentManager->clear(); + + static::assertSame($originalArticleUuid, $trashItem->getResourceId()); + static::assertSame('test-title-de', $trashItem->getResourceTitle()); + + /** @var ArticleDocument $restoredArticle */ + $restoredArticle = $this->articleTrashItemHandler->restore($trashItem); + static::assertSame($originalArticleUuid, $restoredArticle->getUuid()); + + /** @var ArticleDocument $restoredArticleDe */ + $restoredArticleDe = $this->documentManager->find($originalArticleUuid, 'de'); + static::assertSame($originalArticleUuid, $restoredArticleDe->getUuid()); + static::assertSame('test-title-de', $restoredArticleDe->getTitle()); + static::assertSame('/not-existing-path/test-title-de', $restoredArticleDe->getRoutePath()); + static::assertSame('de', $restoredArticleDe->getLocale()); + static::assertSame('default_with_page_tree_route', $restoredArticleDe->getStructureType()); + static::assertSame('sulu_io', $restoredArticleDe->getMainWebspace()); + } }