Merge "objectcache: add some WANObjectCache comments to set() and delete()"
[lhc/web/wiklou.git] / includes / api / ApiLogin.php
1 <?php
2 /**
3 * Copyright © 2006-2007 Yuri Astrakhan "<Firstname><Lastname>@gmail.com",
4 * Daniel Cannon (cannon dot danielc at gmail dot com)
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 along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 * http://www.gnu.org/copyleft/gpl.html
20 *
21 * @file
22 */
23
24 use MediaWiki\Auth\AuthManager;
25 use MediaWiki\Auth\AuthenticationRequest;
26 use MediaWiki\Auth\AuthenticationResponse;
27 use MediaWiki\Logger\LoggerFactory;
28
29 /**
30 * Unit to authenticate log-in attempts to the current wiki.
31 *
32 * @ingroup API
33 */
34 class ApiLogin extends ApiBase {
35
36 public function __construct( ApiMain $main, $action ) {
37 parent::__construct( $main, $action, 'lg' );
38 }
39
40 protected function getExtendedDescription() {
41 if ( $this->getConfig()->get( 'EnableBotPasswords' ) ) {
42 return 'apihelp-login-extended-description';
43 } else {
44 return 'apihelp-login-extended-description-nobotpasswords';
45 }
46 }
47
48 /**
49 * Format a message for the response
50 * @param Message|string|array $message
51 * @return string|array
52 */
53 private function formatMessage( $message ) {
54 $message = Message::newFromSpecifier( $message );
55 $errorFormatter = $this->getErrorFormatter();
56 if ( $errorFormatter instanceof ApiErrorFormatter_BackCompat ) {
57 return ApiErrorFormatter::stripMarkup(
58 $message->useDatabase( false )->inLanguage( 'en' )->text()
59 );
60 } else {
61 return $errorFormatter->formatMessage( $message );
62 }
63 }
64
65 /**
66 * Executes the log-in attempt using the parameters passed. If
67 * the log-in succeeds, it attaches a cookie to the session
68 * and outputs the user id, username, and session token. If a
69 * log-in fails, as the result of a bad password, a nonexistent
70 * user, or any other reason, the host is cached with an expiry
71 * and no log-in attempts will be accepted until that expiry
72 * is reached. The expiry is $this->mLoginThrottle.
73 */
74 public function execute() {
75 // If we're in a mode that breaks the same-origin policy, no tokens can
76 // be obtained
77 if ( $this->lacksSameOriginSecurity() ) {
78 $this->getResult()->addValue( null, 'login', [
79 'result' => 'Aborted',
80 'reason' => $this->formatMessage( 'api-login-fail-sameorigin' ),
81 ] );
82
83 return;
84 }
85
86 $this->requirePostedParameters( [ 'password', 'token' ] );
87
88 $params = $this->extractRequestParams();
89
90 $result = [];
91
92 // Make sure session is persisted
93 $session = MediaWiki\Session\SessionManager::getGlobalSession();
94 $session->persist();
95
96 // Make sure it's possible to log in
97 if ( !$session->canSetUser() ) {
98 $this->getResult()->addValue( null, 'login', [
99 'result' => 'Aborted',
100 'reason' => $this->formatMessage( [
101 'api-login-fail-badsessionprovider',
102 $session->getProvider()->describe( $this->getErrorFormatter()->getLanguage() ),
103 ] )
104 ] );
105
106 return;
107 }
108
109 $authRes = false;
110 $context = new DerivativeContext( $this->getContext() );
111 $loginType = 'N/A';
112
113 // Check login token
114 $token = $session->getToken( '', 'login' );
115 if ( $token->wasNew() || !$params['token'] ) {
116 $authRes = 'NeedToken';
117 } elseif ( !$token->match( $params['token'] ) ) {
118 $authRes = 'WrongToken';
119 }
120
121 // Try bot passwords
122 if (
123 $authRes === false && $this->getConfig()->get( 'EnableBotPasswords' ) &&
124 ( $botLoginData = BotPassword::canonicalizeLoginData( $params['name'], $params['password'] ) )
125 ) {
126 $status = BotPassword::login(
127 $botLoginData[0], $botLoginData[1], $this->getRequest()
128 );
129 if ( $status->isOK() ) {
130 $session = $status->getValue();
131 $authRes = 'Success';
132 $loginType = 'BotPassword';
133 } elseif ( !$botLoginData[2] || $status->hasMessage( 'login-throttled' ) ) {
134 $authRes = 'Failed';
135 $message = $status->getMessage();
136 LoggerFactory::getInstance( 'authentication' )->info(
137 'BotPassword login failed: ' . $status->getWikiText( false, false, 'en' )
138 );
139 }
140 }
141
142 if ( $authRes === false ) {
143 // Simplified AuthManager login, for backwards compatibility
144 $manager = AuthManager::singleton();
145 $reqs = AuthenticationRequest::loadRequestsFromSubmission(
146 $manager->getAuthenticationRequests( AuthManager::ACTION_LOGIN, $this->getUser() ),
147 [
148 'username' => $params['name'],
149 'password' => $params['password'],
150 'domain' => $params['domain'],
151 'rememberMe' => true,
152 ]
153 );
154 $res = AuthManager::singleton()->beginAuthentication( $reqs, 'null:' );
155 switch ( $res->status ) {
156 case AuthenticationResponse::PASS:
157 if ( $this->getConfig()->get( 'EnableBotPasswords' ) ) {
158 $this->addDeprecation( 'apiwarn-deprecation-login-botpw', 'main-account-login' );
159 } else {
160 $this->addDeprecation( 'apiwarn-deprecation-login-nobotpw', 'main-account-login' );
161 }
162 $authRes = 'Success';
163 $loginType = 'AuthManager';
164 break;
165
166 case AuthenticationResponse::FAIL:
167 // Hope it's not a PreAuthenticationProvider that failed...
168 $authRes = 'Failed';
169 $message = $res->message;
170 \MediaWiki\Logger\LoggerFactory::getInstance( 'authentication' )
171 ->info( __METHOD__ . ': Authentication failed: '
172 . $message->inLanguage( 'en' )->plain() );
173 break;
174
175 default:
176 \MediaWiki\Logger\LoggerFactory::getInstance( 'authentication' )
177 ->info( __METHOD__ . ': Authentication failed due to unsupported response type: '
178 . $res->status, $this->getAuthenticationResponseLogData( $res ) );
179 $authRes = 'Aborted';
180 break;
181 }
182 }
183
184 $result['result'] = $authRes;
185 switch ( $authRes ) {
186 case 'Success':
187 $user = $session->getUser();
188
189 ApiQueryInfo::resetTokenCache();
190
191 // Deprecated hook
192 $injected_html = '';
193 Hooks::run( 'UserLoginComplete', [ &$user, &$injected_html, true ] );
194
195 $result['lguserid'] = intval( $user->getId() );
196 $result['lgusername'] = $user->getName();
197 break;
198
199 case 'NeedToken':
200 $result['token'] = $token->toString();
201 $this->addDeprecation( 'apiwarn-deprecation-login-token', 'action=login&!lgtoken' );
202 break;
203
204 case 'WrongToken':
205 break;
206
207 case 'Failed':
208 $result['reason'] = $this->formatMessage( $message );
209 break;
210
211 case 'Aborted':
212 $result['reason'] = $this->formatMessage(
213 $this->getConfig()->get( 'EnableBotPasswords' )
214 ? 'api-login-fail-aborted'
215 : 'api-login-fail-aborted-nobotpw'
216 );
217 break;
218
219 default:
220 ApiBase::dieDebug( __METHOD__, "Unhandled case value: {$authRes}" );
221 }
222
223 $this->getResult()->addValue( null, 'login', $result );
224
225 if ( $loginType === 'LoginForm' && isset( LoginForm::$statusCodes[$authRes] ) ) {
226 $authRes = LoginForm::$statusCodes[$authRes];
227 }
228 LoggerFactory::getInstance( 'authevents' )->info( 'Login attempt', [
229 'event' => 'login',
230 'successful' => $authRes === 'Success',
231 'loginType' => $loginType,
232 'status' => $authRes,
233 ] );
234 }
235
236 public function isDeprecated() {
237 return !$this->getConfig()->get( 'EnableBotPasswords' );
238 }
239
240 public function mustBePosted() {
241 return true;
242 }
243
244 public function isReadMode() {
245 return false;
246 }
247
248 public function getAllowedParams() {
249 return [
250 'name' => null,
251 'password' => [
252 ApiBase::PARAM_TYPE => 'password',
253 ],
254 'domain' => null,
255 'token' => [
256 ApiBase::PARAM_TYPE => 'string',
257 ApiBase::PARAM_REQUIRED => false, // for BC
258 ApiBase::PARAM_SENSITIVE => true,
259 ApiBase::PARAM_HELP_MSG => [ 'api-help-param-token', 'login' ],
260 ],
261 ];
262 }
263
264 protected function getExamplesMessages() {
265 return [
266 'action=login&lgname=user&lgpassword=password'
267 => 'apihelp-login-example-gettoken',
268 'action=login&lgname=user&lgpassword=password&lgtoken=123ABC'
269 => 'apihelp-login-example-login',
270 ];
271 }
272
273 public function getHelpUrls() {
274 return 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Login';
275 }
276
277 /**
278 * Turns an AuthenticationResponse into a hash suitable for passing to Logger
279 * @param AuthenticationResponse $response
280 * @return array
281 */
282 protected function getAuthenticationResponseLogData( AuthenticationResponse $response ) {
283 $ret = [
284 'status' => $response->status,
285 ];
286 if ( $response->message ) {
287 $ret['message'] = $response->message->inLanguage( 'en' )->plain();
288 };
289 $reqs = [
290 'neededRequests' => $response->neededRequests,
291 'createRequest' => $response->createRequest,
292 'linkRequest' => $response->linkRequest,
293 ];
294 foreach ( $reqs as $k => $v ) {
295 if ( $v ) {
296 $v = is_array( $v ) ? $v : [ $v ];
297 $reqClasses = array_unique( array_map( 'get_class', $v ) );
298 sort( $reqClasses );
299 $ret[$k] = implode( ', ', $reqClasses );
300 }
301 }
302 return $ret;
303 }
304 }