User.class.php 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633
  1. <?php
  2. /**
  3. * Define an application user.
  4. * @author valentin carruesco
  5. * @category Core
  6. * @license MIT
  7. */
  8. class User extends Entity {
  9. public $id,$login,$password,$function,$name,$firstname,$mail,$state,$rights,$firms,$superadmin,$token,$preferences,$phone,$mobile,$groups,$ranks,$manager,$service,$origin,$meta;
  10. public $entityLabel = 'Utilisateur';
  11. public $fields = array(
  12. 'id' => 'key',
  13. 'login' => array('label'=>'Identifiant','type'=>'string'),
  14. 'password' => array('label'=>'Mot de passe','type'=>'password'),
  15. 'name' => array('label'=>'Nom','type'=>'string'),
  16. 'function' => array('label'=>'Fonction','type'=>'string'),
  17. 'firstname' => array('label'=>'Prénom','type'=>'text'),
  18. 'token' => array('label'=>'Token','type'=>'string'),
  19. 'mail' => array('label'=>'E-mail','type'=>'mail'),
  20. 'state' => array('label'=>'Etat','type'=>'string'),
  21. 'phone' => array('label'=>'Téléphone','type'=>'phone'),
  22. 'mobile' => array('label'=>'Téléphone portable','type'=>'phone'),
  23. 'manager' => array('label'=>'Manager','type'=>'user'),
  24. 'origin' => array('label'=>'Origine','type'=>'text'),
  25. 'service' => array('label'=>'Service','type'=>'text'),
  26. 'meta' => array('label'=>'Meta informations','type'=>'text')
  27. );
  28. function __construct(){
  29. global $myUser, $myFirm, $conf;
  30. parent::__construct();
  31. $this->token = self::generateToken();
  32. $this->meta = array();
  33. $this->ranks = array();
  34. $this->firms = array();
  35. }
  36. public static function getAll($options = array()){
  37. $users = array();
  38. $options = array_merge(array(
  39. 'right' => true,
  40. 'manager' => true,
  41. 'force' =>false
  42. ),$options);
  43. if(isset($_SESSION['users_'.($options['right']?'rights':'norights')]))
  44. $users = unserialize($_SESSION['users_'.($options['right']?'rights':'norights')]);
  45. if(empty($users) || $options['force']){
  46. $users = array();
  47. //récuperation des users non db (plugin ad ...)
  48. Plugin::callHook('user_get_all',array(&$users,$options));
  49. $logins = array();
  50. //Récuperation des users db
  51. foreach (self::loadAll(array('state'=>User::ACTIVE), array(' name ASC ')) as $baseUser)
  52. $users[$baseUser->login] = $baseUser;
  53. //récuperation des logins concernés pour requettage
  54. foreach($users as $key=>$user){
  55. //chargement des managers pour les origines db
  56. if($options['manager'] && empty($user->origin) && !empty($user->manager) && isset($users[$user->manager])){
  57. $users[$key]->manager = $users[$user->manager];
  58. }
  59. $logins[] = $user->login;
  60. }
  61. if($options['right']){
  62. $ranksId = array();
  63. //Ajout des rangs depuis les liens user/firm/rank
  64. foreach (UserFirmRank::loadAll(array('user:IN'=>$logins ), null, null, array('*'), 1) as $firmRank) {
  65. $rank = $firmRank->join('rank');
  66. $firm = $firmRank->join('firm');
  67. $ranksId[] = $rank->id;
  68. $users[$firmRank->user]->firms[$firm->id] = $firm;
  69. if(!isset($users[$firmRank->user]->ranks[$firmRank->firm])) $users[$firmRank->user]->ranks[$firmRank->firm] = array();
  70. $users[$firmRank->user]->ranks[$firmRank->firm][$rank->id] = $rank;
  71. if(empty($users[$firmRank->user]->superadmin) && $rank->superadmin) $users[$firmRank->user]->superadmin = boolval($rank->superadmin);
  72. }
  73. //récuperation des droits ciblés sur user et rangs pour les user et rang user concernés
  74. $rights = Right::staticQuery('SELECT * FROM {{table}} WHERE
  75. (targetScope="user" AND targetUid IN ('.implode(',',array_fill(0,count($logins),'?')).')) OR (targetScope="rank" AND targetUid IN('.implode(',',array_fill(0,count($ranksId),'?')).')) ',array_merge($logins,$ranksId),true);
  76. $rightsForRank = array();
  77. foreach($rights as $right):
  78. //pour les rangs on remplis un tableau de mapping qu'on redistribuera plus tard
  79. if($right->targetScope =='rank'){
  80. if(!isset($rightsForRank[$right->targetUid])) $rightsForRank[$right->targetUid] = array();
  81. $rightsForRank[$right->targetUid][]= $right;
  82. //pour les users
  83. }else{
  84. //Pour le premier rang qui aborde cette scope on met tous les droits à false
  85. if(!isset($users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid])){
  86. $users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid] = array(
  87. 'read' => false,
  88. 'edit' => false,
  89. 'delete' => false,
  90. 'recursive' => false,
  91. 'configure' => false
  92. );
  93. }
  94. //Puis on complete uniquement les droits à true sur la scope pour chaques rangs additionnels
  95. if($right->read) $users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid]['read'] = true;
  96. if($right->edit) $users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid]['edit'] = true;
  97. if($right->delete) $users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid]['delete'] = true;
  98. if($right->configure) $users[$right->targetUid]->rights[$right->scope][$right->firm][$right->uid]['configure'] = true;
  99. }
  100. endforeach;
  101. //pour chaque user
  102. foreach($users as $key=>$user){
  103. //onrecupere les firms
  104. foreach($user->ranks as $firmrank){
  105. //on liste les rangs du user pour cette firm
  106. foreach($firmrank as $rank){
  107. //si un de ces rangs possede des droits liés
  108. if(!isset($rightsForRank[$rank->id])) continue;
  109. //pour chaque droit liées on les applique au user
  110. foreach($rightsForRank[$rank->id] as $right){
  111. //Pour le premier rang qui aborde cette scope on met tous les droits à false
  112. if(!isset($users[$key]->rights[$right->scope][$right->firm][$right->uid])){
  113. $users[$key]->rights[$right->scope][$right->firm][$right->uid] = array(
  114. 'read' => false,
  115. 'edit' => false,
  116. 'delete' => false,
  117. 'recursive' => false,
  118. 'configure' => false
  119. );
  120. }
  121. //Puis on complete uniquement les droits à true sur la scope pour chaques rangs additionnels
  122. if($right->read) $users[$key]->rights[$right->scope][$right->firm][$right->uid]['read'] = true;
  123. if($right->edit) $users[$key]->rights[$right->scope][$right->firm][$right->uid]['edit'] = true;
  124. if($right->delete) $users[$key]->rights[$right->scope][$right->firm][$right->uid]['delete'] = true;
  125. if($right->configure) $users[$key]->rights[$right->scope][$right->firm][$right->uid]['configure'] = true;
  126. }
  127. }
  128. }
  129. }
  130. }
  131. //sort des users par noms
  132. uasort($users, function($a, $b){ return strcmp($a->name, $b->name); });
  133. $_SESSION['users_'.($options['right']?'rights':'norights')] = serialize($users);
  134. }
  135. return $users;
  136. }
  137. public function __sleep(){
  138. $fields = $this->toArray();
  139. unset($fields['fullName'], $fields['initials']);
  140. return array_merge(array('superadmin','rights','groups','ranks','firms','preferences'),array_keys($fields));
  141. }
  142. //Surchage de toArray pour prendre en compte le fullName et les initials régulierement utilisé en toArray
  143. public function toArray($decoded=false) {
  144. $fields = parent::toArray($decoded);
  145. $fields['fullName'] = $this->fullName();
  146. $fields['initials'] = $this->initials();
  147. unset($fields['password']);
  148. unset($fields['token']);
  149. if(isset($fields['manager']) && is_object($fields['manager'])) $fields['manager'] = $fields['manager']->toArray();
  150. if($decoded){
  151. $fields['fullName'] = html_entity_decode($fields['fullName']);
  152. $fields['initials'] = html_entity_decode($fields['initials']);
  153. }
  154. return $fields;
  155. }
  156. // $this->rights[$right->scope][$right->uid][$right->firm]
  157. public function can($scope,$right,$uid=0,$options = array()){
  158. if($this->superadmin == 1) return true;
  159. global $myFirm;
  160. $firm = is_object($myFirm) && $myFirm->id !=0 ? $myFirm->id : 0;
  161. // Recherche dans les droits ciblé sur la firm courante
  162. if(isset($this->rights[$scope][$firm][$uid][$right]))
  163. return $this->rights[$scope][$firm][$uid][$right]==1;
  164. // Recherche dans les droits non ciblés sur une firm
  165. if(isset($this->rights[$scope][0][$uid][$right]))
  166. return $this->rights[$scope][0][$uid][$right]==1;
  167. if(isset($options['contains']) && $options['contains'] == true){
  168. if(isset($this->rights[$scope])) {
  169. $rightArray = array();
  170. if(isset($this->rights[$scope][0]))
  171. $rightArray =$this->rights[$scope][0];
  172. if(isset($this->rights[$scope][$myFirm->id]))
  173. $rightArray = $this->rights[$scope][$myFirm->id];
  174. foreach($rightArray as $uid=>$rights){
  175. if(!isset($rights[$right]) || !$rights[$right]) continue;
  176. return true;
  177. }
  178. }
  179. }
  180. return false;
  181. }
  182. //Lance les exception appropriées en fonction du droit ou des droits spécifiés
  183. // ex : User::check_access('document','configure');
  184. public static function check_access($scope,$right,$uid=0){
  185. global $myUser;
  186. if(!isset($myUser) || !is_object($myUser) || !$myUser->connected()) throw new Exception("Contrôle d'accès - Vous devez être connecté",401);
  187. if(!$myUser->can($scope,$right,$uid)) throw new Exception("Contrôle d'accès - Permissions insuffisantes ('".$scope.".".$right."')",403);
  188. }
  189. //Vérifie que l'utilisateur courant est lié a l'id du rang spécifié
  190. public function hasRank($rankId){
  191. if($this->superadmin) return true;
  192. $rankIds = array();
  193. global $myFirm;
  194. if(empty($this->ranks) || !isset($this->ranks[$myFirm->id])) return false;
  195. foreach ($this->ranks[$myFirm->id] as $rank)
  196. $rankIds[$rank->id] = true;
  197. return isset($rankIds[$rankId]);
  198. }
  199. public function preference($key=null, $value=null){
  200. global $myUser;
  201. if(!isset($key) && !isset($value)) return $this->preferences;
  202. if(isset($key) && !isset($value)) return isset($this->preferences[$key])?$this->preferences[$key]:'';
  203. if(isset($key) && isset($value)){
  204. $this->preferences[$key] = $value;
  205. $preference = UserPreference::load(array('key'=>$key,'user'=>$this->login));
  206. if(!$preference) $preference = new UserPreference();
  207. $preference->key = $key;
  208. $preference->value = $value;
  209. $preference->user = $this->login;
  210. $preference->save();
  211. if($myUser->login == $this->login) $_SESSION['currentUser'] = serialize($this);
  212. }
  213. }
  214. public function loadRanks(){
  215. //Ajout des rangs depuis les liens user/firm/rank
  216. foreach (UserFirmRank::loadAll(array('user'=>$this->login), null, null, array('*'), 1) as $firmRank) {
  217. $rank = $firmRank->join('rank');
  218. $firm = $firmRank->join('firm');
  219. $this->firms[$firm->id] = $firm;
  220. if(!isset($this->