2011-03-18 16:23:47 +13:00
|
|
|
<?php
|
2016-06-16 16:57:19 +12:00
|
|
|
|
2017-08-09 14:53:38 +12:00
|
|
|
namespace SilverStripe\CMS\Tests\Model;
|
2017-08-09 13:25:12 +12:00
|
|
|
|
2018-04-06 15:53:57 +12:00
|
|
|
use Silverstripe\Assets\Dev\TestAssetStore;
|
|
|
|
use SilverStripe\CMS\Model\RedirectorPage;
|
2017-02-28 15:46:19 +13:00
|
|
|
use SilverStripe\CMS\Model\SiteTree;
|
2018-04-06 15:53:57 +12:00
|
|
|
use SilverStripe\CMS\Model\SiteTreeLink;
|
2016-07-22 11:32:32 +12:00
|
|
|
use SilverStripe\CMS\Model\VirtualPage;
|
2018-04-06 15:53:57 +12:00
|
|
|
use SilverStripe\CMS\Tests\Model\SiteTreeBrokenLinksTest\NotPageObject;
|
2016-08-23 14:36:06 +12:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
2018-04-06 15:53:57 +12:00
|
|
|
use SilverStripe\ORM\DataObject;
|
|
|
|
use SilverStripe\ORM\DB;
|
|
|
|
use SilverStripe\Versioned\Versioned;
|
2017-08-09 13:25:12 +12:00
|
|
|
|
2011-03-18 16:23:47 +13:00
|
|
|
/**
|
2016-01-26 18:38:42 +13:00
|
|
|
* Tests {@see SiteTreeLinkTracking} broken links feature: LinkTracking
|
2011-03-18 16:23:47 +13:00
|
|
|
*/
|
2017-01-26 09:59:25 +13:00
|
|
|
class SiteTreeBrokenLinksTest extends SapphireTest
|
|
|
|
{
|
|
|
|
protected static $fixture_file = 'SiteTreeBrokenLinksTest.yml';
|
|
|
|
|
2018-04-06 15:53:57 +12:00
|
|
|
protected static $extra_dataobjects = [
|
|
|
|
NotPageObject::class,
|
|
|
|
];
|
|
|
|
|
2021-10-28 10:40:52 +13:00
|
|
|
protected function setUp(): void
|
2017-01-26 09:59:25 +13:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
|
|
TestAssetStore::activate('SiteTreeBrokenLinksTest');
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
}
|
|
|
|
|
2021-10-28 10:40:52 +13:00
|
|
|
protected function tearDown(): void
|
2017-01-26 09:59:25 +13:00
|
|
|
{
|
|
|
|
TestAssetStore::reset();
|
|
|
|
parent::tearDown();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBrokenLinksBetweenPages()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $obj */
|
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
$obj->Content = '<a href="[sitetree_link,id=3423423]">this is a broken link</a>';
|
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertTrue($obj->HasBrokenLink, 'Page has a broken link');
|
|
|
|
|
2018-04-06 15:53:57 +12:00
|
|
|
$obj->Content = '<a href="[sitetree_link,id=' . $this->idFromFixture(
|
2018-05-04 01:04:18 +00:00
|
|
|
SiteTree::class,
|
2018-04-06 15:53:57 +12:00
|
|
|
'about'
|
|
|
|
) . ']">this is not a broken link</a>';
|
2017-01-26 09:59:25 +13:00
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertFalse($obj->HasBrokenLink, 'Page does NOT have a broken link');
|
|
|
|
}
|
|
|
|
|
2018-04-06 15:53:57 +12:00
|
|
|
/**
|
|
|
|
* Ensure broken links can be tracked between non-page objects
|
|
|
|
*/
|
|
|
|
public function testBrokenLinksNonPage()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $aboutPage */
|
|
|
|
$aboutPage = $this->objFromFixture(SiteTree::class, 'about');
|
2018-04-06 15:53:57 +12:00
|
|
|
|
|
|
|
/** @var NotPageObject $obj */
|
|
|
|
$obj = $this->objFromFixture(NotPageObject::class, 'object1');
|
|
|
|
$obj->Content = '<a href="[sitetree_link,id=3423423]">this is a broken link</a>';
|
|
|
|
$obj->AnotherContent = '<a href="[sitetree_link,id=' . $aboutPage->ID . ']">this is not a broken link</a>';
|
|
|
|
$obj->write();
|
|
|
|
|
|
|
|
// Two links created for this record
|
|
|
|
$this->assertListEquals(
|
|
|
|
[
|
|
|
|
['LinkedID' => 3423423],
|
|
|
|
['LinkedID' => $aboutPage->ID],
|
|
|
|
],
|
|
|
|
SiteTreeLink::get()->filter([
|
|
|
|
'ParentClass' => NotPageObject::class,
|
|
|
|
'ParentID' => $obj->ID,
|
|
|
|
])
|
|
|
|
);
|
|
|
|
|
|
|
|
// ManyManyThrough relation only links to unbroken pages
|
|
|
|
$this->assertListEquals(
|
|
|
|
[
|
|
|
|
['Title' => 'About'],
|
|
|
|
],
|
|
|
|
$obj->LinkTracking()
|
|
|
|
);
|
|
|
|
|
|
|
|
// About-page backlinks contains this object
|
|
|
|
$this->assertListEquals(
|
|
|
|
[
|
2018-05-04 01:04:18 +00:00
|
|
|
['ID' => $obj->ID],
|
2018-04-06 15:53:57 +12:00
|
|
|
],
|
|
|
|
$aboutPage->BackLinkTracking()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-01-26 09:59:25 +13:00
|
|
|
public function testBrokenAnchorBetweenPages()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $obj */
|
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'content');
|
|
|
|
$target = $this->objFromFixture(SiteTree::class, 'about');
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
$obj->Content = "<a href=\"[sitetree_link,id={$target->ID}]#no-anchor-here\">this is a broken link</a>";
|
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertTrue($obj->HasBrokenLink, 'Page has a broken link');
|
|
|
|
|
|
|
|
$obj->Content = "<a href=\"[sitetree_link,id={$target->ID}]#yes-anchor-here\">this is not a broken link</a>";
|
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertFalse($obj->HasBrokenLink, 'Page does NOT have a broken link');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBrokenVirtualPages()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
$vp = new VirtualPage();
|
|
|
|
|
|
|
|
$vp->CopyContentFromID = $obj->ID;
|
|
|
|
$vp->syncLinkTracking();
|
|
|
|
$this->assertFalse($vp->HasBrokenLink, 'Working virtual page is NOT marked as broken');
|
|
|
|
|
|
|
|
$vp->CopyContentFromID = 12345678;
|
|
|
|
$vp->syncLinkTracking();
|
|
|
|
$this->assertTrue($vp->HasBrokenLink, 'Broken virtual page IS marked as such');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBrokenInternalRedirectorPages()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
$rp = new RedirectorPage();
|
|
|
|
|
|
|
|
$rp->RedirectionType = 'Internal';
|
|
|
|
|
|
|
|
$rp->LinkToID = $obj->ID;
|
|
|
|
$rp->syncLinkTracking();
|
|
|
|
$this->assertFalse($rp->HasBrokenLink, 'Working redirector page is NOT marked as broken');
|
|
|
|
|
|
|
|
$rp->LinkToID = 12345678;
|
|
|
|
$rp->syncLinkTracking();
|
|
|
|
$this->assertTrue($rp->HasBrokenLink, 'Broken redirector page IS marked as such');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDeletingMarksBackLinkedPagesAsBroken()
|
|
|
|
{
|
|
|
|
// Set up two published pages with a link from content -> about
|
2018-05-04 01:04:18 +00:00
|
|
|
$linkDest = $this->objFromFixture(SiteTree::class, 'about');
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $linkSrc */
|
|
|
|
$linkSrc = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
$linkSrc->Content = "<p><a href=\"[sitetree_link,id=$linkDest->ID]\">about us</a></p>";
|
|
|
|
$linkSrc->write();
|
|
|
|
|
|
|
|
// Confirm no broken link
|
|
|
|
$this->assertEquals(0, (int)$linkSrc->HasBrokenLink);
|
|
|
|
|
|
|
|
// Delete page from draft
|
|
|
|
$linkDest->delete();
|
|
|
|
|
|
|
|
// Confirm draft has broken link
|
|
|
|
$linkSrc->flushCache();
|
2018-05-04 01:04:18 +00:00
|
|
|
$linkSrc = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
$this->assertEquals(1, (int)$linkSrc->HasBrokenLink);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testPublishingSourceBeforeDestHasBrokenLink()
|
|
|
|
{
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
|
|
|
|
// Set up two draft pages with a link from content -> about
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $linkDest */
|
|
|
|
$linkDest = $this->objFromFixture(SiteTree::class, 'about');
|
2017-01-26 09:59:25 +13:00
|
|
|
// Ensure that it's not on the published site
|
|
|
|
$linkDest->doUnpublish();
|
|
|
|
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $linkSrc */
|
|
|
|
$linkSrc = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
$linkSrc->Content = "<p><a href=\"[sitetree_link,id=$linkDest->ID]\">about us</a></p>";
|
|
|
|
$linkSrc->write();
|
|
|
|
|
|
|
|
// Publish the source of the link, while the dest is still unpublished.
|
|
|
|
$linkSrc->publishRecursive();
|
|
|
|
|
2017-06-21 16:29:40 +12:00
|
|
|
// Verify that the link is not marked as broken on draft (source of truth)
|
2017-01-26 09:59:25 +13:00
|
|
|
$this->assertEquals(0, (int)$linkSrc->HasBrokenLink);
|
2017-06-21 16:29:40 +12:00
|
|
|
|
|
|
|
// Live doesn't have separate broken link tracking
|
|
|
|
$this->assertEquals(0, DB::query("SELECT \"HasBrokenLink\" FROM \"SiteTree_Live\"
|
2018-05-04 01:04:18 +00:00
|
|
|
WHERE \"ID\" = $linkSrc->ID")->value());
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testRestoreFixesBrokenLinks()
|
|
|
|
{
|
|
|
|
// Create page and virtual page
|
2018-05-04 01:04:18 +00:00
|
|
|
$p = new SiteTree();
|
2017-01-26 09:59:25 +13:00
|
|
|
$p->Title = "source";
|
|
|
|
$p->write();
|
|
|
|
$pageID = $p->ID;
|
|
|
|
$this->assertTrue($p->publishRecursive());
|
|
|
|
|
|
|
|
// Content links are one kind of link to pages
|
2018-05-04 01:04:18 +00:00
|
|
|
$p2 = new SiteTree();
|
2017-01-26 09:59:25 +13:00
|
|
|
$p2->Title = "regular link";
|
|
|
|
$p2->Content = "<a href=\"[sitetree_link,id=$p->ID]\">test</a>";
|
|
|
|
$p2->write();
|
|
|
|
$this->assertTrue($p2->publishRecursive());
|
|
|
|
|
|
|
|
// Redirector links are a third
|
|
|
|
$rp = new RedirectorPage();
|
|
|
|
$rp->Title = "redirector";
|
2018-03-14 16:34:46 +13:00
|
|
|
$rp->RedirectionType = 'Internal';
|
2017-01-26 09:59:25 +13:00
|
|
|
$rp->LinkToID = $p->ID;
|
|
|
|
$rp->write();
|
|
|
|
$this->assertTrue($rp->publishRecursive());
|
|
|
|
|
|
|
|
// Confirm that there are no broken links to begin with
|
|
|
|
$this->assertFalse($p2->HasBrokenLink);
|
|
|
|
$this->assertFalse($rp->HasBrokenLink);
|
|
|
|
|
2017-06-21 16:29:40 +12:00
|
|
|
// Unpublishing doesn't affect broken state on live (draft is source of truth)
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var SiteTree $p2Live */
|
2018-03-14 16:34:46 +13:00
|
|
|
$p2Live = Versioned::get_by_stage(SiteTree::class, Versioned::LIVE)->byID($p2->ID);
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var SiteTree $rpLive */
|
2018-03-14 16:34:46 +13:00
|
|
|
$rpLive = Versioned::get_by_stage(SiteTree::class, Versioned::LIVE)->byID($rp->ID);
|
2017-06-21 16:29:40 +12:00
|
|
|
$this->assertEquals(0, $p2Live->HasBrokenLink);
|
|
|
|
$this->assertEquals(0, $rpLive->HasBrokenLink);
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Delete the source page, confirm that the VP, RP and page 2 have broken links on draft
|
|
|
|
$p->delete();
|
|
|
|
$p2->flushCache();
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var SiteTree $p2 */
|
2017-06-21 16:29:40 +12:00
|
|
|
$p2 = DataObject::get_by_id(SiteTree::class, $p2->ID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$rp->flushCache();
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var RedirectorPage $rp */
|
2017-06-21 16:29:40 +12:00
|
|
|
$rp = DataObject::get_by_id(SiteTree::class, $rp->ID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$this->assertEquals(1, $p2->HasBrokenLink);
|
|
|
|
$this->assertEquals(1, $rp->HasBrokenLink);
|
|
|
|
|
|
|
|
// Restore the page to stage, confirm that this fixes the links
|
2017-06-21 16:29:40 +12:00
|
|
|
/** @var SiteTree $p */
|
|
|
|
$p = Versioned::get_latest_version(SiteTree::class, $pageID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$p->doRestoreToStage();
|
|
|
|
|
|
|
|
$p2->flushCache();
|
2017-06-21 16:29:40 +12:00
|
|
|
$p2 = DataObject::get_by_id(SiteTree::class, $p2->ID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$rp->flushCache();
|
2017-06-21 16:29:40 +12:00
|
|
|
$rp = DataObject::get_by_id(SiteTree::class, $rp->ID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$this->assertFalse((bool)$p2->HasBrokenLink);
|
|
|
|
$this->assertFalse((bool)$rp->HasBrokenLink);
|
|
|
|
|
|
|
|
// Publish and confirm that the p2 and RP broken links are fixed on published
|
|
|
|
$this->assertTrue($p->publishRecursive());
|
2017-06-21 16:29:40 +12:00
|
|
|
$p2Live = Versioned::get_one_by_stage(SiteTree::class, 'Live', '"SiteTree"."ID" = ' . $p2->ID);
|
|
|
|
$rpLive = Versioned::get_one_by_stage(SiteTree::class, 'Live', '"SiteTree"."ID" = ' . $rp->ID);
|
2017-01-26 09:59:25 +13:00
|
|
|
$this->assertFalse((bool)$p2Live->HasBrokenLink);
|
|
|
|
$this->assertFalse((bool)$rpLive->HasBrokenLink);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testRevertToLiveFixesBrokenLinks()
|
|
|
|
{
|
|
|
|
// Create page and virutal page
|
2018-05-04 01:04:18 +00:00
|
|
|
$page = new SiteTree();
|
2017-08-09 10:56:08 +12:00
|
|
|
$page->Title = "source";
|
|
|
|
$page->write();
|
|
|
|
$pageID = $page->ID;
|
|
|
|
$this->assertTrue($page->publishRecursive());
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Content links are one kind of link to pages
|
2018-05-04 01:04:18 +00:00
|
|
|
$page2 = new SiteTree();
|
2017-08-09 10:56:08 +12:00
|
|
|
$page2->Title = "regular link";
|
|
|
|
$page2->Content = "<a href=\"[sitetree_link,id={$pageID}]\">test</a>";
|
|
|
|
$page2->write();
|
|
|
|
$this->assertTrue($page2->publishRecursive());
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Redirector links are a third
|
2017-08-09 10:56:08 +12:00
|
|
|
$redirectorPage = new RedirectorPage();
|
|
|
|
$redirectorPage->Title = "redirector";
|
2018-04-06 15:53:57 +12:00
|
|
|
$redirectorPage->RedirectionType = 'Internal';
|
2017-08-09 10:56:08 +12:00
|
|
|
$redirectorPage->LinkToID = $page->ID;
|
|
|
|
$redirectorPage->write();
|
|
|
|
$this->assertTrue($redirectorPage->publishRecursive());
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Confirm that there are no broken links to begin with
|
2017-08-09 10:56:08 +12:00
|
|
|
$this->assertFalse($page2->HasBrokenLink);
|
|
|
|
$this->assertFalse($redirectorPage->HasBrokenLink);
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Delete from draft and confirm that broken links are marked
|
2017-08-09 10:56:08 +12:00
|
|
|
$page->delete();
|
2017-01-26 09:59:25 +13:00
|
|
|
|
2017-08-09 10:56:08 +12:00
|
|
|
$page2->flushCache();
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var SiteTree $page2 */
|
2017-08-09 10:56:08 +12:00
|
|
|
$page2 = DataObject::get_by_id(SiteTree::class, $page2->ID);
|
|
|
|
$redirectorPage->flushCache();
|
2018-04-06 15:53:57 +12:00
|
|
|
/** @var RedirectorPage $redirectorPage */
|
2017-08-09 10:56:08 +12:00
|
|
|
$redirectorPage = DataObject::get_by_id(SiteTree::class, $redirectorPage->ID);
|
|
|
|
$this->assertEquals(1, $page2->HasBrokenLink);
|
|
|
|
$this->assertEquals(1, $redirectorPage->HasBrokenLink);
|
2017-01-26 09:59:25 +13:00
|
|
|
|
|
|
|
// Call doRevertToLive and confirm that broken links are restored
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $pageLive */
|
2017-08-09 10:56:08 +12:00
|
|
|
$pageLive = Versioned::get_one_by_stage(SiteTree::class, 'Live', '"SiteTree"."ID" = ' . $pageID);
|
|
|
|
$pageLive->doRevertToLive();
|
|
|
|
|
|
|
|
$page2->flushCache();
|
|
|
|
$page2 = DataObject::get_by_id(SiteTree::class, $page2->ID);
|
|
|
|
$redirectorPage->flushCache();
|
|
|
|
$redirectorPage = DataObject::get_by_id(SiteTree::class, $redirectorPage->ID);
|
|
|
|
$this->assertFalse((bool)$page2->HasBrokenLink);
|
|
|
|
$this->assertFalse((bool)$redirectorPage->HasBrokenLink);
|
2017-01-26 09:59:25 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testBrokenAnchorLinksInAPage()
|
|
|
|
{
|
2018-05-04 01:04:18 +00:00
|
|
|
/** @var SiteTree $obj */
|
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'content');
|
2017-01-26 09:59:25 +13:00
|
|
|
$origContent = $obj->Content;
|
|
|
|
|
2018-05-04 01:04:18 +00:00
|
|
|
$obj->Content = $origContent . '<a href="#no-anchor-here">this links to a non-existent in-page anchor or ' .
|
|
|
|
'skiplink</a>';
|
2017-01-26 09:59:25 +13:00
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertTrue($obj->HasBrokenLink, 'Page has a broken anchor/skiplink');
|
|
|
|
|
2018-05-04 01:04:18 +00:00
|
|
|
$obj->Content = $origContent . '<a href="#yes-anchor-here">this links to an existent in-page ' .
|
|
|
|
'anchor/skiplink</a>';
|
2017-01-26 09:59:25 +13:00
|
|
|
$obj->syncLinkTracking();
|
|
|
|
$this->assertFalse($obj->HasBrokenLink, 'Page doesn\'t have a broken anchor or skiplink');
|
|
|
|
}
|
2011-03-18 16:23:47 +13:00
|
|
|
}
|