API: Fixes for AuthManager
[mediawiki.git] / tests / phpunit / includes / api / ApiResultTest.php
blob7c0063d4045be00daffea26a291c29f9cfa8959d
1 <?php
3 /**
4 * @covers ApiResult
5 * @group API
6 */
7 class ApiResultTest extends MediaWikiTestCase {
9 /**
10 * @covers ApiResult
12 public function testStaticDataMethods() {
13 $arr = [];
15 ApiResult::setValue( $arr, 'setValue', '1' );
17 ApiResult::setValue( $arr, null, 'unnamed 1' );
18 ApiResult::setValue( $arr, null, 'unnamed 2' );
20 ApiResult::setValue( $arr, 'deleteValue', '2' );
21 ApiResult::unsetValue( $arr, 'deleteValue' );
23 ApiResult::setContentValue( $arr, 'setContentValue', '3' );
25 $this->assertSame( [
26 'setValue' => '1',
27 'unnamed 1',
28 'unnamed 2',
29 ApiResult::META_CONTENT => 'setContentValue',
30 'setContentValue' => '3',
31 ], $arr );
33 try {
34 ApiResult::setValue( $arr, 'setValue', '99' );
35 $this->fail( 'Expected exception not thrown' );
36 } catch ( RuntimeException $ex ) {
37 $this->assertSame(
38 'Attempting to add element setValue=99, existing value is 1',
39 $ex->getMessage(),
40 'Expected exception'
44 try {
45 ApiResult::setContentValue( $arr, 'setContentValue2', '99' );
46 $this->fail( 'Expected exception not thrown' );
47 } catch ( RuntimeException $ex ) {
48 $this->assertSame(
49 'Attempting to set content element as setContentValue2 when setContentValue ' .
50 'is already set as the content element',
51 $ex->getMessage(),
52 'Expected exception'
56 ApiResult::setValue( $arr, 'setValue', '99', ApiResult::OVERRIDE );
57 $this->assertSame( '99', $arr['setValue'] );
59 ApiResult::setContentValue( $arr, 'setContentValue2', '99', ApiResult::OVERRIDE );
60 $this->assertSame( 'setContentValue2', $arr[ApiResult::META_CONTENT] );
62 $arr = [ 'foo' => 1, 'bar' => 1 ];
63 ApiResult::setValue( $arr, 'top', '2', ApiResult::ADD_ON_TOP );
64 ApiResult::setValue( $arr, null, '2', ApiResult::ADD_ON_TOP );
65 ApiResult::setValue( $arr, 'bottom', '2' );
66 ApiResult::setValue( $arr, 'foo', '2', ApiResult::OVERRIDE );
67 ApiResult::setValue( $arr, 'bar', '2', ApiResult::OVERRIDE | ApiResult::ADD_ON_TOP );
68 $this->assertSame( [ 0, 'top', 'foo', 'bar', 'bottom' ], array_keys( $arr ) );
70 $arr = [];
71 ApiResult::setValue( $arr, 'sub', [ 'foo' => 1 ] );
72 ApiResult::setValue( $arr, 'sub', [ 'bar' => 1 ] );
73 $this->assertSame( [ 'sub' => [ 'foo' => 1, 'bar' => 1 ] ], $arr );
75 try {
76 ApiResult::setValue( $arr, 'sub', [ 'foo' => 2, 'baz' => 2 ] );
77 $this->fail( 'Expected exception not thrown' );
78 } catch ( RuntimeException $ex ) {
79 $this->assertSame(
80 'Conflicting keys (foo) when attempting to merge element sub',
81 $ex->getMessage(),
82 'Expected exception'
86 $arr = [];
87 $title = Title::newFromText( "MediaWiki:Foobar" );
88 $obj = new stdClass;
89 $obj->foo = 1;
90 $obj->bar = 2;
91 ApiResult::setValue( $arr, 'title', $title );
92 ApiResult::setValue( $arr, 'obj', $obj );
93 $this->assertSame( [
94 'title' => (string)$title,
95 'obj' => [ 'foo' => 1, 'bar' => 2, ApiResult::META_TYPE => 'assoc' ],
96 ], $arr );
98 $fh = tmpfile();
99 try {
100 ApiResult::setValue( $arr, 'file', $fh );
101 $this->fail( 'Expected exception not thrown' );
102 } catch ( InvalidArgumentException $ex ) {
103 $this->assertSame(
104 'Cannot add resource(stream) to ApiResult',
105 $ex->getMessage(),
106 'Expected exception'
109 try {
110 ApiResult::setValue( $arr, null, $fh );
111 $this->fail( 'Expected exception not thrown' );
112 } catch ( InvalidArgumentException $ex ) {
113 $this->assertSame(
114 'Cannot add resource(stream) to ApiResult',
115 $ex->getMessage(),
116 'Expected exception'
119 try {
120 $obj->file = $fh;
121 ApiResult::setValue( $arr, 'sub', $obj );
122 $this->fail( 'Expected exception not thrown' );
123 } catch ( InvalidArgumentException $ex ) {
124 $this->assertSame(
125 'Cannot add resource(stream) to ApiResult',
126 $ex->getMessage(),
127 'Expected exception'
130 try {
131 $obj->file = $fh;
132 ApiResult::setValue( $arr, null, $obj );
133 $this->fail( 'Expected exception not thrown' );
134 } catch ( InvalidArgumentException $ex ) {
135 $this->assertSame(
136 'Cannot add resource(stream) to ApiResult',
137 $ex->getMessage(),
138 'Expected exception'
141 fclose( $fh );
143 try {
144 ApiResult::setValue( $arr, 'inf', INF );
145 $this->fail( 'Expected exception not thrown' );
146 } catch ( InvalidArgumentException $ex ) {
147 $this->assertSame(
148 'Cannot add non-finite floats to ApiResult',
149 $ex->getMessage(),
150 'Expected exception'
153 try {
154 ApiResult::setValue( $arr, null, INF );
155 $this->fail( 'Expected exception not thrown' );
156 } catch ( InvalidArgumentException $ex ) {
157 $this->assertSame(
158 'Cannot add non-finite floats to ApiResult',
159 $ex->getMessage(),
160 'Expected exception'
163 try {
164 ApiResult::setValue( $arr, 'nan', NAN );
165 $this->fail( 'Expected exception not thrown' );
166 } catch ( InvalidArgumentException $ex ) {
167 $this->assertSame(
168 'Cannot add non-finite floats to ApiResult',
169 $ex->getMessage(),
170 'Expected exception'
173 try {
174 ApiResult::setValue( $arr, null, NAN );
175 $this->fail( 'Expected exception not thrown' );
176 } catch ( InvalidArgumentException $ex ) {
177 $this->assertSame(
178 'Cannot add non-finite floats to ApiResult',
179 $ex->getMessage(),
180 'Expected exception'
184 ApiResult::setValue( $arr, null, NAN, ApiResult::NO_VALIDATE );
186 try {
187 ApiResult::setValue( $arr, null, NAN, ApiResult::NO_SIZE_CHECK );
188 $this->fail( 'Expected exception not thrown' );
189 } catch ( InvalidArgumentException $ex ) {
190 $this->assertSame(
191 'Cannot add non-finite floats to ApiResult',
192 $ex->getMessage(),
193 'Expected exception'
197 $arr = [];
198 $result2 = new ApiResult( 8388608 );
199 $result2->addValue( null, 'foo', 'bar' );
200 ApiResult::setValue( $arr, 'baz', $result2 );
201 $this->assertSame( [
202 'baz' => [
203 ApiResult::META_TYPE => 'assoc',
204 'foo' => 'bar',
206 ], $arr );
208 $arr = [];
209 ApiResult::setValue( $arr, 'foo', "foo\x80bar" );
210 ApiResult::setValue( $arr, 'bar', "a\xcc\x81" );
211 ApiResult::setValue( $arr, 'baz', 74 );
212 ApiResult::setValue( $arr, null, "foo\x80bar" );
213 ApiResult::setValue( $arr, null, "a\xcc\x81" );
214 $this->assertSame( [
215 'foo' => "foo\xef\xbf\xbdbar",
216 'bar' => "\xc3\xa1",
217 'baz' => 74,
218 0 => "foo\xef\xbf\xbdbar",
219 1 => "\xc3\xa1",
220 ], $arr );
222 $obj = new stdClass;
223 $obj->{'1'} = 'one';
224 $arr = [];
225 ApiResult::setValue( $arr, 'foo', $obj );
226 $this->assertSame( [
227 'foo' => [
228 1 => 'one',
229 ApiResult::META_TYPE => 'assoc',
231 ], $arr );
235 * @covers ApiResult
237 public function testInstanceDataMethods() {
238 $result = new ApiResult( 8388608 );
240 $result->addValue( null, 'setValue', '1' );
242 $result->addValue( null, null, 'unnamed 1' );
243 $result->addValue( null, null, 'unnamed 2' );
245 $result->addValue( null, 'deleteValue', '2' );
246 $result->removeValue( null, 'deleteValue' );
248 $result->addValue( [ 'a', 'b' ], 'deleteValue', '3' );
249 $result->removeValue( [ 'a', 'b', 'deleteValue' ], null, '3' );
251 $result->addContentValue( null, 'setContentValue', '3' );
253 $this->assertSame( [
254 'setValue' => '1',
255 'unnamed 1',
256 'unnamed 2',
257 'a' => [ 'b' => [] ],
258 'setContentValue' => '3',
259 ApiResult::META_TYPE => 'assoc',
260 ApiResult::META_CONTENT => 'setContentValue',
261 ], $result->getResultData() );
262 $this->assertSame( 20, $result->getSize() );
264 try {
265 $result->addValue( null, 'setValue', '99' );
266 $this->fail( 'Expected exception not thrown' );
267 } catch ( RuntimeException $ex ) {
268 $this->assertSame(
269 'Attempting to add element setValue=99, existing value is 1',
270 $ex->getMessage(),
271 'Expected exception'
275 try {
276 $result->addContentValue( null, 'setContentValue2', '99' );
277 $this->fail( 'Expected exception not thrown' );
278 } catch ( RuntimeException $ex ) {
279 $this->assertSame(
280 'Attempting to set content element as setContentValue2 when setContentValue ' .
281 'is already set as the content element',
282 $ex->getMessage(),
283 'Expected exception'
287 $result->addValue( null, 'setValue', '99', ApiResult::OVERRIDE );
288 $this->assertSame( '99', $result->getResultData( [ 'setValue' ] ) );
290 $result->addContentValue( null, 'setContentValue2', '99', ApiResult::OVERRIDE );
291 $this->assertSame( 'setContentValue2',
292 $result->getResultData( [ ApiResult::META_CONTENT ] ) );
294 $result->reset();
295 $this->assertSame( [
296 ApiResult::META_TYPE => 'assoc',
297 ], $result->getResultData() );
298 $this->assertSame( 0, $result->getSize() );
300 $result->addValue( null, 'foo', 1 );
301 $result->addValue( null, 'bar', 1 );
302 $result->addValue( null, 'top', '2', ApiResult::ADD_ON_TOP );
303 $result->addValue( null, null, '2', ApiResult::ADD_ON_TOP );
304 $result->addValue( null, 'bottom', '2' );
305 $result->addValue( null, 'foo', '2', ApiResult::OVERRIDE );
306 $result->addValue( null, 'bar', '2', ApiResult::OVERRIDE | ApiResult::ADD_ON_TOP );
307 $this->assertSame( [ 0, 'top', 'foo', 'bar', 'bottom', ApiResult::META_TYPE ],
308 array_keys( $result->getResultData() ) );
310 $result->reset();
311 $result->addValue( null, 'foo', [ 'bar' => 1 ] );
312 $result->addValue( [ 'foo', 'top' ], 'x', 2, ApiResult::ADD_ON_TOP );
313 $result->addValue( [ 'foo', 'bottom' ], 'x', 2 );
314 $this->assertSame( [ 'top', 'bar', 'bottom' ],
315 array_keys( $result->getResultData( [ 'foo' ] ) ) );
317 $result->reset();
318 $result->addValue( null, 'sub', [ 'foo' => 1 ] );
319 $result->addValue( null, 'sub', [ 'bar' => 1 ] );
320 $this->assertSame( [
321 'sub' => [ 'foo' => 1, 'bar' => 1 ],
322 ApiResult::META_TYPE => 'assoc',
323 ], $result->getResultData() );
325 try {
326 $result->addValue( null, 'sub', [ 'foo' => 2, 'baz' => 2 ] );
327 $this->fail( 'Expected exception not thrown' );
328 } catch ( RuntimeException $ex ) {
329 $this->assertSame(
330 'Conflicting keys (foo) when attempting to merge element sub',
331 $ex->getMessage(),
332 'Expected exception'
336 $result->reset();
337 $title = Title::newFromText( "MediaWiki:Foobar" );
338 $obj = new stdClass;
339 $obj->foo = 1;
340 $obj->bar = 2;
341 $result->addValue( null, 'title', $title );
342 $result->addValue( null, 'obj', $obj );
343 $this->assertSame( [
344 'title' => (string)$title,
345 'obj' => [ 'foo' => 1, 'bar' => 2, ApiResult::META_TYPE => 'assoc' ],
346 ApiResult::META_TYPE => 'assoc',
347 ], $result->getResultData() );
349 $fh = tmpfile();
350 try {
351 $result->addValue( null, 'file', $fh );
352 $this->fail( 'Expected exception not thrown' );
353 } catch ( InvalidArgumentException $ex ) {
354 $this->assertSame(
355 'Cannot add resource(stream) to ApiResult',
356 $ex->getMessage(),
357 'Expected exception'
360 try {
361 $result->addValue( null, null, $fh );
362 $this->fail( 'Expected exception not thrown' );
363 } catch ( InvalidArgumentException $ex ) {
364 $this->assertSame(
365 'Cannot add resource(stream) to ApiResult',
366 $ex->getMessage(),
367 'Expected exception'
370 try {
371 $obj->file = $fh;
372 $result->addValue( null, 'sub', $obj );
373 $this->fail( 'Expected exception not thrown' );
374 } catch ( InvalidArgumentException $ex ) {
375 $this->assertSame(
376 'Cannot add resource(stream) to ApiResult',
377 $ex->getMessage(),
378 'Expected exception'
381 try {
382 $obj->file = $fh;
383 $result->addValue( null, null, $obj );
384 $this->fail( 'Expected exception not thrown' );
385 } catch ( InvalidArgumentException $ex ) {
386 $this->assertSame(
387 'Cannot add resource(stream) to ApiResult',
388 $ex->getMessage(),
389 'Expected exception'
392 fclose( $fh );
394 try {
395 $result->addValue( null, 'inf', INF );
396 $this->fail( 'Expected exception not thrown' );
397 } catch ( InvalidArgumentException $ex ) {
398 $this->assertSame(
399 'Cannot add non-finite floats to ApiResult',
400 $ex->getMessage(),
401 'Expected exception'
404 try {
405 $result->addValue( null, null, INF );
406 $this->fail( 'Expected exception not thrown' );
407 } catch ( InvalidArgumentException $ex ) {
408 $this->assertSame(
409 'Cannot add non-finite floats to ApiResult',
410 $ex->getMessage(),
411 'Expected exception'
414 try {
415 $result->addValue( null, 'nan', NAN );
416 $this->fail( 'Expected exception not thrown' );
417 } catch ( InvalidArgumentException $ex ) {
418 $this->assertSame(
419 'Cannot add non-finite floats to ApiResult',
420 $ex->getMessage(),
421 'Expected exception'
424 try {
425 $result->addValue( null, null, NAN );
426 $this->fail( 'Expected exception not thrown' );
427 } catch ( InvalidArgumentException $ex ) {
428 $this->assertSame(
429 'Cannot add non-finite floats to ApiResult',
430 $ex->getMessage(),
431 'Expected exception'
435 $result->addValue( null, null, NAN, ApiResult::NO_VALIDATE );
437 try {
438 $result->addValue( null, null, NAN, ApiResult::NO_SIZE_CHECK );
439 $this->fail( 'Expected exception not thrown' );
440 } catch ( InvalidArgumentException $ex ) {
441 $this->assertSame(
442 'Cannot add non-finite floats to ApiResult',
443 $ex->getMessage(),
444 'Expected exception'
448 $result->reset();
449 $result->addParsedLimit( 'foo', 12 );
450 $this->assertSame( [
451 'limits' => [ 'foo' => 12 ],
452 ApiResult::META_TYPE => 'assoc',
453 ], $result->getResultData() );
454 $result->addParsedLimit( 'foo', 13 );
455 $this->assertSame( [
456 'limits' => [ 'foo' => 13 ],
457 ApiResult::META_TYPE => 'assoc',
458 ], $result->getResultData() );
459 $this->assertSame( null, $result->getResultData( [ 'foo', 'bar', 'baz' ] ) );
460 $this->assertSame( 13, $result->getResultData( [ 'limits', 'foo' ] ) );
461 try {
462 $result->getResultData( [ 'limits', 'foo', 'bar' ] );
463 $this->fail( 'Expected exception not thrown' );
464 } catch ( InvalidArgumentException $ex ) {
465 $this->assertSame(
466 'Path limits.foo is not an array',
467 $ex->getMessage(),
468 'Expected exception'
472 // Add two values and some metadata, but ensure metadata is not counted
473 $result = new ApiResult( 100 );
474 $obj = [ 'attr' => '12345' ];
475 ApiResult::setContentValue( $obj, 'content', '1234567890' );
476 $this->assertTrue( $result->addValue( null, 'foo', $obj ) );
477 $this->assertSame( 15, $result->getSize() );
479 $result = new ApiResult( 10 );
480 $formatter = new ApiErrorFormatter( $result, Language::factory( 'en' ), 'none', false );
481 $result->setErrorFormatter( $formatter );
482 $this->assertFalse( $result->addValue( null, 'foo', '12345678901' ) );
483 $this->assertTrue( $result->addValue( null, 'foo', '12345678901', ApiResult::NO_SIZE_CHECK ) );
484 $this->assertSame( 0, $result->getSize() );
485 $result->reset();
486 $this->assertTrue( $result->addValue( null, 'foo', '1234567890' ) );
487 $this->assertFalse( $result->addValue( null, 'foo', '1' ) );
488 $result->removeValue( null, 'foo' );
489 $this->assertTrue( $result->addValue( null, 'foo', '1' ) );
491 $result = new ApiResult( 10 );
492 $obj = new ApiResultTestSerializableObject( 'ok' );
493 $obj->foobar = 'foobaz';
494 $this->assertTrue( $result->addValue( null, 'foo', $obj ) );
495 $this->assertSame( 2, $result->getSize() );
497 $result = new ApiResult( 8388608 );
498 $result2 = new ApiResult( 8388608 );
499 $result2->addValue( null, 'foo', 'bar' );
500 $result->addValue( null, 'baz', $result2 );
501 $this->assertSame( [
502 'baz' => [
503 'foo' => 'bar',
504 ApiResult::META_TYPE => 'assoc',
506 ApiResult::META_TYPE => 'assoc',
507 ], $result->getResultData() );
509 $result = new ApiResult( 8388608 );
510 $result->addValue( null, 'foo', "foo\x80bar" );
511 $result->addValue( null, 'bar', "a\xcc\x81" );
512 $result->addValue( null, 'baz', 74 );
513 $result->addValue( null, null, "foo\x80bar" );
514 $result->addValue( null, null, "a\xcc\x81" );
515 $this->assertSame( [
516 'foo' => "foo\xef\xbf\xbdbar",
517 'bar' => "\xc3\xa1",
518 'baz' => 74,
519 0 => "foo\xef\xbf\xbdbar",
520 1 => "\xc3\xa1",
521 ApiResult::META_TYPE => 'assoc',
522 ], $result->getResultData() );
524 $result = new ApiResult( 8388608 );
525 $obj = new stdClass;
526 $obj->{'1'} = 'one';
527 $arr = [];
528 $result->addValue( $arr, 'foo', $obj );
529 $this->assertSame( [
530 'foo' => [
531 1 => 'one',
532 ApiResult::META_TYPE => 'assoc',
534 ApiResult::META_TYPE => 'assoc',
535 ], $result->getResultData() );
539 * @covers ApiResult
541 public function testMetadata() {
542 $arr = [ 'foo' => [ 'bar' => [] ] ];
543 $result = new ApiResult( 8388608 );
544 $result->addValue( null, 'foo', [ 'bar' => [] ] );
546 $expect = [
547 'foo' => [
548 'bar' => [
549 ApiResult::META_INDEXED_TAG_NAME => 'ritn',
550 ApiResult::META_TYPE => 'default',
552 ApiResult::META_INDEXED_TAG_NAME => 'ritn',
553 ApiResult::META_TYPE => 'default',
555 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
556 ApiResult::META_INDEXED_TAG_NAME => 'itn',
557 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar' ],
558 ApiResult::META_TYPE => 'array',
561 ApiResult::setSubelementsList( $arr, 'foo' );
562 ApiResult::setSubelementsList( $arr, [ 'bar', 'baz' ] );
563 ApiResult::unsetSubelementsList( $arr, 'baz' );
564 ApiResult::setIndexedTagNameRecursive( $arr, 'ritn' );
565 ApiResult::setIndexedTagName( $arr, 'itn' );
566 ApiResult::setPreserveKeysList( $arr, 'foo' );
567 ApiResult::setPreserveKeysList( $arr, [ 'bar', 'baz' ] );
568 ApiResult::unsetPreserveKeysList( $arr, 'baz' );
569 ApiResult::setArrayTypeRecursive( $arr, 'default' );
570 ApiResult::setArrayType( $arr, 'array' );
571 $this->assertSame( $expect, $arr );
573 $result->addSubelementsList( null, 'foo' );
574 $result->addSubelementsList( null, [ 'bar', 'baz' ] );
575 $result->removeSubelementsList( null, 'baz' );
576 $result->addIndexedTagNameRecursive( null, 'ritn' );
577 $result->addIndexedTagName( null, 'itn' );
578 $result->addPreserveKeysList( null, 'foo' );
579 $result->addPreserveKeysList( null, [ 'bar', 'baz' ] );
580 $result->removePreserveKeysList( null, 'baz' );
581 $result->addArrayTypeRecursive( null, 'default' );
582 $result->addArrayType( null, 'array' );
583 $this->assertEquals( $expect, $result->getResultData() );
585 $arr = [ 'foo' => [ 'bar' => [] ] ];
586 $expect = [
587 'foo' => [
588 'bar' => [
589 ApiResult::META_TYPE => 'kvp',
590 ApiResult::META_KVP_KEY_NAME => 'key',
592 ApiResult::META_TYPE => 'kvp',
593 ApiResult::META_KVP_KEY_NAME => 'key',
595 ApiResult::META_TYPE => 'BCkvp',
596 ApiResult::META_KVP_KEY_NAME => 'bc',
598 ApiResult::setArrayTypeRecursive( $arr, 'kvp', 'key' );
599 ApiResult::setArrayType( $arr, 'BCkvp', 'bc' );
600 $this->assertSame( $expect, $arr );
604 * @covers ApiResult
606 public function testUtilityFunctions() {
607 $arr = [
608 'foo' => [
609 'bar' => [ '_dummy' => 'foobaz' ],
610 'bar2' => (object)[ '_dummy' => 'foobaz' ],
611 'x' => 'ok',
612 '_dummy' => 'foobaz',
614 'foo2' => (object)[
615 'bar' => [ '_dummy' => 'foobaz' ],
616 'bar2' => (object)[ '_dummy' => 'foobaz' ],
617 'x' => 'ok',
618 '_dummy' => 'foobaz',
620 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
621 ApiResult::META_INDEXED_TAG_NAME => 'itn',
622 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
623 ApiResult::META_TYPE => 'array',
624 '_dummy' => 'foobaz',
625 '_dummy2' => 'foobaz!',
627 $this->assertEquals( [
628 'foo' => [
629 'bar' => [],
630 'bar2' => (object)[],
631 'x' => 'ok',
633 'foo2' => (object)[
634 'bar' => [],
635 'bar2' => (object)[],
636 'x' => 'ok',
638 '_dummy2' => 'foobaz!',
639 ], ApiResult::stripMetadata( $arr ), 'ApiResult::stripMetadata' );
641 $metadata = [];
642 $data = ApiResult::stripMetadataNonRecursive( $arr, $metadata );
643 $this->assertEquals( [
644 'foo' => [
645 'bar' => [ '_dummy' => 'foobaz' ],
646 'bar2' => (object)[ '_dummy' => 'foobaz' ],
647 'x' => 'ok',
648 '_dummy' => 'foobaz',
650 'foo2' => (object)[
651 'bar' => [ '_dummy' => 'foobaz' ],
652 'bar2' => (object)[ '_dummy' => 'foobaz' ],
653 'x' => 'ok',
654 '_dummy' => 'foobaz',
656 '_dummy2' => 'foobaz!',
657 ], $data, 'ApiResult::stripMetadataNonRecursive ($data)' );
658 $this->assertEquals( [
659 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
660 ApiResult::META_INDEXED_TAG_NAME => 'itn',
661 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
662 ApiResult::META_TYPE => 'array',
663 '_dummy' => 'foobaz',
664 ], $metadata, 'ApiResult::stripMetadataNonRecursive ($metadata)' );
666 $metadata = null;
667 $data = ApiResult::stripMetadataNonRecursive( (object)$arr, $metadata );
668 $this->assertEquals( (object)[
669 'foo' => [
670 'bar' => [ '_dummy' => 'foobaz' ],
671 'bar2' => (object)[ '_dummy' => 'foobaz' ],
672 'x' => 'ok',
673 '_dummy' => 'foobaz',
675 'foo2' => (object)[
676 'bar' => [ '_dummy' => 'foobaz' ],
677 'bar2' => (object)[ '_dummy' => 'foobaz' ],
678 'x' => 'ok',
679 '_dummy' => 'foobaz',
681 '_dummy2' => 'foobaz!',
682 ], $data, 'ApiResult::stripMetadataNonRecursive on object ($data)' );
683 $this->assertEquals( [
684 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
685 ApiResult::META_INDEXED_TAG_NAME => 'itn',
686 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
687 ApiResult::META_TYPE => 'array',
688 '_dummy' => 'foobaz',
689 ], $metadata, 'ApiResult::stripMetadataNonRecursive on object ($metadata)' );
693 * @covers ApiResult
694 * @dataProvider provideTransformations
695 * @param string $label
696 * @param array $input
697 * @param array $transforms
698 * @param array|Exception $expect
700 public function testTransformations( $label, $input, $transforms, $expect ) {
701 $result = new ApiResult( false );
702 $result->addValue( null, 'test', $input );
704 if ( $expect instanceof Exception ) {
705 try {
706 $output = $result->getResultData( 'test', $transforms );
707 $this->fail( 'Expected exception not thrown', $label );
708 } catch ( Exception $ex ) {
709 $this->assertEquals( $ex, $expect, $label );
711 } else {
712 $output = $result->getResultData( 'test', $transforms );
713 $this->assertEquals( $expect, $output, $label );
717 public function provideTransformations() {
718 $kvp = function ( $keyKey, $key, $valKey, $value ) {
719 return [
720 $keyKey => $key,
721 $valKey => $value,
722 ApiResult::META_PRESERVE_KEYS => [ $keyKey ],
723 ApiResult::META_CONTENT => $valKey,
724 ApiResult::META_TYPE => 'assoc',
727 $typeArr = [
728 'defaultArray' => [ 2 => 'a', 0 => 'b', 1 => 'c' ],
729 'defaultAssoc' => [ 'x' => 'a', 1 => 'b', 0 => 'c' ],
730 'defaultAssoc2' => [ 2 => 'a', 3 => 'b', 0 => 'c' ],
731 'array' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'array' ],
732 'BCarray' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'BCarray' ],
733 'BCassoc' => [ 'a', 'b', 'c', ApiResult::META_TYPE => 'BCassoc' ],
734 'assoc' => [ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
735 'kvp' => [ 'x' => 'a', 'y' => 'b', 'z' => [ 'c' ], ApiResult::META_TYPE => 'kvp' ],
736 'BCkvp' => [ 'x' => 'a', 'y' => 'b',
737 ApiResult::META_TYPE => 'BCkvp',
738 ApiResult::META_KVP_KEY_NAME => 'key',
740 'kvpmerge' => [ 'x' => 'a', 'y' => [ 'b' ], 'z' => [ 'c' => 'd' ],
741 ApiResult::META_TYPE => 'kvp',
742 ApiResult::META_KVP_MERGE => true,
744 'emptyDefault' => [ '_dummy' => 1 ],
745 'emptyAssoc' => [ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
746 '_dummy' => 1,
747 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
749 $stripArr = [
750 'foo' => [
751 'bar' => [ '_dummy' => 'foobaz' ],
752 'baz' => [
753 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
754 ApiResult::META_INDEXED_TAG_NAME => 'itn',
755 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
756 ApiResult::META_TYPE => 'array',
758 'x' => 'ok',
759 '_dummy' => 'foobaz',
761 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
762 ApiResult::META_INDEXED_TAG_NAME => 'itn',
763 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
764 ApiResult::META_TYPE => 'array',
765 '_dummy' => 'foobaz',
766 '_dummy2' => 'foobaz!',
769 return [
771 'BC: META_BC_BOOLS',
773 'BCtrue' => true,
774 'BCfalse' => false,
775 'true' => true,
776 'false' => false,
777 ApiResult::META_BC_BOOLS => [ 0, 'true', 'false' ],
779 [ 'BC' => [] ],
781 'BCtrue' => '',
782 'true' => true,
783 'false' => false,
784 ApiResult::META_BC_BOOLS => [ 0, 'true', 'false' ],
788 'BC: META_BC_SUBELEMENTS',
790 'bc' => 'foo',
791 'nobc' => 'bar',
792 ApiResult::META_BC_SUBELEMENTS => [ 'bc' ],
794 [ 'BC' => [] ],
796 'bc' => [
797 '*' => 'foo',
798 ApiResult::META_CONTENT => '*',
799 ApiResult::META_TYPE => 'assoc',
801 'nobc' => 'bar',
802 ApiResult::META_BC_SUBELEMENTS => [ 'bc' ],
806 'BC: META_CONTENT',
808 'content' => '!!!',
809 ApiResult::META_CONTENT => 'content',
811 [ 'BC' => [] ],
813 '*' => '!!!',
814 ApiResult::META_CONTENT => '*',
818 'BC: BCkvp type',
820 'foo' => 'foo value',
821 'bar' => 'bar value',
822 '_baz' => 'baz value',
823 ApiResult::META_TYPE => 'BCkvp',
824 ApiResult::META_KVP_KEY_NAME => 'key',
825 ApiResult::META_PRESERVE_KEYS => [ '_baz' ],
827 [ 'BC' => [] ],
829 $kvp( 'key', 'foo', '*', 'foo value' ),
830 $kvp( 'key', 'bar', '*', 'bar value' ),
831 $kvp( 'key', '_baz', '*', 'baz value' ),
832 ApiResult::META_TYPE => 'array',
833 ApiResult::META_KVP_KEY_NAME => 'key',
834 ApiResult::META_PRESERVE_KEYS => [ '_baz' ],
838 'BC: BCarray type',
840 ApiResult::META_TYPE => 'BCarray',
842 [ 'BC' => [] ],
844 ApiResult::META_TYPE => 'default',
848 'BC: BCassoc type',
850 ApiResult::META_TYPE => 'BCassoc',
852 [ 'BC' => [] ],
854 ApiResult::META_TYPE => 'default',
858 'BC: BCkvp exception',
860 ApiResult::META_TYPE => 'BCkvp',
862 [ 'BC' => [] ],
863 new UnexpectedValueException(
864 'Type "BCkvp" used without setting ApiResult::META_KVP_KEY_NAME metadata item'
868 'BC: nobool, no*, nosub',
870 'true' => true,
871 'false' => false,
872 'content' => 'content',
873 ApiResult::META_CONTENT => 'content',
874 'bc' => 'foo',
875 ApiResult::META_BC_SUBELEMENTS => [ 'bc' ],
876 'BCarray' => [ ApiResult::META_TYPE => 'BCarray' ],
877 'BCassoc' => [ ApiResult::META_TYPE => 'BCassoc' ],
878 'BCkvp' => [
879 'foo' => 'foo value',
880 'bar' => 'bar value',
881 '_baz' => 'baz value',
882 ApiResult::META_TYPE => 'BCkvp',
883 ApiResult::META_KVP_KEY_NAME => 'key',
884 ApiResult::META_PRESERVE_KEYS => [ '_baz' ],
887 [ 'BC' => [ 'nobool', 'no*', 'nosub' ] ],
889 'true' => true,
890 'false' => false,
891 'content' => 'content',
892 'bc' => 'foo',
893 'BCarray' => [ ApiResult::META_TYPE => 'default' ],
894 'BCassoc' => [ ApiResult::META_TYPE => 'default' ],
895 'BCkvp' => [
896 $kvp( 'key', 'foo', '*', 'foo value' ),
897 $kvp( 'key', 'bar', '*', 'bar value' ),
898 $kvp( 'key', '_baz', '*', 'baz value' ),
899 ApiResult::META_TYPE => 'array',
900 ApiResult::META_KVP_KEY_NAME => 'key',
901 ApiResult::META_PRESERVE_KEYS => [ '_baz' ],
903 ApiResult::META_CONTENT => 'content',
904 ApiResult::META_BC_SUBELEMENTS => [ 'bc' ],
909 'Types: Normal transform',
910 $typeArr,
911 [ 'Types' => [] ],
913 'defaultArray' => [ 'b', 'c', 'a', ApiResult::META_TYPE => 'array' ],
914 'defaultAssoc' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
915 'defaultAssoc2' => [ 2 => 'a', 3 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
916 'array' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
917 'BCarray' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
918 'BCassoc' => [ 'a', 'b', 'c', ApiResult::META_TYPE => 'assoc' ],
919 'assoc' => [ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
920 'kvp' => [ 'x' => 'a', 'y' => 'b',
921 'z' => [ 'c', ApiResult::META_TYPE => 'array' ],
922 ApiResult::META_TYPE => 'assoc'
924 'BCkvp' => [ 'x' => 'a', 'y' => 'b',
925 ApiResult::META_TYPE => 'assoc',
926 ApiResult::META_KVP_KEY_NAME => 'key',
928 'kvpmerge' => [
929 'x' => 'a',
930 'y' => [ 'b', ApiResult::META_TYPE => 'array' ],
931 'z' => [ 'c' => 'd', ApiResult::META_TYPE => 'assoc' ],
932 ApiResult::META_TYPE => 'assoc',
933 ApiResult::META_KVP_MERGE => true,
935 'emptyDefault' => [ '_dummy' => 1, ApiResult::META_TYPE => 'array' ],
936 'emptyAssoc' => [ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
937 '_dummy' => 1,
938 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
939 ApiResult::META_TYPE => 'assoc',
943 'Types: AssocAsObject',
944 $typeArr,
945 [ 'Types' => [ 'AssocAsObject' => true ] ],
946 (object)[
947 'defaultArray' => [ 'b', 'c', 'a', ApiResult::META_TYPE => 'array' ],
948 'defaultAssoc' => (object)[ 'x' => 'a',
949 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc'
951 'defaultAssoc2' => (object)[ 2 => 'a', 3 => 'b',
952 0 => 'c', ApiResult::META_TYPE => 'assoc'
954 'array' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
955 'BCarray' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
956 'BCassoc' => (object)[ 'a', 'b', 'c', ApiResult::META_TYPE => 'assoc' ],
957 'assoc' => (object)[ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
958 'kvp' => (object)[ 'x' => 'a', 'y' => 'b',
959 'z' => [ 'c', ApiResult::META_TYPE => 'array' ],
960 ApiResult::META_TYPE => 'assoc'
962 'BCkvp' => (object)[ 'x' => 'a', 'y' => 'b',
963 ApiResult::META_TYPE => 'assoc',
964 ApiResult::META_KVP_KEY_NAME => 'key',
966 'kvpmerge' => (object)[
967 'x' => 'a',
968 'y' => [ 'b', ApiResult::META_TYPE => 'array' ],
969 'z' => (object)[ 'c' => 'd', ApiResult::META_TYPE => 'assoc' ],
970 ApiResult::META_TYPE => 'assoc',
971 ApiResult::META_KVP_MERGE => true,
973 'emptyDefault' => [ '_dummy' => 1, ApiResult::META_TYPE => 'array' ],
974 'emptyAssoc' => (object)[ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
975 '_dummy' => 1,
976 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
977 ApiResult::META_TYPE => 'assoc',
981 'Types: ArmorKVP',
982 $typeArr,
983 [ 'Types' => [ 'ArmorKVP' => 'name' ] ],
985 'defaultArray' => [ 'b', 'c', 'a', ApiResult::META_TYPE => 'array' ],
986 'defaultAssoc' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
987 'defaultAssoc2' => [ 2 => 'a', 3 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
988 'array' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
989 'BCarray' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
990 'BCassoc' => [ 'a', 'b', 'c', ApiResult::META_TYPE => 'assoc' ],
991 'assoc' => [ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
992 'kvp' => [
993 $kvp( 'name', 'x', 'value', 'a' ),
994 $kvp( 'name', 'y', 'value', 'b' ),
995 $kvp( 'name', 'z', 'value', [ 'c', ApiResult::META_TYPE => 'array' ] ),
996 ApiResult::META_TYPE => 'array'
998 'BCkvp' => [
999 $kvp( 'key', 'x', 'value', 'a' ),
1000 $kvp( 'key', 'y', 'value', 'b' ),
1001 ApiResult::META_TYPE => 'array',
1002 ApiResult::META_KVP_KEY_NAME => 'key',
1004 'kvpmerge' => [
1005 $kvp( 'name', 'x', 'value', 'a' ),
1006 $kvp( 'name', 'y', 'value', [ 'b', ApiResult::META_TYPE => 'array' ] ),
1008 'name' => 'z',
1009 'c' => 'd',
1010 ApiResult::META_TYPE => 'assoc',
1011 ApiResult::META_PRESERVE_KEYS => [ 'name' ]
1013 ApiResult::META_TYPE => 'array',
1014 ApiResult::META_KVP_MERGE => true,
1016 'emptyDefault' => [ '_dummy' => 1, ApiResult::META_TYPE => 'array' ],
1017 'emptyAssoc' => [ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
1018 '_dummy' => 1,
1019 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
1020 ApiResult::META_TYPE => 'assoc',
1024 'Types: ArmorKVP + BC',
1025 $typeArr,
1026 [ 'BC' => [], 'Types' => [ 'ArmorKVP' => 'name' ] ],
1028 'defaultArray' => [ 'b', 'c', 'a', ApiResult::META_TYPE => 'array' ],
1029 'defaultAssoc' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
1030 'defaultAssoc2' => [ 2 => 'a', 3 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
1031 'array' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
1032 'BCarray' => [ 'x' => 'a', 1 => 'b', 0 => 'c', ApiResult::META_TYPE => 'assoc' ],
1033 'BCassoc' => [ 'a', 'b', 'c', ApiResult::META_TYPE => 'array' ],
1034 'assoc' => [ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
1035 'kvp' => [
1036 $kvp( 'name', 'x', '*', 'a' ),
1037 $kvp( 'name', 'y', '*', 'b' ),
1038 $kvp( 'name', 'z', '*', [ 'c', ApiResult::META_TYPE => 'array' ] ),
1039 ApiResult::META_TYPE => 'array'
1041 'BCkvp' => [
1042 $kvp( 'key', 'x', '*', 'a' ),
1043 $kvp( 'key', 'y', '*', 'b' ),
1044 ApiResult::META_TYPE => 'array',
1045 ApiResult::META_KVP_KEY_NAME => 'key',
1047 'kvpmerge' => [
1048 $kvp( 'name', 'x', '*', 'a' ),
1049 $kvp( 'name', 'y', '*', [ 'b', ApiResult::META_TYPE => 'array' ] ),
1051 'name' => 'z',
1052 'c' => 'd',
1053 ApiResult::META_TYPE => 'assoc',
1054 ApiResult::META_PRESERVE_KEYS => [ 'name' ] ],
1055 ApiResult::META_TYPE => 'array',
1056 ApiResult::META_KVP_MERGE => true,
1058 'emptyDefault' => [ '_dummy' => 1, ApiResult::META_TYPE => 'array' ],
1059 'emptyAssoc' => [ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
1060 '_dummy' => 1,
1061 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
1062 ApiResult::META_TYPE => 'assoc',
1066 'Types: ArmorKVP + AssocAsObject',
1067 $typeArr,
1068 [ 'Types' => [ 'ArmorKVP' => 'name', 'AssocAsObject' => true ] ],
1069 (object)[
1070 'defaultArray' => [ 'b', 'c', 'a', ApiResult::META_TYPE => 'array' ],
1071 'defaultAssoc' => (object)[ 'x' => 'a', 1 => 'b',
1072 0 => 'c', ApiResult::META_TYPE => 'assoc'
1074 'defaultAssoc2' => (object)[ 2 => 'a', 3 => 'b',
1075 0 => 'c', ApiResult::META_TYPE => 'assoc'
1077 'array' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
1078 'BCarray' => [ 'a', 'c', 'b', ApiResult::META_TYPE => 'array' ],
1079 'BCassoc' => (object)[ 'a', 'b', 'c', ApiResult::META_TYPE => 'assoc' ],
1080 'assoc' => (object)[ 2 => 'a', 0 => 'b', 1 => 'c', ApiResult::META_TYPE => 'assoc' ],
1081 'kvp' => [
1082 (object)$kvp( 'name', 'x', 'value', 'a' ),
1083 (object)$kvp( 'name', 'y', 'value', 'b' ),
1084 (object)$kvp( 'name', 'z', 'value', [ 'c', ApiResult::META_TYPE => 'array' ] ),
1085 ApiResult::META_TYPE => 'array'
1087 'BCkvp' => [
1088 (object)$kvp( 'key', 'x', 'value', 'a' ),
1089 (object)$kvp( 'key', 'y', 'value', 'b' ),
1090 ApiResult::META_TYPE => 'array',
1091 ApiResult::META_KVP_KEY_NAME => 'key',
1093 'kvpmerge' => [
1094 (object)$kvp( 'name', 'x', 'value', 'a' ),
1095 (object)$kvp( 'name', 'y', 'value', [ 'b', ApiResult::META_TYPE => 'array' ] ),
1096 (object)[
1097 'name' => 'z',
1098 'c' => 'd',
1099 ApiResult::META_TYPE => 'assoc',
1100 ApiResult::META_PRESERVE_KEYS => [ 'name' ]
1102 ApiResult::META_TYPE => 'array',
1103 ApiResult::META_KVP_MERGE => true,
1105 'emptyDefault' => [ '_dummy' => 1, ApiResult::META_TYPE => 'array' ],
1106 'emptyAssoc' => (object)[ '_dummy' => 1, ApiResult::META_TYPE => 'assoc' ],
1107 '_dummy' => 1,
1108 ApiResult::META_PRESERVE_KEYS => [ '_dummy' ],
1109 ApiResult::META_TYPE => 'assoc',
1113 'Types: BCkvp exception',
1115 ApiResult::META_TYPE => 'BCkvp',
1117 [ 'Types' => [] ],
1118 new UnexpectedValueException(
1119 'Type "BCkvp" used without setting ApiResult::META_KVP_KEY_NAME metadata item'
1124 'Strip: With ArmorKVP + AssocAsObject transforms',
1125 $typeArr,
1126 [ 'Types' => [ 'ArmorKVP' => 'name', 'AssocAsObject' => true ], 'Strip' => 'all' ],
1127 (object)[
1128 'defaultArray' => [ 'b', 'c', 'a' ],
1129 'defaultAssoc' => (object)[ 'x' => 'a', 1 => 'b', 0 => 'c' ],
1130 'defaultAssoc2' => (object)[ 2 => 'a', 3 => 'b', 0 => 'c' ],
1131 'array' => [ 'a', 'c', 'b' ],
1132 'BCarray' => [ 'a', 'c', 'b' ],
1133 'BCassoc' => (object)[ 'a', 'b', 'c' ],
1134 'assoc' => (object)[ 2 => 'a', 0 => 'b', 1 => 'c' ],
1135 'kvp' => [
1136 (object)[ 'name' => 'x', 'value' => 'a' ],
1137 (object)[ 'name' => 'y', 'value' => 'b' ],
1138 (object)[ 'name' => 'z', 'value' => [ 'c' ] ],
1140 'BCkvp' => [
1141 (object)[ 'key' => 'x', 'value' => 'a' ],
1142 (object)[ 'key' => 'y', 'value' => 'b' ],
1144 'kvpmerge' => [
1145 (object)[ 'name' => 'x', 'value' => 'a' ],
1146 (object)[ 'name' => 'y', 'value' => [ 'b' ] ],
1147 (object)[ 'name' => 'z', 'c' => 'd' ],
1149 'emptyDefault' => [],
1150 'emptyAssoc' => (object)[],
1151 '_dummy' => 1,
1156 'Strip: all',
1157 $stripArr,
1158 [ 'Strip' => 'all' ],
1160 'foo' => [
1161 'bar' => [],
1162 'baz' => [],
1163 'x' => 'ok',
1165 '_dummy2' => 'foobaz!',
1169 'Strip: base',
1170 $stripArr,
1171 [ 'Strip' => 'base' ],
1173 'foo' => [
1174 'bar' => [ '_dummy' => 'foobaz' ],
1175 'baz' => [
1176 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
1177 ApiResult::META_INDEXED_TAG_NAME => 'itn',
1178 ApiResult::META_PRESERVE_KEYS => [ 'foo', 'bar', '_dummy2', 0 ],
1179 ApiResult::META_TYPE => 'array',
1181 'x' => 'ok',
1182 '_dummy' => 'foobaz',
1184 '_dummy2' => 'foobaz!',
1188 'Strip: bc',
1189 $stripArr,
1190 [ 'Strip' => 'bc' ],
1192 'foo' => [
1193 'bar' => [],
1194 'baz' => [
1195 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
1196 ApiResult::META_INDEXED_TAG_NAME => 'itn',
1198 'x' => 'ok',
1200 '_dummy2' => 'foobaz!',
1201 ApiResult::META_SUBELEMENTS => [ 'foo', 'bar' ],
1202 ApiResult::META_INDEXED_TAG_NAME => 'itn',
1207 'Custom transform',
1209 'foo' => '?',
1210 'bar' => '?',
1211 '_dummy' => '?',
1212 '_dummy2' => '?',
1213 '_dummy3' => '?',
1214 ApiResult::META_CONTENT => 'foo',
1215 ApiResult::META_PRESERVE_KEYS => [ '_dummy2', '_dummy3' ],
1218 'Custom' => [ $this, 'customTransform' ],
1219 'BC' => [],
1220 'Types' => [],
1221 'Strip' => 'all'
1224 '*' => 'FOO',
1225 'bar' => 'BAR',
1226 'baz' => [ 'a', 'b' ],
1227 '_dummy2' => '_DUMMY2',
1228 '_dummy3' => '_DUMMY3',
1229 ApiResult::META_CONTENT => 'bar',
1237 * Custom transformer for testTransformations
1238 * @param array &$data
1239 * @param array &$metadata
1241 public function customTransform( &$data, &$metadata ) {
1242 // Prevent recursion
1243 if ( isset( $metadata['_added'] ) ) {
1244 $metadata[ApiResult::META_TYPE] = 'array';
1245 return;
1248 foreach ( $data as $k => $v ) {
1249 $data[$k] = strtoupper( $k );
1251 $data['baz'] = [ '_added' => 1, 'z' => 'b', 'y' => 'a' ];
1252 $metadata[ApiResult::META_PRESERVE_KEYS][0] = '_dummy';
1253 $data[ApiResult::META_CONTENT] = 'bar';
1257 * @covers ApiResult
1259 public function testAddMetadataToResultVars() {
1260 $arr = [
1261 'a' => "foo",
1262 'b' => false,
1263 'c' => 10,
1264 'sequential_numeric_keys' => [ 'a', 'b', 'c' ],
1265 'non_sequential_numeric_keys' => [ 'a', 'b', 4 => 'c' ],
1266 'string_keys' => [
1267 'one' => 1,
1268 'two' => 2
1270 'object_sequential_keys' => (object)[ 'a', 'b', 'c' ],
1271 '_type' => "should be overwritten in result",
1273 $this->assertSame( [
1274 ApiResult::META_TYPE => 'kvp',
1275 ApiResult::META_KVP_KEY_NAME => 'key',
1276 ApiResult::META_PRESERVE_KEYS => [
1277 'a', 'b', 'c',
1278 'sequential_numeric_keys', 'non_sequential_numeric_keys',
1279 'string_keys', 'object_sequential_keys'
1281 ApiResult::META_BC_BOOLS => [ 'b' ],
1282 ApiResult::META_INDEXED_TAG_NAME => 'var',
1283 'a' => "foo",
1284 'b' => false,
1285 'c' => 10,
1286 'sequential_numeric_keys' => [
1287 ApiResult::META_TYPE => 'array',
1288 ApiResult::META_BC_BOOLS => [],
1289 ApiResult::META_INDEXED_TAG_NAME => 'value',
1290 0 => 'a',
1291 1 => 'b',
1292 2 => 'c',
1294 'non_sequential_numeric_keys' => [
1295 ApiResult::META_TYPE => 'kvp',
1296 ApiResult::META_KVP_KEY_NAME => 'key',
1297 ApiResult::META_PRESERVE_KEYS => [ 0, 1, 4 ],
1298 ApiResult::META_BC_BOOLS => [],
1299 ApiResult::META_INDEXED_TAG_NAME => 'var',
1300 0 => 'a',
1301 1 => 'b',
1302 4 => 'c',
1304 'string_keys' => [
1305 ApiResult::META_TYPE => 'kvp',
1306 ApiResult::META_KVP_KEY_NAME => 'key',
1307 ApiResult::META_PRESERVE_KEYS => [ 'one', 'two' ],
1308 ApiResult::META_BC_BOOLS => [],
1309 ApiResult::META_INDEXED_TAG_NAME => 'var',
1310 'one' => 1,
1311 'two' => 2,
1313 'object_sequential_keys' => [
1314 ApiResult::META_TYPE => 'kvp',
1315 ApiResult::META_KVP_KEY_NAME => 'key',
1316 ApiResult::META_PRESERVE_KEYS => [ 0, 1, 2 ],
1317 ApiResult::META_BC_BOOLS => [],
1318 ApiResult::META_INDEXED_TAG_NAME => 'var',
1319 0 => 'a',
1320 1 => 'b',
1321 2 => 'c',
1323 ], ApiResult::addMetadataToResultVars( $arr ) );
1327 * @covers ApiResult
1329 public function testDeprecatedFunctions() {
1330 // Ignore ApiResult deprecation warnings during this test
1331 set_error_handler( function ( $errno, $errstr ) use ( &$warnings ) {
1332 if ( preg_match( '/Use of ApiResult::\S+ was deprecated in MediaWiki \d+.\d+\./', $errstr ) ) {
1333 return true;
1335 if ( preg_match( '/Use of ApiMain to ApiResult::__construct ' .
1336 'was deprecated in MediaWiki \d+.\d+\./', $errstr ) ) {
1337 return true;
1339 return false;
1340 } );
1341 $reset = new ScopedCallback( 'restore_error_handler' );
1343 $context = new DerivativeContext( RequestContext::getMain() );
1344 $context->setConfig( new HashConfig( [
1345 'APIModules' => [],
1346 'APIFormatModules' => [],
1347 'APIMaxResultSize' => 42,
1348 ] ) );
1349 $main = new ApiMain( $context );
1350 $result = TestingAccessWrapper::newFromObject( new ApiResult( $main ) );
1351 $this->assertSame( 42, $result->maxSize );
1352 $this->assertSame( $main->getErrorFormatter(), $result->errorFormatter );
1353 $this->assertSame( $main, $result->mainForContinuation );
1355 $result = new ApiResult( 8388608 );
1357 $result->addContentValue( null, 'test', 'content' );
1358 $result->addContentValue( [ 'foo', 'bar' ], 'test', 'content' );
1359 $result->addIndexedTagName( null, 'itn' );
1360 $result->addSubelementsList( null, [ 'sub' ] );
1361 $this->assertSame( [
1362 'foo' => [
1363 'bar' => [
1364 '*' => 'content',
1367 '*' => 'content',
1368 ], $result->getData() );
1370 $arr = [];
1371 ApiResult::setContent( $arr, 'value' );
1372 ApiResult::setContent( $arr, 'value2', 'foobar' );
1373 $this->assertSame( [
1374 ApiResult::META_CONTENT => 'content',
1375 'content' => 'value',
1376 'foobar' => [
1377 ApiResult::META_CONTENT => 'content',
1378 'content' => 'value2',
1380 ], $arr );
1382 $result = new ApiResult( 3 );
1383 $formatter = new ApiErrorFormatter_BackCompat( $result );
1384 $result->setErrorFormatter( $formatter );
1385 $result->disableSizeCheck();
1386 $this->assertTrue( $result->addValue( null, 'foo', '1234567890' ) );
1387 $result->enableSizeCheck();
1388 $this->assertSame( 0, $result->getSize() );
1389 $this->assertFalse( $result->addValue( null, 'foo', '1234567890' ) );
1391 $arr = [ 'foo' => [ 'bar' => 1 ] ];
1392 $result->setIndexedTagName_recursive( $arr, 'itn' );
1393 $this->assertSame( [
1394 'foo' => [
1395 'bar' => 1,
1396 ApiResult::META_INDEXED_TAG_NAME => 'itn'
1398 ], $arr );
1400 $status = Status::newGood();
1401 $status->fatal( 'parentheses', '1' );
1402 $status->fatal( 'parentheses', '2' );
1403 $status->warning( 'parentheses', '3' );
1404 $status->warning( 'parentheses', '4' );
1405 $this->assertSame( [
1407 'type' => 'error',
1408 'message' => 'parentheses',
1409 'params' => [
1410 0 => '1',
1411 ApiResult::META_INDEXED_TAG_NAME => 'param',
1415 'type' => 'error',
1416 'message' => 'parentheses',
1417 'params' => [
1418 0 => '2',
1419 ApiResult::META_INDEXED_TAG_NAME => 'param',
1422 ApiResult::META_INDEXED_TAG_NAME => 'error',
1423 ], $result->convertStatusToArray( $status, 'error' ) );
1424 $this->assertSame( [
1426 'type' => 'warning',
1427 'message' => 'parentheses',
1428 'params' => [
1429 0 => '3',
1430 ApiResult::META_INDEXED_TAG_NAME => 'param',
1434 'type' => 'warning',
1435 'message' => 'parentheses',
1436 'params' => [
1437 0 => '4',
1438 ApiResult::META_INDEXED_TAG_NAME => 'param',
1441 ApiResult::META_INDEXED_TAG_NAME => 'warning',
1442 ], $result->convertStatusToArray( $status, 'warning' ) );
1446 * @covers ApiResult
1448 public function testDeprecatedContinuation() {
1449 // Ignore ApiResult deprecation warnings during this test
1450 set_error_handler( function ( $errno, $errstr ) use ( &$warnings ) {
1451 if ( preg_match( '/Use of ApiResult::\S+ was deprecated in MediaWiki \d+.\d+\./', $errstr ) ) {
1452 return true;
1454 return false;
1455 } );
1457 $reset = new ScopedCallback( 'restore_error_handler' );
1458 $allModules = [
1459 new MockApiQueryBase( 'mock1' ),
1460 new MockApiQueryBase( 'mock2' ),
1461 new MockApiQueryBase( 'mocklist' ),
1463 $generator = new MockApiQueryBase( 'generator' );
1465 $main = new ApiMain( RequestContext::getMain() );
1466 $result = new ApiResult( 8388608 );
1467 $result->setMainForContinuation( $main );
1468 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1469 $this->assertSame( [ false, $allModules ], $ret );
1470 $result->setContinueParam( $allModules[0], 'm1continue', [ 1, 2 ] );
1471 $result->setContinueParam( $allModules[2], 'mlcontinue', 2 );
1472 $result->setGeneratorContinueParam( $generator, 'gcontinue', 3 );
1473 $result->endContinuation( 'raw' );
1474 $result->endContinuation( 'standard' );
1475 $this->assertSame( [
1476 'mlcontinue' => 2,
1477 'm1continue' => '1|2',
1478 'continue' => '||mock2',
1479 ], $result->getResultData( 'continue' ) );
1480 $this->assertSame( null, $result->getResultData( 'batchcomplete' ) );
1481 $this->assertSame( [
1482 'mock1' => [ 'm1continue' => '1|2' ],
1483 'mocklist' => [ 'mlcontinue' => 2 ],
1484 'generator' => [ 'gcontinue' => 3 ],
1485 ], $result->getResultData( 'query-continue' ) );
1486 $main->setContinuationManager( null );
1488 $result = new ApiResult( 8388608 );
1489 $result->setMainForContinuation( $main );
1490 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1491 $this->assertSame( [ false, $allModules ], $ret );
1492 $result->setContinueParam( $allModules[0], 'm1continue', [ 1, 2 ] );
1493 $result->setGeneratorContinueParam( $generator, 'gcontinue', [ 3, 4 ] );
1494 $result->endContinuation( 'raw' );
1495 $result->endContinuation( 'standard' );
1496 $this->assertSame( [
1497 'm1continue' => '1|2',
1498 'continue' => '||mock2|mocklist',
1499 ], $result->getResultData( 'continue' ) );
1500 $this->assertSame( null, $result->getResultData( 'batchcomplete' ) );
1501 $this->assertSame( [
1502 'mock1' => [ 'm1continue' => '1|2' ],
1503 'generator' => [ 'gcontinue' => '3|4' ],
1504 ], $result->getResultData( 'query-continue' ) );
1505 $main->setContinuationManager( null );
1507 $result = new ApiResult( 8388608 );
1508 $result->setMainForContinuation( $main );
1509 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1510 $this->assertSame( [ false, $allModules ], $ret );
1511 $result->setContinueParam( $allModules[2], 'mlcontinue', 2 );
1512 $result->setGeneratorContinueParam( $generator, 'gcontinue', 3 );
1513 $result->endContinuation( 'raw' );
1514 $result->endContinuation( 'standard' );
1515 $this->assertSame( [
1516 'mlcontinue' => 2,
1517 'gcontinue' => 3,
1518 'continue' => 'gcontinue||',
1519 ], $result->getResultData( 'continue' ) );
1520 $this->assertSame( true, $result->getResultData( 'batchcomplete' ) );
1521 $this->assertSame( [
1522 'mocklist' => [ 'mlcontinue' => 2 ],
1523 'generator' => [ 'gcontinue' => 3 ],
1524 ], $result->getResultData( 'query-continue' ) );
1525 $main->setContinuationManager( null );
1527 $result = new ApiResult( 8388608 );
1528 $result->setMainForContinuation( $main );
1529 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1530 $this->assertSame( [ false, $allModules ], $ret );
1531 $result->setGeneratorContinueParam( $generator, 'gcontinue', 3 );
1532 $result->endContinuation( 'raw' );
1533 $result->endContinuation( 'standard' );
1534 $this->assertSame( [
1535 'gcontinue' => 3,
1536 'continue' => 'gcontinue||mocklist',
1537 ], $result->getResultData( 'continue' ) );
1538 $this->assertSame( true, $result->getResultData( 'batchcomplete' ) );
1539 $this->assertSame( [
1540 'generator' => [ 'gcontinue' => 3 ],
1541 ], $result->getResultData( 'query-continue' ) );
1542 $main->setContinuationManager( null );
1544 $result = new ApiResult( 8388608 );
1545 $result->setMainForContinuation( $main );
1546 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1547 $this->assertSame( [ false, $allModules ], $ret );
1548 $result->setContinueParam( $allModules[0], 'm1continue', [ 1, 2 ] );
1549 $result->setContinueParam( $allModules[2], 'mlcontinue', 2 );
1550 $result->endContinuation( 'raw' );
1551 $result->endContinuation( 'standard' );
1552 $this->assertSame( [
1553 'mlcontinue' => 2,
1554 'm1continue' => '1|2',
1555 'continue' => '||mock2',
1556 ], $result->getResultData( 'continue' ) );
1557 $this->assertSame( null, $result->getResultData( 'batchcomplete' ) );
1558 $this->assertSame( [
1559 'mock1' => [ 'm1continue' => '1|2' ],
1560 'mocklist' => [ 'mlcontinue' => 2 ],
1561 ], $result->getResultData( 'query-continue' ) );
1562 $main->setContinuationManager( null );
1564 $result = new ApiResult( 8388608 );
1565 $result->setMainForContinuation( $main );
1566 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1567 $this->assertSame( [ false, $allModules ], $ret );
1568 $result->setContinueParam( $allModules[0], 'm1continue', [ 1, 2 ] );
1569 $result->endContinuation( 'raw' );
1570 $result->endContinuation( 'standard' );
1571 $this->assertSame( [
1572 'm1continue' => '1|2',
1573 'continue' => '||mock2|mocklist',
1574 ], $result->getResultData( 'continue' ) );
1575 $this->assertSame( null, $result->getResultData( 'batchcomplete' ) );
1576 $this->assertSame( [
1577 'mock1' => [ 'm1continue' => '1|2' ],
1578 ], $result->getResultData( 'query-continue' ) );
1579 $main->setContinuationManager( null );
1581 $result = new ApiResult( 8388608 );
1582 $result->setMainForContinuation( $main );
1583 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1584 $this->assertSame( [ false, $allModules ], $ret );
1585 $result->setContinueParam( $allModules[2], 'mlcontinue', 2 );
1586 $result->endContinuation( 'raw' );
1587 $result->endContinuation( 'standard' );
1588 $this->assertSame( [
1589 'mlcontinue' => 2,
1590 'continue' => '-||mock1|mock2',
1591 ], $result->getResultData( 'continue' ) );
1592 $this->assertSame( true, $result->getResultData( 'batchcomplete' ) );
1593 $this->assertSame( [
1594 'mocklist' => [ 'mlcontinue' => 2 ],
1595 ], $result->getResultData( 'query-continue' ) );
1596 $main->setContinuationManager( null );
1598 $result = new ApiResult( 8388608 );
1599 $result->setMainForContinuation( $main );
1600 $ret = $result->beginContinuation( null, $allModules, [ 'mock1', 'mock2' ] );
1601 $this->assertSame( [ false, $allModules ], $ret );
1602 $result->endContinuation( 'raw' );
1603 $result->endContinuation( 'standard' );
1604 $this->assertSame( null, $result->getResultData( 'continue' ) );
1605 $this->assertSame( true, $result->getResultData( 'batchcomplete' ) );
1606 $this->assertSame( null, $result->getResultData( 'query-continue' ) );
1607 $main->setContinuationManager( null );
1609 $result = new ApiResult( 8388608 );
1610 $result->setMainForContinuation( $main );
1611 $ret = $result->beginContinuation( '||mock2', $allModules, [ 'mock1', 'mock2' ] );
1612 $this->assertSame(
1613 [ false, array_values( array_diff_key( $allModules, [ 1 => 1 ] ) ) ],
1614 $ret
1616 $main->setContinuationManager( null );
1618 $result = new ApiResult( 8388608 );
1619 $result->setMainForContinuation( $main );
1620 $ret = $result->beginContinuation( '-||', $allModules, [ 'mock1', 'mock2' ] );
1621 $this->assertSame(
1622 [ true, array_values( array_diff_key( $allModules, [ 0 => 0, 1 => 1 ] ) ) ],
1623 $ret
1625 $main->setContinuationManager( null );
1627 $result = new ApiResult( 8388608 );
1628 $result->setMainForContinuation( $main );
1629 try {
1630 $result->beginContinuation( 'foo', $allModules, [ 'mock1', 'mock2' ] );
1631 $this->fail( 'Expected exception not thrown' );
1632 } catch ( UsageException $ex ) {
1633 $this->assertSame(
1634 'Invalid continue param. You should pass the original value returned by the previous query',
1635 $ex->getMessage(),
1636 'Expected exception'
1639 $main->setContinuationManager( null );
1641 $result = new ApiResult( 8388608 );
1642 $result->setMainForContinuation( $main );
1643 $result->beginContinuation( '||mock2', array_slice( $allModules, 0, 2 ),
1644 [ 'mock1', 'mock2' ] );
1645 try {
1646 $result->setContinueParam( $allModules[1], 'm2continue', 1 );
1647 $this->fail( 'Expected exception not thrown' );
1648 } catch ( UnexpectedValueException $ex ) {
1649 $this->assertSame(
1650 'Module \'mock2\' was not supposed to have been executed, but it was executed anyway',
1651 $ex->getMessage(),
1652 'Expected exception'
1655 try {
1656 $result->setContinueParam( $allModules[2], 'mlcontinue', 1 );
1657 $this->fail( 'Expected exception not thrown' );
1658 } catch ( UnexpectedValueException $ex ) {
1659 $this->assertSame(
1660 'Module \'mocklist\' called ApiContinuationManager::addContinueParam ' .
1661 'but was not passed to ApiContinuationManager::__construct',
1662 $ex->getMessage(),
1663 'Expected exception'
1666 $main->setContinuationManager( null );
1670 public function testObjectSerialization() {
1671 $arr = [];
1672 ApiResult::setValue( $arr, 'foo', (object)[ 'a' => 1, 'b' => 2 ] );
1673 $this->assertSame( [
1674 'a' => 1,
1675 'b' => 2,
1676 ApiResult::META_TYPE => 'assoc',
1677 ], $arr['foo'] );
1679 $arr = [];
1680 ApiResult::setValue( $arr, 'foo', new ApiResultTestStringifiableObject() );
1681 $this->assertSame( 'Ok', $arr['foo'] );
1683 $arr = [];
1684 ApiResult::setValue( $arr, 'foo', new ApiResultTestSerializableObject( 'Ok' ) );
1685 $this->assertSame( 'Ok', $arr['foo'] );
1687 try {
1688 $arr = [];
1689 ApiResult::setValue( $arr, 'foo', new ApiResultTestSerializableObject(
1690 new ApiResultTestStringifiableObject()
1691 ) );
1692 $this->fail( 'Expected exception not thrown' );
1693 } catch ( UnexpectedValueException $ex ) {
1694 $this->assertSame(
1695 'ApiResultTestSerializableObject::serializeForApiResult() ' .
1696 'returned an object of class ApiResultTestStringifiableObject',
1697 $ex->getMessage(),
1698 'Expected exception'
1702 try {
1703 $arr = [];
1704 ApiResult::setValue( $arr, 'foo', new ApiResultTestSerializableObject( NAN ) );
1705 $this->fail( 'Expected exception not thrown' );
1706 } catch ( UnexpectedValueException $ex ) {
1707 $this->assertSame(
1708 'ApiResultTestSerializableObject::serializeForApiResult() ' .
1709 'returned an invalid value: Cannot add non-finite floats to ApiResult',
1710 $ex->getMessage(),
1711 'Expected exception'
1715 $arr = [];
1716 ApiResult::setValue( $arr, 'foo', new ApiResultTestSerializableObject(
1718 'one' => new ApiResultTestStringifiableObject( '1' ),
1719 'two' => new ApiResultTestSerializableObject( 2 ),
1721 ) );
1722 $this->assertSame( [
1723 'one' => '1',
1724 'two' => 2,
1725 ], $arr['foo'] );
1730 class ApiResultTestStringifiableObject {
1731 private $ret;
1733 public function __construct( $ret = 'Ok' ) {
1734 $this->ret = $ret;
1737 public function __toString() {
1738 return $this->ret;
1742 class ApiResultTestSerializableObject {
1743 private $ret;
1745 public function __construct( $ret ) {
1746 $this->ret = $ret;
1749 public function __toString() {
1750 return "Fail";
1753 public function serializeForApiResult() {
1754 return $this->ret;