mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
Merge pull request #7583 from open-sausages/pulls/4/relation-object-navigation
BUG Ensure relObject() safely bails on empty objects
This commit is contained in:
commit
333ecc201c
@ -1017,7 +1017,10 @@ class DataList extends ViewableData implements SS_List, Filterable, Sortable, Li
|
|||||||
public function relation($relationName)
|
public function relation($relationName)
|
||||||
{
|
{
|
||||||
$ids = $this->column('ID');
|
$ids = $this->column('ID');
|
||||||
return singleton($this->dataClass)->$relationName()->forForeignID($ids);
|
$singleton = DataObject::singleton($this->dataClass);
|
||||||
|
/** @var HasManyList|ManyManyList $relation */
|
||||||
|
$relation = $singleton->$relationName($ids);
|
||||||
|
return $relation;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function dbObject($fieldName)
|
public function dbObject($fieldName)
|
||||||
|
@ -1584,10 +1584,14 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
* Returns a one-to-many relation as a HasManyList
|
* Returns a one-to-many relation as a HasManyList
|
||||||
*
|
*
|
||||||
* @param string $componentName Name of the component
|
* @param string $componentName Name of the component
|
||||||
|
* @param int|array $id Optional ID(s) for parent of this relation, if not the current record
|
||||||
* @return HasManyList|UnsavedRelationList The components of the one-to-many relationship.
|
* @return HasManyList|UnsavedRelationList The components of the one-to-many relationship.
|
||||||
*/
|
*/
|
||||||
public function getComponents($componentName)
|
public function getComponents($componentName, $id = null)
|
||||||
{
|
{
|
||||||
|
if (!isset($id)) {
|
||||||
|
$id = $this->ID;
|
||||||
|
}
|
||||||
$result = null;
|
$result = null;
|
||||||
|
|
||||||
$schema = $this->getSchema();
|
$schema = $this->getSchema();
|
||||||
@ -1601,7 +1605,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
}
|
}
|
||||||
|
|
||||||
// If we haven't been written yet, we can't save these relations, so use a list that handles this case
|
// If we haven't been written yet, we can't save these relations, so use a list that handles this case
|
||||||
if (!$this->ID) {
|
if (!$id) {
|
||||||
if (!isset($this->unsavedRelations[$componentName])) {
|
if (!isset($this->unsavedRelations[$componentName])) {
|
||||||
$this->unsavedRelations[$componentName] =
|
$this->unsavedRelations[$componentName] =
|
||||||
new UnsavedRelationList(static::class, $componentName, $componentClass);
|
new UnsavedRelationList(static::class, $componentName, $componentClass);
|
||||||
@ -1620,7 +1624,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
|
|
||||||
return $result
|
return $result
|
||||||
->setDataQueryParam($this->getInheritableQueryParams())
|
->setDataQueryParam($this->getInheritableQueryParams())
|
||||||
->forForeignID($this->ID);
|
->forForeignID($id);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -1799,10 +1803,14 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
/**
|
/**
|
||||||
* Returns a many-to-many component, as a ManyManyList.
|
* Returns a many-to-many component, as a ManyManyList.
|
||||||
* @param string $componentName Name of the many-many component
|
* @param string $componentName Name of the many-many component
|
||||||
|
* @param int|array $id Optional ID for parent of this relation, if not the current record
|
||||||
* @return RelationList|UnsavedRelationList The set of components
|
* @return RelationList|UnsavedRelationList The set of components
|
||||||
*/
|
*/
|
||||||
public function getManyManyComponents($componentName)
|
public function getManyManyComponents($componentName, $id = null)
|
||||||
{
|
{
|
||||||
|
if (!isset($id)) {
|
||||||
|
$id = $this->ID;
|
||||||
|
}
|
||||||
$schema = static::getSchema();
|
$schema = static::getSchema();
|
||||||
$manyManyComponent = $schema->manyManyComponent(static::class, $componentName);
|
$manyManyComponent = $schema->manyManyComponent(static::class, $componentName);
|
||||||
if (!$manyManyComponent) {
|
if (!$manyManyComponent) {
|
||||||
@ -1814,7 +1822,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
}
|
}
|
||||||
|
|
||||||
// If we haven't been written yet, we can't save these relations, so use a list that handles this case
|
// If we haven't been written yet, we can't save these relations, so use a list that handles this case
|
||||||
if (!$this->ID) {
|
if (!$id) {
|
||||||
if (!isset($this->unsavedRelations[$componentName])) {
|
if (!isset($this->unsavedRelations[$componentName])) {
|
||||||
$this->unsavedRelations[$componentName] =
|
$this->unsavedRelations[$componentName] =
|
||||||
new UnsavedRelationList($manyManyComponent['parentClass'], $componentName, $manyManyComponent['childClass']);
|
new UnsavedRelationList($manyManyComponent['parentClass'], $componentName, $manyManyComponent['childClass']);
|
||||||
@ -1846,7 +1854,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
// foreignID set elsewhere.
|
// foreignID set elsewhere.
|
||||||
return $result
|
return $result
|
||||||
->setDataQueryParam($this->getInheritableQueryParams())
|
->setDataQueryParam($this->getInheritableQueryParams())
|
||||||
->forForeignID($this->ID);
|
->forForeignID($id);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -2013,8 +2021,8 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, use the database field's scaffolder
|
// Otherwise, use the database field's scaffolder
|
||||||
} else {
|
} elseif ($object = $this->relObject($fieldName)) {
|
||||||
$field = $this->relObject($fieldName)->scaffoldSearchField();
|
$field = $object->scaffoldSearchField();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Allow fields to opt out of search
|
// Allow fields to opt out of search
|
||||||
@ -2679,82 +2687,69 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
* The path to the related field is specified with dot separated syntax
|
* The path to the related field is specified with dot separated syntax
|
||||||
* (eg: Parent.Child.Child.FieldName).
|
* (eg: Parent.Child.Child.FieldName).
|
||||||
*
|
*
|
||||||
* @param string $fieldPath
|
* If a relation is blank, this will return null instead.
|
||||||
|
* If a relation name is invalid (e.g. non-relation on a parent) this
|
||||||
|
* can throw a LogicException.
|
||||||
|
*
|
||||||
|
* @param string $fieldPath List of paths on this object. All items in this path
|
||||||
|
* must be ViewableData implementors
|
||||||
*
|
*
|
||||||
* @return mixed DBField of the field on the object or a DataList instance.
|
* @return mixed DBField of the field on the object or a DataList instance.
|
||||||
|
* @throws LogicException If accessing invalid relations
|
||||||
*/
|
*/
|
||||||
public function relObject($fieldPath)
|
public function relObject($fieldPath)
|
||||||
{
|
{
|
||||||
$object = null;
|
$object = null;
|
||||||
|
|
||||||
if (strpos($fieldPath, '.') !== false) {
|
|
||||||
$parts = explode('.', $fieldPath);
|
|
||||||
$fieldName = array_pop($parts);
|
|
||||||
|
|
||||||
// Traverse dot syntax
|
|
||||||
$component = $this;
|
$component = $this;
|
||||||
|
|
||||||
foreach ($parts as $relation) {
|
// Parse all relations
|
||||||
if ($component instanceof SS_List) {
|
foreach (explode('.', $fieldPath) as $relation) {
|
||||||
if (method_exists($component, $relation)) {
|
if (!$component) {
|
||||||
$component = $component->$relation();
|
return null;
|
||||||
} else {
|
|
||||||
/** @var DataList $component */
|
|
||||||
$component = $component->relation($relation);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$component = $component->$relation();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$object = $component->dbObject($fieldName);
|
// Inspect relation type
|
||||||
|
if (ClassInfo::hasMethod($component, $relation)) {
|
||||||
|
$component = $component->$relation();
|
||||||
|
} elseif ($component instanceof Relation || $component instanceof DataList) {
|
||||||
|
// $relation could either be a field (aggregate), or another relation
|
||||||
|
$singleton = DataObject::singleton($component->dataClass());
|
||||||
|
$component = $singleton->dbObject($relation) ?: $component->relation($relation);
|
||||||
|
} elseif ($component instanceof DataObject && ($dbObject = $component->dbObject($relation))) {
|
||||||
|
$component = $dbObject;
|
||||||
|
} elseif ($component instanceof ViewableData && $component->hasField($relation)) {
|
||||||
|
$component = $component->obj($relation);
|
||||||
} else {
|
} else {
|
||||||
$object = $this->dbObject($fieldPath);
|
throw new LogicException(
|
||||||
|
"$relation is not a relation/field on ".get_class($component)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return $object;
|
return $component;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Traverses to a field referenced by relationships between data objects, returning the value
|
* Traverses to a field referenced by relationships between data objects, returning the value
|
||||||
* The path to the related field is specified with dot separated syntax (eg: Parent.Child.Child.FieldName)
|
* The path to the related field is specified with dot separated syntax (eg: Parent.Child.Child.FieldName)
|
||||||
*
|
*
|
||||||
* @param $fieldName string
|
* @param string $fieldName string
|
||||||
* @return string | null - will return null on a missing value
|
* @return mixed Will return null on a missing value
|
||||||
*/
|
*/
|
||||||
public function relField($fieldName)
|
public function relField($fieldName)
|
||||||
{
|
{
|
||||||
|
// Navigate to relative parent using relObject() if needed
|
||||||
$component = $this;
|
$component = $this;
|
||||||
|
if (($pos = strrpos($fieldName, '.')) !== false) {
|
||||||
// We're dealing with relations here so we traverse the dot syntax
|
$relation = substr($fieldName, 0, $pos);
|
||||||
if (strpos($fieldName, '.') !== false) {
|
$fieldName = substr($fieldName, $pos + 1);
|
||||||
$relations = explode('.', $fieldName);
|
$component = $this->relObject($relation);
|
||||||
$fieldName = array_pop($relations);
|
|
||||||
foreach ($relations as $relation) {
|
|
||||||
// Inspect $component for element $relation
|
|
||||||
if ($component->hasMethod($relation)) {
|
|
||||||
// Check nested method
|
|
||||||
$component = $component->$relation();
|
|
||||||
} elseif ($component instanceof SS_List) {
|
|
||||||
// Select adjacent relation from DataList
|
|
||||||
/** @var DataList $component */
|
|
||||||
$component = $component->relation($relation);
|
|
||||||
} elseif ($component instanceof DataObject
|
|
||||||
&& ($dbObject = $component->dbObject($relation))
|
|
||||||
) {
|
|
||||||
// Select db object
|
|
||||||
$component = $dbObject;
|
|
||||||
} else {
|
|
||||||
user_error("$relation is not a relation/field on ".get_class($component), E_USER_ERROR);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bail if the component is null
|
// Bail if the component is null
|
||||||
if (!$component) {
|
if (!$component) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
if ($component->hasMethod($fieldName)) {
|
if (ClassInfo::hasMethod($component, $fieldName)) {
|
||||||
return $component->$fieldName();
|
return $component->$fieldName();
|
||||||
}
|
}
|
||||||
return $component->$fieldName;
|
return $component->$fieldName;
|
||||||
@ -3209,14 +3204,14 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
if (is_int($name)) {
|
if (is_int($name)) {
|
||||||
// Format: array('MyFieldName')
|
// Format: array('MyFieldName')
|
||||||
$rewrite[$identifer] = array();
|
$rewrite[$identifer] = array();
|
||||||
} elseif (is_array($specOrName)) {
|
} elseif (is_array($specOrName) && ($relObject = $this->relObject($identifer))) {
|
||||||
// Format: array('MyFieldName' => array(
|
// Format: array('MyFieldName' => array(
|
||||||
// 'filter => 'ExactMatchFilter',
|
// 'filter => 'ExactMatchFilter',
|
||||||
// 'field' => 'NumericField', // optional
|
// 'field' => 'NumericField', // optional
|
||||||
// 'title' => 'My Title', // optional
|
// 'title' => 'My Title', // optional
|
||||||
// ))
|
// ))
|
||||||
$rewrite[$identifer] = array_merge(
|
$rewrite[$identifer] = array_merge(
|
||||||
array('filter' => $this->relObject($identifer)->config()->get('default_search_filter_class')),
|
array('filter' => $relObject->config()->get('default_search_filter_class')),
|
||||||
(array)$specOrName
|
(array)$specOrName
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
@ -6,6 +6,12 @@ use SilverStripe\ORM\FieldType\DBField;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract representation of a DB relation field, either saved or in memory
|
* Abstract representation of a DB relation field, either saved or in memory
|
||||||
|
*
|
||||||
|
* Below methods will be added in 5.x
|
||||||
|
*
|
||||||
|
* @method Relation relation($relationName)
|
||||||
|
* @method Relation forForeignID($id)
|
||||||
|
* @method string dataClass()
|
||||||
*/
|
*/
|
||||||
interface Relation extends SS_List, Filterable, Sortable, Limitable
|
interface Relation extends SS_List, Filterable, Sortable, Limitable
|
||||||
{
|
{
|
||||||
|
@ -62,7 +62,7 @@ abstract class RelationList extends DataList implements Relation
|
|||||||
try {
|
try {
|
||||||
$query->removeFilterOn($currentFilter);
|
$query->removeFilterOn($currentFilter);
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
/* NOP */
|
/* NOP */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -282,13 +282,27 @@ class UnsavedRelationList extends ArrayList implements Relation
|
|||||||
/**
|
/**
|
||||||
* Returns a copy of this list with the relationship linked to the given foreign ID.
|
* Returns a copy of this list with the relationship linked to the given foreign ID.
|
||||||
* @param int|array $id An ID or an array of IDs.
|
* @param int|array $id An ID or an array of IDs.
|
||||||
* @return static
|
* @return Relation
|
||||||
*/
|
*/
|
||||||
public function forForeignID($id)
|
public function forForeignID($id)
|
||||||
{
|
{
|
||||||
$class = singleton($this->baseClass);
|
$singleton = DataObject::singleton($this->baseClass);
|
||||||
$class->ID = 1;
|
/** @var Relation $relation */
|
||||||
return $class->{$this->relationName}()->forForeignID($id);
|
$relation = $singleton->{$this->relationName}($id);
|
||||||
|
return $relation;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $relationName
|
||||||
|
* @return Relation
|
||||||
|
*/
|
||||||
|
public function relation($relationName)
|
||||||
|
{
|
||||||
|
$ids = $this->column('ID');
|
||||||
|
$singleton = DataObject::singleton($this->dataClass);
|
||||||
|
/** @var Relation $relation */
|
||||||
|
$relation = $singleton->$relationName($ids);
|
||||||
|
return $relation;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -299,7 +313,7 @@ class UnsavedRelationList extends ArrayList implements Relation
|
|||||||
*/
|
*/
|
||||||
public function dbObject($fieldName)
|
public function dbObject($fieldName)
|
||||||
{
|
{
|
||||||
return singleton($this->dataClass)->dbObject($fieldName);
|
return DataObject::singleton($this->dataClass)->dbObject($fieldName);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function extractValue($item, $key)
|
protected function extractValue($item, $key)
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace SilverStripe\ORM\Tests;
|
namespace SilverStripe\ORM\Tests;
|
||||||
|
|
||||||
|
use LogicException;
|
||||||
use SilverStripe\Core\Config\Config;
|
use SilverStripe\Core\Config\Config;
|
||||||
use SilverStripe\Dev\SapphireTest;
|
use SilverStripe\Dev\SapphireTest;
|
||||||
use SilverStripe\i18n\i18n;
|
use SilverStripe\i18n\i18n;
|
||||||
@ -1992,7 +1993,7 @@ class DataObjectTest extends SapphireTest
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \LogicException
|
* @expectedException LogicException
|
||||||
*/
|
*/
|
||||||
public function testInvalidate()
|
public function testInvalidate()
|
||||||
{
|
{
|
||||||
@ -2047,11 +2048,17 @@ class DataObjectTest extends SapphireTest
|
|||||||
|
|
||||||
public function testRelField()
|
public function testRelField()
|
||||||
{
|
{
|
||||||
$captain = $this->objFromFixture(DataObjectTest\Player::class, 'captain1');
|
$captain1 = $this->objFromFixture(DataObjectTest\Player::class, 'captain1');
|
||||||
// Test traversal of a single has_one
|
// Test traversal of a single has_one
|
||||||
$this->assertEquals("Team 1", $captain->relField('FavouriteTeam.Title'));
|
$this->assertEquals("Team 1", $captain1->relField('FavouriteTeam.Title'));
|
||||||
// Test direct field access
|
// Test direct field access
|
||||||
$this->assertEquals("Captain", $captain->relField('FirstName'));
|
$this->assertEquals("Captain", $captain1->relField('FirstName'));
|
||||||
|
|
||||||
|
// Test empty link
|
||||||
|
$captain2 = $this->objFromFixture(DataObjectTest\Player::class, 'captain2');
|
||||||
|
$this->assertEmpty($captain2->relField('FavouriteTeam.Title'));
|
||||||
|
$this->assertNull($captain2->relField('FavouriteTeam.ReturnsNull'));
|
||||||
|
$this->assertNull($captain2->relField('FavouriteTeam.ReturnsNull.Title'));
|
||||||
|
|
||||||
$player = $this->objFromFixture(DataObjectTest\Player::class, 'player2');
|
$player = $this->objFromFixture(DataObjectTest\Player::class, 'player2');
|
||||||
// Test that we can traverse more than once, and that arbitrary methods are okay
|
// Test that we can traverse more than once, and that arbitrary methods are okay
|
||||||
@ -2063,24 +2070,39 @@ class DataObjectTest extends SapphireTest
|
|||||||
// Test that relField works on db field manipulations
|
// Test that relField works on db field manipulations
|
||||||
$comment = $this->objFromFixture(DataObjectTest\TeamComment::class, 'comment3');
|
$comment = $this->objFromFixture(DataObjectTest\TeamComment::class, 'comment3');
|
||||||
$this->assertEquals("PHIL IS A UNIQUE GUY, AND COMMENTS ON TEAM2", $comment->relField('Comment.UpperCase'));
|
$this->assertEquals("PHIL IS A UNIQUE GUY, AND COMMENTS ON TEAM2", $comment->relField('Comment.UpperCase'));
|
||||||
|
|
||||||
|
// relField throws exception on invalid properties
|
||||||
|
$this->expectException(LogicException::class);
|
||||||
|
$this->expectExceptionMessage("Not is not a relation/field on " . DataObjectTest\TeamComment::class);
|
||||||
|
$comment->relField('Not.A.Field');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testRelObject()
|
public function testRelObject()
|
||||||
{
|
{
|
||||||
$captain = $this->objFromFixture(DataObjectTest\Player::class, 'captain1');
|
$captain1 = $this->objFromFixture(DataObjectTest\Player::class, 'captain1');
|
||||||
|
|
||||||
// Test traversal of a single has_one
|
// Test traversal of a single has_one
|
||||||
$this->assertInstanceOf(DBVarchar::class, $captain->relObject('FavouriteTeam.Title'));
|
$this->assertInstanceOf(DBVarchar::class, $captain1->relObject('FavouriteTeam.Title'));
|
||||||
$this->assertEquals("Team 1", $captain->relObject('FavouriteTeam.Title')->getValue());
|
$this->assertEquals("Team 1", $captain1->relObject('FavouriteTeam.Title')->getValue());
|
||||||
|
|
||||||
|
// Test empty link
|
||||||
|
$captain2 = $this->objFromFixture(DataObjectTest\Player::class, 'captain2');
|
||||||
|
$this->assertEmpty($captain2->relObject('FavouriteTeam.Title')->getValue());
|
||||||
|
$this->assertNull($captain2->relObject('FavouriteTeam.ReturnsNull.Title'));
|
||||||
|
|
||||||
// Test direct field access
|
// Test direct field access
|
||||||
$this->assertInstanceOf(DBBoolean::class, $captain->relObject('IsRetired'));
|
$this->assertInstanceOf(DBBoolean::class, $captain1->relObject('IsRetired'));
|
||||||
$this->assertEquals(1, $captain->relObject('IsRetired')->getValue());
|
$this->assertEquals(1, $captain1->relObject('IsRetired')->getValue());
|
||||||
|
|
||||||
$player = $this->objFromFixture(DataObjectTest\Player::class, 'player2');
|
$player = $this->objFromFixture(DataObjectTest\Player::class, 'player2');
|
||||||
// Test that we can traverse more than once, and that arbitrary methods are okay
|
// Test that we can traverse more than once, and that arbitrary methods are okay
|
||||||
$this->assertInstanceOf(DBVarchar::class, $player->relObject('Teams.First.Title'));
|
$this->assertInstanceOf(DBVarchar::class, $player->relObject('Teams.First.Title'));
|
||||||
$this->assertEquals("Team 1", $player->relObject('Teams.First.Title')->getValue());
|
$this->assertEquals("Team 1", $player->relObject('Teams.First.Title')->getValue());
|
||||||
|
|
||||||
|
// relObject throws exception on invalid properties
|
||||||
|
$this->expectException(LogicException::class);
|
||||||
|
$this->expectExceptionMessage("Not is not a relation/field on " . DataObjectTest\Player::class);
|
||||||
|
$player->relObject('Not.A.Field');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testLateStaticBindingStyle()
|
public function testLateStaticBindingStyle()
|
||||||
|
@ -80,4 +80,9 @@ class Team extends DataObject implements TestOnly
|
|||||||
{
|
{
|
||||||
return 'dynamicfield';
|
return 'dynamicfield';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function ReturnsNull()
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user