У меня есть две сущности - пользователи и вызовы.Пользователь может участвовать во многих задачах, и задача может иметь много участников (пользователей).Я начал подходить к этой проблеме, создав отношение «многие ко многим» в своем классе «Пользователи»:
/**
* @ORM\ManytoMany(targetEntity="Challenge")
* @ORM\JoinTable(name="users_challenges",joinColumns={@ORM\JoinColumn(name="user_id",referencedColumnName="id")},
* inverseJoinColumns={@ORM\JoinColumn(name="challenge_id",referencedColumnName="id")})
*
*/
protected $challenges;
Однако затем я понял, что мне нужно сохранить атрибут расстояния для комбинации «пользователь / вызов» (как далекопользователь путешествовал в их вызове).Документ Doctrine2 гласит:
«Почему ассоциации« многие ко многим »менее распространены? Потому что часто вы хотите связать дополнительные атрибуты с ассоциацией, и в этом случае вы вводите класс ассоциации.связь «многие ко» исчезает и заменяется ассоциациями «один ко многим» или «многие к одному» между 3 участвующими классами. "
Итак, мой вопрос, какими должны быть эти ассоциации между User, Challenge и UsersChallenges?
ОБНОВЛЕНИЕ
См. Комментарий к первому ответу для ссылок на код сущности.У меня есть метод контроллера ниже, который всегда создает новую запись UsersChallenges, а не обновляет существующую (что я хочу)
public function updateUserDistanceAction()
{
$request = $this->getRequest();
$distance = $request->get('distance');
$challenge_id = $request->get('challenge_id');
if($request->isXmlHttpRequest()) {
$em = $this->getDoctrine()->getEntityManager();
$user = $this->get('security.context')->getToken()->getUser();
$existingChallenges = $user->getChallenges();
$challengeToUpdate = $em->getRepository('GymloopCoreBundle:Challenge')
->find( (int) $challenge_id);
if(!$challengeToUpdate) {
throw $this->createNotFoundException('No challenge found');
}
//does the challengeToUpdate exist in existingChallenges? If yes, update UsersChallenges with the distance
//if not, create a new USersChallenges object, set distance and flush
if ( !$existingChallenges->isEmpty() && $existingChallenges->contains($challengeToUpdate)) {
$userChallenge = $em->getRepository('GymloopCoreBundle:UsersChallenges')
->findOneByChallengeId($challengeToUpdate->getId());
$userChallenge->setDistance( $userChallenge->getDistance() + (int) $distance );
$em->flush();
} else {
$newUserChallenge = new UsersChallenges();
$newUserChallenge->setDistance($distance);
$newUserChallenge->setChallenge($challengeToUpdate);
$newUserChallenge->setUser($user);
$user->addUsersChallenges($newUserChallenge);
$em->persist($user);
$em->persist($newUserChallenge);
$em->flush();
}
//if success
return new Response('success');
//else
}
}