Merge "Make Maintenance::commitTransaction() do slave wait checks"
[lhc/web/wiklou.git] / includes / specials / SpecialChangePassword.php
1 <?php
2 /**
3 * Implements Special:ChangePassword
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * http://www.gnu.org/copyleft/gpl.html
19 *
20 * @file
21 * @ingroup SpecialPage
22 */
23
24 /**
25 * Let users recover their password.
26 *
27 * @ingroup SpecialPage
28 */
29 class SpecialChangePassword extends FormSpecialPage {
30 protected $mUserName;
31 protected $mDomain;
32
33 // Optional Wikitext Message to show above the password change form
34 protected $mPreTextMessage = null;
35
36 // label for old password input
37 protected $mOldPassMsg = null;
38
39 public function __construct() {
40 parent::__construct( 'ChangePassword', 'editmyprivateinfo' );
41 $this->listed( false );
42 }
43
44 /**
45 * Main execution point
46 * @param string|null $par
47 */
48 function execute( $par ) {
49 $this->getOutput()->disallowUserJs();
50
51 parent::execute( $par );
52 }
53
54 protected function checkExecutePermissions( User $user ) {
55 parent::checkExecutePermissions( $user );
56
57 if ( !$this->getRequest()->wasPosted() ) {
58 $this->requireLogin( 'resetpass-no-info' );
59 }
60 }
61
62 /**
63 * Set a message at the top of the Change Password form
64 * @since 1.23
65 * @param Message $msg Message to parse and add to the form header
66 */
67 public function setChangeMessage( Message $msg ) {
68 $this->mPreTextMessage = $msg;
69 }
70
71 /**
72 * Set a message at the top of the Change Password form
73 * @since 1.23
74 * @param string $msg Message label for old/temp password field
75 */
76 public function setOldPasswordMessage( $msg ) {
77 $this->mOldPassMsg = $msg;
78 }
79
80 protected function getFormFields() {
81 $user = $this->getUser();
82 $request = $this->getRequest();
83
84 $oldpassMsg = $this->mOldPassMsg;
85 if ( $oldpassMsg === null ) {
86 $oldpassMsg = $user->isLoggedIn() ? 'oldpassword' : 'resetpass-temp-password';
87 }
88
89 $fields = array(
90 'Name' => array(
91 'type' => 'info',
92 'label-message' => 'username',
93 'default' => $request->getVal( 'wpName', $user->getName() ),
94 ),
95 'Password' => array(
96 'type' => 'password',
97 'label-message' => $oldpassMsg,
98 ),
99 'NewPassword' => array(
100 'type' => 'password',
101 'label-message' => 'newpassword',
102 ),
103 'Retype' => array(
104 'type' => 'password',
105 'label-message' => 'retypenew',
106 ),
107 );
108
109 if ( !$this->getUser()->isLoggedIn() ) {
110 if ( !LoginForm::getLoginToken() ) {
111 LoginForm::setLoginToken();
112 }
113 $fields['LoginOnChangeToken'] = array(
114 'type' => 'hidden',
115 'label' => 'Change Password Token',
116 'default' => LoginForm::getLoginToken(),
117 );
118 }
119
120 $extraFields = array();
121 Hooks::run( 'ChangePasswordForm', array( &$extraFields ) );
122 foreach ( $extraFields as $extra ) {
123 list( $name, $label, $type, $default ) = $extra;
124 $fields[$name] = array(
125 'type' => $type,
126 'name' => $name,
127 'label-message' => $label,
128 'default' => $default,
129 );
130 }
131
132 if ( !$user->isLoggedIn() ) {
133 $fields['Remember'] = array(
134 'type' => 'check',
135 'label' => $this->msg( 'remembermypassword' )
136 ->numParams(
137 ceil( $this->getConfig()->get( 'CookieExpiration' ) / ( 3600 * 24 ) )
138 )->text(),
139 'default' => $request->getVal( 'wpRemember' ),
140 );
141 }
142
143 return $fields;
144 }
145
146 protected function alterForm( HTMLForm $form ) {
147 $form->setId( 'mw-resetpass-form' );
148 $form->setTableId( 'mw-resetpass-table' );
149 $form->setWrapperLegendMsg( 'resetpass_header' );
150 $form->setSubmitTextMsg(
151 $this->getUser()->isLoggedIn()
152 ? 'resetpass-submit-loggedin'
153 : 'resetpass_submit'
154 );
155 $form->addButton( array(
156 'name' => 'wpCancel',
157 'value' => $this->msg( 'resetpass-submit-cancel' )->text()
158 ) );
159 $form->setHeaderText( $this->msg( 'resetpass_text' )->parseAsBlock() );
160 if ( $this->mPreTextMessage instanceof Message ) {
161 $form->addPreText( $this->mPreTextMessage->parseAsBlock() );
162 }
163 $form->addHiddenFields(
164 $this->getRequest()->getValues( 'wpName', 'wpDomain', 'returnto', 'returntoquery' ) );
165 }
166
167 public function onSubmit( array $data ) {
168 global $wgAuth;
169
170 $request = $this->getRequest();
171
172 if ( $request->getCheck( 'wpLoginToken' ) ) {
173 // This comes from Special:Userlogin when logging in with a temporary password
174 return false;
175 }
176
177 if ( !$this->getUser()->isLoggedIn()
178 && $request->getVal( 'wpLoginOnChangeToken' ) !== LoginForm::getLoginToken()
179 ) {
180 // Potential CSRF (bug 62497)
181 return false;
182 }
183
184 if ( $request->getCheck( 'wpCancel' ) ) {
185 $returnto = $request->getVal( 'returnto' );
186 $titleObj = $returnto !== null ? Title::newFromText( $returnto ) : null;
187 if ( !$titleObj instanceof Title ) {
188 $titleObj = Title::newMainPage();
189 }
190 $query = $request->getVal( 'returntoquery' );
191 $this->getOutput()->redirect( $titleObj->getFullURL( $query ) );
192
193 return true;
194 }
195
196 $this->mUserName = $request->getVal( 'wpName', $this->getUser()->getName() );
197 $this->mDomain = $wgAuth->getDomain();
198
199 if ( !$wgAuth->allowPasswordChange() ) {
200 throw new ErrorPageError( 'changepassword', 'resetpass_forbidden' );
201 }
202
203 $status = $this->attemptReset( $data['Password'], $data['NewPassword'], $data['Retype'] );
204
205 return $status;
206 }
207
208 public function onSuccess() {
209 if ( $this->getUser()->isLoggedIn() ) {
210 $this->getOutput()->wrapWikiMsg(
211 "<div class=\"successbox\">\n$1\n</div>",
212 'changepassword-success'
213 );
214 $this->getOutput()->returnToMain();
215 } else {
216 $request = $this->getRequest();
217 LoginForm::setLoginToken();
218 $token = LoginForm::getLoginToken();
219 $data = array(
220 'action' => 'submitlogin',
221 'wpName' => $this->mUserName,
222 'wpDomain' => $this->mDomain,
223 'wpLoginToken' => $token,
224 'wpPassword' => $request->getVal( 'wpNewPassword' ),
225 ) + $request->getValues( 'wpRemember', 'returnto', 'returntoquery' );
226 $login = new LoginForm( new DerivativeRequest( $request, $data, true ) );
227 $login->setContext( $this->getContext() );
228 $login->execute( null );
229 }
230 }
231
232 /**
233 * Checks the new password if it meets the requirements for passwords and set
234 * it as a current password, otherwise set the passed Status object to fatal
235 * and doesn't change anything
236 *
237 * @param string $oldpass The current (temporary) password.
238 * @param string $newpass The password to set.
239 * @param string $retype The string of the retype password field to check with newpass
240 * @return Status
241 */
242 protected function attemptReset( $oldpass, $newpass, $retype ) {
243 $isSelf = ( $this->mUserName === $this->getUser()->getName() );
244 if ( $isSelf ) {
245 $user = $this->getUser();
246 } else {
247 $user = User::newFromName( $this->mUserName );
248 }
249
250 if ( !$user || $user->isAnon() ) {
251 return Status::newFatal( $this->msg( 'nosuchusershort', $this->mUserName ) );
252 }
253
254 if ( $newpass !== $retype ) {
255 Hooks::run( 'PrefsPasswordAudit', array( $user, $newpass, 'badretype' ) );
256 return Status::newFatal( $this->msg( 'badretype' ) );
257 }
258
259 $throttleCount = LoginForm::incLoginThrottle( $this->mUserName );
260 if ( $throttleCount === true ) {
261 $lang = $this->getLanguage();
262 $throttleInfo = $this->getConfig()->get( 'PasswordAttemptThrottle' );
263 return Status::newFatal( $this->msg( 'changepassword-throttled' )
264 ->params( $lang->formatDuration( $throttleInfo['seconds'] ) )
265 );
266 }
267
268 // @todo Make these separate messages, since the message is written for both cases
269 if ( !$user->checkTemporaryPassword( $oldpass ) && !$user->checkPassword( $oldpass ) ) {
270 Hooks::run( 'PrefsPasswordAudit', array( $user, $newpass, 'wrongpassword' ) );
271 return Status::newFatal( $this->msg( 'resetpass-wrong-oldpass' ) );
272 }
273
274 // User is resetting their password to their old password
275 if ( $oldpass === $newpass ) {
276 return Status::newFatal( $this->msg( 'resetpass-recycled' ) );
277 }
278
279 // Do AbortChangePassword after checking mOldpass, so we don't leak information
280 // by possibly aborting a new password before verifying the old password.
281 $abortMsg = 'resetpass-abort-generic';
282 if ( !Hooks::run( 'AbortChangePassword', array( $user, $oldpass, $newpass, &$abortMsg ) ) ) {
283 Hooks::run( 'PrefsPasswordAudit', array( $user, $newpass, 'abortreset' ) );
284 return Status::newFatal( $this->msg( $abortMsg ) );
285 }
286
287 // Please reset throttle for successful logins, thanks!
288 if ( $throttleCount ) {
289 LoginForm::clearLoginThrottle( $this->mUserName );
290 }
291
292 try {
293 $user->setPassword( $newpass );
294 Hooks::run( 'PrefsPasswordAudit', array( $user, $newpass, 'success' ) );
295 } catch ( PasswordError $e ) {
296 Hooks::run( 'PrefsPasswordAudit', array( $user, $newpass, 'error' ) );
297 return Status::newFatal( new RawMessage( $e->getMessage() ) );
298 }
299
300 if ( $isSelf ) {
301 // This is needed to keep the user connected since
302 // changing the password also modifies the user's token.
303 $remember = $this->getRequest()->getCookie( 'Token' ) !== null;
304 $user->setCookies( null, null, $remember );
305 }
306 $user->saveSettings();
307 $this->resetPasswordExpiration( $user );
308 return Status::newGood();
309 }
310
311 public function requiresUnblock() {
312 return false;
313 }
314
315 protected function getGroupName() {
316 return 'users';
317 }
318
319 /**
320 * For resetting user password expiration, until AuthManager comes along
321 * @param User $user
322 */
323 private function resetPasswordExpiration( User $user ) {
324 global $wgPasswordExpirationDays;
325 $newExpire = null;
326 if ( $wgPasswordExpirationDays ) {
327 $newExpire = wfTimestamp(
328 TS_MW,
329 time() + ( $wgPasswordExpirationDays * 24 * 3600 )
330 );
331 }
332 // Give extensions a chance to force an expiration
333 Hooks::run( 'ResetPasswordExpiration', array( $this, &$newExpire ) );
334 $dbw = wfGetDB( DB_MASTER );
335 $dbw->update(
336 'user',
337 array( 'user_password_expires' => $dbw->timestampOrNull( $newExpire ) ),
338 array( 'user_id' => $user->getID() ),
339 __METHOD__
340 );
341 }
342
343 protected function getDisplayFormat() {
344 return 'ooui';
345 }
346 }