From 0149457fba010e31bc5c7205b5f7262d28b4cfcb Mon Sep 17 00:00:00 2001 From: Mathieu Jaumotte Date: Sun, 21 Mar 2021 14:06:37 +0100 Subject: [PATCH] Duplicate module Signed-off-by: Mathieu Jaumotte --- .../Actions/Remove/PersonMove.php | 1 - .../Controller/PersonDuplicateController.php | 161 +++++++++++------- .../Entity/PersonNotDuplicate.php | 2 +- .../Form/PersonConfimDuplicateType.php | 2 +- .../PersonNotDuplicateRepository.php | 41 +++++ .../views/PersonDuplicate/confirm.html.twig | 4 +- .../views/PersonDuplicate/view.html.twig | 141 ++++++++++----- .../Search/SimilarPersonMatcher.php | 35 +++- .../ChillPersonBundle/config/routes.yaml | 8 +- .../config/services/controller.yaml | 2 + .../translations/messages.en.yml | 2 + .../translations/messages.fr.yml | 4 + 12 files changed, 292 insertions(+), 111 deletions(-) create mode 100644 src/Bundle/ChillPersonBundle/Repository/PersonNotDuplicateRepository.php diff --git a/src/Bundle/ChillPersonBundle/Actions/Remove/PersonMove.php b/src/Bundle/ChillPersonBundle/Actions/Remove/PersonMove.php index fb7e6b9b4..bb5d9bdac 100644 --- a/src/Bundle/ChillPersonBundle/Actions/Remove/PersonMove.php +++ b/src/Bundle/ChillPersonBundle/Actions/Remove/PersonMove.php @@ -94,7 +94,6 @@ class PersonMove foreach ($metadata->getAssociationMappings() as $field => $mapping) { if ($mapping['targetEntity'] === Person::class) { - if (\in_array($metadata->getName(), $toDelete)) { $sql = $this->createDeleteSQL($metadata, $from, $field); $event = new ActionEvent($from->getId(), $metadata->getName(), $sql, diff --git a/src/Bundle/ChillPersonBundle/Controller/PersonDuplicateController.php b/src/Bundle/ChillPersonBundle/Controller/PersonDuplicateController.php index 4438bde8d..a7b9a6523 100644 --- a/src/Bundle/ChillPersonBundle/Controller/PersonDuplicateController.php +++ b/src/Bundle/ChillPersonBundle/Controller/PersonDuplicateController.php @@ -2,13 +2,17 @@ namespace Chill\PersonBundle\Controller; +use Chill\PersonBundle\Actions\Remove\PersonMove; +use Chill\PersonBundle\Entity\AccompanyingPeriod; use Chill\PersonBundle\Entity\Person; use Chill\PersonBundle\Entity\PersonNotDuplicate; use Chill\PersonBundle\Form\PersonConfimDuplicateType; +use Chill\PersonBundle\Privacy\PrivacyEvent; use Chill\PersonBundle\Repository\PersonRepository; use Chill\PersonBundle\Search\SimilarPersonMatcher; use http\Exception\InvalidArgumentException; use Symfony\Bundle\FrameworkBundle\Controller\Controller; +use Symfony\Component\EventDispatcher\EventDispatcherInterface; use Symfony\Component\HttpFoundation\Request; use Symfony\Component\Translation\TranslatorInterface; @@ -29,14 +33,28 @@ class PersonDuplicateController extends Controller */ private $personRepository; + /** + * @var \Chill\PersonBundle\Actions\Remove\PersonMove + */ + private $personMove; + + /** + * @var \Symfony\Component\EventDispatcher\EventDispatcherInterface + */ + private $eventDispatcher; + public function __construct( SimilarPersonMatcher $similarPersonMatcher, TranslatorInterface $translator, - PersonRepository $personRepository + PersonRepository $personRepository, + PersonMove $personMove, + EventDispatcherInterface $eventDispatcher ) { $this->similarPersonMatcher = $similarPersonMatcher; $this->translator = $translator; $this->personRepository = $personRepository; + $this->personMove = $personMove; + $this->eventDispatcher = $eventDispatcher; } public function viewAction($person_id) @@ -47,88 +65,88 @@ class PersonDuplicateController extends Controller . " found on this server"); } - $duplicatePersons = $this->similarPersonMatcher->matchPerson($person, 0.5); + $duplicatePersons = $this->similarPersonMatcher-> + matchPerson($person, 0.5, SimilarPersonMatcher::SIMILAR_SEARCH_ORDER_BY_ALPHABETICAL); + + $notDuplicatePersons = $this->getDoctrine()->getRepository(PersonNotDuplicate::class) + ->findNotDuplicatePerson($person); return $this->render('ChillPersonBundle:PersonDuplicate:view.html.twig', [ - "person" => $person, - 'duplicatePersons' => $duplicatePersons + 'person' => $person, + 'duplicatePersons' => $duplicatePersons, + 'notDuplicatePersons' => $notDuplicatePersons, ]); } - public function confirmAction($person_id, $person2_id, Request $request) + public function confirmAction($person1_id, $person2_id, Request $request) { - if ($person_id === $person2_id) { - throw new InvalidArgumentException('Can not merge same person'); - } - - if ($person_id > $person2_id) { - $tmpId = $person2_id; - $person2_id = $person_id; - $person_id = $tmpId; - unset($tmpId); - } - - $person = $this->_getPerson($person_id); - if ($person === null) { - throw $this->createNotFoundException("Person with id $person_id not" - . " found on this server"); - } - - $person2 = $this->_getPerson($person2_id); - if ($person2 === null) { - throw $this->createNotFoundException("Person with id $person2_id not" - . " found on this server"); - } + [$person1, $person2] = $this->_getPersonsByPriority($person1_id, $person2_id); $form = $this->createForm(PersonConfimDuplicateType::class); $form->handleRequest($request); if ($form->isSubmitted() && $form->isValid()) { - dd('todo'); + $event = new PrivacyEvent($person1, array( + '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' => $person, + 'person' => $person1, 'person2' => $person2, 'form' => $form->createView(), ]); } - public function notDuplicateAction($person_id, $person2_id) + public function notDuplicateAction($person1_id, $person2_id) { - if ($person_id === $person2_id) { - throw new InvalidArgumentException('Can not merge same person'); + [$person1, $person2] = $this->_getPersonsByPriority($person1_id, $person2_id); + + $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(); } - if ($person_id > $person2_id) { - $tmpId = $person2_id; - $person2_id = $person_id; - $person_id = $tmpId; - unset($tmpId); + 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); + + $personNotDuplicate = $this->getDoctrine()->getRepository(PersonNotDuplicate::class) + ->findOneBy(['person1' => $person1, 'person2' => $person2]); + + if ($personNotDuplicate instanceof PersonNotDuplicate) { + $this->getDoctrine()->getManager()->remove($personNotDuplicate); + $this->getDoctrine()->getManager()->flush(); } - $person = $this->_getPerson($person_id); - if ($person === null) { - throw $this->createNotFoundException("Person with id $person_id not" - . " found on this server"); - } - - $person2 = $this->_getPerson($person2_id); - if ($person2 === null) { - throw $this->createNotFoundException("Person with id $person2_id not" - . " found on this server"); - } - - $personNotDuplicate = new PersonNotDuplicate(); - $personNotDuplicate->setPerson1($person); - $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' => $person->getId()]); + return $this->redirectToRoute('chill_person_duplicate_view', ['person_id' => $person1->getId()]); } /** @@ -138,4 +156,31 @@ class PersonDuplicateController extends Controller { 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 ($person1 === null) { + throw $this->createNotFoundException("Person with id $person1_id not" + . " found on this server"); + } + + if ($person2 === null) { + throw $this->createNotFoundException("Person with id $person2_id not" + . " found on this server"); + } + + return [$person1, $person2]; + } } diff --git a/src/Bundle/ChillPersonBundle/Entity/PersonNotDuplicate.php b/src/Bundle/ChillPersonBundle/Entity/PersonNotDuplicate.php index e8b79698f..ce2a1c26f 100644 --- a/src/Bundle/ChillPersonBundle/Entity/PersonNotDuplicate.php +++ b/src/Bundle/ChillPersonBundle/Entity/PersonNotDuplicate.php @@ -9,7 +9,7 @@ use Chill\MainBundle\Entity\User; * PersonNotDuplicate * * @ORM\Table(name="chill_person_not_duplicate") - * @ORM\Entity() + * @ORM\Entity(repositoryClass="Chill\PersonBundle\Repository\PersonNotDuplicateRepository") */ class PersonNotDuplicate { diff --git a/src/Bundle/ChillPersonBundle/Form/PersonConfimDuplicateType.php b/src/Bundle/ChillPersonBundle/Form/PersonConfimDuplicateType.php index acc385f43..fd7fa3e50 100644 --- a/src/Bundle/ChillPersonBundle/Form/PersonConfimDuplicateType.php +++ b/src/Bundle/ChillPersonBundle/Form/PersonConfimDuplicateType.php @@ -16,7 +16,7 @@ class PersonConfimDuplicateType extends AbstractType { $builder ->add('confirm', CheckboxType::class, [ - 'label' => 'Je confirme la fusion de ces 2 personnes', + 'label' => 'I confirm the merger of these 2 people', 'mapped' => false, ]); } diff --git a/src/Bundle/ChillPersonBundle/Repository/PersonNotDuplicateRepository.php b/src/Bundle/ChillPersonBundle/Repository/PersonNotDuplicateRepository.php new file mode 100644 index 000000000..8ab711b15 --- /dev/null +++ b/src/Bundle/ChillPersonBundle/Repository/PersonNotDuplicateRepository.php @@ -0,0 +1,41 @@ +createQueryBuilder('pnd'); + $qb->select('pnd') + ->where('pnd.person1 = :person OR pnd.person2 = :person') + ; + $qb->setParameter('person', $person); + $result = $qb->getQuery()->getResult(); + + $persons = []; + foreach ($result as $row) { + if ($row->getPerson1() === $person) { + $persons[] = $row->getPerson2(); + } elseif ($row->getPerson2() === $person) { + $persons[] = $row->getPerson1(); + } + } + + return $persons; + } +} diff --git a/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/confirm.html.twig b/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/confirm.html.twig index fcd314a62..f6155eead 100644 --- a/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/confirm.html.twig +++ b/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/confirm.html.twig @@ -6,12 +6,12 @@ ' ' ~ person.lastName }}{% endblock %} {% block personcontent %} -

Ancien dossier

+

{{ 'Old person'|trans }}

{{ person2 }} -

Nouveau Dossier

+

{{ 'New person'|trans }}

{{ person }} diff --git a/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/view.html.twig b/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/view.html.twig index 691f55e1c..ba5bd13f6 100644 --- a/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/view.html.twig +++ b/src/Bundle/ChillPersonBundle/Resources/views/PersonDuplicate/view.html.twig @@ -8,9 +8,12 @@ {% block personcontent %} -

{{ title|default('Person duplicate')|trans }}

- + +{% if duplicatePersons|length > 0 %} +

{{ title|default('Person duplicate')|trans }}

+ +
@@ -20,46 +23,104 @@ -{% for duplicatePerson in duplicatePersons %} - - + + + + + + {% endfor %} +{% endif %} + + + +{% if notDuplicatePersons|length > 0 %} +

{{ 'Person flaged as duplicate' | trans }}

+ +
{% trans %}Name{% endtrans %}
- {% set is_open = duplicatePerson.isOpen() %} - - {{ duplicatePerson|chill_entity_render_box }} - {% apply spaceless %} - {% if chill_person.fields.accompanying_period == 'visible' %} - {% if is_open == false %} - - {% else %} - + {% for duplicatePerson in duplicatePersons %} +
+ {% set is_open = duplicatePerson.isOpen() %} + + {{ duplicatePerson|chill_entity_render_box }} + {% apply spaceless %} + {% if chill_person.fields.accompanying_period == 'visible' %} + {% if is_open == false %} + + {% else %} + + {% endif %} {% endif %} + {% endapply %} + + + {% if duplicatePerson.birthdate is not null %} + {{ duplicatePerson.birthdate|format_date('long') }} + {% else %} + {{ 'Unknown date of birth'|trans }} + {% endif %} + + {% if duplicatePerson.nationality is not null %} + {{ duplicatePerson.nationality.name|localize_translatable_string }} + {% else %} + {{ 'Without nationality'|trans }} + {% endif %} + +
    +
  • +
  • +
  • +
+
+ + + + + + + + + + {% for notDuplicatePerson in notDuplicatePersons %} + + + - - - - -{% endfor %} + + + + + {% endfor %} +
{% trans %}Name{% endtrans %}{% trans %}Date of birth{% endtrans %}{% trans %}Nationality{% endtrans %} 
+ {% set is_open = notDuplicatePerson.isOpen() %} + + {{ notDuplicatePerson|chill_entity_render_box }} + {% apply spaceless %} + {% if chill_person.fields.accompanying_period == 'visible' %} + {% if is_open == false %} + + {% else %} + + {% endif %} + {% endif %} + {% endapply %} + + + {% if notDuplicatePerson.birthdate is not null %} + {{ notDuplicatePerson.birthdate|format_date('long') }} + {% else %} + {{ 'Unknown date of birth'|trans }} {% endif %} - {% endapply %} - - - {% if duplicatePerson.birthdate is not null %} - {{ duplicatePerson.birthdate|format_date('long') }} - {% else %} - {{ 'Unknown date of birth'|trans }} - {% endif %} - - {% if duplicatePerson.nationality is not null %} - {{ duplicatePerson.nationality.name|localize_translatable_string }} - {% else %} - {{ 'Without nationality'|trans }} - {% endif %} - -
    -
  • -
  • -
  • -
-
+ {% if notDuplicatePerson.nationality is not null %} + {{ notDuplicatePerson.nationality.name|localize_translatable_string }} + {% else %} + {{ 'Without nationality'|trans }} + {% endif %} + +
    +
  • +
  • +
+
+{% endif %} {% endblock %} diff --git a/src/Bundle/ChillPersonBundle/Search/SimilarPersonMatcher.php b/src/Bundle/ChillPersonBundle/Search/SimilarPersonMatcher.php index 4b22bd656..89de2cd09 100644 --- a/src/Bundle/ChillPersonBundle/Search/SimilarPersonMatcher.php +++ b/src/Bundle/ChillPersonBundle/Search/SimilarPersonMatcher.php @@ -18,6 +18,7 @@ */ namespace Chill\PersonBundle\Search; +use Chill\PersonBundle\Entity\PersonNotDuplicate; use Doctrine\ORM\EntityManagerInterface; use Chill\PersonBundle\Entity\Person; use Chill\MainBundle\Security\Authorization\AuthorizationHelper; @@ -32,6 +33,10 @@ use Chill\PersonBundle\Security\Authorization\PersonVoter; */ class SimilarPersonMatcher { + CONST SIMILAR_SEARCH_ORDER_BY_ALPHABETICAL = 'alphabetical'; + + CONST SIMILAR_SEARCH_ORDER_BY_SIMILARITY = 'similarity'; + /** * * @var EntityManagerInterface @@ -61,32 +66,50 @@ class SimilarPersonMatcher } - public function matchPerson(Person $person, $precision = 0.15) + public function matchPerson(Person $person, $precision = 0.15, $orderBy = self::SIMILAR_SEARCH_ORDER_BY_SIMILARITY) { $centers = $this->authorizationHelper->getReachableCenters( $this->tokenStorage->getToken()->getUser(), new Role(PersonVoter::SEE) ); - + $dql = 'SELECT p from ChillPersonBundle:Person p ' . ' WHERE (' . ' SIMILARITY(p.fullnameCanonical, UNACCENT(LOWER(:fullName))) >= :precision ' - . ' OR SIMILARITY(p.fullnameCanonical, UNACCENT(LOWER(:fullNameInverted))) >= :precision ' . ' ) ' . ' AND p.center IN (:centers)' - . ' AND p.id != :personId' - . ' ORDER BY SIMILARITY(p.fullnameCanonical, UNACCENT(LOWER(:fullName))) DESC ' + . ' AND p.id != :personId ' ; + + $notDuplicatePersons = $this->em->getRepository(PersonNotDuplicate::class) + ->findNotDuplicatePerson($person); + + if (count($notDuplicatePersons)) { + $dql .= ' AND p.id not in (:notDuplicatePersons)'; + } + + switch ($orderBy) { + case self::SIMILAR_SEARCH_ORDER_BY_ALPHABETICAL: + $dql .= ' ORDER BY p.fullnameCanonical ASC '; + break; + case self::SIMILAR_SEARCH_ORDER_BY_SIMILARITY: + default : + $dql .= ' ORDER BY SIMILARITY(p.fullnameCanonical, UNACCENT(LOWER(:fullName))) DESC '; + } + $query = $this->em ->createQuery($dql) ->setParameter('fullName', $person->getFirstName() . ' ' . $person->getLastName()) - ->setParameter('fullNameInverted', $person->getLastName() . ' ' . $person->getFirstName()) ->setParameter('centers', $centers) ->setParameter('personId', $person->getId()) ->setParameter('precision', $precision) ; + if (count($notDuplicatePersons)) { + $query->setParameter('notDuplicatePersons', $notDuplicatePersons); + } + return $query->getResult(); } } diff --git a/src/Bundle/ChillPersonBundle/config/routes.yaml b/src/Bundle/ChillPersonBundle/config/routes.yaml index 9db6ec12f..0af12d76e 100644 --- a/src/Bundle/ChillPersonBundle/config/routes.yaml +++ b/src/Bundle/ChillPersonBundle/config/routes.yaml @@ -112,7 +112,7 @@ chill_person_closingmotive_admin: label: 'person_admin.closing motives' chill_person_duplicate_confirm: - path: /{_locale}/person/{person_id}/duplicate/{person2_id}/confirm + path: /{_locale}/person/{person1_id}/duplicate/{person2_id}/confirm controller: Chill\PersonBundle\Controller\PersonDuplicateController::confirmAction chill_person_maritalstatus_admin: @@ -125,5 +125,9 @@ chill_person_maritalstatus_admin: label: 'person_admin.marital status' chill_person_duplicate_not_duplicate: - path: /{_locale}/person/{person_id}/duplicate/{person2_id}/not-duplicate + path: /{_locale}/person/{person1_id}/duplicate/{person2_id}/not-duplicate controller: Chill\PersonBundle\Controller\PersonDuplicateController::notDuplicateAction + +chill_person_remove_duplicate_not_duplicate: + path: /{_locale}/person/{person1_id}/duplicate/{person2_id}/remove-not-duplicate + controller: Chill\PersonBundle\Controller\PersonDuplicateController::removeNotDuplicateAction diff --git a/src/Bundle/ChillPersonBundle/config/services/controller.yaml b/src/Bundle/ChillPersonBundle/config/services/controller.yaml index a6e446ebf..13713f3fd 100644 --- a/src/Bundle/ChillPersonBundle/config/services/controller.yaml +++ b/src/Bundle/ChillPersonBundle/config/services/controller.yaml @@ -28,4 +28,6 @@ services: $similarPersonMatcher: '@Chill\PersonBundle\Search\SimilarPersonMatcher' $translator: '@Symfony\Component\Translation\TranslatorInterface' $personRepository: '@Chill\PersonBundle\Repository\PersonRepository' + $personMove: '@Chill\PersonBundle\Actions\Remove\PersonMove' + $eventDispatcher: '@Symfony\Component\EventDispatcher\EventDispatcherInterface' tags: ['controller.service_arguments'] diff --git a/src/Bundle/ChillPersonBundle/translations/messages.en.yml b/src/Bundle/ChillPersonBundle/translations/messages.en.yml index cd9212d5d..0c19e8a50 100644 --- a/src/Bundle/ChillPersonBundle/translations/messages.en.yml +++ b/src/Bundle/ChillPersonBundle/translations/messages.en.yml @@ -67,6 +67,8 @@ Reset: 'Remise à zéro' 'Closing motive': 'Motif de clôture' 'Person details': 'Détails de la personne' 'Person duplicate': 'Find duplicate' +Old person: Old person +New person: New person Create an accompanying period: Create an accompanying period 'Create': Create diff --git a/src/Bundle/ChillPersonBundle/translations/messages.fr.yml b/src/Bundle/ChillPersonBundle/translations/messages.fr.yml index 354625f05..b2451c7ae 100644 --- a/src/Bundle/ChillPersonBundle/translations/messages.fr.yml +++ b/src/Bundle/ChillPersonBundle/translations/messages.fr.yml @@ -66,6 +66,10 @@ Married: Marié(e) 'Contact information': 'Informations de contact' 'Administrative information': Administratif File number: Dossier n° +Old person: Ancien dossier +New person: Nouveau dossier +I confirm the merger of these 2 people : Je confime la fusion de ces 2 dossiers +Person flaged as duplicate: Dossiers marqués comme faux-positif # addresses part address_street_address_1: Adresse ligne 1