2009-05-06 08:36:16 +02:00
|
|
|
<?php
|
|
|
|
|
2016-10-14 03:30:05 +02:00
|
|
|
namespace SilverStripe\ORM\Tests;
|
|
|
|
|
2024-09-23 04:31:50 +02:00
|
|
|
use SilverStripe\Core\Validation\ValidationException;
|
2017-03-21 04:22:23 +01:00
|
|
|
use SilverStripe\Versioned\Versioned;
|
2016-08-19 00:51:35 +02:00
|
|
|
use SilverStripe\Dev\SapphireTest;
|
2016-06-15 06:03:16 +02:00
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
class HierarchyTest extends SapphireTest
|
|
|
|
{
|
|
|
|
protected static $fixture_file = 'HierarchyTest.yml';
|
|
|
|
|
2020-04-20 19:58:09 +02:00
|
|
|
protected static $extra_dataobjects = [
|
2016-12-16 05:34:21 +01:00
|
|
|
HierarchyTest\TestObject::class,
|
|
|
|
HierarchyTest\HideTestObject::class,
|
|
|
|
HierarchyTest\HideTestSubObject::class,
|
2021-09-24 14:17:25 +02:00
|
|
|
HierarchyTest\HierarchyOnSubclassTestObject::class,
|
|
|
|
HierarchyTest\HierarchyOnSubclassTestSubObject::class,
|
2020-04-20 19:58:09 +02:00
|
|
|
];
|
2016-12-16 05:34:21 +01:00
|
|
|
|
2017-06-22 12:50:45 +02:00
|
|
|
public static function getExtraDataObjects()
|
2017-03-21 04:22:23 +01:00
|
|
|
{
|
|
|
|
// Prevent setup breaking if versioned module absent
|
|
|
|
if (class_exists(Versioned::class)) {
|
|
|
|
return parent::getExtraDataObjects();
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2021-10-27 04:39:47 +02:00
|
|
|
protected function setUp(): void
|
2017-03-21 04:22:23 +01:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
|
|
|
|
// Note: Soft support for versioned module optionality
|
|
|
|
if (!class_exists(Versioned::class)) {
|
|
|
|
$this->markTestSkipped('HierarchyTest requires the Versioned extension');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-16 05:34:21 +01:00
|
|
|
/**
|
|
|
|
* Test the Hierarchy prevents infinite loops.
|
|
|
|
*/
|
|
|
|
public function testPreventLoop()
|
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
$this->expectException(ValidationException::class);
|
|
|
|
$this->expectExceptionMessage(sprintf(
|
|
|
|
'Infinite loop found within the "%s" hierarchy',
|
|
|
|
HierarchyTest\TestObject::class
|
|
|
|
));
|
2016-12-16 05:34:21 +01:00
|
|
|
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2');
|
|
|
|
$obj2aa = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2aa');
|
|
|
|
|
|
|
|
$obj2->ParentID = $obj2aa->ID;
|
|
|
|
$obj2->write();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test Hierarchy::AllHistoricalChildren().
|
|
|
|
*/
|
|
|
|
public function testAllHistoricalChildren()
|
|
|
|
{
|
|
|
|
// Delete some objs
|
|
|
|
$this->objFromFixture(HierarchyTest\TestObject::class, 'obj2b')->delete();
|
|
|
|
$this->objFromFixture(HierarchyTest\TestObject::class, 'obj3a')->delete();
|
|
|
|
$this->objFromFixture(HierarchyTest\TestObject::class, 'obj3')->delete();
|
|
|
|
|
|
|
|
// Check that obj1-3 appear at the top level of the AllHistoricalChildren tree
|
|
|
|
$this->assertEquals(
|
2020-04-20 19:58:09 +02:00
|
|
|
["Obj 1", "Obj 2", "Obj 3"],
|
2017-03-29 06:23:49 +02:00
|
|
|
HierarchyTest\TestObject::singleton()->AllHistoricalChildren()->column('Title')
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
// Check numHistoricalChildren
|
2017-03-29 06:23:49 +02:00
|
|
|
$this->assertEquals(3, HierarchyTest\TestObject::singleton()->numHistoricalChildren());
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
// Check that both obj 2 children are returned
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2');
|
|
|
|
$this->assertEquals(
|
2020-04-20 19:58:09 +02:00
|
|
|
["Obj 2a", "Obj 2b"],
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2->AllHistoricalChildren()->column('Title')
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check numHistoricalChildren
|
|
|
|
$this->assertEquals(2, $obj2->numHistoricalChildren());
|
|
|
|
|
|
|
|
|
|
|
|
// Obj 3 has been deleted; let's bring it back from the grave
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj3 */
|
|
|
|
$obj3 = Versioned::get_including_deleted(
|
|
|
|
HierarchyTest\TestObject::class,
|
|
|
|
"\"Title\" = 'Obj 3'"
|
|
|
|
)->First();
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
// Check that all obj 3 children are returned
|
|
|
|
$this->assertEquals(
|
2020-04-20 19:58:09 +02:00
|
|
|
["Obj 3a", "Obj 3b", "Obj 3c", "Obj 3d"],
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj3->AllHistoricalChildren()->column('Title')
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check numHistoricalChildren
|
|
|
|
$this->assertEquals(4, $obj3->numHistoricalChildren());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testNumChildren()
|
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj1 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj1 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj1');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2 */
|
|
|
|
$obj2 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2');
|
|
|
|
/** @var HierarchyTest\TestObject $obj3 */
|
|
|
|
$obj3 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj3');
|
|
|
|
/** @var HierarchyTest\TestObject $obj2a */
|
|
|
|
$obj2a = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2a');
|
|
|
|
/** @var HierarchyTest\TestObject $obj2b */
|
|
|
|
$obj2b = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2b');
|
|
|
|
/** @var HierarchyTest\TestObject $obj3a */
|
|
|
|
$obj3a = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj3a');
|
|
|
|
/** @var HierarchyTest\TestObject $obj3b */
|
|
|
|
$obj3b = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj3d');
|
|
|
|
|
|
|
|
$this->assertEquals(0, $obj1->numChildren());
|
|
|
|
$this->assertEquals(2, $obj2->numChildren());
|
|
|
|
$this->assertEquals(4, $obj3->numChildren());
|
|
|
|
$this->assertEquals(2, $obj2a->numChildren());
|
|
|
|
$this->assertEquals(0, $obj2b->numChildren());
|
|
|
|
$this->assertEquals(2, $obj3a->numChildren());
|
|
|
|
$this->assertEquals(0, $obj3b->numChildren());
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj1Child1 = new HierarchyTest\TestObject();
|
|
|
|
$obj1Child1->ParentID = $obj1->ID;
|
|
|
|
$obj1Child1->write();
|
|
|
|
$this->assertEquals(
|
|
|
|
$obj1->numChildren(false),
|
|
|
|
1,
|
|
|
|
'numChildren() caching can be disabled through method parameter'
|
|
|
|
);
|
|
|
|
$obj1Child2 = new HierarchyTest\TestObject();
|
|
|
|
$obj1Child2->ParentID = $obj1->ID;
|
|
|
|
$obj1Child2->write();
|
|
|
|
$obj1->flushCache();
|
|
|
|
$this->assertEquals(
|
|
|
|
$obj1->numChildren(),
|
|
|
|
2,
|
|
|
|
'numChildren() caching can be disabled by flushCache()'
|
|
|
|
);
|
|
|
|
}
|
2021-09-24 14:17:25 +02:00
|
|
|
|
|
|
|
public function testNumChildrenHierarchyOnSubclass()
|
|
|
|
{
|
|
|
|
/** @var HierarchyTest\HierarchyOnSubclassTestObject $obj5 */
|
|
|
|
$obj5 = $this->objFromFixture(HierarchyTest\HierarchyOnSubclassTestObject::class, 'obj5');
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$obj5->hasMethod('numChildren'),
|
|
|
|
'numChildren() cannot be called on object without Hierarchy extension'
|
|
|
|
);
|
|
|
|
|
|
|
|
/** @var HierarchyTest\HierarchyOnSubclassTestSubObject $obj5a */
|
|
|
|
$obj5a = $this->objFromFixture(HierarchyTest\HierarchyOnSubclassTestSubObject::class, 'obj5a');
|
|
|
|
/** @var HierarchyTest\HierarchyOnSubclassTestSubObject $obj5b */
|
|
|
|
$obj5b = $this->objFromFixture(HierarchyTest\HierarchyOnSubclassTestSubObject::class, 'obj5b');
|
|
|
|
|
|
|
|
$this->assertEquals(2, $obj5a->numChildren());
|
|
|
|
$this->assertEquals(1, $obj5b->numChildren());
|
|
|
|
|
|
|
|
$obj5bChild2 = new HierarchyTest\HierarchyOnSubclassTestSubObject();
|
|
|
|
$obj5bChild2->ParentID = $obj5b->ID;
|
|
|
|
$obj5bChild2->write();
|
|
|
|
$this->assertEquals(
|
|
|
|
$obj5b->numChildren(false),
|
|
|
|
2,
|
|
|
|
'numChildren() caching can be disabled through method parameter'
|
|
|
|
);
|
|
|
|
$obj5bChild3 = new HierarchyTest\HierarchyOnSubclassTestSubObject();
|
|
|
|
$obj5bChild3->ParentID = $obj5b->ID;
|
|
|
|
$obj5bChild3->write();
|
|
|
|
$obj5b->flushCache();
|
|
|
|
$this->assertEquals(
|
|
|
|
$obj5b->numChildren(),
|
|
|
|
3,
|
|
|
|
'numChildren() caching can be disabled by flushCache()'
|
|
|
|
);
|
|
|
|
}
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
public function testLoadDescendantIDListIntoArray()
|
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2a */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2a = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2a');
|
|
|
|
$obj2b = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2b');
|
|
|
|
$obj2aa = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2aa');
|
|
|
|
$obj2ab = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2ab');
|
|
|
|
|
|
|
|
$obj2IdList = $obj2->getDescendantIDList();
|
|
|
|
$obj2aIdList = $obj2a->getDescendantIDList();
|
|
|
|
|
|
|
|
$this->assertContains($obj2a->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2b->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2aa->ID, $obj2IdList);
|
|
|
|
$this->assertContains($obj2ab->ID, $obj2IdList);
|
2022-04-14 03:12:59 +02:00
|
|
|
$this->assertEquals(4, count($obj2IdList ?? []));
|
2016-12-16 05:34:21 +01:00
|
|
|
|
|
|
|
$this->assertContains($obj2aa->ID, $obj2aIdList);
|
|
|
|
$this->assertContains($obj2ab->ID, $obj2aIdList);
|
2022-04-14 03:12:59 +02:00
|
|
|
$this->assertEquals(2, count($obj2aIdList ?? []));
|
2016-12-16 05:34:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The "only deleted from stage" argument to liveChildren() should exclude
|
|
|
|
* any page that has been moved to another location on the stage site
|
|
|
|
*/
|
|
|
|
public function testLiveChildrenOnlyDeletedFromStage()
|
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj1 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj1 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj1');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2a */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2a = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2a');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2b */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2b = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2b');
|
|
|
|
|
|
|
|
// Get a published set of objects for our fixture
|
|
|
|
$obj1->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2a->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
$obj2b->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
|
|
|
|
// Then delete 2a from stage and move 2b to a sub-node of 1.
|
|
|
|
$obj2a->delete();
|
|
|
|
$obj2b->ParentID = $obj1->ID;
|
|
|
|
$obj2b->write();
|
|
|
|
|
|
|
|
// Get live children, excluding pages that have been moved on the stage site
|
|
|
|
$children = $obj2->liveChildren(true, true)->column("Title");
|
|
|
|
|
|
|
|
// 2a has been deleted from stage and should be shown
|
|
|
|
$this->assertContains("Obj 2a", $children);
|
|
|
|
|
|
|
|
// 2b has merely been moved to a different parent and so shouldn't be shown
|
|
|
|
$this->assertNotContains("Obj 2b", $children);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testBreadcrumbs()
|
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj1 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj1 = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj1');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2a */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2a = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2a');
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\TestObject $obj2aa */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj2aa = $this->objFromFixture(HierarchyTest\TestObject::class, 'obj2aa');
|
|
|
|
|
|
|
|
$this->assertEquals('Obj 1', $obj1->getBreadcrumbs());
|
|
|
|
$this->assertEquals('Obj 2 » Obj 2a', $obj2a->getBreadcrumbs());
|
|
|
|
$this->assertEquals('Obj 2 » Obj 2a » Obj 2aa', $obj2aa->getBreadcrumbs());
|
|
|
|
}
|
|
|
|
|
2021-09-24 12:29:43 +02:00
|
|
|
public function testNoHideFromHierarchy()
|
2016-12-16 05:34:21 +01:00
|
|
|
{
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\HideTestObject $obj4 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj4 = $this->objFromFixture(HierarchyTest\HideTestObject::class, 'obj4');
|
|
|
|
$obj4->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
|
|
|
|
foreach ($obj4->stageChildren() as $child) {
|
|
|
|
$child->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
}
|
|
|
|
$this->assertEquals($obj4->stageChildren()->Count(), 2);
|
|
|
|
$this->assertEquals($obj4->liveChildren()->Count(), 2);
|
|
|
|
}
|
|
|
|
|
2021-09-24 12:29:43 +02:00
|
|
|
public function testHideFromHierarchy()
|
2016-12-16 05:34:21 +01:00
|
|
|
{
|
2022-11-15 06:20:54 +01:00
|
|
|
HierarchyTest\HideTestObject::config()->merge(
|
2016-12-16 05:34:21 +01:00
|
|
|
'hide_from_hierarchy',
|
2017-03-29 06:23:49 +02:00
|
|
|
[ HierarchyTest\HideTestSubObject::class ]
|
2016-12-16 05:34:21 +01:00
|
|
|
);
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\HideTestObject $obj4 */
|
2016-12-16 05:34:21 +01:00
|
|
|
$obj4 = $this->objFromFixture(HierarchyTest\HideTestObject::class, 'obj4');
|
|
|
|
$obj4->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
|
|
|
|
// load without using stage children otherwise it'll bbe filtered before it's publish
|
|
|
|
// we need to publish all of them, and expect liveChildren to return some.
|
|
|
|
$children = HierarchyTest\HideTestObject::get()
|
|
|
|
->filter('ParentID', (int)$obj4->ID)
|
|
|
|
->exclude('ID', (int)$obj4->ID);
|
|
|
|
|
2017-03-29 06:23:49 +02:00
|
|
|
/** @var HierarchyTest\HideTestObject $child */
|
2016-12-16 05:34:21 +01:00
|
|
|
foreach ($children as $child) {
|
|
|
|
$child->copyVersionToStage(Versioned::DRAFT, Versioned::LIVE);
|
|
|
|
}
|
|
|
|
$this->assertEquals($obj4->stageChildren()->Count(), 1);
|
|
|
|
$this->assertEquals($obj4->liveChildren()->Count(), 1);
|
|
|
|
}
|
2011-03-23 04:32:24 +01:00
|
|
|
}
|