diff --git a/VERSIONS_WIP/64203 b/VERSIONS_WIP/64203 new file mode 100644 index 0000000000000000000000000000000000000000..6aa97dafcf2fe4c4d304fd1463b2faaebc0bb67b --- /dev/null +++ b/VERSIONS_WIP/64203 @@ -0,0 +1,2 @@ + - ticket #64203 : Administration : ajout d'une interface de gestion des doublons d'utilsateurs. + \ No newline at end of file diff --git a/application/modules/admin/controllers/UsersController.php b/application/modules/admin/controllers/UsersController.php index 72f3753164d161305e965e69a8add42b392a0f95..c9570c46edc08a7f812659989e93216d3f9e4438 100644 --- a/application/modules/admin/controllers/UsersController.php +++ b/application/modules/admin/controllers/UsersController.php @@ -1,6 +1,6 @@ <?php /** - * Copyright (c) 2012, Agence Française Informatique (AFI). All rights reserved. + * Copyright (c) 2012 Agence Française Informatique (AFI). All rights reserved. * * BOKEH is free software; you can redistribute it and/or modify * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by @@ -62,5 +62,88 @@ class Admin_UsersController extends ZendAfi_Controller_Action { $settings = Class_User_Settings::newWith($user); $settings->set($this->_getParam('key'), $this->_getParam('value')); $user->save(); - } + } + + + public function manageDoubleAction() { + $this->view->titre = $this->_('Gestion des doublons d\'abonnés'); + $this->view->double_manager = new Class_User_DbDoubleFinder; + + if(!$this->view->double_manager->hasDouble()) { + $this->_helper->notify($this->_('Aucun doublon d\'abonné dans votre base de données')); + return $this->_redirectToIndex(); + } + } + + + public function manageDoubleManualAction() { + $this->view->titre = $this->_('Gestion manuelle des doublons d\'abonnés'); + $this->view->double_manager = new Class_User_DbDoubleFinder; + + if(!$this->view->double_manager->hasDouble()) { + $this->_helper->notify($this->_('Aucun doublon d\'abonné dans votre base de données')); + return $this->_redirectToIndex(); + } + } + + + public function deleteDoubleAction() { + $total = (int) $this->_getParam('total', 0); + $cursor = (int) $this->_getParam('cursor', 0); + + $db_double_finder = new Class_User_DbDoubleFinder; + $new_cursor = $db_double_finder->dedupeUsers($cursor); + + $continue = $new_cursor + ? true + : false; + + if(!$continue) + $this->_helper->notify($this->_('Le dédoublonnage automatique des doublons est terminé.')); + + $done = $total - $db_double_finder->countDouble(); + + return $this->_helper->progressbar($total, $done, $new_cursor, $continue); + } + + + public function manageDoubleUserAction() { + if(!$id_user = $this->_getParam('id_user', null)) { + $this->_helper->notify($this->_('ID_USER invalide')); + return $this->_redirectToIndex(); + } + + if(!$user = Class_Users::find($id_user)) { + $this->_helper->notify($this->_('Utilisateur invalide')); + return $this->_redirectToIndex(); + } + + $this->view->double_manager = new Class_User_DbDoubleFinder; + $this->view->users = $this->view->double_manager->findDoubleFor($user); + $this->view->titre = $this->_('Gestion manuelle des doublon pour l\'ID ABON : %s', $user->getIdabon()); + $this->view->id_user = $id_user; + $this->view->next = $this->view->double_manager->getNextDouble($id_user); + $this->view->previous = $this->view->double_manager->getPreviousDouble($id_user); + } + + + public function manageDoubleMergeAction() { + + if ((!$users_from = $this->_getParam('id_user_from',null)) + || (!$user_to = Class_Users::find($this->_getParam('id_user_to',null)))) { + $this->_helper->notify($this->_('id_users invalides')); + return $this->_redirectToIndex(); + } + $double_manager = new Class_User_DbDoubleFinder; + $next = $double_manager->getNextDouble($this->_getParam('id_user')); + + if (!$double_manager->mergeUsersInto($user_to, Class_Users::findAllBy(['id_user' => explode('_',$users_from)]))) { + $this->_helper->notify($this->_('Echec de la migration')); + return $this->_redirectToIndex(); + } + + $this->_helper->notify($this->_('La migration pour l\'utilisateur %s a été effectué.',$user_to->getLogin())); + return $this->_redirect('/admin/users/manage-double-user/id_user/'.$next); + } + } diff --git a/application/modules/admin/views/scripts/users/index.phtml b/application/modules/admin/views/scripts/users/index.phtml index 22a26d3d9f8ea923421f4bfe4e7b0f15c0467d06..25072612d2a4aaee996c7a49ed623697121d1da2 100644 --- a/application/modules/admin/views/scripts/users/index.phtml +++ b/application/modules/admin/views/scripts/users/index.phtml @@ -1,43 +1,26 @@ <?php if(Class_Users::getIdentity()->isAdmin()) echo $this->Button_New((new Class_Entity()) - ->setText($this->_('Ajouter un utilisateur'))); + ->setText($this->_('Ajouter un utilisateur'))); -$map =[['url' => ['module' => 'admin', - 'controller' => 'users', - 'action' => 'edit', - 'id' => '%s'], - 'icon' => 'edit', - 'label' => $this->_('modifier')], +$double_finder = (new Class_User_DbDoubleFinder); +$has_double = $double_finder->hasDouble(); - ['url' => ['module' => 'admin', - 'controller' => 'users', - 'action' => 'delete', - 'id' => '%s'], - 'icon' => 'delete', - 'label' => $this->_('supprimer')], - - ['url' => ['module' => 'opac', - 'controller' => 'blog', - 'action' => 'viewauteur', - 'id' => '%s'], - 'icon' => 'star', - 'label' => $this->_('avi(s)'), - 'condition' => 'hasAvis', - 'anchorOptions' => ['target' => '_blank']], - - ['url' => ['module' => 'opac', - 'controller' => 'panier', - 'action' => 'viewauteur', - 'id' => '%s'], - 'icon' => 'basket', - 'label' => $this->_('panier(s)'), - 'condition' => 'hasPaniers', - 'anchorOptions' => ['target' => '_blank']]]; - -$actions = function($user) use($map) { - return $this->renderModelActions($user,$map); -}; +echo $this->button( + (new Class_Entity()) + ->setText($this->_('Gérer les doublons')) + ->setAttribs($has_double + ? ['title' => $this->_('Il y a %s doublons', $double_finder->countDouble())] + : ['title' => $this->_('Il n\'y a pas de doublon'), + 'disabled' => 'disabled']) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double'], + null, true)) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'users'), + ['style' => 'filter: invert();']))); echo $this->Admin_SearchUsers($this->users, @@ -45,4 +28,7 @@ echo $this->Admin_SearchUsers($this->users, $this->form, $this->page, $this->params, - $actions); + function($model) + { + return $this->renderPluginsActions($model); + }); diff --git a/application/modules/admin/views/scripts/users/manage-double-manual.phtml b/application/modules/admin/views/scripts/users/manage-double-manual.phtml new file mode 100644 index 0000000000000000000000000000000000000000..a9dac65ca9474f5c661077176837203c802bba60 --- /dev/null +++ b/application/modules/admin/views/scripts/users/manage-double-manual.phtml @@ -0,0 +1,13 @@ +<?php +echo $this->tag('p', + $this->_('Vous avez %s comptes abonnés possédant au moins un doublon dans votre base de données qui ne peuvent pas être supprimés automatiquement.', + $this->double_manager->countDouble())); + +echo $this->tag('p', + $this->tag('small', + $this->_('Requête d\'identification des doublons : %s', $this->tag('b', $this->double_manager->getRequest())))); + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['action' => 'manage-double-user', + 'id_user' => $this->double_manager->getFirstDouble()])) + ->setText($this->_('Gérer manuellement les doublons'))); diff --git a/application/modules/admin/views/scripts/users/manage-double-user.phtml b/application/modules/admin/views/scripts/users/manage-double-user.phtml new file mode 100644 index 0000000000000000000000000000000000000000..8b9e94368b55cc11e61d851b47e05bebda7616b6 --- /dev/null +++ b/application/modules/admin/views/scripts/users/manage-double-user.phtml @@ -0,0 +1,99 @@ +<?php + +echo $this->tag('p', + $this->_('Il reste %s comptes abonnés possédant au moins un doublon dans votre base de données.', + $this->tag('b', $this->double_manager->countDouble()))); + + +$description = (new Class_TableDescription('double-users')) + ->addColumn($this->_('ID'), ['attribute' => 'id_user']) + ->addColumn($this->_('Login'), ['attribute' => 'login']) + ->addColumn($this->_('Nom'), ['attribute' => 'nom']) + ->addColumn($this->_('Prenom'), ['attribute' => 'prenom']) + ->addColumn($this->_('Email'), ['attribute' => 'mail']) + ->addColumn($this->_('Bibliothèque'), ['attribute' => 'libelleBib']) + ->addColumn($this->_('Expiration de l\'abo'), function($user) + { + return Class_Date::humanDate($user->getDateFin(), 'dd MMMM yyyy'); + }) + ->addColumn($this->_('Exporté par le SIGB'), function($model) + { + return $model->getStatut() + ? $this->_('Non') + : $this->_('Oui'); + }) + ->addRowAction(function ($model) + { + $id_users = (new Storm_Model_Collection($this->users)) + ->collect('id_user') + ->getArrayCopy(); + $id_users = array_diff($id_users, [$model->getId()]); + return $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double-merge', + 'id_user_to' => $model->getIdUser(), + 'id_user_from' => implode('_', $id_users)])) + + ->setText($this->_('Conserver')));}); + +echo $this->renderTable($description, + $this->users, + ['sorter' => true]); + +$attribs = []; +if ($this->previous == '') + $attribs = ['disabled' => 'disabled']; + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'index' + ])) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'back'), + ['style' => 'filter: invert();'])) + ->setText($this->_('Retour à la gestion des utilisateurs'))); + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double' + ])) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'back'), + ['style' => 'filter: invert();'])) + ->setText($this->_('Retour à la gestion des doublons'))); + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double-user', + 'id_user' => $this->previous] + )) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'down'), + ['style' => 'filter: invert(); transform: rotate(90deg);'])) + + ->setText($this->_('Précédent')) + ->setAttribs($attribs)); + +$attribs = []; +if ($this->next == '') + $attribs = ['disabled' => 'disabled']; + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double-user', + 'id_user' => $this->next] + )) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'down'), + ['style' => 'filter: invert(); transform: rotate(270deg);'])) + + ->setText($this->_('Ignorer et continuer'))); diff --git a/application/modules/admin/views/scripts/users/manage-double.phtml b/application/modules/admin/views/scripts/users/manage-double.phtml new file mode 100644 index 0000000000000000000000000000000000000000..36d21803f3af86eca94fcc23cb8992ea879ea055 --- /dev/null +++ b/application/modules/admin/views/scripts/users/manage-double.phtml @@ -0,0 +1,45 @@ +<?php +echo $this->tag('p', + $this->_('Vous avez %s comptes abonnés possédant au moins un doublon dans votre base de données.', + $this->tag('b', $this->double_manager->countDouble()))); + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['module' => 'admin', + 'controller' => 'users', + 'action' => 'index' + ])) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'back'), + ['style' => 'filter: invert();'])) + ->setText($this->_('Retour à la gestion des utilisateurs'))); + +echo $this->button((new Class_ButtonDescription()) + ->beProgressbar($this, Class_Url::relative('/admin/users/delete-double'), + $this->double_manager->countDouble(), + $this->_('Lancer le dédoublonnage automatique'))); + + +echo $this->button((new Class_Entity()) + ->setUrl($this->url(['action' => 'manage-double-user', + 'id_user' => $this->double_manager->getFirstDouble()])) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'users'), + ['style' => 'filter: invert();'])) + + ->setText($this->_('Gérer manuellement les doublons'))); + + +echo $this->button((new Class_Entity()) + ->setAttribs([ + 'onclick' => '$(this).next().toggle();']) + ->setImage($this->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'help'), + ['style' => 'filter: invert();'])) + ->setText($this->_('Voir la requete'))); + +echo $this->tag('pre', + $this->double_manager->getRequest(), + ['style' => 'display: none;']); diff --git a/library/Class/Admin/Skin.php b/library/Class/Admin/Skin.php index 9765371c682d45fa802f4dbee0944f269bea4b99..c63b83785e96abe8cca3333a7bcd2b5875deeab7 100644 --- a/library/Class/Admin/Skin.php +++ b/library/Class/Admin/Skin.php @@ -143,7 +143,8 @@ class Class_Admin_Skin { public function renderButtonCssOn($script_loader) { - return $this->_renderCssOn($script_loader, 'buttons.css'); + return $this->_renderCssOn($script_loader, 'buttons.css') + ->renderJQueryCssOn($script_loader); } diff --git a/library/Class/ButtonDescription.php b/library/Class/ButtonDescription.php new file mode 100644 index 0000000000000000000000000000000000000000..c55089245620a6a5ece5d93cc4b541b3910a7faf --- /dev/null +++ b/library/Class/ButtonDescription.php @@ -0,0 +1,41 @@ +<?php +/** + * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved. + * + * BOKEH is free software; you can redistribute it and/or modify + * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by + * the Free Software Foundation. + * + * There are special exceptions to the terms and conditions of the AGPL as it + * is applied to this software (see README file). + * + * BOKEH is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU AFFERO GENERAL PUBLIC LICENSE for more details. + * + * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE + * along with BOKEH; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + + +class Class_ButtonDescription extends Class_Entity { + public function beProgressbar($view, $url, $total, $text) { + + Class_ScriptLoader::getInstance()->addAdminScript('progressbar.js'); + $id = md5(implode(',',$this->_attribs)); + $this->setAttribs([ + 'id' => $id, + 'onclick' => sprintf('progressbar(\'%s\',\'%s\', %d);', + $id, $url,$total) + ]) + ->setImage($view->tagImg(Class_Admin_Skin::current() + ->getIconUrl('actions', + 'test'), + ['style' => 'filter: invert();'])) + ->setText($text.$view->tag('span', '')); + return $this; + } +} +?> \ No newline at end of file diff --git a/library/Class/User/Datas.php b/library/Class/User/Datas.php index 2f80c2d4b076ad809c7783d68a63773024c30e5c..aad8a96adaac3817a0a88d96e091c3c2ef2dda30 100644 --- a/library/Class/User/Datas.php +++ b/library/Class/User/Datas.php @@ -20,20 +20,58 @@ */ -class Class_User_Datas extends Class_Entity { - protected $_relations = []; +class Class_User_Datas { + protected + $_attribs = [], + $_relations = []; + + + public static function hasDataInIds($ids) { + $instance = new static(null); + return $instance->hasDataIn($ids); + } + public function __construct($user) { $this->setUser($user); - $this->_relations = [new Class_User_DatasRelation('notices_paniers', 'id_user'), + $this->_relations = [ + new Class_User_DatasRelation('notices_paniers', 'id_user'), + new Class_User_DatasRelation('user_group_memberships', 'user_id'), new Class_User_DatasRelationAvisNotice(), new Class_User_DatasRelation('cms_avis', 'id_user'), + new Class_User_DatasRelation('linked_cards', 'parent_id'), + new Class_User_DatasRelation('linked_cards', 'child_id'), + new Class_User_DatasRelation('session_activity_inscriptions', 'stagiaire_id'), new Class_User_DatasRelation('suggestion_achat', 'user_id'), - new Class_User_DatasRelation('session_formation_inscriptions', 'stagiaire_id'), - new Class_User_DatasRelation('user_group_memberships', 'user_id'), new Class_User_DatasRelation('formulaires', 'id_user'), - new Class_User_DatasRelation('multimedia_devicehold', 'id_user')]; + new Class_User_DatasRelation('multimedia_devicehold', 'id_user') + ]; + } + + + public function hasDataIn($ids) { + foreach($this->_relations as $relation) + if (0 < $this->_countIdsContainningData($relation, $ids)) + return true; + + return false; + } + + + protected function _countIdsContainningData($relation, $ids) { + return $relation->countIdsContainingData($ids); + } + + + public function setUser($user) { + $this->_attribs['User'] = $user; + return $this; + } + + + public function getUser() { + return $this->_attribs['User']; } @@ -85,11 +123,25 @@ class Class_User_Datas extends Class_Entity { return current($targets); } + + + public function setError($key) { + $this->_attribs['Error'] = $key; + return $this; + } + + + public function getError() { + return $this->_attribs['Error']; + } } -class Class_User_DatasRelation extends Class_Entity { + +class Class_User_DatasRelation { + protected $_attribs; + public function __construct($table, $key) { $this->setTable($table) ->setKey($key) @@ -97,6 +149,39 @@ class Class_User_DatasRelation extends Class_Entity { } + public function setTable($table) { + $this->_attribs['Table'] = $table; + return $this; + } + + + public function getTable() { + return $this->_attribs['Table']; + } + + + public function setSql($sql) { + $this->_attribs['Sql'] = $sql; + return $this; + } + + + public function getSql() { + return $this->_attribs['Sql']; + } + + + public function setKey($key) { + $this->_attribs['Key'] = $key; + return $this; + } + + + public function getKey() { + return $this->_attribs['Key']; + } + + public function countFor($user) { return $this ->getSql() @@ -105,6 +190,16 @@ class Class_User_DatasRelation extends Class_Entity { } + public function countIdsContainingData($ids) { + return $this + ->getSql() + ->fetchOne(sprintf('select count(id) from %s where %s in (%s)', + $this->getTable(), + $this->getKey(), + implode(',', $ids))); + } + + public function giveFromTo($from, $to) { $this ->getSql() diff --git a/library/Class/User/DbDoubleFinder.php b/library/Class/User/DbDoubleFinder.php new file mode 100644 index 0000000000000000000000000000000000000000..e99195cbd1ae11039e494597301899d12c1b31f8 --- /dev/null +++ b/library/Class/User/DbDoubleFinder.php @@ -0,0 +1,236 @@ +<?php +/** + * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved. + * + * BOKEH is free software; you can redistribute it and/or modify + * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by + * the Free Software Foundation. + * + * There are special exceptions to the terms and conditions of the AGPL as it + * is applied to this software (see README file). + * + * BOKEH is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU AFFERO GENERAL PUBLIC LICENSE for more details. + * + * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE + * along with BOKEH; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + + +class Class_User_DbDoubleFinder { + + const LIMIT_DEDUPE = 10; + + + public function getRequest($cursor = 0) { + return sprintf('select min(id_user) as id_user, password, login, idabon, ordreabon, count(*) as doublon from bib_admin_users where role_level = 2 and id_user > %s group by idabon, ordreabon, login, password having doublon > 1 order by id_user asc', + $cursor); + } + + + public function getDouble($cursor = 0) { + return Zend_Registry::get('sql') + ->fetchAll($this->getRequest($cursor)); + } + + + public function hasDouble() { + return $this->countDouble() > 0; + } + + + public function countDouble() { + return count($this->getDouble()); + } + + + public function getFirstDoubleInLimit($cursor = 0) { + $double = $this->getDouble($cursor); + return array_slice($double, 0, static::LIMIT_DEDUPE); + } + + + public function findDoubleFor($user) { + return $this->_findUsersWithParams($user->getIdabon(), + $user->getLogin(), + $user->getOrdreabon(), + $user->getPassword()); + } + + + public function hasDoubleFor($user) { + return 1 < count($this->findDoubleFor($user)); + } + + + public function getFirstDouble() { + $ids = $this->_getUserIdsFromDouble(); + return array_shift($ids); + } + + + public function getNextDouble($id_user) { + $ids = $this->_getUserIdsFromDouble(); + + if(false === ($position = array_search($id_user, $ids))) + return ''; + + $next = $position + 1; + return isset($ids[$next]) + ? $ids[$next] + : ''; + } + + + public function getPreviousDouble($id_user) { + $ids = $this->_getUserIdsFromDouble(); + + if(false === ($position = array_search($id_user, $ids))) + return ''; + + $previous = $position - 1; + return isset($ids[$previous]) + ? $ids[$previous] + : ''; + } + + + public function dedupeUsers($cursor = 0) { + $double = $this->getFirstDoubleInLimit($cursor); + foreach($double as $data) + $this->_autoMergeOrDelete($this->_findUsersFromDouble($data)); + + $last = array_pop($double); + + if(empty($last)) + return null; + + return isset($last['id_user']) + ? (int) $last['id_user'] + : null; + } + + + protected function _autoMergeOrDelete($users) { + if(!$user = $this->_findUserToKeep($users)) + return $this->_autoDeleteDoubles($users); + + return $this->mergeUsersInto($user, $users); + } + + + public function mergeUsersInto($user, $users) { + $users = array_filter($users, function($instance) use ($user) + { + return $instance->getId() !== $user->getId(); + }); + + foreach ($users as $deletable_user) { + $datas = new Class_User_Datas($deletable_user); + + if (!$datas->giveTo(['id_user' => $user->getIdUser()], true)) + return; + + $deletable_user->delete(); + } + + return true; + } + + + protected function _findUsersFromDouble($data) { + return $this->_findUsersWithParams($data['idabon'], + $data['login'], + $data['ordreabon'], + $data['password']); + } + + + protected function _findUsersWithParams($idabon, $login, $ordreabon, $password) { + return Class_Users::findAllBy(['idabon' => $idabon, + 'password' => $password, + 'login' => $login, + 'ordreabon' => $ordreabon]); + } + + + protected function _findUserToKeep($users) { + $last_name = ''; + $first_name =''; + $status = null; + + $keep_user = null; + foreach($users as $user) { + if ((!$last_name = $this->_userAsSame($user->getNom(), $last_name)) + || (!$first_name = $this->_userAsSame($user->getPrenom(), $first_name))) + return; + + if (!$user->hasDateFin()) + continue; + + if ($user->hasToBeDelete()) + continue; + + if ($keep_user) + return; + + $keep_user = $user; + } + + return $this->_hasGreaterDates($keep_user, $users); + } + + + protected function _userAsSame($name, $expected_name) { + $name = Class_Indexation::getInstance()->alphaMaj($name); + + if ($expected_name == '') + $expected_name = $name; + + return $expected_name == $name + ? $name + : false; + } + + + protected function _hasGreaterDates($user, $users) { + if(!$user) + return null; + + foreach($users as $other) { + if (strtotime($other->getDateFin()) > strtotime($user->getDateFin())) + return null; + } + + return $user; + } + + + protected function _autoDeleteDoubles($users) { + if(!$ids = $this->_getDoubleUsersWithOutDatas($users)) + return; + + return Zend_Registry::get('sql') + ->query(sprintf('delete from bib_admin_users where id_user in (%s)', implode(',',$ids))); + } + + + protected function _getDoubleUsersWithOutDatas($users) { + $ids = (new Storm_Model_Collection($users)) + ->collect('id_user') + ->getArrayCopy(); + + if(Class_User_Datas::hasDataInIds($ids)) + return null; + + return $ids; + } + + + protected function _getUserIdsFromDouble() { + return array_column($this->getDouble(), 'id_user'); + } +} \ No newline at end of file diff --git a/library/Class/Users.php b/library/Class/Users.php index c5c17052a5216d40e2db294fbf960da88f0108cd..70e0e32881c4b74d63031722387c83317e7d86c4 100644 --- a/library/Class/Users.php +++ b/library/Class/Users.php @@ -1859,4 +1859,9 @@ class Class_Users extends Storm_Model_Abstract { public function hasPaniers() { return (0 < $this->numberOfPaniers()); } + + + public function hasToBeDelete() { + return static::STATUT_TO_BE_DELETED == $this->getStatut(); + } } diff --git a/library/ZendAfi/Controller/Action/Helper/Progressbar.php b/library/ZendAfi/Controller/Action/Helper/Progressbar.php new file mode 100644 index 0000000000000000000000000000000000000000..df7391cc8f97f5869cd47d4323cae4d87ef28bd4 --- /dev/null +++ b/library/ZendAfi/Controller/Action/Helper/Progressbar.php @@ -0,0 +1,38 @@ +<?php +/** + * Copyright (c) 2012-2017, Agence Française Informatique (AFI). All rights reserved. + * + * BOKEH is free software; you can redistribute it and/or modify + * it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE as published by + * the Free Software Foundation. + * + * There are special exceptions to the terms and conditions of the AGPL as it + * is applied to this software (see README file). + * + * BOKEH is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU AFFERO GENERAL PUBLIC LICENSE for more details. + * + * You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE + * along with BOKEH; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + + +class ZendAfi_Controller_Action_Helper_Progressbar extends Zend_Controller_Action_Helper_Abstract { + public function progressbar($total, $done, $cursor, $run) { + $this->getActionController() + ->getHelper('json') + ->direct( ['total' => $total, + 'done' => $done, + 'cursor' => $cursor, + 'run' => $run]); + + } + + + public function direct($total, $done, $cursor, $run) { + return $this->progressbar($total, $done, $cursor, $run); + } +} diff --git a/library/ZendAfi/Controller/Plugin/Manager/User.php b/library/ZendAfi/Controller/Plugin/Manager/User.php index 34148948ee8ec00ad75a0bdbdf07fbc4ba1397ee..40fd1f0d35f295c02cada011f1291acf45c05961 100644 --- a/library/ZendAfi/Controller/Plugin/Manager/User.php +++ b/library/ZendAfi/Controller/Plugin/Manager/User.php @@ -21,6 +21,53 @@ class ZendAfi_Controller_Plugin_Manager_User extends ZendAfi_Controller_Plugin_Manager_Manager { + public function getActions($model) { + return [['url' => ['module' => 'admin', + 'controller' => 'users', + 'action' => 'edit', + 'id' => '%s'], + 'icon' => 'edit', + 'label' => $this->_('modifier')], + + ['url' => ['module' => 'admin', + 'controller' => 'users', + 'action' => 'delete', + 'id' => '%s'], + 'icon' => 'delete', + 'label' => $this->_('supprimer')], + + ['url' => ['module' => 'opac', + 'controller' => 'blog', + 'action' => 'viewauteur', + 'id' => '%s'], + 'icon' => 'star', + 'label' => $this->_('avi(s)'), + 'condition' => 'hasAvis', + 'anchorOptions' => ['target' => '_blank']], + + ['url' => ['module' => 'opac', + 'controller' => 'panier', + 'action' => 'viewauteur', + 'id' => '%s'], + 'icon' => 'basket', + 'label' => $this->_('panier(s)'), + 'condition' => 'hasPaniers', + 'anchorOptions' => ['target' => '_blank']], + + ['url' => ['module' => 'admin', + 'controller' => 'users', + 'action' => 'manage-double-user', + 'id_user' => '%s'], + 'icon' => 'help', + 'label' => $this->_('Doublons'), + 'condition' => function ($model) { + return (new Class_User_DbDoubleFinder())->hasDoubleFor($model); + } + ] + ]; + } + + protected function _getPost() { $post = $this->_request->getPost(); $post['user_groups'] = array_filter( diff --git a/library/ZendAfi/View/Helper/Admin/Button.php b/library/ZendAfi/View/Helper/Admin/Button.php index d6d364d4fb4686c09dd81626dc516c000ec378eb..91f9d69613da7807f7533c7669b57def006f8858 100644 --- a/library/ZendAfi/View/Helper/Admin/Button.php +++ b/library/ZendAfi/View/Helper/Admin/Button.php @@ -76,4 +76,5 @@ class ZendAfi_View_Helper_Admin_Button extends ZendAfi_View_Helper_Button { return $button->getElement(); } + } \ No newline at end of file diff --git a/public/admin/js/onload_utils.js b/public/admin/js/onload_utils.js index 696095416f11c87c76fb55e50f6d9380f4a50c4c..d11eacdfc50612ff8f94b153ffe14b80afc66aea 100644 --- a/public/admin/js/onload_utils.js +++ b/public/admin/js/onload_utils.js @@ -150,3 +150,5 @@ var updateSelectWidget = function(element) { url = url + '/categories/' + categories_ids; $('.selected_articles_widget a').attr('href', url); } + + diff --git a/public/admin/js/progressbar.js b/public/admin/js/progressbar.js new file mode 100644 index 0000000000000000000000000000000000000000..a1c0bb9c384a01612a65bce71986541fd55cfa65 --- /dev/null +++ b/public/admin/js/progressbar.js @@ -0,0 +1,28 @@ +var progressbar = function(id, url, total, cursor = 0) { + var run; + var tag = $("#" + id + " span"); + $.ajax({ + type: 'GET', + url: url, + data: {'total': total, + 'cursor': cursor}, + success: function(data) + { + var percent = data.done * 100 / total; + var new_cursor = data.cursor; + run = data.run; + percent = percent ? percent : 1; + tag.progressbar({value: percent}); + + if(run) + progressbar(id, url, total, new_cursor); + + }, + complete: function(event, ui) { + if(run) + return; + tag.progressbar({value: 100}); + + location.reload(); + }}); +} diff --git a/public/admin/skins/bokeh74/buttons.css b/public/admin/skins/bokeh74/buttons.css index 7e78097b72029592ea93b8eaed8fa8c471a31dde..ffa3ba453b01f66066a6306779f517bdd8c6dda9 100644 --- a/public/admin/skins/bokeh74/buttons.css +++ b/public/admin/skins/bokeh74/buttons.css @@ -26,6 +26,7 @@ transition: background 0.4s; border: none; cursor: pointer; + vertical-align: top; } .admin-button > img { @@ -44,4 +45,10 @@ div.admin-buttons { .admin-button:disabled > img { filter: opacity(0.2); +} + + +.admin-button span.ui-progressbar { + display: block; + margin: 5px; } \ No newline at end of file diff --git a/public/admin/skins/bokeh74/global.css b/public/admin/skins/bokeh74/global.css index 59733f9854b7238266e11d111e938522a4fe4e17..f8fe1afe0084d8a12026e48af5cf3ec39585c935 100755 --- a/public/admin/skins/bokeh74/global.css +++ b/public/admin/skins/bokeh74/global.css @@ -161,6 +161,11 @@ td[id*="menu_item"] { box-shadow: 1px 1px 5px var(--widget-shadow); } +.modules pre { + box-shadow: inset 1px 1px 5px var(--widget-shadow); +} + + /* Font */ body, body * { @@ -994,3 +999,9 @@ table#logs img { .modules a[data-order$="desc"] { background-image: url(); } + +.modules pre { + margin: 1ex 1em; + padding: 1ex 1em; + overflow: auto; +} diff --git a/public/admin/skins/bokeh74/jquery.css b/public/admin/skins/bokeh74/jquery.css index bf00c573346b7de0b5f37d445c1b1bd5487aaaf4..8225cfd2499f3e73a07bffd2121aa3932ca5c3d1 100644 --- a/public/admin/skins/bokeh74/jquery.css +++ b/public/admin/skins/bokeh74/jquery.css @@ -151,3 +151,13 @@ body .ui-tabs a.errors.ui-tabs-anchor { color: white; background: var(--error-text); } + +body .ui-progressbar .ui-progressbar-value { + background : var(--success-background); + margin: 0; + +} + +body .ui-progressbar { + height: 1em; +} diff --git a/public/opac/js/subModal.js b/public/opac/js/subModal.js index ebc20192df84a48d45fe89850890d53d766860ab..4445f2cc5283d2cf7709e54bccbbb206b1d30fd4 100644 --- a/public/opac/js/subModal.js +++ b/public/opac/js/subModal.js @@ -5,8 +5,10 @@ var current_anchor; window.initializePopups = function() { - $('[data-popup="true"]') - .unbind('click').click(function(event){ + $('[data-popup="true"], [data-popup="1"]') + .unbind('onclick') + .unbind('click') + .click(function(event){ event.preventDefault(); current_anchor=$(this); addLoadingClass(); diff --git a/tests/application/modules/admin/controllers/UsersControllerTest.php b/tests/application/modules/admin/controllers/UsersControllerTest.php index d65c549915a9d87fd3ad6c7bd0f66db2d544243d..c1d09857f9f967336a5f9ebc744e74c422a705eb 100644 --- a/tests/application/modules/admin/controllers/UsersControllerTest.php +++ b/tests/application/modules/admin/controllers/UsersControllerTest.php @@ -27,12 +27,12 @@ abstract class UsersControllerWithMarcusTestCase extends AbstractControllerTestC $group_vodeclic = $this->fixture('Class_UserGroup', ['id' => 20, 'libelle' => 'Multimedia']) - ->addRight(Class_UserGroup::RIGHT_ACCES_VODECLIC); + ->addRight(Class_UserGroup::RIGHT_ACCES_VODECLIC); $group_referent = $this->fixture('Class_UserGroup', ['id' => 22, 'libelle' => 'Referent']) - ->addRight(Class_UserGroup::RIGHT_USER_DOMAINES_SUPPRESSION_LIMIT); + ->addRight(Class_UserGroup::RIGHT_USER_DOMAINES_SUPPRESSION_LIMIT); $group_stagiaires = $this->fixture('Class_UserGroup',['id' => 25, 'libelle' => 'Stagiaires']); @@ -86,14 +86,26 @@ abstract class UsersControllerWithMarcusTestCase extends AbstractControllerTestC $this->dispatch('/admin/users/edit/id/10', true); } + } + + + class UsersControllerIndexTest extends UsersControllerWithMarcusTestCase { public function setUp() { parent::setUp(); Zend_Registry::set('sql', $this->mock() + ->whenCalled('fetchAll') + ->with('select min(id_user) as id_user, password, login, idabon, ordreabon, count(*) as doublon from bib_admin_users where role_level = 2 and id_user > 0 group by idabon, ordreabon, login, password having doublon > 1 order by id_user asc') + ->answers([]) + + ->whenCalled('fetchAll') + ->with('select idabon, count(*) as doublon from bib_admin_users where role_level = 2 group by idabon having doublon > 1;') + ->answers([]) + ->whenCalled('fetchAllByColumn') ->with('select distinct(id_user) as id from notices_avis') ->answers([2233, 987398])); @@ -119,6 +131,13 @@ class UsersControllerIndexTest extends UsersControllerWithMarcusTestCase { ->whenCalled('hasIdentity') ->answers($user) + ->whenCalled('findAllBy') + ->with(['idabon' => '', + 'password' => 'francis', + 'login' => 'francis', + 'ordreabon' => '']) + ->answers($francis) + ->whenCalled('findAllBy') ->with(['role_level' => 2, 'order' => 'nom asc', @@ -131,7 +150,6 @@ class UsersControllerIndexTest extends UsersControllerWithMarcusTestCase { 'order' => 'nom asc', 'where' => '(STR_TO_DATE(date_fin, \'%Y-%m-%d\') >= CURDATE()) AND (id_user in (2233,987398)) AND (login LIKE "%francis%" OR nom LIKE "%francis%" OR prenom LIKE "%francis%" OR pseudo LIKE "%francis%" OR mail LIKE "%francis%" OR idabon LIKE "%francis%") AND (role_level <= 7)']) ->answers(55) - ->beStrict(); $this->dispatch('/admin/users?search_id_site=all&search_role_level=2&search_valid_subscription=1&search_review=1&search_search_for=\"\'fra"n\'cis"', true); @@ -733,10 +751,10 @@ class UsersControllerReferentIndexTest extends UsersControllerWithMarcusTestCase $this->user_loader = Storm_Test_ObjectWrapper::onLoaderOfModel('Class_Users'); $this->user_loader->whenCalled('getIdentity') - ->answers($user = Class_Users::newInstanceWithId(2) - ->setLogin('referent') - ->setRoleLevel(ZendAfi_Acl_AdminControllerRoles::MODO_PORTAIL) - ->setPseudo('referent')); + ->answers($user = Class_Users::newInstanceWithId(2) + ->setLogin('referent') + ->setRoleLevel(ZendAfi_Acl_AdminControllerRoles::MODO_PORTAIL) + ->setPseudo('referent')); $this->addUserToRightsReferent($user); @@ -881,15 +899,15 @@ abstract class Admin_UsersControllerEditAdminTestCase extends Admin_AbstractCont 'libelle' => 'Region Savoie']); $this->_lib_ac = $this->fixture('Class_Bib', - ['id' => 7, - 'libelle' => 'AC', - 'id_zone' => 1]); + ['id' => 7, + 'libelle' => 'AC', + 'id_zone' => 1]); $this->_lib_po = $this->fixture('Class_Bib', - ['id' => 15, - 'libelle' => 'PO', - 'id_zone' => 1, - 'redmine_api_key' => '123abc']); + ['id' => 15, + 'libelle' => 'PO', + 'id_zone' => 1, + 'redmine_api_key' => '123abc']); $this->fixture('Class_Bib', ['id' => 13, @@ -946,7 +964,7 @@ class Admin_UsersControllerFormEditAdminTest extends Admin_UsersControllerEditAd /** @test */ public function withModoPortailRedmineLibrarySelectorShouldBePresent() { - $this->assertXPath('//form//select[@name="redmine_library"]', $this->_response->getBody()); + $this->assertXPath('//form//select[@name="redmine_library"]', $this->_response->getBody()); } @@ -1165,7 +1183,7 @@ class UsersControllerWithAdminPortalTest extends Admin_AbstractControllerTestCas } - /** @test */ + /** @test */ public function linkToShowTumPanierShouldNotBePresent() { $this->assertNotXPath('//a[contains(@href, "/panier/viewauteur/id/3")]'); } @@ -1176,3 +1194,555 @@ class UsersControllerWithAdminPortalTest extends Admin_AbstractControllerTestCas $this->assertXPath('//th/a[contains(@href, "/search_order/nom/")]'); } } + + + + +abstract class UsersControllerDoubleTestCase extends Admin_AbstractControllerTestCase { + protected $_storm_default_to_volatile = true; + + + public function setUp() { + parent::setUp(); + $mock_sql = + $this->mock() + ->whenCalled('fetchAll') + ->with('select min(id_user) as id_user, password, login, idabon, ordreabon, count(*) as doublon from bib_admin_users where role_level = 2 and id_user > 0 group by idabon, ordreabon, login, password having doublon > 1 order by id_user asc') + ->answers([ + ['id_user' => '25', + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '89', + 'ordreabon' => '18', + 'doublon' => '2'], + + ['id_user' => '2', + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '49', + 'ordreabon' => '1', + 'doublon' => '2'], + + ['id_user' => '100', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => '89', + 'ordreabon' => '18', + 'doublon' => '3'], + + ['id_user' => '103', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => '90', + 'ordreabon' => '18', + 'doublon' => '3'], + + ['id_user' => '106', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => '91', + 'ordreabon' => '18', + 'doublon' => '3'], + + ['id_user' => '110', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => '100', + 'ordreabon' => '18', + 'doublon' => '3'] + + ]) + + ->whenCalled('fetchOne') + ->with('select count(id) from notices_paniers where id_user in (25,654,655)') + ->answers(1) + + ->whenCalled('fetchOne') + ->with('select count(id) from notices_paniers where id_user in (100,101,102)') + ->answers(1) + + ->whenCalled('query') + ->with('delete from bib_admin_users where id_user in (2,29)') + ->willDo(function() + { + Class_Users::find(2)->delete(); + Class_Users::find(29)->delete(); + }) + + ->whenCalled('fetchOne') + ->with('select count(id) from notices_paniers where id_user in (103,104,105)') + ->answers(1) + + ->whenCalled('fetchOne') + ->with('select count(id) from notices_paniers where id_user in (106,107,108)') + ->answers(1) + + ->whenCalled('fetchOne') + ->with('select count(id) from notices_paniers where id_user in (110,111)') + ->answers(1) + + ->whenCalled('fetchOne') + ->answers(0) + + ->whenCalled('query') + ->answers(true) + + ->whenCalled('query') + ->with('update notices_paniers set id_user=654 where id_user=25') + ->willDo(function() { + return Class_PanierNotice::find(4)->setIdUser(654)->save(); + }) + ->whenCalled('query') + ->with('update notices_paniers set id_user=100 where id_user=101') + ->willDo(function() { + return Class_PanierNotice::find(101)->setIdUser(100)->save(); + }) + ->whenCalled('query') + ->with('update notices_paniers set id_user=100 where id_user=102') + ->willDo(function() { + return Class_PanierNotice::find(102)->setIdUser(100)->save(); + }); + + Zend_Registry::set('sql', $mock_sql); + + $this->fixture('Class_Users', + ['id' => 2, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '49', + 'ordreabon' => 1]); + + $this->fixture('Class_Users', + ['id' => 29, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '49', + 'ordreabon' => 1]); + + $this->fixture('Class_Users', + ['id' => 123, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '49', + 'ordreabon' => 2]); + + $this->fixture('Class_Users', + ['id' => 25, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '89', + 'ordreabon' => 18]); + + $this->fixture('Class_Users', + ['id' => 654, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '89', + 'ordreabon' => 18]); + + $this->fixture('Class_Users', + ['id' => 655, + 'login' => 'Ret', + 'password' => 'urn', + 'idabon' => '89', + 'ordreabon' => 18]); + + $this->createUsersWithStatusDifferentAndSameNames(89, [100,101,102], [0,1,1]); + $this->createUsersWithStatusDifferentAndSameNames(90, [103,104,105], [1,1,1]); + $this->createUsersWithStatusDifferentAndSameNames(91, [106,107,108], [0,0,1]); + $this->createUsersWithEmptyValidityDate(); + $this->fixture('Class_PanierNotice', + ['id' => 4, + 'libelle' => 's', + 'id_user' => 25]); + } + + + protected function createUsersWithEmptyValidityDate() { + $this->fixture('Class_Users', + ['id' => 110, + 'nom' => 'Abatzi', + 'prenom' => 'rita', + 'statut' => 1, + 'login' => 'rita', + 'date_fin' => null, + 'password' => 'rita', + 'idabon' => 100, + 'ordreabon' => 18]); + + $this->fixture('Class_Users', + ['id' => 111, + 'nom' => 'Abatzi', + 'prenom' => 'rita', + 'statut' => 0, + 'date_fin' => null, + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => 100, + 'ordreabon' => 18]); + + $this->fixture('Class_PanierNotice', + ['id' => 40, + 'libelle' => 's', + 'id_user' => 110]); + + } + + + protected function createUsersWithStatusDifferentAndSameNames($id_abon, $ids, $status) { + $this->fixture('Class_Users', + ['id' => $ids[0], + 'nom' => 'test', + 'prenom' => 'test', + 'statut' => $status[0], + 'login' => 'rita', + 'date_fin' => '2017-01-08', + 'password' => 'rita', + 'idabon' => $id_abon, + 'ordreabon' => 18]); + + $this->fixture('Class_Users', + ['id' => $ids[1], + 'nom' => 'test', + 'prenom' => 'test', + 'statut' => $status[1], + 'date_fin' => '2017-01-07', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => $id_abon, + 'ordreabon' => 18]); + + $this->fixture('Class_Users', + ['id' => $ids[2], + 'nom' => 'test', + 'prenom' => 'test', + 'statut' => $status[2], + 'date_fin' => '2016-09-07', + 'login' => 'rita', + 'password' => 'rita', + 'idabon' => $id_abon, + 'ordreabon' => 18]); + + $this->fixture('Class_PanierNotice', + ['id' => $ids[0], + 'libelle' => 'mybasket', + 'id_user' => $ids[0]]); + + $this->fixture('Class_PanierNotice', + ['id' => $ids[1], + 'libelle' => 'mybasket', + 'id_user' => $ids[1]]); + + $this->fixture('Class_PanierNotice', + ['id' => $ids[2], + 'libelle' => 'mybasket', + 'id_user' => $ids[2]]); + } +} + + + + +class UsersControllerManageDoubleTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double', true); + } + + + /** @test */ + public function sixDoubleAccountsShouldBeFounded() { + $this->assertXPathContentContains('//p', 'Vous avez <b>6</b> comptes abonnés possédant au moins un doublon',$this->_response->getBody()); + } + + + /** @test */ + public function purgeButtonShouldBePresent() { + $this->assertXPathContentContains('//button[contains(@onclick, "/admin/users/delete-double")]', 'Lancer le dédoublonnage automatique'); + } +} + + + +class UsersControllerDeleteDoubleEndCursorTest extends Admin_AbstractControllerTestCase { + + public function setUp() { + parent::setUp(); + Zend_Registry::set('sql', $this->mock() + ->whenCalled('fetchAll') + ->answers([])); + + $this->dispatch('/admin/users/delete-double/total/6/cursor/1000', true); + } + + + /** @test */ + public function jsonShouldBeRendered() { + $this->assertEquals(['total' => 6, + 'done' => 6, + 'cursor' => null, + 'run' => false], json_decode($this->_response->getBody(),true)); + } +} + + + + + +class UsersControllerDeleteDoubleTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/delete-double/total/6', true); + } + + + /** @test */ + public function jsonShouldBeRendered() { + $this->assertEquals(['total' => 6, + 'done' => 0, + 'cursor' => 110, + 'run' => true], json_decode($this->_response->getBody(),true)); + } + + + /** @test */ + public function userId123456ShouldNotBeDeleted() { + $this->assertNotNull(Class_Users::find(123)); + } + + + /** @test */ + public function userId2ShouldHaveBeendDeleted() { + $this->assertNull(Class_Users::find(2)); + } + + + /** @test */ + public function userId101And102ShouldHaveBeendDeleted() { + $this->assertNull(Class_Users::find(101)); + $this->assertNull(Class_Users::find(102)); + $this->assertNotNull(Class_Users::find(100)); + } + + + /** @test */ + public function userId103And104ShouldNotHaveBeendDeleted() { + $this->assertNotNull(Class_Users::find(103)); + $this->assertNotNull(Class_Users::find(104)); + $this->assertNotNull(Class_Users::find(105)); + } + + + /** @test */ + public function userId106And107ShouldNotHaveBeendDeleted() { + $this->assertNotNull(Class_Users::find(106)); + $this->assertNotNull(Class_Users::find(107)); + $this->assertNotNull(Class_Users::find(108)); + } + + + /** @test */ + public function userId110And111ShouldNotHaveBeendDeleted() { + $this->assertNotNull(Class_Users::find(110)); + $this->assertNotNull(Class_Users::find(111)); + } + + + /** @test */ + public function userId100ShouldHaveBaskets() { + $this->assertEquals(3,Class_PanierNotice::countBy(['id_user' => 100,'id' =>[100,101,102]])); + } + + + /** @test */ + public function userId29ShouldHaveBeendDeleted() { + $this->assertNull(Class_Users::find(29)); + } + + + /** @test */ + public function userId25ShouldNotBeDeleted() { + $this->assertNotNull(Class_Users::find(25)); + } + + + /** @test */ + public function userId654ShouldNotBeDeleted() { + $this->assertNotNull(Class_Users::find(654)); + } +} + + + + +class UsersControllerManageDoubleManualTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double-manual', true); + } + + + /** @test */ + public function shouldDisplayContinuButton() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-user/id_user/25")]'); + } +} + + + + +class UsersControllerManageDoubleUserTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double-user/id_user/25', true); + } + + + /** @test */ + public function shouldDisplayUsers() { + $this->assertXpath('//div'); + } + + + /** @test */ + public function backButtonShouldBeDisplayed() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double")]',$this->_response->getBody()); + } + + + /** @test */ + public function buttonMergeShouldBe654to25() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-merge/id_user/25/id_user_to/25/id_user_from/654")]',$this->_response->getBody()); + } + + + /** @test */ + public function buttonMergeShouldBe25to654() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-merge/id_user/25/id_user_to/654/id_user_from/25_655")]',$this->_response->getBody()); + } + + + /** @test */ + public function buttonIgnoreAndContinueShouldHaveUserId2() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-user/id_user/2")]',$this->_response->getBody()); + } + + + /** @test */ + public function buttonPreviousShouldNotBeAvailable() { + $this->assertXpathContentContains('//button[contains(@disabled, "disabled")]','Précédent',$this->_response->getBody()); + } +} + + + + +class UsersControllerManageNextDoubleUserTest extends UsersControllerDoubleTestCase { + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double-user/id_user/2', true); + } + + + /** @test*/ + public function buttonMergeShouldBe2to29() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-merge/id_user/2/id_user_to/29/id_user_from/2")]',$this->_response->getBody()); + } + + + /** @test */ + public function buttonPreviousAndNextShouldBeAvailable() { + $this->assertNotXPath('//button[contains(@disabled, "disabled")]'); + } + + + /** @test */ + public function buttonPreviousShouldHaveUserId25() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double-user/id_user/25")]',$this->_response->getBody()); + } +} + + + + +class UsersControllerManageDoubleMergeTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double-merge/id_user_from/25/id_user_to/654', true); + } + + + /** @test */ + public function user25ShouldBeDeleted() { + $this->assertNull(Class_Users::find(25)); + } + + + /** @test */ + public function panier4ShouldHaveIdUser654() { + $this->assertEquals(654,Class_PanierNotice::find(4)->getIdUser()); + } +} + + + + +class UsersControllerManageDoubleMergeWithMultipleSourcesTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/manage-double-merge/id_user_from/25_655/id_user_to/654', true); + } + + + /** @test */ + public function user25ShouldBeDeleted() { + $this->assertNull(Class_Users::find(25)); + } + + + /** @test */ + public function user655ShouldBeDeleted() { + $this->assertNull(Class_Users::find(655)); + } +} + + + + +class UsersControllerIndexWithDoubleTest extends UsersControllerDoubleTestCase { + + public function setUp() { + parent::setUp(); + $this->dispatch('/admin/users/index', true); + } + + + /** @test */ + public function buttonManageDoubleShouldBePresent() { + $this->assertXpath('//button[contains(@onclick, "/admin/users/manage-double")]'); + } + + + /** @test */ + public function linkToManageDoubleOfUserId25ShouldBePresent() { + $this->assertXpath('//td//a[contains(@href, "/admin/users/manage-double-user/id_user/25")]'); + } + + + /** @test */ + public function idUser123ShouldBePresent() { + $this->assertXPath('//td//a[contains(@href, "/admin/users/edit/id/123")]'); + } + + + /** @test */ + public function idUser123ShouldNotHaveLinkToManageDouble() { + $this->assertNotXPath('//td//a[contains(@href, "/admin/users/manage-double-user/id_user/123")]'); + } +} diff --git a/tests/library/Class/UserDatasTest.php b/tests/library/Class/UserDatasTest.php index 8260fc02ff1d15d7d32e50d0d6cbfa2066d5914e..f67537ef647f221f4485a7a0b24f8f374306a0a4 100644 --- a/tests/library/Class/UserDatasTest.php +++ b/tests/library/Class/UserDatasTest.php @@ -96,10 +96,12 @@ class UserDatasTest extends UserDatasTestCase { ['notices_avis', 'id_user'], ['cms_avis', 'id_user'], ['suggestion_achat', 'user_id'], - ['session_formation_inscriptions', 'stagiaire_id'], + ['session_activity_inscriptions', 'stagiaire_id'], ['user_group_memberships', 'user_id'], ['formulaires', 'id_user'], - ['multimedia_devicehold', 'id_user'] + ['multimedia_devicehold', 'id_user'], + ['linked_cards', 'parent_id'], + ['linked_cards', 'child_id'] ]; foreach($queries as $query)