API: Fixes for AuthManager
[mediawiki.git] / tests / phpunit / includes / api / ApiOptionsTest.php
blob0a577c1cb6929fffadd8eca52d0215987e10402b
1 <?php
3 /**
4 * @group API
5 * @group Database
6 * @group medium
8 * @covers ApiOptions
9 */
10 class ApiOptionsTest extends MediaWikiLangTestCase {
12 /** @var PHPUnit_Framework_MockObject_MockObject */
13 private $mUserMock;
14 /** @var ApiOptions */
15 private $mTested;
16 private $mSession;
17 /** @var DerivativeContext */
18 private $mContext;
20 private static $Success = [ 'options' => 'success' ];
22 protected function setUp() {
23 parent::setUp();
25 $this->mUserMock = $this->getMockBuilder( 'User' )
26 ->disableOriginalConstructor()
27 ->getMock();
29 // Set up groups and rights
30 $this->mUserMock->expects( $this->any() )
31 ->method( 'getEffectiveGroups' )->will( $this->returnValue( [ '*', 'user' ] ) );
32 $this->mUserMock->expects( $this->any() )
33 ->method( 'isAllowed' )->will( $this->returnValue( true ) );
35 // Set up callback for User::getOptionKinds
36 $this->mUserMock->expects( $this->any() )
37 ->method( 'getOptionKinds' )->will( $this->returnCallback( [ $this, 'getOptionKinds' ] ) );
39 // No actual DB data
40 $this->mUserMock->expects( $this->any() )
41 ->method( 'getInstanceForUpdate' )->will( $this->returnValue( $this->mUserMock ) );
43 // Create a new context
44 $this->mContext = new DerivativeContext( new RequestContext() );
45 $this->mContext->getContext()->setTitle( Title::newFromText( 'Test' ) );
46 $this->mContext->setUser( $this->mUserMock );
48 $main = new ApiMain( $this->mContext );
50 // Empty session
51 $this->mSession = [];
53 $this->mTested = new ApiOptions( $main, 'options' );
55 $this->mergeMwGlobalArrayValue( 'wgHooks', [
56 'GetPreferences' => [
57 [ $this, 'hookGetPreferences' ]
59 ] );
62 public function hookGetPreferences( $user, &$preferences ) {
63 $preferences = [];
65 foreach ( [ 'name', 'willBeNull', 'willBeEmpty', 'willBeHappy' ] as $k ) {
66 $preferences[$k] = [
67 'type' => 'text',
68 'section' => 'test',
69 'label' => '&#160;',
73 $preferences['testmultiselect'] = [
74 'type' => 'multiselect',
75 'options' => [
76 'Test' => [
77 '<span dir="auto">Some HTML here for option 1</span>' => 'opt1',
78 '<span dir="auto">Some HTML here for option 2</span>' => 'opt2',
79 '<span dir="auto">Some HTML here for option 3</span>' => 'opt3',
80 '<span dir="auto">Some HTML here for option 4</span>' => 'opt4',
83 'section' => 'test',
84 'label' => '&#160;',
85 'prefix' => 'testmultiselect-',
86 'default' => [],
89 return true;
92 /**
93 * @param IContextSource $context
94 * @param array|null $options
96 * @return array
98 public function getOptionKinds( IContextSource $context, $options = null ) {
99 // Match with above.
100 $kinds = [
101 'name' => 'registered',
102 'willBeNull' => 'registered',
103 'willBeEmpty' => 'registered',
104 'willBeHappy' => 'registered',
105 'testmultiselect-opt1' => 'registered-multiselect',
106 'testmultiselect-opt2' => 'registered-multiselect',
107 'testmultiselect-opt3' => 'registered-multiselect',
108 'testmultiselect-opt4' => 'registered-multiselect',
109 'special' => 'special',
112 if ( $options === null ) {
113 return $kinds;
116 $mapping = [];
117 foreach ( $options as $key => $value ) {
118 if ( isset( $kinds[$key] ) ) {
119 $mapping[$key] = $kinds[$key];
120 } elseif ( substr( $key, 0, 7 ) === 'userjs-' ) {
121 $mapping[$key] = 'userjs';
122 } else {
123 $mapping[$key] = 'unused';
127 return $mapping;
130 private function getSampleRequest( $custom = [] ) {
131 $request = [
132 'token' => '123ABC',
133 'change' => null,
134 'optionname' => null,
135 'optionvalue' => null,
138 return array_merge( $request, $custom );
141 private function executeQuery( $request ) {
142 $this->mContext->setRequest( new FauxRequest( $request, true, $this->mSession ) );
143 $this->mTested->execute();
145 return $this->mTested->getResult()->getResultData( null, [ 'Strip' => 'all' ] );
149 * @expectedException UsageException
151 public function testNoToken() {
152 $request = $this->getSampleRequest( [ 'token' => null ] );
154 $this->executeQuery( $request );
157 public function testAnon() {
158 $this->mUserMock->expects( $this->once() )
159 ->method( 'isAnon' )
160 ->will( $this->returnValue( true ) );
162 try {
163 $request = $this->getSampleRequest();
165 $this->executeQuery( $request );
166 } catch ( UsageException $e ) {
167 $this->assertEquals( 'notloggedin', $e->getCodeString() );
168 $this->assertEquals( 'Anonymous users cannot change preferences', $e->getMessage() );
170 return;
172 $this->fail( "UsageException was not thrown" );
175 public function testNoOptionname() {
176 try {
177 $request = $this->getSampleRequest( [ 'optionvalue' => '1' ] );
179 $this->executeQuery( $request );
180 } catch ( UsageException $e ) {
181 $this->assertEquals( 'nooptionname', $e->getCodeString() );
182 $this->assertEquals( 'The optionname parameter must be set', $e->getMessage() );
184 return;
186 $this->fail( "UsageException was not thrown" );
189 public function testNoChanges() {
190 $this->mUserMock->expects( $this->never() )
191 ->method( 'resetOptions' );
193 $this->mUserMock->expects( $this->never() )
194 ->method( 'setOption' );
196 $this->mUserMock->expects( $this->never() )
197 ->method( 'saveSettings' );
199 try {
200 $request = $this->getSampleRequest();
202 $this->executeQuery( $request );
203 } catch ( UsageException $e ) {
204 $this->assertEquals( 'nochanges', $e->getCodeString() );
205 $this->assertEquals( 'No changes were requested', $e->getMessage() );
207 return;
209 $this->fail( "UsageException was not thrown" );
212 public function testReset() {
213 $this->mUserMock->expects( $this->once() )
214 ->method( 'resetOptions' )
215 ->with( $this->equalTo( [ 'all' ] ) );
217 $this->mUserMock->expects( $this->never() )
218 ->method( 'setOption' );
220 $this->mUserMock->expects( $this->once() )
221 ->method( 'saveSettings' );
223 $request = $this->getSampleRequest( [ 'reset' => '' ] );
225 $response = $this->executeQuery( $request );
227 $this->assertEquals( self::$Success, $response );
230 public function testResetKinds() {
231 $this->mUserMock->expects( $this->once() )
232 ->method( 'resetOptions' )
233 ->with( $this->equalTo( [ 'registered' ] ) );
235 $this->mUserMock->expects( $this->never() )
236 ->method( 'setOption' );
238 $this->mUserMock->expects( $this->once() )
239 ->method( 'saveSettings' );
241 $request = $this->getSampleRequest( [ 'reset' => '', 'resetkinds' => 'registered' ] );
243 $response = $this->executeQuery( $request );
245 $this->assertEquals( self::$Success, $response );
248 public function testOptionWithValue() {
249 $this->mUserMock->expects( $this->never() )
250 ->method( 'resetOptions' );
252 $this->mUserMock->expects( $this->once() )
253 ->method( 'setOption' )
254 ->with( $this->equalTo( 'name' ), $this->equalTo( 'value' ) );
256 $this->mUserMock->expects( $this->once() )
257 ->method( 'saveSettings' );
259 $request = $this->getSampleRequest( [ 'optionname' => 'name', 'optionvalue' => 'value' ] );
261 $response = $this->executeQuery( $request );
263 $this->assertEquals( self::$Success, $response );
266 public function testOptionResetValue() {
267 $this->mUserMock->expects( $this->never() )
268 ->method( 'resetOptions' );
270 $this->mUserMock->expects( $this->once() )
271 ->method( 'setOption' )
272 ->with( $this->equalTo( 'name' ), $this->identicalTo( null ) );
274 $this->mUserMock->expects( $this->once() )
275 ->method( 'saveSettings' );
277 $request = $this->getSampleRequest( [ 'optionname' => 'name' ] );
278 $response = $this->executeQuery( $request );
280 $this->assertEquals( self::$Success, $response );
283 public function testChange() {
284 $this->mUserMock->expects( $this->never() )
285 ->method( 'resetOptions' );
287 $this->mUserMock->expects( $this->at( 2 ) )
288 ->method( 'getOptions' );
290 $this->mUserMock->expects( $this->at( 5 ) )
291 ->method( 'setOption' )
292 ->with( $this->equalTo( 'willBeNull' ), $this->identicalTo( null ) );
294 $this->mUserMock->expects( $this->at( 6 ) )
295 ->method( 'getOptions' );
297 $this->mUserMock->expects( $this->at( 7 ) )
298 ->method( 'setOption' )
299 ->with( $this->equalTo( 'willBeEmpty' ), $this->equalTo( '' ) );
301 $this->mUserMock->expects( $this->at( 8 ) )
302 ->method( 'getOptions' );
304 $this->mUserMock->expects( $this->at( 9 ) )
305 ->method( 'setOption' )
306 ->with( $this->equalTo( 'willBeHappy' ), $this->equalTo( 'Happy' ) );
308 $this->mUserMock->expects( $this->once() )
309 ->method( 'saveSettings' );
311 $request = $this->getSampleRequest( [
312 'change' => 'willBeNull|willBeEmpty=|willBeHappy=Happy'
313 ] );
315 $response = $this->executeQuery( $request );
317 $this->assertEquals( self::$Success, $response );
320 public function testResetChangeOption() {
321 $this->mUserMock->expects( $this->once() )
322 ->method( 'resetOptions' );
324 $this->mUserMock->expects( $this->at( 5 ) )
325 ->method( 'getOptions' );
327 $this->mUserMock->expects( $this->at( 6 ) )
328 ->method( 'setOption' )
329 ->with( $this->equalTo( 'willBeHappy' ), $this->equalTo( 'Happy' ) );
331 $this->mUserMock->expects( $this->at( 7 ) )
332 ->method( 'getOptions' );
334 $this->mUserMock->expects( $this->at( 8 ) )
335 ->method( 'setOption' )
336 ->with( $this->equalTo( 'name' ), $this->equalTo( 'value' ) );
338 $this->mUserMock->expects( $this->once() )
339 ->method( 'saveSettings' );
341 $args = [
342 'reset' => '',
343 'change' => 'willBeHappy=Happy',
344 'optionname' => 'name',
345 'optionvalue' => 'value'
348 $response = $this->executeQuery( $this->getSampleRequest( $args ) );
350 $this->assertEquals( self::$Success, $response );
353 public function testMultiSelect() {
354 $this->mUserMock->expects( $this->never() )
355 ->method( 'resetOptions' );
357 $this->mUserMock->expects( $this->at( 4 ) )
358 ->method( 'setOption' )
359 ->with( $this->equalTo( 'testmultiselect-opt1' ), $this->identicalTo( true ) );
361 $this->mUserMock->expects( $this->at( 5 ) )
362 ->method( 'setOption' )
363 ->with( $this->equalTo( 'testmultiselect-opt2' ), $this->identicalTo( null ) );
365 $this->mUserMock->expects( $this->at( 6 ) )
366 ->method( 'setOption' )
367 ->with( $this->equalTo( 'testmultiselect-opt3' ), $this->identicalTo( false ) );
369 $this->mUserMock->expects( $this->at( 7 ) )
370 ->method( 'setOption' )
371 ->with( $this->equalTo( 'testmultiselect-opt4' ), $this->identicalTo( false ) );
373 $this->mUserMock->expects( $this->once() )
374 ->method( 'saveSettings' );
376 $request = $this->getSampleRequest( [
377 'change' => 'testmultiselect-opt1=1|testmultiselect-opt2|'
378 . 'testmultiselect-opt3=|testmultiselect-opt4=0'
379 ] );
381 $response = $this->executeQuery( $request );
383 $this->assertEquals( self::$Success, $response );
386 public function testSpecialOption() {
387 $this->mUserMock->expects( $this->never() )
388 ->method( 'resetOptions' );
390 $this->mUserMock->expects( $this->never() )
391 ->method( 'saveSettings' );
393 $request = $this->getSampleRequest( [
394 'change' => 'special=1'
395 ] );
397 $response = $this->executeQuery( $request );
399 $this->assertEquals( [
400 'options' => 'success',
401 'warnings' => [
402 'options' => [
403 'warnings' => "Validation error for 'special': cannot be set by this module"
406 ], $response );
409 public function testUnknownOption() {
410 $this->mUserMock->expects( $this->never() )
411 ->method( 'resetOptions' );
413 $this->mUserMock->expects( $this->never() )
414 ->method( 'saveSettings' );
416 $request = $this->getSampleRequest( [
417 'change' => 'unknownOption=1'
418 ] );
420 $response = $this->executeQuery( $request );
422 $this->assertEquals( [
423 'options' => 'success',
424 'warnings' => [
425 'options' => [
426 'warnings' => "Validation error for 'unknownOption': not a valid preference"
429 ], $response );
432 public function testUserjsOption() {
433 $this->mUserMock->expects( $this->never() )
434 ->method( 'resetOptions' );
436 $this->mUserMock->expects( $this->once() )
437 ->method( 'setOption' )
438 ->with( $this->equalTo( 'userjs-option' ), $this->equalTo( '1' ) );
440 $this->mUserMock->expects( $this->once() )
441 ->method( 'saveSettings' );
443 $request = $this->getSampleRequest( [
444 'change' => 'userjs-option=1'
445 ] );
447 $response = $this->executeQuery( $request );
449 $this->assertEquals( self::$Success, $response );