3 * Implements Special:ChangePassword
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.
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.
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
21 * @ingroup SpecialPage
25 * Let users recover their password.
27 * @ingroup SpecialPage
29 class SpecialChangePassword
extends FormSpecialPage
{
33 // Optional Wikitext Message to show above the password change form
34 protected $mPreTextMessage = null;
36 // label for old password input
37 protected $mOldPassMsg = null;
39 public function __construct() {
40 parent
::__construct( 'ChangePassword', 'editmyprivateinfo' );
41 $this->listed( false );
44 public function doesWrites() {
49 * Main execution point
50 * @param string|null $par
52 function execute( $par ) {
53 $this->getOutput()->disallowUserJs();
55 parent
::execute( $par );
58 protected function checkExecutePermissions( User
$user ) {
59 parent
::checkExecutePermissions( $user );
61 if ( !$this->getRequest()->wasPosted() ) {
62 $this->requireLogin( 'resetpass-no-info' );
67 * Set a message at the top of the Change Password form
69 * @param Message $msg Message to parse and add to the form header
71 public function setChangeMessage( Message
$msg ) {
72 $this->mPreTextMessage
= $msg;
76 * Set a message at the top of the Change Password form
78 * @param string $msg Message label for old/temp password field
80 public function setOldPasswordMessage( $msg ) {
81 $this->mOldPassMsg
= $msg;
84 protected function getFormFields() {
85 $user = $this->getUser();
86 $request = $this->getRequest();
88 $oldpassMsg = $this->mOldPassMsg
;
89 if ( $oldpassMsg === null ) {
90 $oldpassMsg = $user->isLoggedIn() ?
'oldpassword' : 'resetpass-temp-password';
96 'label-message' => 'username',
97 'default' => $request->getVal( 'wpName', $user->getName() ),
100 'type' => 'password',
101 'label-message' => $oldpassMsg,
103 'NewPassword' => array(
104 'type' => 'password',
105 'label-message' => 'newpassword',
108 'type' => 'password',
109 'label-message' => 'retypenew',
113 if ( !$this->getUser()->isLoggedIn() ) {
114 if ( !LoginForm
::getLoginToken() ) {
115 LoginForm
::setLoginToken();
117 $fields['LoginOnChangeToken'] = array(
119 'label' => 'Change Password Token',
120 'default' => LoginForm
::getLoginToken(),
124 $extraFields = array();
125 Hooks
::run( 'ChangePasswordForm', array( &$extraFields ) );
126 foreach ( $extraFields as $extra ) {
127 list( $name, $label, $type, $default ) = $extra;
128 $fields[$name] = array(
131 'label-message' => $label,
132 'default' => $default,
136 if ( !$user->isLoggedIn() ) {
137 $fields['Remember'] = array(
139 'label' => $this->msg( 'remembermypassword' )
141 ceil( $this->getConfig()->get( 'CookieExpiration' ) / ( 3600 * 24 ) )
143 'default' => $request->getVal( 'wpRemember' ),
150 protected function alterForm( HTMLForm
$form ) {
151 $form->setId( 'mw-resetpass-form' );
152 $form->setTableId( 'mw-resetpass-table' );
153 $form->setWrapperLegendMsg( 'resetpass_header' );
154 $form->setSubmitTextMsg(
155 $this->getUser()->isLoggedIn()
156 ?
'resetpass-submit-loggedin'
159 $form->addButton( array(
160 'name' => 'wpCancel',
161 'value' => $this->msg( 'resetpass-submit-cancel' )->text()
163 $form->setHeaderText( $this->msg( 'resetpass_text' )->parseAsBlock() );
164 if ( $this->mPreTextMessage
instanceof Message
) {
165 $form->addPreText( $this->mPreTextMessage
->parseAsBlock() );
167 $form->addHiddenFields(
168 $this->getRequest()->getValues( 'wpName', 'wpDomain', 'returnto', 'returntoquery' ) );
171 public function onSubmit( array $data ) {
174 $request = $this->getRequest();
176 if ( $request->getCheck( 'wpLoginToken' ) ) {
177 // This comes from Special:Userlogin when logging in with a temporary password
181 if ( !$this->getUser()->isLoggedIn()
182 && $request->getVal( 'wpLoginOnChangeToken' ) !== LoginForm
::getLoginToken()
184 // Potential CSRF (bug 62497)
188 if ( $request->getCheck( 'wpCancel' ) ) {
189 $returnto = $request->getVal( 'returnto' );
190 $titleObj = $returnto !== null ? Title
::newFromText( $returnto ) : null;
191 if ( !$titleObj instanceof Title
) {
192 $titleObj = Title
::newMainPage();
194 $query = $request->getVal( 'returntoquery' );
195 $this->getOutput()->redirect( $titleObj->getFullURL( $query ) );
200 $this->mUserName
= $request->getVal( 'wpName', $this->getUser()->getName() );
201 $this->mDomain
= $wgAuth->getDomain();
203 if ( !$wgAuth->allowPasswordChange() ) {
204 throw new ErrorPageError( 'changepassword', 'resetpass_forbidden' );
207 $status = $this->attemptReset( $data['Password'], $data['NewPassword'], $data['Retype'] );
212 public function onSuccess() {
213 if ( $this->getUser()->isLoggedIn() ) {
214 $this->getOutput()->wrapWikiMsg(
215 "<div class=\"successbox\">\n$1\n</div>",
216 'changepassword-success'
218 $this->getOutput()->returnToMain();
220 $request = $this->getRequest();
221 LoginForm
::setLoginToken();
222 $token = LoginForm
::getLoginToken();
224 'action' => 'submitlogin',
225 'wpName' => $this->mUserName
,
226 'wpDomain' => $this->mDomain
,
227 'wpLoginToken' => $token,
228 'wpPassword' => $request->getVal( 'wpNewPassword' ),
229 ) +
$request->getValues( 'wpRemember', 'returnto', 'returntoquery' );
230 $login = new LoginForm( new DerivativeRequest( $request, $data, true ) );
231 $login->setContext( $this->getContext() );
232 $login->execute( null );
237 * Checks the new password if it meets the requirements for passwords and set
238 * it as a current password, otherwise set the passed Status object to fatal
239 * and doesn't change anything
241 * @param string $oldpass The current (temporary) password.
242 * @param string $newpass The password to set.
243 * @param string $retype The string of the retype password field to check with newpass
246 protected function attemptReset( $oldpass, $newpass, $retype ) {
247 $isSelf = ( $this->mUserName
=== $this->getUser()->getName() );
249 $user = $this->getUser();
251 $user = User
::newFromName( $this->mUserName
);
254 if ( !$user ||
$user->isAnon() ) {
255 return Status
::newFatal( $this->msg( 'nosuchusershort', $this->mUserName
) );
258 if ( $newpass !== $retype ) {
259 Hooks
::run( 'PrefsPasswordAudit', array( $user, $newpass, 'badretype' ) );
260 return Status
::newFatal( $this->msg( 'badretype' ) );
263 $throttleCount = LoginForm
::incLoginThrottle( $this->mUserName
);
264 if ( $throttleCount === true ) {
265 $lang = $this->getLanguage();
266 $throttleInfo = $this->getConfig()->get( 'PasswordAttemptThrottle' );
267 return Status
::newFatal( $this->msg( 'changepassword-throttled' )
268 ->params( $lang->formatDuration( $throttleInfo['seconds'] ) )
272 // @todo Make these separate messages, since the message is written for both cases
273 if ( !$user->checkTemporaryPassword( $oldpass ) && !$user->checkPassword( $oldpass ) ) {
274 Hooks
::run( 'PrefsPasswordAudit', array( $user, $newpass, 'wrongpassword' ) );
275 return Status
::newFatal( $this->msg( 'resetpass-wrong-oldpass' ) );
278 // User is resetting their password to their old password
279 if ( $oldpass === $newpass ) {
280 return Status
::newFatal( $this->msg( 'resetpass-recycled' ) );
283 // Do AbortChangePassword after checking mOldpass, so we don't leak information
284 // by possibly aborting a new password before verifying the old password.
285 $abortMsg = 'resetpass-abort-generic';
286 if ( !Hooks
::run( 'AbortChangePassword', array( $user, $oldpass, $newpass, &$abortMsg ) ) ) {
287 Hooks
::run( 'PrefsPasswordAudit', array( $user, $newpass, 'abortreset' ) );
288 return Status
::newFatal( $this->msg( $abortMsg ) );
291 // Please reset throttle for successful logins, thanks!
292 if ( $throttleCount ) {
293 LoginForm
::clearLoginThrottle( $this->mUserName
);
297 $user->setPassword( $newpass );
298 Hooks
::run( 'PrefsPasswordAudit', array( $user, $newpass, 'success' ) );
299 } catch ( PasswordError
$e ) {
300 Hooks
::run( 'PrefsPasswordAudit', array( $user, $newpass, 'error' ) );
301 return Status
::newFatal( new RawMessage( $e->getMessage() ) );
305 // This is needed to keep the user connected since
306 // changing the password also modifies the user's token.
307 $remember = $this->getRequest()->getCookie( 'Token' ) !== null;
308 $user->setCookies( null, null, $remember );
310 $user->saveSettings();
311 $this->resetPasswordExpiration( $user );
312 return Status
::newGood();
315 public function requiresUnblock() {
319 protected function getGroupName() {
324 * For resetting user password expiration, until AuthManager comes along
327 private function resetPasswordExpiration( User
$user ) {
328 global $wgPasswordExpirationDays;
330 if ( $wgPasswordExpirationDays ) {
331 $newExpire = wfTimestamp(
333 time() +
( $wgPasswordExpirationDays * 24 * 3600 )
336 // Give extensions a chance to force an expiration
337 Hooks
::run( 'ResetPasswordExpiration', array( $this, &$newExpire ) );
338 $dbw = wfGetDB( DB_MASTER
);
341 array( 'user_password_expires' => $dbw->timestampOrNull( $newExpire ) ),
342 array( 'user_id' => $user->getID() ),
347 protected function getDisplayFormat() {