Joining-Table with Metadata Impairs Getters/Setters - Doctrine 2

我的梦境 提交于 2019-12-25 09:00:24

问题


I'm writing a feature which calls for the records of my joining table to carry extra metadata (Joining-Table with Metadata). I've attempted to implement this in accordance with this section of the Doctrine documentation.

See below for example Entity definitions.

The challenge now is that getGroups and setGroups do not yield/set Group entities (& the same is true from the Group instance perspective), but they yield GroupUser entities.

This adds a substantial delay to process of managing this relationships, which so far have been extremely smooth - for example, I cannot simply add, remove, or check for the existence of a Group to the collection which getGroups yields.

Can anyone identity any errors in my implementation, or else recommend a more fluid way of implementing this concept?

Thanks in advance for any input.

EDIT:

My main concern is this: using this implementation, retrieving a collection of Users from a Group entity requires this Entity method's mediation:

public function getUsers() {
    return $this->users->map(function($groupUser){
        return $groupUser->getUser();
    });
}

I'm concerned that this could imply a major performance hit down the road. Am I incorrect?

Furthermore, how does one re-implement the setUsers method?

Group entity:

<?php

/**
 * @Entity
 * @Table(name="group")
 */

class Group {
    /**
     * @Column(type="integer", nullable=false)
     * @Id
     */
    protected $id = null;

    /**
     * @OneToMany(targetEntity="GroupUser", mappedBy="group")
     * @var \Doctrine\Common\Collections\Collection
     */
    protected $users;
}

User entity:

<?php

/**
 * @Entity
 * @Table(name="user")
 */
class User {
    /**
     * @Column(type="integer", nullable=false)
     * @Id
     */
    protected $id = null;

    /**
     * @OneToMany(targetEntity="GroupUser", mappedBy="user")
     * @var \Doctrine\Common\Collections\Collection
     */
    protected $groups;
}

Joining entity:

<?php
/**
 * @Entity
 * @Table(name="group_user") 
 */
class GroupUser {

    /**
     * @Id
     * @ManyToOne(targetEntity="User", inversedBy="groups")
     * @JoinColumn(name="userId", referencedColumnName="id")
     */
    protected $user;

    /**
     * @Id
     * @ManyToOne(targetEntity="Group", inversedBy="users")
     * @JoinColumn(name="groupId", referencedColumnName="id")
     */
    protected $group;

    /**
     * @Column(type="integer")
     */
    protected $relationship;
}

Related -

  • Same goal, slightly different approach, which consistently produced errors once the resulting collections were manipulated: http://www.doctrine-project.org/jira/browse/DDC-1323
  • Supports the approach, no technical details: Doctrine 2 join table + extra fields

回答1:


I've found just two examples (see question) of entity definitions for this specific type of relationship, however no example code for how they're used. As such it was fairly unclear how fluid (or otherwise) the resulting setters & getters could be expected to be. Hopefully this code will help clear up the approach for anyone else making a similar attempt.

The ideal solution under the circumstances (thanks #doctrine @ freenode) was to implement a custom repository - a more flexible & efficient place for creating & managing the association.

Example Custom Repository for Join-Table with Metadata Class - Solution accompanies code in original question

<?php

use Doctrine\ORM\EntityRepository;

class GroupUserRepository extends EntityRepository {
    /**
     * @param \User $user
     * @param \Group $group
     * @param integer $type One of the integer class constants defined by GroupUser
     * @param string $role Optional string defining user's role in the group.
     * @return \GroupUser
     */
    public function addUserToGroup(User $user, Group $group, $relationship, $role = '') {
        $groupUser = $this->findOneBy(array('user' => $user->getId(), 'group' => $group->getId()));
        if(!$groupUser) {
            $groupUser = new GroupUser();
            $groupUser->setGroup($group);
            $groupUser->setUser($user);
            $groupUser->setRole($role);
            $groupUser->setRelationship($relationship);
            $this->_em->persist($groupUser);
        }
        return $groupUser;
    }

    /**
     * @param \User $user
     * @param \Group $group
     * @return null
     */
    public function removeUserFromGroup(User $user, Group $group) {
        $groupUser = $this->findOneBy(array('user' => $user->getId(), 'group' => $group->getId()));
        if($groupUser)
            $this->_em->remove($groupUser);
    }
}

Then, from the join-table class, modify the Entity meta-data accordingly to specify the custom repository.

<?php
/**
 * @Entity(repositoryClass="\Path\To\GroupUserRepository")
 */
class GroupUser {
    // ...   
}

This causes the custom repository to yield in place of the default one, making a proxy method from the Entity class simple.

<?php
/**
 * @Entity
 */
class Group {
    /**
     * @param \User $user
     * @param integer $relationship One of the integer class constants defined by GroupUser
     * @param string $role Optional string defining user's role in the group.
     * @return \GroupUser
     */
    public function addUser(User $user, $relationship, $role = '') {
        return $this->_em->getRepository('GroupUser')
                ->addUserToGroup($user, $this, $relationship, $role);
    }
}

And things are about as manageable as they were before.



来源:https://stackoverflow.com/questions/9204873/joining-table-with-metadata-impairs-getters-setters-doctrine-2

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!