mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 06:05:56 +00:00
2163 lines
83 KiB
PHP
2163 lines
83 KiB
PHP
<?php
|
|
|
|
namespace SilverStripe\CMS\Tests\Model;
|
|
|
|
use LogicException;
|
|
use Psr\SimpleCache\CacheInterface;
|
|
use ReflectionMethod;
|
|
use SilverStripe\CMS\Model\RedirectorPage;
|
|
use SilverStripe\CMS\Model\SiteTree;
|
|
use SilverStripe\CMS\Model\VirtualPage;
|
|
use SilverStripe\CMS\Tests\Controllers\SiteTreeTest_NamespaceMapTestController;
|
|
use SilverStripe\CMS\Tests\CMSEditLinkExtensionTest\BelongsToPage;
|
|
use SilverStripe\CMS\Tests\CMSEditLinkExtensionTest\PageWithChild;
|
|
use SilverStripe\CMS\Tests\Model\SiteTreeBrokenLinksTest\NotPageObject;
|
|
use SilverStripe\CMS\Tests\Page\SiteTreeTest_NamespaceMapTest;
|
|
use SilverStripe\Control\ContentNegotiator;
|
|
use SilverStripe\Control\Controller;
|
|
use SilverStripe\Control\Director;
|
|
use SilverStripe\Core\Config\Config;
|
|
use SilverStripe\Core\Injector\Injector;
|
|
use SilverStripe\Core\Manifest\VersionProvider;
|
|
use SilverStripe\Dev\SapphireTest;
|
|
use SilverStripe\i18n\i18n;
|
|
use SilverStripe\ORM\DataObject;
|
|
use SilverStripe\ORM\DB;
|
|
use SilverStripe\ORM\FieldType\DBDatetime;
|
|
use SilverStripe\ORM\ValidationException;
|
|
use SilverStripe\Security\Group;
|
|
use SilverStripe\Security\InheritedPermissions;
|
|
use SilverStripe\Security\Member;
|
|
use SilverStripe\Security\Permission;
|
|
use SilverStripe\Security\Security;
|
|
use SilverStripe\SiteConfig\SiteConfig;
|
|
use SilverStripe\Subsites\Extensions\SiteTreeSubsites;
|
|
use SilverStripe\Versioned\Versioned;
|
|
use SilverStripe\View\Parsers\HtmlDiff;
|
|
use SilverStripe\View\Parsers\ShortcodeParser;
|
|
use SilverStripe\View\Parsers\URLSegmentFilter;
|
|
use SilverStripe\View\Shortcodes\EmbedShortcodeProvider;
|
|
use TractorCow\Fluent\Extension\FluentSiteTreeExtension;
|
|
use Page;
|
|
use PageController;
|
|
|
|
use const RESOURCES_DIR;
|
|
|
|
class SiteTreeTest extends SapphireTest
|
|
{
|
|
protected static $fixture_file = [
|
|
'SiteTreeTest.yml',
|
|
'CMSEditLinkExtensionTest/fixtures.yml',
|
|
];
|
|
|
|
protected static $illegal_extensions = [
|
|
SiteTree::class => [
|
|
SiteTreeSubsites::class,
|
|
FluentSiteTreeExtension::class,
|
|
],
|
|
];
|
|
|
|
protected static $extra_dataobjects = [
|
|
SiteTreeTest_ClassA::class,
|
|
SiteTreeTest_ClassB::class,
|
|
SiteTreeTest_ClassC::class,
|
|
SiteTreeTest_ClassD::class,
|
|
SiteTreeTest_ClassCext::class,
|
|
SiteTreeTest_NotRoot::class,
|
|
SiteTreeTest_StageStatusInherit::class,
|
|
SiteTreeTest_DataObject::class,
|
|
PageWithChild::class,
|
|
BelongsToPage::class,
|
|
NotPageObject::class,
|
|
];
|
|
|
|
public function reservedSegmentsProvider()
|
|
{
|
|
return [
|
|
// segments reserved by rules
|
|
['Admin', 'admin-2'],
|
|
['Dev', 'dev-2'],
|
|
['Robots in disguise', 'robots-in-disguise'],
|
|
// segments reserved by folder name
|
|
['assets', 'assets-2'],
|
|
['notafoldername', 'notafoldername'],
|
|
];
|
|
}
|
|
|
|
public function testCreateDefaultpages()
|
|
{
|
|
$remove = SiteTree::get();
|
|
if ($remove) {
|
|
foreach ($remove as $page) {
|
|
$page->delete();
|
|
}
|
|
}
|
|
// Make sure the table is empty
|
|
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 0);
|
|
|
|
// Disable the creation
|
|
SiteTree::config()->create_default_pages = false;
|
|
singleton(SiteTree::class)->requireDefaultRecords();
|
|
|
|
// The table should still be empty
|
|
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 0);
|
|
|
|
// Enable the creation
|
|
SiteTree::config()->create_default_pages = true;
|
|
singleton(SiteTree::class)->requireDefaultRecords();
|
|
|
|
// The table should now have three rows (home, about-us, contact-us)
|
|
$this->assertEquals(DB::query('SELECT COUNT("ID") FROM "SiteTree"')->value(), 3);
|
|
}
|
|
|
|
/**
|
|
* Test generation of the URLSegment values.
|
|
* - Turns things into lowercase-hyphen-format
|
|
* - Generates from Title by default, unless URLSegment is explicitly set
|
|
* - Resolves duplicates by appending a number
|
|
* - renames classes with a class name conflict
|
|
*/
|
|
public function testURLGeneration()
|
|
{
|
|
$expectedURLs = [
|
|
'home' => 'home',
|
|
'staff' => 'my-staff',
|
|
'about' => 'about-us',
|
|
'staffduplicate' => 'my-staff-2',
|
|
'product1' => '1-1-test-product',
|
|
'product2' => 'another-product',
|
|
'product3' => 'another-product-2',
|
|
'product4' => 'another-product-3',
|
|
'object' => 'object',
|
|
'controller' => 'controller',
|
|
'numericonly' => '1930',
|
|
'numeric0' => '0',
|
|
];
|
|
|
|
foreach ($expectedURLs as $fixture => $urlSegment) {
|
|
$obj = $this->objFromFixture(SiteTree::class, $fixture);
|
|
$this->assertEquals($urlSegment, $obj->URLSegment);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check if reserved URL's are properly appended with a number at top level
|
|
* @dataProvider reservedSegmentsProvider
|
|
*/
|
|
public function testDisallowedURLGeneration($title, $urlSegment)
|
|
{
|
|
$page = new SiteTree(['Title' => $title]);
|
|
$id = $page->write();
|
|
$page = SiteTree::get()->byID($id);
|
|
$this->assertEquals($urlSegment, $page->URLSegment);
|
|
}
|
|
|
|
/**
|
|
* Check if reserved URL's are not appended with a number on a child page
|
|
* It's okay to have a URL like domain.com/my-page/admin as it won't interfere with domain.com/admin
|
|
* @dataProvider reservedSegmentsProvider
|
|
*/
|
|
public function testDisallowedChildURLGeneration($title, $urlSegment)
|
|
{
|
|
// Using the same dataprovider, strip out the -2 from the admin and dev segment
|
|
$urlSegment = str_replace('-2', '', $urlSegment ?? '');
|
|
$page = new SiteTree(['Title' => $title, 'ParentID' => 1]);
|
|
$id = $page->write();
|
|
$page = SiteTree::get()->byID($id);
|
|
$this->assertEquals($urlSegment, $page->URLSegment);
|
|
}
|
|
|
|
/**
|
|
* For legacy resources dir values ("resources"), check that URLSegments get a -2 appended
|
|
*/
|
|
public function testLegacyResourcesDirValuesHaveIncrementedValueAppended()
|
|
{
|
|
if (RESOURCES_DIR !== 'resources') {
|
|
// This test only runs on the CMS build because it doesn't have a `resources-dir` flag on its
|
|
// composer.json file
|
|
$this->markTestSkipped('This legacy test requires RESOURCES_DIR to be "resources"');
|
|
}
|
|
|
|
$page = new SiteTree(['Title' => 'Resources']);
|
|
$id = $page->write();
|
|
$page = SiteTree::get()->byID($id);
|
|
$this->assertSame('resources-2', $page->URLSegment);
|
|
}
|
|
|
|
/**
|
|
* For new/configured resources dir values ("_resources"), check that URLSegments have the leading underscore
|
|
* removed
|
|
*/
|
|
public function testDefaultResourcesDirHasLeadingUnderscoreRemovedAndResourcesIsUsed()
|
|
{
|
|
if (RESOURCES_DIR === 'resources') {
|
|
// This test won't runs on the CMS build because it doesn't have a `resources-dir` flag on its
|
|
// composer.json file. It will run on the recipe-cms build.
|
|
$this->markTestSkipped('This test requires RESOURCES_DIR to be something other than "resources"');
|
|
}
|
|
|
|
$page = new SiteTree(['Title' => '_Resources']);
|
|
$id = $page->write();
|
|
$page = SiteTree::get()->byID($id);
|
|
$this->assertSame('resources', $page->URLSegment);
|
|
}
|
|
|
|
/**
|
|
* Test that publication copies data to SiteTree_Live
|
|
*/
|
|
public function testPublishCopiesToLiveTable()
|
|
{
|
|
$obj = $this->objFromFixture(SiteTree::class, 'about');
|
|
$obj->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$createdID = DB::query(
|
|
"SELECT \"ID\" FROM \"SiteTree_Live\" WHERE \"URLSegment\" = '$obj->URLSegment'"
|
|
)->value();
|
|
$this->assertEquals($obj->ID, $createdID);
|
|
}
|
|
|
|
/**
|
|
* Test that field which are set and then cleared are also transferred to the published site.
|
|
*/
|
|
public function testPublishDeletedFields()
|
|
{
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$obj = $this->objFromFixture(SiteTree::class, 'about');
|
|
$obj->Title = "asdfasdf";
|
|
$obj->write();
|
|
$this->assertTrue($obj->publishRecursive());
|
|
|
|
$this->assertEquals(
|
|
'asdfasdf',
|
|
DB::query("SELECT \"Title\" FROM \"SiteTree_Live\" WHERE \"ID\" = '$obj->ID'")->value()
|
|
);
|
|
|
|
$obj->Title = null;
|
|
$obj->write();
|
|
$this->assertTrue($obj->publishRecursive());
|
|
|
|
$this->assertNotNull(DB::query("SELECT \"Title\" FROM \"SiteTree_Live\" WHERE \"ID\" = '$obj->ID'")->value());
|
|
}
|
|
|
|
public function testParentNodeCachedInMemory()
|
|
{
|
|
$parent = new SiteTree();
|
|
$parent->Title = 'Section Title';
|
|
$child = new SiteTree();
|
|
$child->Title = 'Page Title';
|
|
$child->setParent($parent);
|
|
|
|
$this->assertInstanceOf(SiteTree::class, $child->Parent);
|
|
$this->assertEquals("Section Title", $child->Parent->Title);
|
|
}
|
|
|
|
public function testParentModelReturnType()
|
|
{
|
|
$parent = new SiteTreeTest_PageNode();
|
|
$child = new SiteTreeTest_PageNode();
|
|
|
|
$child->setParent($parent);
|
|
$this->assertInstanceOf(SiteTreeTest_PageNode::class, $child->Parent);
|
|
}
|
|
|
|
/**
|
|
* Confirm that DataObject::get_one() gets records from SiteTree_Live
|
|
*/
|
|
public function testGetOneFromLive()
|
|
{
|
|
$s = new SiteTree();
|
|
$s->Title = "V1";
|
|
$s->URLSegment = "get-one-test-page";
|
|
$s->write();
|
|
$s->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
$s->Title = "V2";
|
|
$s->write();
|
|
|
|
$oldMode = Versioned::get_reading_mode();
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
|
|
$checkSiteTree = DataObject::get_one(SiteTree::class, [
|
|
'"SiteTree"."URLSegment"' => 'get-one-test-page',
|
|
]);
|
|
$this->assertEquals("V1", $checkSiteTree->Title);
|
|
|
|
Versioned::set_reading_mode($oldMode);
|
|
}
|
|
|
|
public function testChidrenOfRootAreTopLevelPages()
|
|
{
|
|
$pages = SiteTree::get();
|
|
foreach ($pages as $page) {
|
|
$page->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
}
|
|
unset($pages);
|
|
|
|
/* If we create a new SiteTree object with ID = 0 */
|
|
$obj = new SiteTree();
|
|
/* Then its children should be the top-level pages */
|
|
$stageChildren = $obj->stageChildren()->map('ID', 'Title');
|
|
$liveChildren = $obj->liveChildren()->map('ID', 'Title');
|
|
$allChildren = $obj->AllChildrenIncludingDeleted()->map('ID', 'Title');
|
|
|
|
$this->assertContains('Home', $stageChildren);
|
|
$this->assertContains('Products', $stageChildren);
|
|
$this->assertNotContains('Staff', $stageChildren);
|
|
|
|
$this->assertContains('Home', $liveChildren);
|
|
$this->assertContains('Products', $liveChildren);
|
|
$this->assertNotContains('Staff', $liveChildren);
|
|
|
|
$this->assertContains('Home', $allChildren);
|
|
$this->assertContains('Products', $allChildren);
|
|
$this->assertNotContains('Staff', $allChildren);
|
|
}
|
|
|
|
public function testCanSaveBlankToHasOneRelations()
|
|
{
|
|
/* DataObject::write() should save to a has_one relationship if you set a field called (relname)ID */
|
|
$page = new SiteTree();
|
|
$parentID = $this->idFromFixture(SiteTree::class, 'home');
|
|
$page->ParentID = $parentID;
|
|
$page->write();
|
|
$this->assertEquals(
|
|
$parentID,
|
|
DB::query("SELECT \"ParentID\" FROM \"SiteTree\" WHERE \"ID\" = $page->ID")->value()
|
|
);
|
|
|
|
/* You should then be able to save a null/0/'' value to the relation */
|
|
$page->ParentID = null;
|
|
$page->write();
|
|
$this->assertEquals(0, DB::query("SELECT \"ParentID\" FROM \"SiteTree\" WHERE \"ID\" = $page->ID")->value());
|
|
}
|
|
|
|
public function testStageStates()
|
|
{
|
|
// newly created page
|
|
$createdPage = new SiteTree();
|
|
$createdPage->write();
|
|
$this->assertTrue($createdPage->isOnDraft());
|
|
$this->assertFalse($createdPage->isPublished());
|
|
$this->assertTrue($createdPage->isOnDraftOnly());
|
|
$this->assertTrue($createdPage->isModifiedOnDraft());
|
|
|
|
// published page
|
|
$publishedPage = new SiteTree();
|
|
$publishedPage->write();
|
|
$publishedPage->copyVersionToStage('Stage', 'Live');
|
|
$this->assertTrue($publishedPage->isOnDraft());
|
|
$this->assertTrue($publishedPage->isPublished());
|
|
$this->assertFalse($publishedPage->isOnDraftOnly());
|
|
$this->assertFalse($publishedPage->isOnLiveOnly());
|
|
$this->assertFalse($publishedPage->isModifiedOnDraft());
|
|
|
|
// published page, deleted from stage
|
|
$deletedFromDraftPage = new SiteTree();
|
|
$deletedFromDraftPage->write();
|
|
$deletedFromDraftPage->copyVersionToStage('Stage', 'Live');
|
|
$deletedFromDraftPage->deleteFromStage('Stage');
|
|
$this->assertFalse($deletedFromDraftPage->isArchived());
|
|
$this->assertFalse($deletedFromDraftPage->isOnDraft());
|
|
$this->assertTrue($deletedFromDraftPage->isPublished());
|
|
$this->assertFalse($deletedFromDraftPage->isOnDraftOnly());
|
|
$this->assertTrue($deletedFromDraftPage->isOnLiveOnly());
|
|
$this->assertFalse($deletedFromDraftPage->isModifiedOnDraft());
|
|
|
|
// published page, deleted from live
|
|
$deletedFromLivePage = new SiteTree();
|
|
$deletedFromLivePage->write();
|
|
$deletedFromLivePage->copyVersionToStage('Stage', 'Live');
|
|
$deletedFromLivePage->deleteFromStage('Live');
|
|
$this->assertFalse($deletedFromLivePage->isArchived());
|
|
$this->assertTrue($deletedFromLivePage->isOnDraft());
|
|
$this->assertFalse($deletedFromLivePage->isPublished());
|
|
$this->assertTrue($deletedFromLivePage->isOnDraftOnly());
|
|
$this->assertFalse($deletedFromLivePage->isOnLiveOnly());
|
|
$this->assertTrue($deletedFromLivePage->isModifiedOnDraft());
|
|
|
|
// published page, deleted from both stages
|
|
$deletedFromAllStagesPage = new SiteTree();
|
|
$deletedFromAllStagesPage->write();
|
|
$deletedFromAllStagesPage->copyVersionToStage('Stage', 'Live');
|
|
$deletedFromAllStagesPage->deleteFromStage('Stage');
|
|
$deletedFromAllStagesPage->deleteFromStage('Live');
|
|
$this->assertTrue($deletedFromAllStagesPage->isArchived());
|
|
$this->assertFalse($deletedFromAllStagesPage->isOnDraft());
|
|
$this->assertFalse($deletedFromAllStagesPage->isPublished());
|
|
$this->assertFalse($deletedFromAllStagesPage->isOnDraftOnly());
|
|
$this->assertFalse($deletedFromAllStagesPage->isOnLiveOnly());
|
|
$this->assertFalse($deletedFromAllStagesPage->isModifiedOnDraft());
|
|
|
|
// published page, modified
|
|
$modifiedOnDraftPage = new SiteTree();
|
|
$modifiedOnDraftPage->write();
|
|
$modifiedOnDraftPage->copyVersionToStage('Stage', 'Live');
|
|
$modifiedOnDraftPage->Content = 'modified';
|
|
$modifiedOnDraftPage->write();
|
|
$this->assertFalse($modifiedOnDraftPage->isArchived());
|
|
$this->assertTrue($modifiedOnDraftPage->isOnDraft());
|
|
$this->assertTrue($modifiedOnDraftPage->isPublished());
|
|
$this->assertFalse($modifiedOnDraftPage->isOnDraftOnly());
|
|
$this->assertFalse($modifiedOnDraftPage->isOnLiveOnly());
|
|
$this->assertTrue($modifiedOnDraftPage->isModifiedOnDraft());
|
|
}
|
|
|
|
/**
|
|
* Test that a page can be completely deleted and restored to the stage site
|
|
*/
|
|
public function testRestoreToStage()
|
|
{
|
|
$page = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageID = $page->ID;
|
|
$page->delete();
|
|
$this->assertTrue(!DataObject::get_by_id(SiteTree::class, $pageID));
|
|
|
|
$deletedPage = Versioned::get_latest_version(SiteTree::class, $pageID);
|
|
$resultPage = $deletedPage->doRestoreToStage();
|
|
|
|
$requeriedPage = DataObject::get_by_id(SiteTree::class, $pageID);
|
|
|
|
$this->assertEquals($pageID, $resultPage->ID);
|
|
$this->assertEquals($pageID, $requeriedPage->ID);
|
|
$this->assertEquals('About Us', $requeriedPage->Title);
|
|
$this->assertInstanceOf(SiteTree::class, $requeriedPage);
|
|
|
|
|
|
$page2 = $this->objFromFixture(SiteTree::class, 'products');
|
|
$page2ID = $page2->ID;
|
|
$page2->doUnpublish();
|
|
$page2->delete();
|
|
|
|
// Check that if we restore while on the live site that the content still gets pushed to
|
|
// stage
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
$deletedPage = Versioned::get_latest_version(SiteTree::class, $page2ID);
|
|
$deletedPage->doRestoreToStage();
|
|
$this->assertFalse(
|
|
(bool)Versioned::get_one_by_stage(SiteTree::class, Versioned::LIVE, "\"SiteTree\".\"ID\" = " . $page2ID)
|
|
);
|
|
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
$requeriedPage = DataObject::get_by_id(SiteTree::class, $page2ID);
|
|
$this->assertEquals('Products', $requeriedPage->Title);
|
|
$this->assertInstanceOf(SiteTree::class, $requeriedPage);
|
|
}
|
|
|
|
public function testNoCascadingDeleteWithoutID()
|
|
{
|
|
Config::inst()->set('SiteTree', 'enforce_strict_hierarchy', true);
|
|
$count = SiteTree::get()->count();
|
|
$this->assertNotEmpty($count);
|
|
$obj = new SiteTree();
|
|
$this->assertFalse($obj->exists());
|
|
$fail = true;
|
|
try {
|
|
$obj->delete();
|
|
} catch (LogicException $e) {
|
|
$fail = false;
|
|
}
|
|
if ($fail) {
|
|
$this->fail('Failed to throw delete exception');
|
|
}
|
|
$this->assertCount($count, SiteTree::get());
|
|
}
|
|
|
|
public function testGetByLink()
|
|
{
|
|
$home = $this->objFromFixture(SiteTree::class, 'home');
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$product = $this->objFromFixture(SiteTree::class, 'product1');
|
|
$numeric0 = $this->objFromFixture(SiteTree::class, 'numeric0');
|
|
|
|
SiteTree::config()->nested_urls = false;
|
|
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link('/', false)->ID);
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link('/home/', false)->ID);
|
|
$this->assertEquals($about->ID, SiteTree::get_by_link($about->Link(), false)->ID);
|
|
$this->assertEquals($staff->ID, SiteTree::get_by_link($staff->Link(), false)->ID);
|
|
$this->assertEquals($product->ID, SiteTree::get_by_link($product->Link(), false)->ID);
|
|
$this->assertEquals($numeric0->ID, SiteTree::get_by_link($numeric0->Link(), false)->ID);
|
|
|
|
Config::modify()->set(SiteTree::class, 'nested_urls', true);
|
|
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link('/', false)->ID);
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link('/home/', false)->ID);
|
|
$this->assertEquals($about->ID, SiteTree::get_by_link($about->Link(), false)->ID);
|
|
$this->assertEquals($staff->ID, SiteTree::get_by_link($staff->Link(), false)->ID);
|
|
$this->assertEquals($product->ID, SiteTree::get_by_link($product->Link(), false)->ID);
|
|
$this->assertEquals($numeric0->ID, SiteTree::get_by_link($numeric0->Link(), false)->ID);
|
|
|
|
$this->assertEquals(
|
|
$staff->ID,
|
|
SiteTree::get_by_link('/my-staff/', false)->ID,
|
|
'Assert a unique URLSegment can be used for b/c.'
|
|
);
|
|
}
|
|
|
|
public function testGetByLinkAbsolute()
|
|
{
|
|
$home = $this->objFromFixture(SiteTree::class, 'home');
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$product = $this->objFromFixture(SiteTree::class, 'product1');
|
|
$numeric0 = $this->objFromFixture(SiteTree::class, 'numeric0');
|
|
|
|
$base = 'https://example.test/';
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link(Controller::join_links($base, '/'), false)->ID);
|
|
$this->assertEquals($home->ID, SiteTree::get_by_link(Controller::join_links($base, '/home/'), false)->ID);
|
|
$this->assertEquals($about->ID, SiteTree::get_by_link(Controller::join_links($base, $about->Link()), false)->ID);
|
|
$this->assertEquals($staff->ID, SiteTree::get_by_link(Controller::join_links($base, $staff->Link()), false)->ID);
|
|
$this->assertEquals($product->ID, SiteTree::get_by_link(Controller::join_links($base, $product->Link()), false)->ID);
|
|
$this->assertEquals($numeric0->ID, SiteTree::get_by_link(Controller::join_links($base, $numeric0->Link()), false)->ID);
|
|
}
|
|
|
|
public function testRelativeLink()
|
|
{
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$numeric0 = $this->objFromFixture(SiteTree::class, 'numeric0');
|
|
|
|
Config::modify()->set(SiteTree::class, 'nested_urls', true);
|
|
|
|
$this->assertEquals(
|
|
'about-us',
|
|
$about->RelativeLink(),
|
|
'Matches URLSegment on top level without parameters'
|
|
);
|
|
$this->assertEquals(
|
|
'about-us/my-staff',
|
|
$staff->RelativeLink(),
|
|
'Matches URLSegment plus parent on second level without parameters'
|
|
);
|
|
$this->assertEquals(
|
|
Controller::normaliseTrailingSlash('0/'),
|
|
$numeric0->RelativeLink(),
|
|
'Matches URLSegment for segment = 0'
|
|
);
|
|
$this->assertEquals(
|
|
'about-us/edit',
|
|
$about->RelativeLink('edit'),
|
|
'Matches URLSegment plus parameter on top level'
|
|
);
|
|
$this->assertEquals(
|
|
'about-us/tom&jerry',
|
|
$about->RelativeLink('tom&jerry'),
|
|
'Doesnt url encode parameter'
|
|
);
|
|
}
|
|
|
|
public function testPageLevel()
|
|
{
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$this->assertEquals(1, $about->getPageLevel());
|
|
$this->assertEquals(2, $staff->getPageLevel());
|
|
}
|
|
|
|
public function testAbsoluteLiveLink()
|
|
{
|
|
$parent = $this->objFromFixture(SiteTree::class, 'about');
|
|
$child = $this->objFromFixture(SiteTree::class, 'staff');
|
|
|
|
Config::modify()->set(SiteTree::class, 'nested_urls', true);
|
|
|
|
$child->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
$parent->URLSegment = 'changed-on-live';
|
|
$parent->write();
|
|
$parent->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
$parent->URLSegment = 'changed-on-draft';
|
|
$parent->write();
|
|
|
|
$this->assertStringEndsWith('changed-on-live/my-staff', $child->getAbsoluteLiveLink(false));
|
|
$this->assertStringEndsWith('changed-on-live/my-staff?stage=Live', $child->getAbsoluteLiveLink());
|
|
}
|
|
|
|
public function testDuplicateChildrenRetainSort()
|
|
{
|
|
$parent = new SiteTree();
|
|
$parent->Title = 'Parent';
|
|
$parent->write();
|
|
|
|
$child1 = new SiteTree();
|
|
$child1->ParentID = $parent->ID;
|
|
$child1->Title = 'Child 1';
|
|
$child1->Sort = 2;
|
|
$child1->write();
|
|
|
|
$child2 = new SiteTree();
|
|
$child2->ParentID = $parent->ID;
|
|
$child2->Title = 'Child 2';
|
|
$child2->Sort = 1;
|
|
$child2->write();
|
|
|
|
$duplicateParent = $parent->duplicateWithChildren();
|
|
$duplicateChildren = $duplicateParent->AllChildren()->toArray();
|
|
$this->assertCount(2, $duplicateChildren);
|
|
|
|
$duplicateChild2 = array_shift($duplicateChildren);
|
|
$duplicateChild1 = array_shift($duplicateChildren);
|
|
|
|
|
|
$this->assertEquals('Child 1', $duplicateChild1->Title);
|
|
$this->assertEquals('Child 2', $duplicateChild2->Title);
|
|
|
|
// assertGreaterThan works by having the LOWER value first
|
|
$this->assertGreaterThan($duplicateChild2->Sort, $duplicateChild1->Sort);
|
|
}
|
|
|
|
public function testDeleteFromStageOperatesRecursively()
|
|
{
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', false);
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageStaff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$pageStaffDuplicate = $this->objFromFixture(SiteTree::class, 'staffduplicate');
|
|
|
|
$pageAbout->delete();
|
|
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageAbout->ID));
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaff->ID) instanceof SiteTree);
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaffDuplicate->ID) instanceof SiteTree);
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', true);
|
|
}
|
|
|
|
public function testDeleteFromStageOperatesRecursivelyStrict()
|
|
{
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageStaff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$pageStaffDuplicate = $this->objFromFixture(SiteTree::class, 'staffduplicate');
|
|
|
|
$pageAbout->delete();
|
|
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageAbout->ID));
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageStaff->ID));
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageStaffDuplicate->ID));
|
|
}
|
|
|
|
public function testDuplicate()
|
|
{
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$dupe = $pageAbout->duplicate();
|
|
$this->assertEquals($pageAbout->Title, $dupe->Title);
|
|
$this->assertNotEquals($pageAbout->URLSegment, $dupe->URLSegment);
|
|
$this->assertNotEquals($pageAbout->Sort, $dupe->Sort);
|
|
}
|
|
|
|
public function testDeleteFromLiveOperatesRecursively()
|
|
{
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', false);
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageAbout->publishRecursive();
|
|
$pageStaff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$pageStaff->publishRecursive();
|
|
$pageStaffDuplicate = $this->objFromFixture(SiteTree::class, 'staffduplicate');
|
|
$pageStaffDuplicate->publishRecursive();
|
|
|
|
$parentPage = $this->objFromFixture(SiteTree::class, 'about');
|
|
|
|
$parentPage->doUnpublish();
|
|
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageAbout->ID));
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaff->ID) instanceof SiteTree);
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaffDuplicate->ID) instanceof SiteTree);
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', true);
|
|
}
|
|
|
|
public function testUnpublishDoesNotDeleteChildrenWithLooseHierachyOn()
|
|
{
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', false);
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageAbout->publishRecursive();
|
|
$pageStaff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$pageStaff->publishRecursive();
|
|
$pageStaffDuplicate = $this->objFromFixture(SiteTree::class, 'staffduplicate');
|
|
$pageStaffDuplicate->publishRecursive();
|
|
|
|
$parentPage = $this->objFromFixture(SiteTree::class, 'about');
|
|
$parentPage->doUnpublish();
|
|
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageAbout->ID));
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaff->ID) instanceof SiteTree);
|
|
$this->assertTrue(DataObject::get_by_id(SiteTree::class, $pageStaffDuplicate->ID) instanceof SiteTree);
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
Config::modify()->set(SiteTree::class, 'enforce_strict_hierarchy', true);
|
|
}
|
|
|
|
public function testDeleteFromLiveOperatesRecursivelyStrict()
|
|
{
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
$pageAbout = $this->objFromFixture(SiteTree::class, 'about');
|
|
$pageAbout->publishRecursive();
|
|
$pageStaff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$pageStaff->publishRecursive();
|
|
$pageStaffDuplicate = $this->objFromFixture(SiteTree::class, 'staffduplicate');
|
|
$pageStaffDuplicate->publishRecursive();
|
|
|
|
$parentPage = $this->objFromFixture(SiteTree::class, 'about');
|
|
$parentPage->doUnpublish();
|
|
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageAbout->ID));
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageStaff->ID));
|
|
$this->assertNull(DataObject::get_by_id(SiteTree::class, $pageStaffDuplicate->ID));
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
}
|
|
|
|
/**
|
|
* Simple test to confirm that querying from a particular archive date doesn't throw
|
|
* an error
|
|
*/
|
|
public function testReadArchiveDate()
|
|
{
|
|
DBDatetime::set_mock_now('2009-07-02 14:05:07');
|
|
$oldPage = new SiteTree();
|
|
$oldPage->Title = 'A really old page';
|
|
$oldPage->write();
|
|
DBDatetime::clear_mock_now();
|
|
|
|
$date = '2009-07-02 14:05:07';
|
|
Versioned::reading_archived_date($date);
|
|
$result = SiteTree::get()->where([
|
|
'"SiteTree"."ParentID"' => 0,
|
|
]);
|
|
$this->assertCount(1, $result, '"A really old page" should be returned');
|
|
}
|
|
|
|
public function testEditPermissions()
|
|
{
|
|
$editor = $this->objFromFixture(Member::class, "editor");
|
|
|
|
$home = $this->objFromFixture(SiteTree::class, "home");
|
|
$staff = $this->objFromFixture(SiteTree::class, "staff");
|
|
$products = $this->objFromFixture(SiteTree::class, "products");
|
|
$product1 = $this->objFromFixture(SiteTree::class, "product1");
|
|
$product4 = $this->objFromFixture(SiteTree::class, "product4");
|
|
|
|
// Test logged out users cannot edit
|
|
$this->logOut();
|
|
$this->assertFalse($staff->canEdit());
|
|
|
|
// Can't edit a page that is locked to admins
|
|
$this->assertFalse($home->canEdit($editor));
|
|
|
|
// Can edit a page that is locked to editors
|
|
$this->assertTrue($products->canEdit($editor));
|
|
|
|
// Can edit a child of that page that inherits
|
|
$this->assertTrue($product1->canEdit($editor));
|
|
|
|
// Can't edit a child of that page that has its permissions overridden
|
|
$this->assertFalse($product4->canEdit($editor));
|
|
}
|
|
|
|
public function testCanEditWithAccessToAllSections()
|
|
{
|
|
$page = new SiteTree();
|
|
$page->write();
|
|
$allSectionMember = $this->objFromFixture(Member::class, 'allsections');
|
|
$securityAdminMember = $this->objFromFixture(Member::class, 'securityadmin');
|
|
|
|
$this->assertTrue($page->canEdit($allSectionMember));
|
|
$this->assertFalse($page->canEdit($securityAdminMember));
|
|
}
|
|
|
|
public function testCreatePermissions()
|
|
{
|
|
// Test logged out users cannot create
|
|
$this->logOut();
|
|
$this->assertFalse(singleton(SiteTree::class)->canCreate());
|
|
|
|
// Login with another permission
|
|
$this->logInWithPermission('DUMMY');
|
|
$this->assertFalse(singleton(SiteTree::class)->canCreate());
|
|
|
|
// Login with basic CMS permission
|
|
$perms = SiteConfig::config()->required_permission;
|
|
$this->logInWithPermission(reset($perms));
|
|
$this->assertTrue(singleton(SiteTree::class)->canCreate());
|
|
|
|
// Test creation underneath a parent which this user doesn't have access to
|
|
$parent = $this->objFromFixture(SiteTree::class, 'about');
|
|
$this->assertFalse(singleton(SiteTree::class)->canCreate(null, ['Parent' => $parent]));
|
|
|
|
// Test creation underneath a parent which doesn't allow a certain child
|
|
$parentB = new SiteTreeTest_ClassB();
|
|
$parentB->Title = 'Only Allows SiteTreeTest_ClassC';
|
|
$parentB->write();
|
|
$this->assertTrue(singleton(SiteTreeTest_ClassA::class)->canCreate(null));
|
|
$this->assertFalse(singleton(SiteTreeTest_ClassA::class)->canCreate(null, ['Parent' => $parentB]));
|
|
$this->assertTrue(singleton(SiteTreeTest_ClassC::class)->canCreate(null, ['Parent' => $parentB]));
|
|
|
|
// Test creation underneath a parent which doesn't exist in the database. This should
|
|
// fall back to checking whether the user can create pages at the root of the site
|
|
$this->assertTrue(singleton(SiteTree::class)->canCreate(null, ['Parent' => singleton(SiteTree::class)]));
|
|
|
|
//Test we don't check for allowedChildren on parent context if it's not SiteTree instance
|
|
$this->assertTrue(
|
|
singleton(SiteTree::class)->canCreate(
|
|
null,
|
|
['Parent' => $this->objFromFixture(SiteTreeTest_DataObject::class, 'relations')]
|
|
)
|
|
);
|
|
}
|
|
|
|
public function testEditPermissionsOnDraftVsLive()
|
|
{
|
|
// Create an inherit-permission page
|
|
$page = new SiteTree();
|
|
$page->write();
|
|
$page->CanEditType = "Inherit";
|
|
$page->publishRecursive();
|
|
$pageID = $page->ID;
|
|
|
|
// Lock down the site config
|
|
$sc = $page->SiteConfig;
|
|
$sc->CanEditType = 'OnlyTheseUsers';
|
|
$sc->EditorGroups()->add($this->idFromFixture(Group::class, 'admins'));
|
|
$sc->write();
|
|
|
|
// Confirm that Member.editor can't edit the page
|
|
$member = $this->objFromFixture(Member::class, 'editor');
|
|
Security::setCurrentUser($member);
|
|
$this->assertFalse($page->canEdit());
|
|
|
|
// Change the page to be editable by Group.editors, but do not publish
|
|
$admin = $this->objFromFixture(Member::class, 'admin');
|
|
Security::setCurrentUser($admin);
|
|
$page->CanEditType = 'OnlyTheseUsers';
|
|
$page->EditorGroups()->add($this->idFromFixture(Group::class, 'editors'));
|
|
$page->write();
|
|
|
|
// Clear permission cache
|
|
/** @var InheritedPermissions $checker */
|
|
$checker = SiteTree::getPermissionChecker();
|
|
$checker->clearCache();
|
|
|
|
// Confirm that Member.editor can now edit the page
|
|
$member = $this->objFromFixture(Member::class, 'editor');
|
|
Security::setCurrentUser($member);
|
|
$this->assertTrue($page->canEdit());
|
|
|
|
// Publish the changes to the page
|
|
$admin = $this->objFromFixture(Member::class, 'admin');
|
|
Security::setCurrentUser($admin);
|
|
$page->publishRecursive();
|
|
|
|
// Confirm that Member.editor can still edit the page
|
|
$member = $this->objFromFixture(Member::class, 'editor');
|
|
Security::setCurrentUser($member);
|
|
$this->assertTrue($page->canEdit());
|
|
}
|
|
|
|
public function testCompareVersions()
|
|
{
|
|
// Necessary to avoid
|
|
$oldCleanerClass = HtmlDiff::$html_cleaner_class;
|
|
HtmlDiff::$html_cleaner_class = SiteTreeTest_NullHtmlCleaner::class;
|
|
|
|
$page = new SiteTree();
|
|
$page->write();
|
|
$this->assertEquals(1, $page->Version);
|
|
|
|
// Use inline element to avoid double wrapping applied to
|
|
// blocklevel elements depending on HTMLCleaner implementation:
|
|
// <ins><p> gets converted to <ins><p><inst>
|
|
$page->Content = "<span>This is a test</span>";
|
|
$page->write();
|
|
$this->assertEquals(2, $page->Version);
|
|
|
|
$diff = $page->compareVersions(1, 2);
|
|
|
|
$processedContent = trim($diff->Content ?? '');
|
|
$processedContent = preg_replace('/\s*</', '<', $processedContent ?? '');
|
|
$processedContent = preg_replace('/>\s*/', '>', $processedContent ?? '');
|
|
$this->assertEquals("<ins><span>This is a test</span></ins>", $processedContent);
|
|
|
|
HtmlDiff::$html_cleaner_class = $oldCleanerClass;
|
|
}
|
|
|
|
public function testAuthorIDAndPublisherIDFilledOutOnPublish()
|
|
{
|
|
// Ensure that we have a member ID who is doing all this work
|
|
$member = $this->objFromFixture(Member::class, "admin");
|
|
$this->logInAs($member);
|
|
|
|
// Write the page
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$about->Title = "Another title";
|
|
$about->write();
|
|
|
|
// Check the version created
|
|
$savedVersion = DB::prepared_query(
|
|
"SELECT \"AuthorID\", \"PublisherID\" FROM \"SiteTree_Versions\"
|
|
WHERE \"RecordID\" = ? ORDER BY \"Version\" DESC",
|
|
[$about->ID]
|
|
)->record();
|
|
$this->assertEquals($member->ID, $savedVersion['AuthorID']);
|
|
$this->assertEquals(0, $savedVersion['PublisherID']);
|
|
|
|
// Publish the page
|
|
$about->publishRecursive();
|
|
$publishedVersion = DB::prepared_query(
|
|
"SELECT \"AuthorID\", \"PublisherID\" FROM \"SiteTree_Versions\"
|
|
WHERE \"RecordID\" = ? ORDER BY \"Version\" DESC",
|
|
[$about->ID]
|
|
)->record();
|
|
|
|
// Check the version created
|
|
$this->assertEquals($member->ID, $publishedVersion['AuthorID']);
|
|
$this->assertEquals($member->ID, $publishedVersion['PublisherID']);
|
|
}
|
|
|
|
public function testLinkShortcodeHandler()
|
|
{
|
|
$aboutPage = $this->objFromFixture(SiteTree::class, 'about');
|
|
$redirectPage = $this->objFromFixture(RedirectorPage::class, 'external');
|
|
|
|
$parser = new ShortcodeParser();
|
|
$parser->register('sitetree_link', [SiteTree::class, 'link_shortcode_handler']);
|
|
|
|
$aboutShortcode = sprintf('[sitetree_link,id=%d]', $aboutPage->ID);
|
|
$aboutEnclosed = sprintf('[sitetree_link,id=%d]Example Content[/sitetree_link]', $aboutPage->ID);
|
|
|
|
$aboutShortcodeExpected = $aboutPage->Link();
|
|
$aboutEnclosedExpected = sprintf('<a href="%s">Example Content</a>', $aboutPage->Link());
|
|
|
|
$this->assertEquals(
|
|
$aboutShortcodeExpected,
|
|
$parser->parse($aboutShortcode),
|
|
'Test that simple linking works.'
|
|
);
|
|
$this->assertEquals(
|
|
$aboutEnclosedExpected,
|
|
$parser->parse($aboutEnclosed),
|
|
'Test enclosed content is linked.'
|
|
);
|
|
|
|
$aboutPage->delete();
|
|
|
|
$this->assertEquals(
|
|
$aboutShortcodeExpected,
|
|
$parser->parse($aboutShortcode),
|
|
'Test that deleted pages still link.'
|
|
);
|
|
$this->assertEquals($aboutEnclosedExpected, $parser->parse($aboutEnclosed));
|
|
|
|
$aboutShortcode = '[sitetree_link,id="-1"]';
|
|
$aboutEnclosed = '[sitetree_link,id="-1"]Example Content[/sitetree_link]';
|
|
|
|
$this->assertEquals('', $parser->parse($aboutShortcode), 'Test empty result if no suitable matches.');
|
|
$this->assertEquals('', $parser->parse($aboutEnclosed));
|
|
|
|
$redirectShortcode = sprintf('[sitetree_link,id=%d]', $redirectPage->ID);
|
|
$redirectEnclosed = sprintf('[sitetree_link,id=%d]Example Content[/sitetree_link]', $redirectPage->ID);
|
|
$redirectExpected = 'http://www.google.com?a&b';
|
|
|
|
$this->assertEquals($redirectExpected, $parser->parse($redirectShortcode));
|
|
$this->assertEquals(
|
|
sprintf('<a href="%s">Example Content</a>', $redirectExpected),
|
|
$parser->parse($redirectEnclosed)
|
|
);
|
|
|
|
$this->assertEquals('', $parser->parse('[sitetree_link]'), 'Test that invalid ID attributes are not parsed.');
|
|
$this->assertEquals('', $parser->parse('[sitetree_link,id="text"]'));
|
|
$this->assertEquals('', $parser->parse('[sitetree_link]Example Content[/sitetree_link]'));
|
|
}
|
|
|
|
public function testIsCurrent()
|
|
{
|
|
$aboutPage = $this->objFromFixture(SiteTree::class, 'about');
|
|
$productPage = $this->objFromFixture(SiteTree::class, 'products');
|
|
|
|
Director::set_current_page($aboutPage);
|
|
$this->assertTrue($aboutPage->isCurrent(), 'Assert that basic isCurrent checks works.');
|
|
$this->assertFalse($productPage->isCurrent());
|
|
|
|
$this->assertTrue(
|
|
DataObject::get_one(SiteTree::class, [
|
|
'"SiteTree"."Title"' => 'About Us',
|
|
])->isCurrent(),
|
|
'Assert that isCurrent works on another instance with the same ID.'
|
|
);
|
|
|
|
Director::set_current_page($newPage = new SiteTree());
|
|
$this->assertTrue($newPage->isCurrent(), 'Assert that isCurrent works on unsaved pages.');
|
|
}
|
|
|
|
public function testIsSection()
|
|
{
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$ceo = $this->objFromFixture(SiteTree::class, 'ceo');
|
|
|
|
Director::set_current_page($about);
|
|
$this->assertTrue($about->isSection());
|
|
$this->assertFalse($staff->isSection());
|
|
$this->assertFalse($ceo->isSection());
|
|
|
|
Director::set_current_page($staff);
|
|
$this->assertTrue($about->isSection());
|
|
$this->assertTrue($staff->isSection());
|
|
$this->assertFalse($ceo->isSection());
|
|
|
|
Director::set_current_page($ceo);
|
|
$this->assertTrue($about->isSection());
|
|
$this->assertTrue($staff->isSection());
|
|
$this->assertTrue($ceo->isSection());
|
|
}
|
|
|
|
public function testURLSegmentReserved()
|
|
{
|
|
$siteTree = new SiteTree(['URLSegment' => 'admin']);
|
|
$segment = $siteTree->validURLSegment();
|
|
|
|
$this->assertFalse($segment);
|
|
}
|
|
|
|
public function testURLSegmentAutoUpdate()
|
|
{
|
|
$sitetree = new SiteTree();
|
|
$sitetree->Title = _t(
|
|
'SilverStripe\\CMS\\Controllers\\CMSMain.NEWPAGE',
|
|
'New {pagetype}',
|
|
['pagetype' => $sitetree->i18n_singular_name()]
|
|
);
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'new-page',
|
|
$sitetree->URLSegment,
|
|
'Sets based on default title on first save'
|
|
);
|
|
|
|
$sitetree->Title = 'Changed';
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'changed',
|
|
$sitetree->URLSegment,
|
|
'Auto-updates when set to default title'
|
|
);
|
|
|
|
$sitetree->Title = 'Changed again';
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'changed',
|
|
$sitetree->URLSegment,
|
|
'Does not auto-update once title has been changed'
|
|
);
|
|
}
|
|
|
|
public function testURLSegmentAutoUpdateLocalized()
|
|
{
|
|
$oldLocale = i18n::get_locale();
|
|
i18n::set_locale('de_DE');
|
|
|
|
$sitetree = new SiteTree();
|
|
$sitetree->Title = _t(
|
|
'SilverStripe\\CMS\\Controllers\\CMSMain.NEWPAGE',
|
|
'New {pagetype}',
|
|
['pagetype' => $sitetree->i18n_singular_name()]
|
|
);
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'neue-seite',
|
|
$sitetree->URLSegment,
|
|
'Sets based on default title on first save'
|
|
);
|
|
|
|
$sitetree->Title = 'Changed';
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'changed',
|
|
$sitetree->URLSegment,
|
|
'Auto-updates when set to default title'
|
|
);
|
|
|
|
$sitetree->Title = 'Changed again';
|
|
$sitetree->write();
|
|
$this->assertEquals(
|
|
'changed',
|
|
$sitetree->URLSegment,
|
|
'Does not auto-update once title has been changed'
|
|
);
|
|
|
|
i18n::set_locale($oldLocale);
|
|
}
|
|
|
|
/**
|
|
* @covers \SilverStripe\CMS\Model\SiteTree::validURLSegment
|
|
*/
|
|
public function testValidURLSegmentURLSegmentConflicts()
|
|
{
|
|
$sitetree = new SiteTree();
|
|
SiteTree::config()->nested_urls = false;
|
|
|
|
$sitetree->URLSegment = 'home';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'URLSegment conflicts are recognised');
|
|
$sitetree->URLSegment = 'home-noconflict';
|
|
$this->assertTrue($sitetree->validURLSegment());
|
|
|
|
$sitetree->ParentID = $this->idFromFixture(SiteTree::class, 'about');
|
|
$sitetree->URLSegment = 'home';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'Conflicts are still recognised with a ParentID value');
|
|
|
|
Config::modify()->set(SiteTree::class, 'nested_urls', true);
|
|
|
|
$sitetree->ParentID = 0;
|
|
$sitetree->URLSegment = 'home';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'URLSegment conflicts are recognised');
|
|
|
|
$sitetree->ParentID = $this->idFromFixture(SiteTree::class, 'about');
|
|
$this->assertTrue($sitetree->validURLSegment(), 'URLSegments can be the same across levels');
|
|
|
|
$sitetree->URLSegment = 'my-staff';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'Nested URLSegment conflicts are recognised');
|
|
$sitetree->URLSegment = 'my-staff-noconflict';
|
|
$this->assertTrue($sitetree->validURLSegment());
|
|
}
|
|
|
|
/**
|
|
* @covers \SilverStripe\CMS\Model\SiteTree::validURLSegment
|
|
*/
|
|
public function testValidURLSegmentClassNameConflicts()
|
|
{
|
|
$sitetree = new SiteTree();
|
|
$sitetree->URLSegment = Controller::class;
|
|
|
|
$this->assertTrue($sitetree->validURLSegment(), 'Class names are no longer conflicts');
|
|
}
|
|
|
|
/**
|
|
* @covers \SilverStripe\CMS\Model\SiteTree::validURLSegment
|
|
*/
|
|
public function testValidURLSegmentControllerConflicts()
|
|
{
|
|
Config::modify()->set(SiteTree::class, 'nested_urls', true);
|
|
|
|
$sitetree = new SiteTree();
|
|
$sitetree->ParentID = $this->idFromFixture(SiteTreeTest_Conflicted::class, 'parent');
|
|
|
|
$sitetree->URLSegment = 'index';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'index is not a valid URLSegment');
|
|
|
|
$sitetree->URLSegment = 'conflicted-action';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'allowed_actions conflicts are recognised');
|
|
|
|
$sitetree->URLSegment = 'conflicted-template';
|
|
$this->assertFalse($sitetree->validURLSegment(), 'Action-specific template conflicts are recognised');
|
|
|
|
$sitetree->URLSegment = 'valid';
|
|
$this->assertTrue($sitetree->validURLSegment(), 'Valid URLSegment values are allowed');
|
|
}
|
|
|
|
public function testURLSegmentPrioritizesExtensionVotes()
|
|
{
|
|
$sitetree = new SiteTree();
|
|
$sitetree->URLSegment = 'unique-segment';
|
|
$this->assertTrue($sitetree->validURLSegment());
|
|
|
|
SiteTree::add_extension(SiteTreeTest_Extension::class);
|
|
$sitetree = new SiteTree();
|
|
$sitetree->URLSegment = 'unique-segment';
|
|
$this->assertFalse($sitetree->validURLSegment());
|
|
SiteTree::remove_extension(SiteTreeTest_Extension::class);
|
|
}
|
|
|
|
public function testURLSegmentMultiByte()
|
|
{
|
|
URLSegmentFilter::config()->set('default_allow_multibyte', true);
|
|
$sitetree = new SiteTree();
|
|
$sitetree->write();
|
|
|
|
$sitetree->URLSegment = 'brötchen';
|
|
$sitetree->write();
|
|
$sitetree = DataObject::get_by_id(SiteTree::class, $sitetree->ID, false);
|
|
$this->assertEquals($sitetree->URLSegment, rawurlencode('brötchen'));
|
|
|
|
$sitetree->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
$sitetree = DataObject::get_by_id(SiteTree::class, $sitetree->ID, false);
|
|
$this->assertEquals($sitetree->URLSegment, rawurlencode('brötchen'));
|
|
$sitetreeLive = Versioned::get_one_by_stage(
|
|
SiteTree::class,
|
|
Versioned::LIVE,
|
|
'"SiteTree"."ID" = ' . $sitetree->ID,
|
|
false
|
|
);
|
|
$this->assertEquals($sitetreeLive->URLSegment, rawurlencode('brötchen'));
|
|
}
|
|
|
|
public function testVersionsAreCreated()
|
|
{
|
|
$p = new SiteTree();
|
|
$p->Content = "one";
|
|
$p->write();
|
|
$this->assertEquals(1, $p->Version);
|
|
|
|
// No changes don't bump version
|
|
$p->write();
|
|
$this->assertEquals(1, $p->Version);
|
|
|
|
$p->Content = "two";
|
|
$p->write();
|
|
$this->assertEquals(2, $p->Version);
|
|
|
|
// Only change meta-data don't bump version
|
|
$p->HasBrokenLink = true;
|
|
$p->write();
|
|
$p->HasBrokenLink = false;
|
|
$p->write();
|
|
$this->assertEquals(2, $p->Version);
|
|
|
|
$p->Content = "three";
|
|
$p->write();
|
|
$this->assertEquals(3, $p->Version);
|
|
}
|
|
|
|
public function testHidePagetypes()
|
|
{
|
|
SiteTree::config()->set('hide_pagetypes', ['Page']);
|
|
$classes = SiteTree::page_type_classes();
|
|
$this->assertNotContains('Page', $classes);
|
|
}
|
|
|
|
public function testPageTypeClasses()
|
|
{
|
|
$classes = SiteTree::page_type_classes();
|
|
$this->assertNotContains(SiteTree::class, $classes, 'Page types do not include base class');
|
|
$this->assertContains('Page', $classes, 'Page types do contain subclasses');
|
|
|
|
// Testing what happens in an incorrect config value is set - hide_ancestor should be a string
|
|
Config::modify()->set(SiteTreeTest_ClassA::class, 'hide_ancestor', true);
|
|
$newClasses = SiteTree::page_type_classes();
|
|
$this->assertEquals(
|
|
$classes,
|
|
$newClasses,
|
|
'Setting hide_ancestor to a boolean (incorrect) value caused a page class to be hidden'
|
|
);
|
|
|
|
// Testing what happens if a valid config value is set
|
|
Config::modify()->set(SiteTreeTest_ClassA::class, 'hide_ancestor', 'Page');
|
|
$classes = SiteTree::page_type_classes();
|
|
$this->assertNotContains('Page', $classes);
|
|
}
|
|
|
|
/**
|
|
* Tests that core subclasses of SiteTree are included in allowedChildren() by default, but not instances of
|
|
* HiddenClass
|
|
*/
|
|
public function testAllowedChildrenContainsCoreSubclassesButNotHiddenClass()
|
|
{
|
|
$page = new SiteTree();
|
|
$allowedChildren = $page->allowedChildren();
|
|
|
|
$this->assertContains(
|
|
VirtualPage::class,
|
|
$allowedChildren,
|
|
'Includes core subclasses by default'
|
|
);
|
|
|
|
$this->assertNotContains(
|
|
SiteTreeTest_ClassE::class,
|
|
$allowedChildren,
|
|
'HiddenClass instances should not be returned'
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Tests that various types of SiteTree classes will or will not be returned from the allowedChildren method
|
|
* @dataProvider allowedChildrenProvider
|
|
* @param string $className
|
|
* @param array $expected
|
|
* @param string $assertionMessage
|
|
*/
|
|
public function testAllowedChildren($className, $expected, $assertionMessage)
|
|
{
|
|
$class = new $className;
|
|
$this->assertEquals($expected, $class->allowedChildren(), $assertionMessage);
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
public function allowedChildrenProvider()
|
|
{
|
|
return [
|
|
[
|
|
// Class name
|
|
SiteTreeTest_ClassA::class,
|
|
// Expected
|
|
[ SiteTreeTest_ClassB::class ],
|
|
// Assertion message
|
|
'Direct setting of allowed children',
|
|
],
|
|
[
|
|
SiteTreeTest_ClassB::class,
|
|
[ SiteTreeTest_ClassC::class, SiteTreeTest_ClassCext::class ],
|
|
'Includes subclasses',
|
|
],
|
|
[
|
|
SiteTreeTest_ClassC::class,
|
|
[],
|
|
'Null setting',
|
|
],
|
|
[
|
|
SiteTreeTest_ClassD::class,
|
|
[SiteTreeTest_ClassC::class],
|
|
'Excludes subclasses if class is prefixed by an asterisk',
|
|
],
|
|
];
|
|
}
|
|
|
|
public function testAllowedChildrenValidation()
|
|
{
|
|
$page = new SiteTree();
|
|
$page->write();
|
|
$classA = new SiteTreeTest_ClassA();
|
|
$classA->write();
|
|
$classB = new SiteTreeTest_ClassB();
|
|
$classB->write();
|
|
$classC = new SiteTreeTest_ClassC();
|
|
$classC->write();
|
|
$classD = new SiteTreeTest_ClassD();
|
|
$classD->write();
|
|
$classCext = new SiteTreeTest_ClassCext();
|
|
$classCext->write();
|
|
|
|
$classB->ParentID = $page->ID;
|
|
$valid = $classB->validate();
|
|
$this->assertTrue($valid->isValid(), "Does allow children on unrestricted parent");
|
|
|
|
$classB->ParentID = $classA->ID;
|
|
$valid = $classB->validate();
|
|
$this->assertTrue($valid->isValid(), "Does allow child specifically allowed by parent");
|
|
|
|
$classC->ParentID = $classA->ID;
|
|
$valid = $classC->validate();
|
|
$this->assertFalse($valid->isValid(), "Doesnt allow child on parents specifically restricting children");
|
|
|
|
$classB->ParentID = $classC->ID;
|
|
$valid = $classB->validate();
|
|
$this->assertFalse($valid->isValid(), "Doesnt allow child on parents disallowing all children");
|
|
|
|
$classB->ParentID = $classCext->ID;
|
|
$valid = $classB->validate();
|
|
$this->assertTrue($valid->isValid(), "Extensions of allowed classes are incorrectly reported as invalid");
|
|
|
|
$classCext->ParentID = $classD->ID;
|
|
$valid = $classCext->validate();
|
|
$this->assertFalse(
|
|
$valid->isValid(),
|
|
"Doesnt allow child where only parent class is allowed on parent node, and asterisk prefixing is used"
|
|
);
|
|
}
|
|
|
|
public function testClassDropdown()
|
|
{
|
|
$sitetree = new SiteTree();
|
|
$method = new ReflectionMethod($sitetree, 'getClassDropdown');
|
|
$method->setAccessible(true);
|
|
|
|
Security::setCurrentUser(null);
|
|
$this->assertArrayNotHasKey(SiteTreeTest_ClassA::class, $method->invoke($sitetree));
|
|
|
|
$this->loginWithPermission('ADMIN');
|
|
$this->assertArrayHasKey(SiteTreeTest_ClassA::class, $method->invoke($sitetree));
|
|
|
|
$this->loginWithPermission('CMS_ACCESS_CMSMain');
|
|
$this->assertArrayHasKey(SiteTreeTest_ClassA::class, $method->invoke($sitetree));
|
|
|
|
$this->logInWithPermission('ADMIN');
|
|
$rootPage = $this->objFromFixture(SiteTree::class, 'home');
|
|
$nonRootPage = $this->objFromFixture(SiteTree::class, 'staff');
|
|
|
|
$this->assertArrayNotHasKey(SiteTreeTest_NotRoot::class, $method->invoke($rootPage));
|
|
$this->assertArrayHasKey(SiteTreeTest_NotRoot::class, $method->invoke($nonRootPage));
|
|
|
|
Security::setCurrentUser(null);
|
|
}
|
|
|
|
public function testCanBeRoot()
|
|
{
|
|
$page = new SiteTree();
|
|
$page->ParentID = 0;
|
|
$page->write();
|
|
|
|
$notRootPage = new SiteTreeTest_NotRoot();
|
|
$notRootPage->ParentID = 0;
|
|
$isDetected = false;
|
|
try {
|
|
$notRootPage->write();
|
|
} catch (ValidationException $e) {
|
|
$this->assertStringContainsString('is not allowed on the root level', $e->getMessage());
|
|
$isDetected = true;
|
|
}
|
|
|
|
if (!$isDetected) {
|
|
$this->fail('Fails validation with $can_be_root=false');
|
|
}
|
|
}
|
|
|
|
public function testModifyStatusFlagByInheritance()
|
|
{
|
|
$node = new SiteTreeTest_StageStatusInherit();
|
|
$treeTitle = $node->getTreeTitle();
|
|
$this->assertStringContainsString('InheritedTitle', $treeTitle);
|
|
$this->assertStringContainsString('inherited-class', $treeTitle);
|
|
}
|
|
|
|
public function testMenuTitleIsUnsetWhenEqualsTitle()
|
|
{
|
|
$page = new SiteTree();
|
|
$page->Title = 'orig';
|
|
$page->MenuTitle = 'orig';
|
|
$page->write();
|
|
|
|
// change menu title
|
|
$page->MenuTitle = 'changed';
|
|
$page->write();
|
|
$page = SiteTree::get()->byID($page->ID);
|
|
$this->assertEquals('changed', $page->getField('MenuTitle'));
|
|
|
|
// change menu title back
|
|
$page->MenuTitle = 'orig';
|
|
$page->write();
|
|
$page = SiteTree::get()->byID($page->ID);
|
|
$this->assertEquals(null, $page->getField('MenuTitle'));
|
|
}
|
|
|
|
public function testMetaTagGeneratorDisabling()
|
|
{
|
|
$generator = Config::inst()->get(SiteTree::class, 'meta_generator');
|
|
|
|
// Stub to ensure customisations don't affect the test
|
|
Config::modify()->set(SiteTree::class, 'meta_generator', 'SilverStripe - https://www.silverstripe.org');
|
|
|
|
$page = new SiteTreeTest_PageNode();
|
|
|
|
$meta = $page->MetaTags();
|
|
$this->assertStringContainsString('meta name="generator"', $meta, 'Should include generator tag');
|
|
$this->assertStringContainsString(
|
|
'content="SilverStripe - https://www.silverstripe.org',
|
|
$meta,
|
|
'Should contain default meta generator info'
|
|
);
|
|
|
|
// test proper escaping of quotes in attribute value
|
|
Config::modify()->set(SiteTree::class, 'meta_generator', 'Generator with "quotes" in it');
|
|
$meta = $page->MetaTags();
|
|
$this->assertStringContainsString(
|
|
'content="Generator with "quotes" in it',
|
|
$meta,
|
|
'test proper escaping of values from Config'
|
|
);
|
|
|
|
// test empty generator - no tag should appear at all
|
|
Config::modify()->set(SiteTree::class, 'meta_generator', '');
|
|
$meta = $page->MetaTags();
|
|
$this->assertStringNotContainsString(
|
|
'meta name="generator"',
|
|
$meta,
|
|
'test blank value means no tag generated'
|
|
);
|
|
}
|
|
|
|
|
|
public function testGetBreadcrumbItems()
|
|
{
|
|
$page = $this->objFromFixture(SiteTree::class, "breadcrumbs");
|
|
$this->assertEquals(1, $page->getBreadcrumbItems()->count(), "Only display current page.");
|
|
|
|
// Test breadcrumb order
|
|
$page = $this->objFromFixture(SiteTree::class, "breadcrumbs5");
|
|
$breadcrumbs = $page->getBreadcrumbItems();
|
|
$this->assertEquals($breadcrumbs->count(), 5, "Display all breadcrumbs");
|
|
$this->assertEquals($breadcrumbs->first()->Title, "Breadcrumbs", "Breadcrumbs should be the first item.");
|
|
$this->assertEquals($breadcrumbs->last()->Title, "Breadcrumbs 5", "Breadcrumbs 5 should be last item.");
|
|
|
|
// Test breadcrumb max depth
|
|
$breadcrumbs = $page->getBreadcrumbItems(2);
|
|
$this->assertEquals($breadcrumbs->count(), 2, "Max depth should limit the breadcrumbs to 2 items.");
|
|
$this->assertEquals($breadcrumbs->first()->Title, "Breadcrumbs 4", "First item should be Breadrcumbs 4.");
|
|
$this->assertEquals($breadcrumbs->last()->Title, "Breadcrumbs 5", "Breadcrumbs 5 should be last.");
|
|
}
|
|
|
|
/**
|
|
* Tests SiteTree::MetaTags
|
|
* Note that this test makes no assumption on the closing of tags (other than <title></title>)
|
|
*/
|
|
public function testMetaTags()
|
|
{
|
|
$this->logInWithPermission('ADMIN');
|
|
$page = $this->objFromFixture(SiteTree::class, 'metapage');
|
|
|
|
// Test with title
|
|
$meta = $page->MetaTags();
|
|
$charset = Config::inst()->get(ContentNegotiator::class, 'encoding');
|
|
$this->assertStringContainsString('<meta http-equiv="Content-Type" content="text/html; charset=' . $charset . '"', $meta);
|
|
$this->assertStringContainsString('<meta name="description" content="The <br /> and <br> tags"', $meta);
|
|
$this->assertStringContainsString('<link rel="canonical" href="http://www.mysite.com/html-and-xml"', $meta);
|
|
$this->assertStringContainsString('<meta name="x-page-id" content="' . $page->ID.'"', $meta);
|
|
$this->assertStringContainsString('<meta name="x-cms-edit-link" content="' . $page->getCMSEditLink().'"', $meta);
|
|
$this->assertStringContainsString('<title>HTML & XML</title>', $meta);
|
|
|
|
// Test without title
|
|
$meta = $page->MetaTags(false);
|
|
$this->assertStringNotContainsString('<title>', $meta);
|
|
|
|
$meta = $page->MetaTags('false');
|
|
$this->assertStringNotContainsString('<title>', $meta);
|
|
}
|
|
|
|
public function testMetaComponents()
|
|
{
|
|
$this->logInWithPermission('ADMIN');
|
|
/** @var SiteTree $page */
|
|
$page = $this->objFromFixture(SiteTree::class, 'metapage');
|
|
|
|
$charset = Config::inst()->get(ContentNegotiator::class, 'encoding');
|
|
|
|
$mockVersionProvider = $this->getMockBuilder(VersionProvider::class)
|
|
->setMethods(['getModuleVersion'])
|
|
->getMock();
|
|
$mockVersionProvider->method('getModuleVersion')->willReturn('4.50.99');
|
|
$page->setVersionProvider($mockVersionProvider);
|
|
|
|
$expected = [
|
|
'title' => [
|
|
'tag' => 'title',
|
|
'content' => "HTML & XML",
|
|
],
|
|
'generator' => [
|
|
'attributes' => [
|
|
'name' => 'generator',
|
|
'content' => sprintf(
|
|
'%s %s',
|
|
Config::inst()->get(SiteTree::class, 'meta_generator'),
|
|
'4.50'
|
|
)
|
|
],
|
|
],
|
|
'contentType' => [
|
|
'attributes' => [
|
|
'http-equiv' => 'Content-Type',
|
|
'content' => "text/html; charset=$charset",
|
|
],
|
|
],
|
|
'description' => [
|
|
'attributes' => [
|
|
'name' => 'description',
|
|
'content' => 'The <br /> and <br> tags'
|
|
]
|
|
],
|
|
'pageId' => [
|
|
'attributes' => [
|
|
'name' => 'x-page-id',
|
|
'content' => $page->ID
|
|
],
|
|
],
|
|
'cmsEditLink' => [
|
|
'attributes' => [
|
|
'name' => 'x-cms-edit-link',
|
|
'content' => $page->getCMSEditLink()
|
|
]
|
|
]
|
|
];
|
|
|
|
$this->assertEquals($expected, $page->MetaComponents());
|
|
|
|
// test the meta generator tag version can be configured off
|
|
Config::modify()->set(SiteTree::class, 'show_meta_generator_version', false);
|
|
$content = $expected['generator']['attributes']['content'];
|
|
$expected['generator']['attributes']['content'] = str_replace(' 4.50', '', $content ?? '');
|
|
$this->assertEquals($expected, $page->MetaComponents());
|
|
}
|
|
|
|
/**
|
|
* Test that orphaned pages are handled correctly
|
|
*/
|
|
public function testOrphanedPages()
|
|
{
|
|
$origStage = Versioned::get_reading_mode();
|
|
|
|
// Setup user who can view draft content, but lacks cms permission.
|
|
// To users such as this, orphaned pages should be inaccessible. canView for these pages is only
|
|
// necessary for admin / cms users, who require this permission to edit / rearrange these pages.
|
|
$permission = new Permission();
|
|
$permission->Code = 'VIEW_DRAFT_CONTENT';
|
|
$group = new Group(['Title' => 'Staging Users']);
|
|
$group->write();
|
|
$group->Permissions()->add($permission);
|
|
$member = new Member();
|
|
$member->Email = 'someguy@example.com';
|
|
$member->write();
|
|
$member->Groups()->add($group);
|
|
|
|
// both pages are viewable in stage
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$this->assertFalse($about->isOrphaned());
|
|
$this->assertFalse($staff->isOrphaned());
|
|
$this->assertTrue($about->canView($member));
|
|
$this->assertTrue($staff->canView($member));
|
|
|
|
// Publishing only the child page to live should orphan the live record, but not the staging one
|
|
$staff->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
$this->assertFalse($staff->isOrphaned());
|
|
$this->assertTrue($staff->canView($member));
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff'); // Live copy of page
|
|
$this->assertTrue($staff->isOrphaned()); // because parent isn't published
|
|
$this->assertFalse($staff->canView($member));
|
|
|
|
// Publishing the parent page should restore visibility
|
|
Versioned::set_stage(Versioned::DRAFT);
|
|
$about = $this->objFromFixture(SiteTree::class, 'about');
|
|
$about->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
Versioned::set_stage(Versioned::LIVE);
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$this->assertFalse($staff->isOrphaned());
|
|
$this->assertTrue($staff->canView($member));
|
|
|
|
// Removing staging page should not prevent live page being visible
|
|
$about->deleteFromStage('Stage');
|
|
$staff->deleteFromStage('Stage');
|
|
$staff = $this->objFromFixture(SiteTree::class, 'staff');
|
|
$this->assertFalse($staff->isOrphaned());
|
|
$this->assertTrue($staff->canView($member));
|
|
|
|
// Cleanup
|
|
Versioned::set_reading_mode($origStage);
|
|
}
|
|
|
|
/**
|
|
* Test archived page behaviour
|
|
*/
|
|
public function testArchivedPages()
|
|
{
|
|
$this->logInWithPermission('ADMIN');
|
|
|
|
/** @var SiteTree $page */
|
|
$page = $this->objFromFixture(SiteTree::class, 'home');
|
|
$this->assertTrue($page->canAddChildren());
|
|
$this->assertTrue($page->isOnDraft());
|
|
$this->assertFalse($page->isPublished());
|
|
|
|
// Publish
|
|
$page->publishRecursive();
|
|
$this->assertTrue($page->canAddChildren());
|
|
$this->assertTrue($page->isOnDraft());
|
|
$this->assertTrue($page->isPublished());
|
|
|
|
// Archive
|
|
$page->doArchive();
|
|
$this->assertFalse($page->canAddChildren());
|
|
$this->assertFalse($page->isOnDraft());
|
|
$this->assertTrue($page->isArchived());
|
|
$this->assertFalse($page->isPublished());
|
|
}
|
|
|
|
public function testCanNot()
|
|
{
|
|
// Test that
|
|
$this->logInWithPermission('ADMIN');
|
|
$page = new SiteTreeTest_AdminDenied();
|
|
$this->assertFalse($page->canCreate());
|
|
$this->assertFalse($page->canEdit());
|
|
$this->assertFalse($page->canDelete());
|
|
$this->assertFalse($page->canAddChildren());
|
|
$this->assertFalse($page->canView());
|
|
}
|
|
|
|
public function testCanPublish()
|
|
{
|
|
$page = new SiteTreeTest_ClassD();
|
|
$this->logOut();
|
|
|
|
// Test that false overrides any can_publish = true
|
|
SiteTreeTest_ExtensionA::$can_publish = true;
|
|
SiteTreeTest_ExtensionB::$can_publish = false;
|
|
$this->assertFalse($page->canPublish());
|
|
SiteTreeTest_ExtensionA::$can_publish = false;
|
|
SiteTreeTest_ExtensionB::$can_publish = true;
|
|
$this->assertFalse($page->canPublish());
|
|
|
|
// Test null extensions fall back to canEdit()
|
|
SiteTreeTest_ExtensionA::$can_publish = null;
|
|
SiteTreeTest_ExtensionB::$can_publish = null;
|
|
$page->canEditValue = true;
|
|
$this->assertTrue($page->canPublish());
|
|
$page->canEditValue = false;
|
|
$this->assertFalse($page->canPublish());
|
|
}
|
|
|
|
/**
|
|
* Test url rewriting extensions
|
|
*/
|
|
public function testLinkExtension()
|
|
{
|
|
Director::config()->set('alternate_base_url', 'http://www.baseurl.com');
|
|
$page = new SiteTreeTest_ClassD();
|
|
$page->URLSegment = 'classd';
|
|
$page->write();
|
|
$this->assertEquals(
|
|
'http://www.updatedhost.com/classd/myaction?extra=1',
|
|
$page->Link('myaction')
|
|
);
|
|
$this->assertEquals(
|
|
'http://www.updatedhost.com/classd/myaction?extra=1',
|
|
$page->AbsoluteLink('myaction')
|
|
);
|
|
$this->assertEquals(
|
|
'classd/myaction',
|
|
$page->RelativeLink('myaction')
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Test that the controller name for a Page instance can be gathered by appending "Controller" to the Page
|
|
* class name in a PSR-2 compliant manner.
|
|
*/
|
|
public function testGetControllerName()
|
|
{
|
|
$page = new Page();
|
|
$this->assertSame(PageController::class, $page->getControllerName());
|
|
}
|
|
|
|
|
|
/**
|
|
* Test that the controller name for a SiteTree instance can be gathered when set directly via config var
|
|
*/
|
|
public function testGetControllerNameFromConfig()
|
|
{
|
|
Config::inst()->set(SiteTree::class, 'controller_name', 'This\\Is\\A\\New\\Controller');
|
|
$page = new SiteTree();
|
|
$this->assertSame('This\\Is\\A\\New\\Controller', $page->getControllerName());
|
|
}
|
|
|
|
/**
|
|
* Test that the controller name for a Namespaced SiteTree instance can be gathered when set via namespace map
|
|
*/
|
|
public function testGetControllerNameFromNamespaceMappingConfig()
|
|
{
|
|
Config::inst()->merge(SiteTree::class, 'namespace_mapping', [
|
|
'SilverStripe\\CMS\\Tests\\Page' => 'SilverStripe\\CMS\\Tests\\Controllers',
|
|
]);
|
|
|
|
$namespacedSiteTree = new SiteTreeTest_NamespaceMapTest();
|
|
$this->assertSame(SiteTreeTest_NamespaceMapTestController::class, $namespacedSiteTree->getControllerName());
|
|
}
|
|
|
|
public function testTreeTitleCache()
|
|
{
|
|
$siteTree = new SiteTree();
|
|
$user = $this->objFromFixture(Member::class, 'allsections');
|
|
Security::setCurrentUser($user);
|
|
$pageClass = array_values(SiteTree::page_type_classes())[0];
|
|
|
|
$mockPageMissesCache = $this->getMockBuilder($pageClass)
|
|
->setMethods(['canCreate'])
|
|
->getMock();
|
|
$mockPageMissesCache
|
|
->expects($this->exactly(3))
|
|
->method('canCreate');
|
|
|
|
$mockPageHitsCache = $this->getMockBuilder($pageClass)
|
|
->setMethods(['canCreate'])
|
|
->getMock();
|
|
$mockPageHitsCache
|
|
->expects($this->never())
|
|
->method('canCreate');
|
|
|
|
// Initially, cache misses (1)
|
|
Injector::inst()->registerService($mockPageMissesCache, $pageClass);
|
|
$title = $siteTree->getTreeTitle();
|
|
$this->assertNotNull($title);
|
|
|
|
// Now it hits
|
|
Injector::inst()->registerService($mockPageHitsCache, $pageClass);
|
|
$title = $siteTree->getTreeTitle();
|
|
$this->assertNotNull($title);
|
|
|
|
|
|
// Mutating member record invalidates cache. Misses (2)
|
|
$user->FirstName = 'changed';
|
|
$user->write();
|
|
Injector::inst()->registerService($mockPageMissesCache, $pageClass);
|
|
$title = $siteTree->getTreeTitle();
|
|
$this->assertNotNull($title);
|
|
|
|
// Now it hits again
|
|
Injector::inst()->registerService($mockPageHitsCache, $pageClass);
|
|
$title = $siteTree->getTreeTitle();
|
|
$this->assertNotNull($title);
|
|
|
|
// Different user. Misses. (3)
|
|
$user = $this->objFromFixture(Member::class, 'editor');
|
|
Security::setCurrentUser($user);
|
|
Injector::inst()->registerService($mockPageMissesCache, $pageClass);
|
|
$title = $siteTree->getTreeTitle();
|
|
$this->assertNotNull($title);
|
|
}
|
|
|
|
public function testDependentPagesOnUnsavedRecord()
|
|
{
|
|
$record = new SiteTree();
|
|
$pages = $record->DependentPages();
|
|
$this->assertCount(0, $pages, 'Unsaved pages should have no dependent pages');
|
|
}
|
|
|
|
public function testOnBeforeWriteClearsEmbedShortcodeCache()
|
|
{
|
|
/** @var CacheInterface $cache */
|
|
$url = 'http://www.test-service.com/abc123';
|
|
$content = '<p>Some content with an [embed url="' . $url . '" thumbnail="https://example.com/mythumb.jpg" ' .
|
|
'class="leftAlone ss-htmleditorfield-file embed" width="480" height="270"]' . $url . '[/embed]</p>';
|
|
$embedHtml = '<iframe myattr="something" />';
|
|
|
|
// use reflection to access private methods
|
|
$provider = new EmbedShortcodeProvider();
|
|
$reflector = new \ReflectionClass(EmbedShortcodeProvider::class);
|
|
$method = $reflector->getMethod('getCache');
|
|
$method->setAccessible(true);
|
|
$cache = $method->invokeArgs($provider, []);
|
|
$method = $reflector->getMethod('deriveCacheKey');
|
|
$method->setAccessible(true);
|
|
$class = 'leftAlone ss-htmleditorfield-file embed';
|
|
$width = '480';
|
|
$height = '270';
|
|
$key = $method->invokeArgs($provider, [$url, $class, $width, $height]);
|
|
|
|
// Set cache (VersionedCacheAdapter) on both DRAFT and LIVE
|
|
foreach ([Versioned::DRAFT, Versioned::LIVE] as $stage) {
|
|
Versioned::withVersionedMode(function () use ($cache, $key, $embedHtml, $stage) {
|
|
Versioned::set_reading_mode("Stage.$stage");
|
|
$cache->set($key, $embedHtml);
|
|
});
|
|
}
|
|
|
|
// Create new page on DRAFT
|
|
$page = new SiteTree();
|
|
$page->Content = $content;
|
|
$page->write();
|
|
|
|
// Assert both DRAFT and LIVE caches were cleared on DRAFT $page->write()
|
|
foreach ([Versioned::DRAFT, Versioned::LIVE] as $stage) {
|
|
Versioned::withVersionedMode(function () use ($cache, $key, $stage) {
|
|
Versioned::set_reading_mode("Stage.$stage");
|
|
$this->assertFalse($cache->has($key));
|
|
});
|
|
}
|
|
}
|
|
|
|
public function testGetCMSActions()
|
|
{
|
|
// Create new page on DRAFT
|
|
$page = new SiteTree();
|
|
$page->Content = md5(rand(0, PHP_INT_MAX));
|
|
$page->write();
|
|
|
|
// BEGIN DRAFT
|
|
$actions = $page->getCMSActions();
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('MajorActions.action_save'),
|
|
'save action present for a saved draft page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('MajorActions.action_publish'),
|
|
'publish action present for a saved draft page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive'),
|
|
'archive action present for a saved draft page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign'),
|
|
'addtocampaign action present for a saved draft page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_unpublish'),
|
|
'no unpublish action present for a saved draft page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback'),
|
|
'no rollback action present for a saved draft page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('MajorActions.action_restore'),
|
|
'no restore action present for a saved draft page'
|
|
);
|
|
// END DRAFT
|
|
|
|
// BEGIN PUBLISHED
|
|
$page->publishRecursive();
|
|
$actions = $page->getCMSActions();
|
|
$this->assertNull(
|
|
$actions->fieldByName('MajorActions.action_save'),
|
|
'no save action present for a published page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive'),
|
|
'no archive action present for a saved draft page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback'),
|
|
'rollback action present for a published page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_unpublish'),
|
|
'no unpublish action present for a published page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign'),
|
|
'addtocampaign action present for a published page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('MajorActions.action_restore'),
|
|
'no restore action present for a published page'
|
|
);
|
|
// END PUBLISHED
|
|
|
|
// BEGIN DRAFT AFTER PUBLISHED
|
|
$page->Content = md5(rand(0, PHP_INT_MAX));
|
|
$page->write();
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('MajorActions.action_save'),
|
|
'save action present for a changed published page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_unpublish'),
|
|
'unpublish action present for a changed published page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive'),
|
|
'archive action present for a changed published page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback'),
|
|
'rollback action present for a changed published page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign'),
|
|
'addtocampaign action present for a changed published page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('MajorActions.action_restore'),
|
|
'no restore action present for a changed published page'
|
|
);
|
|
// END DRAFT AFTER PUBLISHED
|
|
|
|
// BEGIN ARCHIVED
|
|
$page->doArchive();
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertNull(
|
|
$actions->fieldByName('MajorActions.action_save'),
|
|
'no save action present for a archived page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_unpublish'),
|
|
'no unpublish action present for a archived page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive'),
|
|
'no archive action present for a archived page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback'),
|
|
'no rollback action present for a archived page'
|
|
);
|
|
$this->assertNull(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign'),
|
|
'no addtocampaign action present for a archived page'
|
|
);
|
|
$this->assertNotNull(
|
|
$actions->fieldByName('MajorActions.action_restore'),
|
|
'restore action present for a archived page'
|
|
);
|
|
// END ARCHIVED
|
|
}
|
|
|
|
public function testGetCMSActionsWithoutForms()
|
|
{
|
|
// Create new page on DRAFT
|
|
$page = new SiteTree();
|
|
$page->Content = md5(rand(0, PHP_INT_MAX));
|
|
$page->write();
|
|
|
|
// BEGIN DRAFT
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('MajorActions.action_save')->getForm(),
|
|
'save action has no form when page is draft'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('MajorActions.action_publish')->getForm(),
|
|
'publish action has no form when page is draft'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive')->getForm(),
|
|
'archive action has no form when page is draft'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign')->getForm(),
|
|
'addtocampaign action has no form when page is draft'
|
|
);
|
|
// END DRAFT
|
|
|
|
// BEGIN PUBLISHED
|
|
$page->publishRecursive();
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback')->getForm(),
|
|
'rollback action has no form when page is published'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign')->getForm(),
|
|
'addtocampaign action has no form when page is published'
|
|
);
|
|
// END PUBLISHED
|
|
|
|
// BEGIN DRAFT AFTER PUBLISHED
|
|
$page->Content = md5(rand(0, PHP_INT_MAX));
|
|
$page->write();
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('MajorActions.action_save')->getForm(),
|
|
'save action has no form when page is draft after published'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_unpublish')->getForm(),
|
|
'unpublish action has no form when page is draft after published'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_archive')->getForm(),
|
|
'archive action has no form when page is draft after published'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_rollback')->getForm(),
|
|
'rollback action has no form when page is draft after published'
|
|
);
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('ActionMenus.MoreOptions.action_addtocampaign')->getForm(),
|
|
'addtocampaign action has no form when page is draft after published'
|
|
);
|
|
// END DRAFT AFTER PUBLISHED
|
|
|
|
// BEGIN ARCHIVED
|
|
$page->doArchive();
|
|
$actions = $page->getCMSActions();
|
|
|
|
$this->assertEmpty(
|
|
$actions->fieldByName('MajorActions.action_restore')->getForm(),
|
|
'retore action has no form when page archived'
|
|
);
|
|
// END ARCHIVED
|
|
}
|
|
|
|
public function testGetCMSEditLinkForManagedDataObject()
|
|
{
|
|
$page = $this->objFromFixture(PageWithChild::class, 'root');
|
|
$child = $this->objFromFixture(BelongsToPage::class, 'one');
|
|
$this->assertSame(
|
|
"http://localhost/admin/pages/edit/show/$page->ID/field/ChildObjects/item/$child->ID",
|
|
$page->getCMSEditLinkForManagedDataObject($child, 'Parent')
|
|
);
|
|
}
|
|
|
|
public function testCMSEditLink()
|
|
{
|
|
$page = $this->objFromFixture(PageWithChild::class, 'root');
|
|
$child = $this->objFromFixture(BelongsToPage::class, 'one');
|
|
$this->assertSame(
|
|
"http://localhost/admin/pages/edit/show/$page->ID",
|
|
$page->getCMSEditLink()
|
|
);
|
|
$this->assertSame(
|
|
"http://localhost/admin/pages/edit/show/$page->ID/field/ChildObjects/item/$child->ID",
|
|
$child->getCMSEditLink()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideSanitiseExtraMeta
|
|
*/
|
|
public function testSanitiseExtraMeta(string $extraMeta, string $expected, string $message): void
|
|
{
|
|
$siteTree = new SiteTree();
|
|
$siteTree->ExtraMeta = $extraMeta;
|
|
$siteTree->write();
|
|
$this->assertSame($expected, $siteTree->ExtraMeta, $message);
|
|
}
|
|
|
|
public function provideSanitiseExtraMeta(): array
|
|
{
|
|
return [
|
|
[
|
|
'<link rel="canonical" accesskey="X" sometrigger="alert(1)" />',
|
|
'<link rel="canonical" sometrigger="alert(1)">',
|
|
'accesskey attribute is removed'
|
|
],
|
|
[
|
|
'<link rel="canonical" onclick="alert(1)" /><meta name="x" onerror="alert(0)">',
|
|
'<link rel="canonical"><meta name="x">',
|
|
'Attributes starting with "on" are removed'
|
|
],
|
|
[
|
|
'<link rel="canonical" onclick=alert(1) /><meta name="x" onerror=\'alert(0)\'>',
|
|
'<link rel="canonical"><meta name="x">',
|
|
'Attributes with different quote styles are removed'
|
|
],
|
|
[
|
|
'<link rel="canonical" ONCLICK=alert(1) /><meta name="x" oNeRrOr=\'alert(0)\'>',
|
|
'<link rel="canonical"><meta name="x">',
|
|
'Mixed case attributes are removed'
|
|
],
|
|
[
|
|
'<link rel="canonical" accesskey="X" onclick="alert(1)" name="x" />',
|
|
'<link rel="canonical" name="x">',
|
|
'Multiple attributes are removed'
|
|
],
|
|
[
|
|
'<link rel="canonical" href="valid" ;;// somethingdodgy <onmouseover=alert(1)>',
|
|
'<link rel="canonical" href="valid" somethingdodgy=""><onmouseover></onmouseover>',
|
|
'Invalid HTML is converted to valid HTML and parsed'
|
|
],
|
|
];
|
|
}
|
|
|
|
public function testOnAfterRevertToLive()
|
|
{
|
|
// Create new page and publish it
|
|
$page = SiteTree::create();
|
|
$page->Content = 'Test content';
|
|
$id = $page->write();
|
|
$page->publishRecursive();
|
|
|
|
// Add link to non-page object
|
|
/** @var NotPageObject $obj */
|
|
$obj = $this->objFromFixture(NotPageObject::class, 'object1');
|
|
$obj->Content = '<a href="[sitetree_link,id='. $id .']">Link to Page</a>';
|
|
$obj->write();
|
|
|
|
// Test that method doesn't throw exception
|
|
$this->expectNotToPerformAssertions();
|
|
$method = new ReflectionMethod(SiteTree::class, 'onAfterRevertToLive');
|
|
$method->setAccessible(true);
|
|
$method->invoke($page);
|
|
}
|
|
}
|