123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 |
- <?php
- use MediaWiki\Auth\AuthenticationRequest;
- use MediaWiki\Auth\AuthenticationResponse;
- use MediaWiki\Auth\AuthManager;
- use MediaWiki\Session\SessionManager;
- /**
- * Special change to change credentials (such as the password).
- *
- * Also does most of the work for SpecialRemoveCredentials.
- */
- class SpecialChangeCredentials extends AuthManagerSpecialPage {
- protected static $allowedActions = [ AuthManager::ACTION_CHANGE ];
- protected static $messagePrefix = 'changecredentials';
- /** Change action needs user data; remove action does not */
- protected static $loadUserData = true;
- public function __construct( $name = 'ChangeCredentials' ) {
- parent::__construct( $name, 'editmyprivateinfo' );
- }
- protected function getGroupName() {
- return 'users';
- }
- public function isListed() {
- $this->loadAuth( '' );
- return (bool)$this->authRequests;
- }
- public function doesWrites() {
- return true;
- }
- protected function getDefaultAction( $subPage ) {
- return AuthManager::ACTION_CHANGE;
- }
- protected function getPreservedParams( $withToken = false ) {
- $request = $this->getRequest();
- $params = parent::getPreservedParams( $withToken );
- $params += [
- 'returnto' => $request->getVal( 'returnto' ),
- 'returntoquery' => $request->getVal( 'returntoquery' ),
- ];
- return $params;
- }
- public function execute( $subPage ) {
- $this->setHeaders();
- $this->outputHeader();
- $this->loadAuth( $subPage );
- if ( !$subPage ) {
- $this->showSubpageList();
- return;
- }
- if ( !$this->authRequests ) {
- // messages used: changecredentials-invalidsubpage, removecredentials-invalidsubpage
- $this->showSubpageList( $this->msg( static::$messagePrefix . '-invalidsubpage', $subPage ) );
- return;
- }
- $this->getOutput()->addBacklinkSubtitle( $this->getPageTitle() );
- $status = $this->trySubmit();
- if ( $status === false || !$status->isOK() ) {
- $this->displayForm( $status );
- return;
- }
- $response = $status->getValue();
- switch ( $response->status ) {
- case AuthenticationResponse::PASS:
- $this->success();
- break;
- case AuthenticationResponse::FAIL:
- $this->displayForm( Status::newFatal( $response->message ) );
- break;
- default:
- throw new LogicException( 'invalid AuthenticationResponse' );
- }
- }
- protected function loadAuth( $subPage, $authAction = null, $reset = false ) {
- parent::loadAuth( $subPage, $authAction );
- if ( $subPage ) {
- $this->authRequests = array_filter( $this->authRequests, function ( $req ) use ( $subPage ) {
- return $req->getUniqueId() === $subPage;
- } );
- if ( count( $this->authRequests ) > 1 ) {
- throw new LogicException( 'Multiple AuthenticationRequest objects with same ID!' );
- }
- }
- }
- protected function getAuthFormDescriptor( $requests, $action ) {
- if ( !static::$loadUserData ) {
- return [];
- } else {
- $descriptor = parent::getAuthFormDescriptor( $requests, $action );
- $any = false;
- foreach ( $descriptor as &$field ) {
- if ( $field['type'] === 'password' && $field['name'] !== 'retype' ) {
- $any = true;
- if ( isset( $field['cssclass'] ) ) {
- $field['cssclass'] .= ' mw-changecredentials-validate-password';
- } else {
- $field['cssclass'] = 'mw-changecredentials-validate-password';
- }
- }
- }
- if ( $any ) {
- $this->getOutput()->addModules( 'mediawiki.misc-authed-ooui' );
- }
- return $descriptor;
- }
- }
- protected function getAuthForm( array $requests, $action ) {
- $form = parent::getAuthForm( $requests, $action );
- $req = reset( $requests );
- $info = $req->describeCredentials();
- $form->addPreText(
- Html::openElement( 'dl' )
- . Html::element( 'dt', [], $this->msg( 'credentialsform-provider' )->text() )
- . Html::element( 'dd', [], $info['provider'] )
- . Html::element( 'dt', [], $this->msg( 'credentialsform-account' )->text() )
- . Html::element( 'dd', [], $info['account'] )
- . Html::closeElement( 'dl' )
- );
- // messages used: changecredentials-submit removecredentials-submit
- $form->setSubmitTextMsg( static::$messagePrefix . '-submit' );
- $form->showCancel()->setCancelTarget( $this->getReturnUrl() ?: Title::newMainPage() );
- return $form;
- }
- protected function needsSubmitButton( array $requests ) {
- // Change/remove forms show are built from a single AuthenticationRequest and do not allow
- // for redirect flow; they always need a submit button.
- return true;
- }
- public function handleFormSubmit( $data ) {
- // remove requests do not accept user input
- $requests = $this->authRequests;
- if ( static::$loadUserData ) {
- $requests = AuthenticationRequest::loadRequestsFromSubmission( $this->authRequests, $data );
- }
- $response = $this->performAuthenticationStep( $this->authAction, $requests );
- // we can't handle FAIL or similar as failure here since it might require changing the form
- return Status::newGood( $response );
- }
- /**
- * @param Message|null $error
- */
- protected function showSubpageList( $error = null ) {
- $out = $this->getOutput();
- if ( $error ) {
- $out->addHTML( $error->parse() );
- }
- $groupedRequests = [];
- foreach ( $this->authRequests as $req ) {
- $info = $req->describeCredentials();
- $groupedRequests[(string)$info['provider']][] = $req;
- }
- $linkRenderer = $this->getLinkRenderer();
- $out->addHTML( Html::openElement( 'dl' ) );
- foreach ( $groupedRequests as $group => $members ) {
- $out->addHTML( Html::element( 'dt', [], $group ) );
- foreach ( $members as $req ) {
- /** @var AuthenticationRequest $req */
- $info = $req->describeCredentials();
- $out->addHTML( Html::rawElement( 'dd', [],
- $linkRenderer->makeLink(
- $this->getPageTitle( $req->getUniqueId() ),
- $info['account']
- )
- ) );
- }
- }
- $out->addHTML( Html::closeElement( 'dl' ) );
- }
- protected function success() {
- $session = $this->getRequest()->getSession();
- $user = $this->getUser();
- $out = $this->getOutput();
- $returnUrl = $this->getReturnUrl();
- // change user token and update the session
- SessionManager::singleton()->invalidateSessionsForUser( $user );
- $session->setUser( $user );
- $session->resetId();
- if ( $returnUrl ) {
- $out->redirect( $returnUrl );
- } else {
- // messages used: changecredentials-success removecredentials-success
- $out->wrapWikiMsg( "<div class=\"successbox\">\n$1\n</div>", static::$messagePrefix
- . '-success' );
- $out->returnToMain();
- }
- }
- /**
- * @return string|null
- */
- protected function getReturnUrl() {
- $request = $this->getRequest();
- $returnTo = $request->getText( 'returnto' );
- $returnToQuery = $request->getText( 'returntoquery', '' );
- if ( !$returnTo ) {
- return null;
- }
- $title = Title::newFromText( $returnTo );
- return $title->getFullUrlForRedirect( $returnToQuery );
- }
- protected function getRequestBlacklist() {
- return $this->getConfig()->get( 'ChangeCredentialsBlacklist' );
- }
- }
|