mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-06-07 18:44:08 +00:00
317 lines
10 KiB
PHP
317 lines
10 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
/*
|
|
* Chill is a software for social workers
|
|
*
|
|
* For the full copyright and license information, please view
|
|
* the LICENSE file that was distributed with this source code.
|
|
*/
|
|
|
|
namespace Chill\PersonBundle\Controller;
|
|
|
|
use Chill\ActivityBundle\Entity\Activity;
|
|
use Chill\DocStoreBundle\Entity\PersonDocument;
|
|
use Chill\EventBundle\Entity\Participation;
|
|
use Chill\PersonBundle\Actions\Remove\PersonMove;
|
|
use Chill\PersonBundle\Entity\Person;
|
|
use Chill\PersonBundle\Entity\PersonNotDuplicate;
|
|
use Chill\PersonBundle\Form\PersonConfimDuplicateType;
|
|
use Chill\PersonBundle\Form\PersonFindManuallyDuplicateType;
|
|
use Chill\PersonBundle\Privacy\PrivacyEvent;
|
|
use Chill\PersonBundle\Repository\PersonNotDuplicateRepository;
|
|
use Chill\PersonBundle\Repository\PersonRepository;
|
|
use Chill\PersonBundle\Search\SimilarPersonMatcher;
|
|
use Chill\TaskBundle\Entity\SingleTask;
|
|
use http\Exception\InvalidArgumentException;
|
|
use Symfony\Bundle\FrameworkBundle\Controller\Controller;
|
|
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
|
|
use Symfony\Component\HttpFoundation\Request;
|
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
|
|
|
use function count;
|
|
|
|
class PersonDuplicateController extends Controller
|
|
{
|
|
/**
|
|
* @var \Symfony\Component\EventDispatcher\EventDispatcherInterface
|
|
*/
|
|
private $eventDispatcher;
|
|
|
|
/**
|
|
* @var \Chill\PersonBundle\Actions\Remove\PersonMove
|
|
*/
|
|
private $personMove;
|
|
|
|
/**
|
|
* @var \Chill\PersonBundle\Repository\PersonRepository
|
|
*/
|
|
private $personRepository;
|
|
|
|
/**
|
|
* @var \Chill\PersonBundle\Search\SimilarPersonMatcher
|
|
*/
|
|
private $similarPersonMatcher;
|
|
|
|
/**
|
|
* @var \Symfony\Component\Translation\TranslatorInterface
|
|
*/
|
|
private $translator;
|
|
|
|
public function __construct(
|
|
SimilarPersonMatcher $similarPersonMatcher,
|
|
TranslatorInterface $translator,
|
|
PersonRepository $personRepository,
|
|
PersonMove $personMove,
|
|
EventDispatcherInterface $eventDispatcher
|
|
) {
|
|
$this->similarPersonMatcher = $similarPersonMatcher;
|
|
$this->translator = $translator;
|
|
$this->personRepository = $personRepository;
|
|
$this->personMove = $personMove;
|
|
$this->eventDispatcher = $eventDispatcher;
|
|
}
|
|
|
|
public function confirmAction($person1_id, $person2_id, Request $request)
|
|
{
|
|
if ($person1_id === $person2_id) {
|
|
throw new InvalidArgumentException('Can not merge same person');
|
|
}
|
|
|
|
$person1 = $this->_getPerson($person1_id);
|
|
$person2 = $this->_getPerson($person2_id);
|
|
|
|
$person1->counters = $this->_getCounters($person1_id);
|
|
$person2->counters = $this->_getCounters($person2_id);
|
|
|
|
if (null === $person1) {
|
|
throw $this->createNotFoundException("Person with id {$person1_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
$this->denyAccessUnlessGranted(
|
|
'CHILL_PERSON_DUPLICATE',
|
|
$person1,
|
|
'You are not allowed to see this person.'
|
|
);
|
|
|
|
if (null === $person2) {
|
|
throw $this->createNotFoundException("Person with id {$person2_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
$form = $this->createForm(PersonConfimDuplicateType::class);
|
|
|
|
$form->handleRequest($request);
|
|
|
|
if ($form->isSubmitted() && $form->isValid()) {
|
|
$event = new PrivacyEvent($person1, [
|
|
'element_class' => Person::class,
|
|
'action' => 'move',
|
|
]);
|
|
$event->addPerson($person2);
|
|
$this->eventDispatcher->dispatch(PrivacyEvent::PERSON_PRIVACY_EVENT, $event);
|
|
|
|
$sqls = $this->personMove->getSQL($person2, $person1);
|
|
|
|
$connection = $this->getDoctrine()->getConnection();
|
|
|
|
$connection->beginTransaction();
|
|
|
|
foreach ($sqls as $sql) {
|
|
$connection->executeQuery($sql);
|
|
}
|
|
$connection->commit();
|
|
|
|
return $this->redirectToRoute('chill_person_duplicate_view', ['person_id' => $person1->getId()]);
|
|
}
|
|
|
|
return $this->render('ChillPersonBundle:PersonDuplicate:confirm.html.twig', [
|
|
'person' => $person1,
|
|
'person2' => $person2,
|
|
'form' => $form->createView(),
|
|
]);
|
|
}
|
|
|
|
public function findManuallyDuplicateAction($person_id, Request $request)
|
|
{
|
|
$person = $this->_getPerson($person_id);
|
|
|
|
if (null === $person) {
|
|
throw $this->createNotFoundException("Person with id {$person_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
$this->denyAccessUnlessGranted(
|
|
'CHILL_PERSON_DUPLICATE',
|
|
$person,
|
|
'You are not allowed to see this person.'
|
|
);
|
|
|
|
$form = $this->createForm(PersonFindManuallyDuplicateType::class);
|
|
|
|
$form->handleRequest($request);
|
|
|
|
if ($form->isSubmitted() && $form->isValid()) {
|
|
$person2 = $form->get('person')->getData();
|
|
|
|
if (null === $person2) {
|
|
throw $this->createNotFoundException("Person with id {$person2->getId}() not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
$direction = $form->get('direction')->getData();
|
|
|
|
if ('starting' === $direction) {
|
|
$params = [
|
|
'person1_id' => $person->getId(),
|
|
'person2_id' => $person2->getId(),
|
|
];
|
|
} else {
|
|
$params = [
|
|
'person1_id' => $person2->getId(),
|
|
'person2_id' => $person->getId(),
|
|
];
|
|
}
|
|
|
|
return $this->redirectToRoute('chill_person_duplicate_confirm', $params);
|
|
}
|
|
|
|
return $this->render('ChillPersonBundle:PersonDuplicate:find_manually.html.twig', [
|
|
'person' => $person,
|
|
'form' => $form->createView(),
|
|
]);
|
|
}
|
|
|
|
public function notDuplicateAction($person1_id, $person2_id)
|
|
{
|
|
[$person1, $person2] = $this->_getPersonsByPriority($person1_id, $person2_id);
|
|
|
|
$this->denyAccessUnlessGranted(
|
|
'CHILL_PERSON_DUPLICATE',
|
|
$person1,
|
|
'You are not allowed to see this person.'
|
|
);
|
|
|
|
$personNotDuplicate = $this->getDoctrine()->getRepository(PersonNotDuplicate::class)
|
|
->findOneBy(['person1' => $person1, 'person2' => $person2]);
|
|
|
|
if (!$personNotDuplicate instanceof PersonNotDuplicate) {
|
|
$personNotDuplicate = new PersonNotDuplicate();
|
|
$personNotDuplicate->setPerson1($person1);
|
|
$personNotDuplicate->setPerson2($person2);
|
|
$personNotDuplicate->setUser($this->getUser());
|
|
|
|
$this->getDoctrine()->getManager()->persist($personNotDuplicate);
|
|
$this->getDoctrine()->getManager()->flush();
|
|
}
|
|
|
|
return $this->redirectToRoute('chill_person_duplicate_view', ['person_id' => $person1->getId()]);
|
|
}
|
|
|
|
public function removeNotDuplicateAction($person1_id, $person2_id)
|
|
{
|
|
[$person1, $person2] = $this->_getPersonsByPriority($person1_id, $person2_id);
|
|
|
|
$this->denyAccessUnlessGranted(
|
|
'CHILL_PERSON_DUPLICATE',
|
|
$person1,
|
|
'You are not allowed to see this person.'
|
|
);
|
|
|
|
$personNotDuplicate = $this->getDoctrine()->getRepository(PersonNotDuplicate::class)
|
|
->findOneBy(['person1' => $person1, 'person2' => $person2]);
|
|
|
|
if ($personNotDuplicate instanceof PersonNotDuplicate) {
|
|
$this->getDoctrine()->getManager()->remove($personNotDuplicate);
|
|
$this->getDoctrine()->getManager()->flush();
|
|
}
|
|
|
|
return $this->redirectToRoute('chill_person_duplicate_view', ['person_id' => $person1->getId()]);
|
|
}
|
|
|
|
public function viewAction($person_id, PersonNotDuplicateRepository $personNotDuplicateRepository)
|
|
{
|
|
$person = $this->_getPerson($person_id);
|
|
|
|
if (null === $person) {
|
|
throw $this->createNotFoundException("Person with id {$person_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
$this->denyAccessUnlessGranted(
|
|
'CHILL_PERSON_DUPLICATE',
|
|
$person,
|
|
'You are not allowed to see this person.'
|
|
);
|
|
|
|
$duplicatePersons = $this->similarPersonMatcher->
|
|
matchPerson($person, 0.5, SimilarPersonMatcher::SIMILAR_SEARCH_ORDER_BY_ALPHABETICAL, false);
|
|
|
|
$notDuplicatePersons = $personNotDuplicateRepository->findNotDuplicatePerson($person);
|
|
|
|
return $this->render('ChillPersonBundle:PersonDuplicate:view.html.twig', [
|
|
'person' => $person,
|
|
'duplicatePersons' => $duplicatePersons,
|
|
'notDuplicatePersons' => $notDuplicatePersons,
|
|
]);
|
|
}
|
|
|
|
private function _getCounters($id): ?array
|
|
{
|
|
$em = $this->getDoctrine()->getManager();
|
|
|
|
$nb_activity = $em->getRepository(Activity::class)->findBy(['person' => $id]);
|
|
$nb_document = $em->getRepository(PersonDocument::class)->findBy(['person' => $id]);
|
|
// $nb_event = $em->getRepository(Participation::class)->findBy(['person' => $id]);
|
|
$nb_task = $em->getRepository(SingleTask::class)->countByParameters(['person' => $id]);
|
|
$person = $em->getRepository(Person::class)->findOneBy(['id' => $id]);
|
|
|
|
return [
|
|
'nb_activity' => count($nb_activity),
|
|
'nb_document' => count($nb_document),
|
|
// 'nb_event' => count($nb_event),
|
|
'nb_task' => $nb_task,
|
|
'nb_addresses' => count($person->getAddresses()),
|
|
];
|
|
}
|
|
|
|
/**
|
|
* easy getting a person by his id.
|
|
*
|
|
* @param mixed $id
|
|
*/
|
|
private function _getPerson($id): ?Person
|
|
{
|
|
return $this->personRepository->find($id);
|
|
}
|
|
|
|
private function _getPersonsByPriority($person1_id, $person2_id)
|
|
{
|
|
if ($person1_id === $person2_id) {
|
|
throw new InvalidArgumentException('Can not merge same person');
|
|
}
|
|
|
|
if ($person1_id > $person2_id) {
|
|
$person1 = $this->_getPerson($person2_id);
|
|
$person2 = $this->_getPerson($person1_id);
|
|
} else {
|
|
$person1 = $this->_getPerson($person1_id);
|
|
$person2 = $this->_getPerson($person2_id);
|
|
}
|
|
|
|
if (null === $person1) {
|
|
throw $this->createNotFoundException("Person with id {$person1_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
if (null === $person2) {
|
|
throw $this->createNotFoundException("Person with id {$person2_id} not"
|
|
. ' found on this server');
|
|
}
|
|
|
|
return [$person1, $person2];
|
|
}
|
|
}
|