-
Notifications
You must be signed in to change notification settings - Fork 16
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
NEW Linkfield ownership #101
Conversation
a28643e
to
b8ff21d
Compare
Just had an offline chat with Guy - we could get rid of DataObject extension and just put a |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have reviewed this assuming that the approach is fundamentally solid.
Another way we could approach this is to reconsider how has_many relations work.
- We could add a new
OwnerRelation
DB field. This would go hand-in hand with theOwnerID
andOwnerClass
fields that will already exist from the polymorphic has_one. - When the link field saves the link, it saves the name of the relation that owns it. This way we have a way to identify which has_many or has_one was used to save the given record.
- We could implement this somewhere in framework, and then in
DataObject::getComponents()
if this special new relation type is being used, add a->filter("{$hasOneName}Relation", $hasManyName)
so it only fetches the records for that relation. - If we don't implement any of this in framework, we can add an extension hook to
DataObject::getComponents()
(there's already one ingetManyManyComponents()
so it makes sense to add one here too) and add that filter via the extension hook.
There's probably a bunch of stuff about that approach that seems obvious to me that isn't obvious to anyone else so please do ask clarification questions.
SilverStripe\ORM\DataObject: | ||
extensions: | ||
- SilverStripe\LinkField\Extensions\DataObjectExtension |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let people apply it themselves to any classes where they want to use links (the same way elemental doesn't apply its extension to DataObject
or to SiteTree
by default).
private static array $db = [ | ||
'OwnerID' => 'Int', | ||
'OwnerClassName' => 'Varchar', | ||
]; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
private static array $db = [ | |
'OwnerID' => 'Int', | |
'OwnerClassName' => 'Varchar', | |
]; | |
private static array $jas_one = [ | |
'Owner' => DataObject::class, | |
]; |
This creates the same db fields (but with the correct types, dbforeignkey and dbclassname) and gives us the ability to go $this->Owner()
This should be declared directly on Link
and LinkArea
though. No need for it here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also should probably be Parent
instead of Owner
* This is only intended to be added to Link and LinkArea | ||
* Implemented as an extension rather than base class so it doesn't create an extra base table that needs to be joined | ||
*/ | ||
class LinkObjectExtension extends DataExtension |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Don't make this an extension. If you want to share the can* methods just use a trait which is more performant. There's nothing here that needs to be in an extension.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The private static $db cannot be used as a trait, because it'll clash with the private static $db on Link. You'll get this error:
PHP Fatal error: Link and LinkObjectTrait define the same property ($db) in the composition of Link. However, the definition differs and is considered incompatible.
The Extension class is basically a heavier weight "Silverstripe trait" and will nicely merge private statics
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The private static $db cannot be used as a trait, because it'll clash with the private static $db on Link.
Correct. As suggested in #101 (comment) the has_one
should be declared on the distinct classes instead. Ideally, so will the can* methods but I understand not wanting to repeat them
private function getOwningDataObject(): ?DataObject | ||
{ | ||
// Thismethod is not called getOwner() because of existing Extension::getOwner() method | ||
// | ||
// If this is a Link, and LinkArea is set on it return that | ||
if (is_a($this->owner, Link::class, true)) { | ||
$linkArea = $this->owner->LinkArea(); | ||
if ($linkArea && $linkArea->exists()) { | ||
return $linkArea; | ||
} | ||
} | ||
// Otherwise look for the ownerID + ownerClassName | ||
// These are set in DataObjectExtension::onAfterWrite() | ||
$ownerID = $this->owner->OwnerID; | ||
$ownerClassName = $this->owner->OwnerClassName; | ||
if ($ownerID === 0 || $ownerClassName === '') { | ||
return null; | ||
} | ||
return $ownerClassName::get()->byID($ownerID); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This won't be needed on LinkArea
at all because of the has_one.
On Link
, the Owner
relation will be the LinkArea
, so this method will still be useful there. But it should be like this:
public function Owner()
{
$owner = $this->getComponent('Owner');
// If $owner is the LinkArea, return the underlying record that owns the area.
if ($owner instanceof LinkArea) {
$owner = $owner->Owner();
}
return $owner;
}
Unfortunately we can't call this getOwner()
because the magic that turns relations into methods doesn't check for getters.
With this method, we can rely on $this->Owner()
in Link
always being whatever ultimately owns the link - the LinkArea
record is completely transparent.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This won't be needed on LinkArea at all because of the has_one.
I'm not sure? The has_one is on an artibary DataObject (usually Page), so the LinkArea needs this as a way to find it's Owner/Parent?
On Link, the Owner relation will be the LinkArea
Not if it's not using a LinkArea, and instead it's a has_one on an arbitary DataObject (usually Page).
General idea for this PR was that it allows Links to be used as has_one's on Page's, or as has_many's via LinkArea's
if ($extended !== null) { | ||
return $extended; | ||
} | ||
$owner = $this->getOwningDataObject(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
$owner = $this->getOwningDataObject(); | |
$owner = $this->Owner(); |
]; | ||
|
||
private static array $extensions = [ | ||
Versioned::class, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why versioned?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Linked issue says "versioned operations" - you need to make Link Versioned to do versioned operations :-)
|
||
private static array $extensions = [ | ||
Versioned::class, | ||
LinkObjectExtension::class, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LinkObjectExtension::class, |
Either add all the can* methods in here or apply them as a trait.
]; | ||
|
||
private static array $extensions = [ | ||
Versioned::class, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why versioned?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Replied above on Link
// skip for the has_one:LinkArea relation on Link | ||
if (is_a($this->owner, Link::class) && $relation === 'LinkArea') { | ||
continue; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// skip for the has_one:LinkArea relation on Link | |
if (is_a($this->owner, Link::class) && $relation === 'LinkArea') { | |
continue; | |
} |
This extension shouldn't be applied directly to Link
or to DataObject
, so we don't need this - unless we expect a link to ever have a child link? But I'd say we don't want to support nesting links.
if ($relationObj->OwnerID !== $this->owner->ID) { | ||
$relationObj->OwnerID = $this->owner->ID; | ||
$doWrite = true; | ||
} | ||
if ($relationObj->OwnerClassName !== $this->owner->ClassName) { | ||
$relationObj->OwnerClassName = $this->owner->ClassName; | ||
$doWrite = true; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't like that we're essentially storing the has_one in two places in the DB here. We're really working around a fundamental limitation in framework.
I would prefer that we either:
- Explicitly document that people have to use
belongs_to
for their links and link areas, OR - Implement a way in framework to say "these has_one relations are the same relation, so only store it once!" effectively treating one of them as a belongs_to, OR
- Make framework store has_many in its own table like many_many, which would allow using a single belongs_to in
Link
to an arbitrary number of has_many or has_one relations on other classes (which means we wouldn't needLinkArea
either) (would be a new major of framework, or a new relation type, or a new behaviour that only happens when framework identifies that the has_many is trying to link up to a belongs_to))
All of the above do away with needing this extension altogether.
Have chatted offline with Guy re his ideas around adding a "third has-many column" that is essentially a replacement for dot-notation. If that works it would remove the need for the intermediary LinkArea table, which would be great performance wise. So he'll explore that further and if that works we'll ditch this approach |
Closing in favor of #102 |
Issue #5
Also implements #12
In peer review because looking for initial feedback - have I overlooked something really obvious?
This is a draft PR that's just testing out the idea of using a LinkArea to manage the has_many relationships similar to ElementalArea in elemental. Advantage of using the extra layer is that it allows you to effectively add multiple "has_many" relations of Link::class and have them correctly assigned to each of the has_many relations.
Documentation would instruct you to add you Links and LinkArea's to $has_one + $owns + $cascade_deletes + $cascase_duplicates
A Page in a project would look something like this if it had two Link's and two LinkArea's
Adding multiple has_many's of the same type normally isn't possible because you'll have something like this:
And you'll get $myDataObject->PageID = 123 - but there's no way to know if it should belong to the 'MyThings' or the 'MoreThings' relationship on Page
The official way to solve "multiple has_manys of the same type" this is with dot notation where the the corresponding has_one relationship is suffixed to the classname of has_many relationship. In this instance it would look something like this (excuse my use of RelOne and RelTwo, I couldn't think of proper naming here):
However in the linkfield context we cannot use dot-notation, because "MyDataObject" would be "Link" and we can't set the $has_one's on Link since it has no knowledge of project code. We'd need to ask people to add the $has_ones to Link via extensions which is very clunky.
Note there's somethings to do with unit testing that needs to be extracted from #90 before closing that PR