Merge branch 'master' into fusionax
[platal.git] / classes / user.php
1 <?php
2 /***************************************************************************
3 * Copyright (C) 2003-2009 Polytechnique.org *
4 * http://opensource.polytechnique.org/ *
5 * *
6 * This program is free software; you can redistribute it and/or modify *
7 * it under the terms of the GNU General Public License as published by *
8 * the Free Software Foundation; either version 2 of the License, or *
9 * (at your option) any later version. *
10 * *
11 * This program is distributed in the hope that it will be useful, *
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
14 * GNU General Public License for more details. *
15 * *
16 * You should have received a copy of the GNU General Public License *
17 * along with this program; if not, write to the Free Software *
18 * Foundation, Inc., *
19 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA *
20 ***************************************************************************/
21
22 class User extends PlUser
23 {
24 // Additional fields (non core)
25 protected $promo = null;
26
27 public function promo()
28 {
29 return $this->promo;
30 }
31
32 // Implementation of the login to uid method.
33 protected function getLogin($login)
34 {
35 global $globals;
36
37 if (!$login) {
38 throw new UserNotFoundException();
39 }
40
41 // If $data is an integer, fetches directly the result.
42 if (is_numeric($login)) {
43 $res = XDB::query("SELECT user_id FROM auth_user_md5 WHERE user_id = {?}", $login);
44 if ($res->numRows()) {
45 return $res->fetchOneCell();
46 }
47
48 throw new UserNotFoundException();
49 }
50
51 // Checks whether $login is a valid hruid or not.
52 $res = XDB::query("SELECT user_id FROM auth_user_md5 WHERE hruid = {?}", $login);
53 if ($res->numRows()) {
54 return $res->fetchOneCell();
55 }
56
57 // From now, $login can only by an email alias, or an email redirection.
58 // If it doesn't look like a valid address, appends the plat/al's main domain.
59 $login = trim(strtolower($login));
60 if (strstr($login, '@') === false) {
61 $login = $login . '@' . $globals->mail->domain;
62 }
63
64 // Checks if $login is a valid alias on the main domains.
65 list($mbox, $fqdn) = explode('@', $login);
66 if ($fqdn == $globals->mail->domain || $fqdn == $globals->mail->domain2) {
67 $res = XDB::query("SELECT u.user_id
68 FROM auth_user_md5 AS u
69 INNER JOIN aliases AS a ON (a.id = u.user_id AND a.type IN ('alias', 'a_vie'))
70 WHERE a.alias = {?}", $mbox);
71 if ($res->numRows()) {
72 return $res->fetchOneCell();
73 }
74
75 if (preg_match('/^(.*)\.([0-9]{4})$/u', $mbox, $matches)) {
76 $res = XDB::query("SELECT u.user_id
77 FROM auth_user_md5 AS u
78 INNER JOIN aliases AS a ON (a.id = u.user_id AND a.type IN ('alias', 'a_vie'))
79 WHERE a.alias = {?} AND u.promo = {?}", $matches[1], $matches[2]);
80 if ($res->numRows() == 1) {
81 return $res->fetchOneCell();
82 }
83 }
84
85 throw new UserNotFoundException();
86 }
87
88 // Looks for $login as an email alias from the dedicated alias domain.
89 if ($fqdn == $globals->mail->alias_dom || $fqdn == $globals->mail->alias_dom2) {
90 $res = XDB::query("SELECT redirect
91 FROM virtual_redirect
92 INNER JOIN virtual USING(vid)
93 WHERE alias = {?}", $mbox . '@' . $globals->mail->alias_dom);
94 if ($redir = $res->fetchOneCell()) {
95 // We now have a valid alias, which has to be translated to an hruid.
96 list($alias, $alias_fqdn) = explode('@', $redir);
97 $res = XDB::query("SELECT u.user_id
98 FROM auth_user_md5 AS u
99 LEFT JOIN aliases AS a ON (a.id = u.user_id AND a.type IN ('alias', 'a_vie'))
100 WHERE a.alias = {?}", $alias);
101 if ($res->numRows()) {
102 return $res->fetchOneCell();
103 }
104 }
105
106 throw new UserNotFoundException();
107 }
108
109 // Otherwise, we do suppose $login is an email redirection.
110 $res = XDB::query("SELECT u.user_id
111 FROM auth_user_md5 AS u
112 LEFT JOIN emails AS e ON (e.uid = u.user_id)
113 WHERE e.email = {?}", $login);
114 if ($res->numRows() == 1) {
115 return $res->fetchOneCell();
116 }
117
118 throw new UserNotFoundException($res->fetchColumn(1));
119 }
120
121 // Implementation of the data loader.
122 protected function loadMainFields()
123 {
124 if ($this->hruid !== null && $this->forlife !== null
125 && $this->bestalias !== null && $this->display_name !== null
126 && $this->full_name !== null && $this->promo !== null && $this->perms !== null
127 && $this->gender !== null && $this->email_format !== null) {
128 return;
129 }
130
131 global $globals;
132 $res = XDB::query("SELECT u.hruid, d.promo,
133 CONCAT(af.alias, '@{$globals->mail->domain}') AS forlife,
134 CONCAT(ab.alias, '@{$globals->mail->domain}') AS bestalias,
135 CONCAT(u.prenom, ' ', IF(u.nom_usage <> '', u.nom_usage, u.nom)) AS full_name,
136 IF(u.prenom != '', u.prenom, u.nom) AS display_name,
137 FIND_IN_SET('femme', u.flags) AS gender,
138 q.core_mail_fmt AS email_format,
139 u.perms
140 FROM auth_user_md5 AS u
141 INNER JOIN profile_display AS d ON (d.pid = u.user_id)
142 LEFT JOIN auth_user_quick AS q ON (q.user_id = u.user_id)
143 LEFT JOIN aliases AS af ON (af.id = u.user_id AND af.type = 'a_vie')
144 LEFT JOIN aliases AS ab ON (ab.id = u.user_id AND FIND_IN_SET('bestalias', ab.flags))
145 WHERE u.user_id = {?}", $this->user_id);
146 $this->fillFromArray($res->fetchOneAssoc());
147 }
148
149 // Specialization of the fillFromArray method, to implement hacks to enable
150 // lazy loading of user's main properties from the session.
151 // TODO(vzanotti): remove the conversion hacks once the old codebase will
152 // stop being used actively.
153 protected function fillFromArray(array $values)
154 {
155 // It might happen that the 'user_id' field is called uid in some places
156 // (eg. in sessions), so we hard link uid to user_id to prevent useless
157 // SQL requests.
158 if (!isset($values['user_id']) && isset($values['uid'])) {
159 $values['user_id'] = $values['uid'];
160 }
161
162 // Also, if display_name and full_name are not known, but the user's
163 // surname and last name are, we can construct the former two.
164 if (isset($values['prenom']) && isset($values['nom'])) {
165 if (!isset($values['display_name'])) {
166 $values['display_name'] = ($values['prenom'] ? $values['prenom'] : $values['nom']);
167 }
168 if (!isset($values['full_name'])) {
169 $values['full_name'] = $values['prenom'] . ' ' . $values['nom'];
170 }
171 }
172
173 // We also need to convert the gender (usually named "femme"), and the
174 // email format parameter (valued "texte" instead of "text").
175 if (isset($values['femme'])) {
176 $values['gender'] = (bool) $values['femme'];
177 }
178 if (isset($values['mail_fmt'])) {
179 $values['email_format'] = $values['mail_fmt'];
180 }
181 if (isset($values['email_format'])) {
182 $values['email_format'] = ($values['email_format'] ? self::FORMAT_HTML : self::FORMAT_TEXT);
183 }
184
185 parent::fillFromArray($values);
186 }
187
188 // Specialization of the buildPerms method
189 // This function build 'generic' permissions for the user. It does not take
190 // into account page specific permissions (e.g X.net group permissions)
191 protected function buildPerms()
192 {
193 if (!is_null($this->perm_flags)) {
194 return;
195 }
196 if ($this->perms === null) {
197 $this->loadMainFields();
198 }
199 $this->perm_flags = self::makePerms($this->perms);
200 }
201
202 // Return the password of the user
203 public function password()
204 {
205 return XDB::fetchOneCell('SELECT u.password
206 FROM auth_user_md5 AS u
207 WHERE u.user_id = {?}', $this->id());
208 }
209
210 // Return permission flags for a given permission level.
211 public static function makePerms($perms)
212 {
213 $flags = new PlFlagSet();
214 if (is_null($flags) || $perms == 'disabled' || $perms == 'ext') {
215 return $flags;
216 }
217 $flags->addFlag(PERMS_USER);
218 if ($perms == 'admin') {
219 $flags->addFlag(PERMS_ADMIN);
220 }
221 return $flags;
222 }
223
224 // Implementation of the default user callback.
225 public static function _default_user_callback($login, $results)
226 {
227 $result_count = count($results);
228 if ($result_count == 0 || !S::admin()) {
229 Platal::page()->trigError("Il n'y a pas d'utilisateur avec l'identifiant : $login");
230 } else {
231 Platal::page()->trigError("Il y a $result_count utilisateurs avec cet identifiant : " . join(', ', $results));
232 }
233 }
234
235 // Implementation of the static email locality checker.
236 public static function isForeignEmailAddress($email)
237 {
238 global $globals;
239 if (strpos($email, '@') === false) {
240 return false;
241 }
242
243 list($user, $dom) = explode('@', $email);
244 return $dom != $globals->mail->domain &&
245 $dom != $globals->mail->domain2 &&
246 $dom != $globals->mail->alias_dom &&
247 $dom != $globals->mail->alias_dom2;
248 }
249 }
250
251 // vim:set et sw=4 sts=4 sws=4 foldmethod=marker enc=utf-8:
252 ?>