Fix DatabaseSqlite IDEA warnings
[mediawiki.git] / tests / phpunit / includes / WatchedItemQueryServiceUnitTest.php
blob92446ed950a91e5b69edf53bedcf3cfdd3615cad
1 <?php
3 /**
4 * @covers WatchedItemQueryService
5 */
6 class WatchedItemQueryServiceUnitTest extends PHPUnit_Framework_TestCase {
8 /**
9 * @return PHPUnit_Framework_MockObject_MockObject|Database
11 private function getMockDb() {
12 $mock = $this->getMockBuilder( Database::class )
13 ->disableOriginalConstructor()
14 ->getMock();
16 $mock->expects( $this->any() )
17 ->method( 'makeList' )
18 ->with(
19 $this->isType( 'array' ),
20 $this->isType( 'int' )
22 ->will( $this->returnCallback( function( $a, $conj ) {
23 $sqlConj = $conj === LIST_AND ? ' AND ' : ' OR ';
24 return join( $sqlConj, array_map( function( $s ) {
25 return '(' . $s . ')';
26 }, $a
27 ) );
28 } ) );
30 $mock->expects( $this->any() )
31 ->method( 'addQuotes' )
32 ->will( $this->returnCallback( function( $value ) {
33 return "'$value'";
34 } ) );
36 $mock->expects( $this->any() )
37 ->method( 'timestamp' )
38 ->will( $this->returnArgument( 0 ) );
40 $mock->expects( $this->any() )
41 ->method( 'bitAnd' )
42 ->willReturnCallback( function( $a, $b ) {
43 return "($a & $b)";
44 } );
46 return $mock;
49 /**
50 * @param $mockDb
51 * @return PHPUnit_Framework_MockObject_MockObject|LoadBalancer
53 private function getMockLoadBalancer( $mockDb ) {
54 $mock = $this->getMockBuilder( LoadBalancer::class )
55 ->disableOriginalConstructor()
56 ->getMock();
57 $mock->expects( $this->any() )
58 ->method( 'getConnectionRef' )
59 ->with( DB_SLAVE )
60 ->will( $this->returnValue( $mockDb ) );
61 return $mock;
64 /**
65 * @param int $id
66 * @return PHPUnit_Framework_MockObject_MockObject|User
68 private function getMockNonAnonUserWithId( $id ) {
69 $mock = $this->getMock( User::class );
70 $mock->expects( $this->any() )
71 ->method( 'isAnon' )
72 ->will( $this->returnValue( false ) );
73 $mock->expects( $this->any() )
74 ->method( 'getId' )
75 ->will( $this->returnValue( $id ) );
76 return $mock;
79 /**
80 * @param int $id
81 * @return PHPUnit_Framework_MockObject_MockObject|User
83 private function getMockUnrestrictedNonAnonUserWithId( $id ) {
84 $mock = $this->getMockNonAnonUserWithId( $id );
85 $mock->expects( $this->any() )
86 ->method( 'isAllowed' )
87 ->will( $this->returnValue( true ) );
88 $mock->expects( $this->any() )
89 ->method( 'isAllowedAny' )
90 ->will( $this->returnValue( true ) );
91 $mock->expects( $this->any() )
92 ->method( 'useRCPatrol' )
93 ->will( $this->returnValue( true ) );
94 return $mock;
97 /**
98 * @param int $id
99 * @param string $notAllowedAction
100 * @return PHPUnit_Framework_MockObject_MockObject|User
102 private function getMockNonAnonUserWithIdAndRestrictedPermissions( $id, $notAllowedAction ) {
103 $mock = $this->getMockNonAnonUserWithId( $id );
105 $mock->expects( $this->any() )
106 ->method( 'isAllowed' )
107 ->will( $this->returnCallback( function( $action ) use ( $notAllowedAction ) {
108 return $action !== $notAllowedAction;
109 } ) );
110 $mock->expects( $this->any() )
111 ->method( 'isAllowedAny' )
112 ->will( $this->returnCallback( function() use ( $notAllowedAction ) {
113 $actions = func_get_args();
114 return !in_array( $notAllowedAction, $actions );
115 } ) );
117 return $mock;
121 * @param int $id
122 * @return PHPUnit_Framework_MockObject_MockObject|User
124 private function getMockNonAnonUserWithIdAndNoPatrolRights( $id ) {
125 $mock = $this->getMockNonAnonUserWithId( $id );
127 $mock->expects( $this->any() )
128 ->method( 'isAllowed' )
129 ->will( $this->returnValue( true ) );
130 $mock->expects( $this->any() )
131 ->method( 'isAllowedAny' )
132 ->will( $this->returnValue( true ) );
134 $mock->expects( $this->any() )
135 ->method( 'useRCPatrol' )
136 ->will( $this->returnValue( false ) );
137 $mock->expects( $this->any() )
138 ->method( 'useNPPatrol' )
139 ->will( $this->returnValue( false ) );
141 return $mock;
144 private function getMockAnonUser() {
145 $mock = $this->getMock( User::class );
146 $mock->expects( $this->any() )
147 ->method( 'isAnon' )
148 ->will( $this->returnValue( true ) );
149 return $mock;
152 private function getFakeRow( array $rowValues ) {
153 $fakeRow = new stdClass();
154 foreach ( $rowValues as $valueName => $value ) {
155 $fakeRow->$valueName = $value;
157 return $fakeRow;
160 public function testGetWatchedItemsWithRecentChangeInfo() {
161 $mockDb = $this->getMockDb();
162 $mockDb->expects( $this->once() )
163 ->method( 'select' )
164 ->with(
165 [ 'recentchanges', 'watchlist', 'page' ],
167 'rc_id',
168 'rc_namespace',
169 'rc_title',
170 'rc_timestamp',
171 'rc_type',
172 'rc_deleted',
173 'wl_notificationtimestamp',
174 'rc_cur_id',
175 'rc_this_oldid',
176 'rc_last_oldid',
179 'wl_user' => 1,
180 '(rc_this_oldid=page_latest) OR (rc_type=3)',
182 $this->isType( 'string' ),
185 'watchlist' => [
186 'INNER JOIN',
188 'wl_namespace=rc_namespace',
189 'wl_title=rc_title'
192 'page' => [
193 'LEFT JOIN',
194 'rc_cur_id=page_id',
198 ->will( $this->returnValue( [
199 $this->getFakeRow( [
200 'rc_id' => 1,
201 'rc_namespace' => 0,
202 'rc_title' => 'Foo1',
203 'rc_timestamp' => '20151212010101',
204 'rc_type' => RC_NEW,
205 'rc_deleted' => 0,
206 'wl_notificationtimestamp' => '20151212010101',
207 ] ),
208 $this->getFakeRow( [
209 'rc_id' => 2,
210 'rc_namespace' => 1,
211 'rc_title' => 'Foo2',
212 'rc_timestamp' => '20151212010102',
213 'rc_type' => RC_NEW,
214 'rc_deleted' => 0,
215 'wl_notificationtimestamp' => null,
216 ] ),
217 ] ) );
219 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
220 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
222 $items = $queryService->getWatchedItemsWithRecentChangeInfo( $user );
224 $this->assertInternalType( 'array', $items );
225 $this->assertCount( 2, $items );
227 foreach ( $items as list( $watchedItem, $recentChangeInfo ) ) {
228 $this->assertInstanceOf( WatchedItem::class, $watchedItem );
229 $this->assertInternalType( 'array', $recentChangeInfo );
232 $this->assertEquals(
233 new WatchedItem( $user, new TitleValue( 0, 'Foo1' ), '20151212010101' ),
234 $items[0][0]
236 $this->assertEquals(
238 'rc_id' => 1,
239 'rc_namespace' => 0,
240 'rc_title' => 'Foo1',
241 'rc_timestamp' => '20151212010101',
242 'rc_type' => RC_NEW,
243 'rc_deleted' => 0,
245 $items[0][1]
248 $this->assertEquals(
249 new WatchedItem( $user, new TitleValue( 1, 'Foo2' ), null ),
250 $items[1][0]
252 $this->assertEquals(
254 'rc_id' => 2,
255 'rc_namespace' => 1,
256 'rc_title' => 'Foo2',
257 'rc_timestamp' => '20151212010102',
258 'rc_type' => RC_NEW,
259 'rc_deleted' => 0,
261 $items[1][1]
265 public function getWatchedItemsWithRecentChangeInfoOptionsProvider() {
266 return [
268 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_FLAGS ] ],
269 [ 'rc_type', 'rc_minor', 'rc_bot' ],
274 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_USER ] ],
275 [ 'rc_user_text' ],
280 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_USER_ID ] ],
281 [ 'rc_user' ],
286 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_COMMENT ] ],
287 [ 'rc_comment' ],
292 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_PATROL_INFO ] ],
293 [ 'rc_patrolled', 'rc_log_type' ],
298 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_SIZES ] ],
299 [ 'rc_old_len', 'rc_new_len' ],
304 [ 'includeFields' => [ WatchedItemQueryService::INCLUDE_LOG_INFO ] ],
305 [ 'rc_logid', 'rc_log_type', 'rc_log_action', 'rc_params' ],
310 [ 'namespaceIds' => [ 0, 1 ] ],
312 [ 'wl_namespace' => [ 0, 1 ] ],
316 [ 'namespaceIds' => [ 0, "1; DROP TABLE watchlist;\n--" ] ],
318 [ 'wl_namespace' => [ 0, 1 ] ],
322 [ 'rcTypes' => [ RC_EDIT, RC_NEW ] ],
324 [ 'rc_type' => [ RC_EDIT, RC_NEW ] ],
328 [ 'dir' => WatchedItemQueryService::DIR_OLDER ],
331 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ]
334 [ 'dir' => WatchedItemQueryService::DIR_NEWER ],
337 [ 'ORDER BY' => [ 'rc_timestamp', 'rc_id' ] ]
340 [ 'dir' => WatchedItemQueryService::DIR_OLDER, 'start' => '20151212010101' ],
342 [ "rc_timestamp <= '20151212010101'" ],
343 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ]
346 [ 'dir' => WatchedItemQueryService::DIR_OLDER, 'end' => '20151212010101' ],
348 [ "rc_timestamp >= '20151212010101'" ],
349 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ]
353 'dir' => WatchedItemQueryService::DIR_OLDER,
354 'start' => '20151212020101',
355 'end' => '20151212010101'
358 [ "rc_timestamp <= '20151212020101'", "rc_timestamp >= '20151212010101'" ],
359 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ]
362 [ 'dir' => WatchedItemQueryService::DIR_NEWER, 'start' => '20151212010101' ],
364 [ "rc_timestamp >= '20151212010101'" ],
365 [ 'ORDER BY' => [ 'rc_timestamp', 'rc_id' ] ]
368 [ 'dir' => WatchedItemQueryService::DIR_NEWER, 'end' => '20151212010101' ],
370 [ "rc_timestamp <= '20151212010101'" ],
371 [ 'ORDER BY' => [ 'rc_timestamp', 'rc_id' ] ]
375 'dir' => WatchedItemQueryService::DIR_NEWER,
376 'start' => '20151212010101',
377 'end' => '20151212020101'
380 [ "rc_timestamp >= '20151212010101'", "rc_timestamp <= '20151212020101'" ],
381 [ 'ORDER BY' => [ 'rc_timestamp', 'rc_id' ] ]
384 [ 'limit' => 10 ],
387 [ 'LIMIT' => 10 ],
390 [ 'limit' => "10; DROP TABLE watchlist;\n--" ],
393 [ 'LIMIT' => 10 ],
396 [ 'filters' => [ WatchedItemQueryService::FILTER_MINOR ] ],
398 [ 'rc_minor != 0' ],
402 [ 'filters' => [ WatchedItemQueryService::FILTER_NOT_MINOR ] ],
404 [ 'rc_minor = 0' ],
408 [ 'filters' => [ WatchedItemQueryService::FILTER_BOT ] ],
410 [ 'rc_bot != 0' ],
414 [ 'filters' => [ WatchedItemQueryService::FILTER_NOT_BOT ] ],
416 [ 'rc_bot = 0' ],
420 [ 'filters' => [ WatchedItemQueryService::FILTER_ANON ] ],
422 [ 'rc_user = 0' ],
426 [ 'filters' => [ WatchedItemQueryService::FILTER_NOT_ANON ] ],
428 [ 'rc_user != 0' ],
432 [ 'filters' => [ WatchedItemQueryService::FILTER_PATROLLED ] ],
434 [ 'rc_patrolled != 0' ],
438 [ 'filters' => [ WatchedItemQueryService::FILTER_NOT_PATROLLED ] ],
440 [ 'rc_patrolled = 0' ],
444 [ 'filters' => [ WatchedItemQueryService::FILTER_UNREAD ] ],
446 [ 'rc_timestamp >= wl_notificationtimestamp' ],
450 [ 'filters' => [ WatchedItemQueryService::FILTER_NOT_UNREAD ] ],
452 [ 'wl_notificationtimestamp IS NULL OR rc_timestamp < wl_notificationtimestamp' ],
456 [ 'onlyByUser' => 'SomeOtherUser' ],
458 [ 'rc_user_text' => 'SomeOtherUser' ],
462 [ 'notByUser' => 'SomeOtherUser' ],
464 [ "rc_user_text != 'SomeOtherUser'" ],
468 [ 'startFrom' => [ '20151212010101', 123 ], 'dir' => WatchedItemQueryService::DIR_OLDER ],
471 "(rc_timestamp < '20151212010101') OR ((rc_timestamp = '20151212010101') AND (rc_id <= 123))"
473 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ],
476 [ 'startFrom' => [ '20151212010101', 123 ], 'dir' => WatchedItemQueryService::DIR_NEWER ],
479 "(rc_timestamp > '20151212010101') OR ((rc_timestamp = '20151212010101') AND (rc_id >= 123))"
481 [ 'ORDER BY' => [ 'rc_timestamp', 'rc_id' ] ],
485 'startFrom' => [ '20151212010101', "123; DROP TABLE watchlist;\n--" ],
486 'dir' => WatchedItemQueryService::DIR_OLDER
490 "(rc_timestamp < '20151212010101') OR ((rc_timestamp = '20151212010101') AND (rc_id <= 123))"
492 [ 'ORDER BY' => [ 'rc_timestamp DESC', 'rc_id DESC' ] ],
498 * @dataProvider getWatchedItemsWithRecentChangeInfoOptionsProvider
500 public function testGetWatchedItemsWithRecentChangeInfo_optionsAndEmptyResult(
501 array $options,
502 array $expectedExtraFields,
503 array $expectedExtraConds,
504 array $expectedDbOptions
506 $expectedFields = array_merge(
508 'rc_id',
509 'rc_namespace',
510 'rc_title',
511 'rc_timestamp',
512 'rc_type',
513 'rc_deleted',
514 'wl_notificationtimestamp',
516 'rc_cur_id',
517 'rc_this_oldid',
518 'rc_last_oldid',
520 $expectedExtraFields
522 $expectedConds = array_merge(
523 [ 'wl_user' => 1, '(rc_this_oldid=page_latest) OR (rc_type=3)', ],
524 $expectedExtraConds
527 $mockDb = $this->getMockDb();
528 $mockDb->expects( $this->once() )
529 ->method( 'select' )
530 ->with(
531 [ 'recentchanges', 'watchlist', 'page' ],
532 $expectedFields,
533 $expectedConds,
534 $this->isType( 'string' ),
535 $expectedDbOptions,
537 'watchlist' => [
538 'INNER JOIN',
540 'wl_namespace=rc_namespace',
541 'wl_title=rc_title'
544 'page' => [
545 'LEFT JOIN',
546 'rc_cur_id=page_id',
550 ->will( $this->returnValue( [] ) );
552 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
553 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
555 $items = $queryService->getWatchedItemsWithRecentChangeInfo( $user, $options );
557 $this->assertEmpty( $items );
560 public function filterPatrolledOptionProvider() {
561 return [
562 [ WatchedItemQueryService::FILTER_PATROLLED ],
563 [ WatchedItemQueryService::FILTER_NOT_PATROLLED ],
568 * @dataProvider filterPatrolledOptionProvider
570 public function testGetWatchedItemsWithRecentChangeInfo_filterPatrolledAndUserWithNoPatrolRights(
571 $filtersOption
573 $mockDb = $this->getMockDb();
574 $mockDb->expects( $this->once() )
575 ->method( 'select' )
576 ->with(
577 [ 'recentchanges', 'watchlist', 'page' ],
578 $this->isType( 'array' ),
579 [ 'wl_user' => 1, '(rc_this_oldid=page_latest) OR (rc_type=3)' ],
580 $this->isType( 'string' ),
581 $this->isType( 'array' ),
582 $this->isType( 'array' )
584 ->will( $this->returnValue( [] ) );
586 $user = $this->getMockNonAnonUserWithIdAndNoPatrolRights( 1 );
588 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
589 $items = $queryService->getWatchedItemsWithRecentChangeInfo(
590 $user,
591 [ 'filters' => [ $filtersOption ] ]
594 $this->assertEmpty( $items );
597 public function mysqlIndexOptimizationProvider() {
598 return [
600 'mysql',
602 [ "rc_timestamp > ''" ],
605 'mysql',
606 [ 'start' => '20151212010101', 'dir' => WatchedItemQueryService::DIR_OLDER ],
607 [ "rc_timestamp <= '20151212010101'" ],
610 'mysql',
611 [ 'end' => '20151212010101', 'dir' => WatchedItemQueryService::DIR_OLDER ],
612 [ "rc_timestamp >= '20151212010101'" ],
615 'postgres',
623 * @dataProvider mysqlIndexOptimizationProvider
625 public function testGetWatchedItemsWithRecentChangeInfo_mysqlIndexOptimization(
626 $dbType,
627 array $options,
628 array $expectedExtraConds
630 $commonConds = [ 'wl_user' => 1, '(rc_this_oldid=page_latest) OR (rc_type=3)' ];
631 $conds = array_merge( $commonConds, $expectedExtraConds );
633 $mockDb = $this->getMockDb();
634 $mockDb->expects( $this->once() )
635 ->method( 'select' )
636 ->with(
637 [ 'recentchanges', 'watchlist', 'page' ],
638 $this->isType( 'array' ),
639 $conds,
640 $this->isType( 'string' ),
641 $this->isType( 'array' ),
642 $this->isType( 'array' )
644 ->will( $this->returnValue( [] ) );
645 $mockDb->expects( $this->any() )
646 ->method( 'getType' )
647 ->will( $this->returnValue( $dbType ) );
649 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
650 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
652 $items = $queryService->getWatchedItemsWithRecentChangeInfo( $user, $options );
654 $this->assertEmpty( $items );
657 public function userPermissionRelatedExtraChecksProvider() {
658 return [
661 'deletedhistory',
663 '(rc_type != ' . RC_LOG . ') OR ((rc_deleted & ' . LogPage::DELETED_ACTION . ') != ' .
664 LogPage::DELETED_ACTION . ')'
669 'suppressrevision',
671 '(rc_type != ' . RC_LOG . ') OR (' .
672 '(rc_deleted & ' . ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ') != ' .
673 ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ')'
678 'viewsuppressed',
680 '(rc_type != ' . RC_LOG . ') OR (' .
681 '(rc_deleted & ' . ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ') != ' .
682 ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ')'
686 [ 'onlyByUser' => 'SomeOtherUser' ],
687 'deletedhistory',
689 'rc_user_text' => 'SomeOtherUser',
690 '(rc_deleted & ' . Revision::DELETED_USER . ') != ' . Revision::DELETED_USER,
691 '(rc_type != ' . RC_LOG . ') OR ((rc_deleted & ' . LogPage::DELETED_ACTION . ') != ' .
692 LogPage::DELETED_ACTION . ')'
696 [ 'onlyByUser' => 'SomeOtherUser' ],
697 'suppressrevision',
699 'rc_user_text' => 'SomeOtherUser',
700 '(rc_deleted & ' . ( Revision::DELETED_USER | Revision::DELETED_RESTRICTED ) . ') != ' .
701 ( Revision::DELETED_USER | Revision::DELETED_RESTRICTED ),
702 '(rc_type != ' . RC_LOG . ') OR (' .
703 '(rc_deleted & ' . ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ') != ' .
704 ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ')'
708 [ 'onlyByUser' => 'SomeOtherUser' ],
709 'viewsuppressed',
711 'rc_user_text' => 'SomeOtherUser',
712 '(rc_deleted & ' . ( Revision::DELETED_USER | Revision::DELETED_RESTRICTED ) . ') != ' .
713 ( Revision::DELETED_USER | Revision::DELETED_RESTRICTED ),
714 '(rc_type != ' . RC_LOG . ') OR (' .
715 '(rc_deleted & ' . ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ') != ' .
716 ( LogPage::DELETED_ACTION | LogPage::DELETED_RESTRICTED ) . ')'
723 * @dataProvider userPermissionRelatedExtraChecksProvider
725 public function testGetWatchedItemsWithRecentChangeInfo_userPermissionRelatedExtraChecks(
726 array $options,
727 $notAllowedAction,
728 array $expectedExtraConds
730 $commonConds = [ 'wl_user' => 1, '(rc_this_oldid=page_latest) OR (rc_type=3)' ];
731 $conds = array_merge( $commonConds, $expectedExtraConds );
733 $mockDb = $this->getMockDb();
734 $mockDb->expects( $this->once() )
735 ->method( 'select' )
736 ->with(
737 [ 'recentchanges', 'watchlist', 'page' ],
738 $this->isType( 'array' ),
739 $conds,
740 $this->isType( 'string' ),
741 $this->isType( 'array' ),
742 $this->isType( 'array' )
744 ->will( $this->returnValue( [] ) );
746 $user = $this->getMockNonAnonUserWithIdAndRestrictedPermissions( 1, $notAllowedAction );
748 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
749 $items = $queryService->getWatchedItemsWithRecentChangeInfo( $user, $options );
751 $this->assertEmpty( $items );
754 public function testGetWatchedItemsWithRecentChangeInfo_allRevisionsOptionAndEmptyResult() {
755 $mockDb = $this->getMockDb();
756 $mockDb->expects( $this->once() )
757 ->method( 'select' )
758 ->with(
759 [ 'recentchanges', 'watchlist' ],
761 'rc_id',
762 'rc_namespace',
763 'rc_title',
764 'rc_timestamp',
765 'rc_type',
766 'rc_deleted',
767 'wl_notificationtimestamp',
769 'rc_cur_id',
770 'rc_this_oldid',
771 'rc_last_oldid',
773 [ 'wl_user' => 1, ],
774 $this->isType( 'string' ),
777 'watchlist' => [
778 'INNER JOIN',
780 'wl_namespace=rc_namespace',
781 'wl_title=rc_title'
786 ->will( $this->returnValue( [] ) );
788 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
789 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
791 $items = $queryService->getWatchedItemsWithRecentChangeInfo( $user, [ 'allRevisions' => true ] );
793 $this->assertEmpty( $items );
796 public function getWatchedItemsWithRecentChangeInfoInvalidOptionsProvider() {
797 return [
799 [ 'rcTypes' => [ 1337 ] ],
800 'Bad value for parameter $options[\'rcTypes\']',
803 [ 'rcTypes' => [ 'edit' ] ],
804 'Bad value for parameter $options[\'rcTypes\']',
807 [ 'rcTypes' => [ RC_EDIT, 1337 ] ],
808 'Bad value for parameter $options[\'rcTypes\']',
811 [ 'dir' => 'foo' ],
812 'Bad value for parameter $options[\'dir\']',
815 [ 'start' => '20151212010101' ],
816 'Bad value for parameter $options[\'dir\']: must be provided',
819 [ 'end' => '20151212010101' ],
820 'Bad value for parameter $options[\'dir\']: must be provided',
823 [ 'startFrom' => [ '20151212010101', 123 ] ],
824 'Bad value for parameter $options[\'dir\']: must be provided',
827 [ 'dir' => WatchedItemQueryService::DIR_OLDER, 'startFrom' => '20151212010101' ],
828 'Bad value for parameter $options[\'startFrom\']: must be a two-element array',
831 [ 'dir' => WatchedItemQueryService::DIR_OLDER, 'startFrom' => [ '20151212010101' ] ],
832 'Bad value for parameter $options[\'startFrom\']: must be a two-element array',
836 'dir' => WatchedItemQueryService::DIR_OLDER,
837 'startFrom' => [ '20151212010101', 123, 'foo' ]
839 'Bad value for parameter $options[\'startFrom\']: must be a two-element array',
842 [ 'watchlistOwner' => $this->getMockUnrestrictedNonAnonUserWithId( 2 ) ],
843 'Bad value for parameter $options[\'watchlistOwnerToken\']',
846 [ 'watchlistOwner' => 'Other User', 'watchlistOwnerToken' => 'some-token' ],
847 'Bad value for parameter $options[\'watchlistOwner\']',
853 * @dataProvider getWatchedItemsWithRecentChangeInfoInvalidOptionsProvider
855 public function testGetWatchedItemsWithRecentChangeInfo_invalidOptions(
856 array $options,
857 $expectedInExceptionMessage
859 $mockDb = $this->getMockDb();
860 $mockDb->expects( $this->never() )
861 ->method( $this->anything() );
863 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
864 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
866 $this->setExpectedException( InvalidArgumentException::class, $expectedInExceptionMessage );
867 $queryService->getWatchedItemsWithRecentChangeInfo( $user, $options );
870 public function testGetWatchedItemsWithRecentChangeInfo_usedInGeneratorOptionAndEmptyResult() {
871 $mockDb = $this->getMockDb();
872 $mockDb->expects( $this->once() )
873 ->method( 'select' )
874 ->with(
875 [ 'recentchanges', 'watchlist', 'page' ],
877 'rc_id',
878 'rc_namespace',
879 'rc_title',
880 'rc_timestamp',
881 'rc_type',
882 'rc_deleted',
883 'wl_notificationtimestamp',
884 'rc_cur_id',
886 [ 'wl_user' => 1, '(rc_this_oldid=page_latest) OR (rc_type=3)' ],
887 $this->isType( 'string' ),
890 'watchlist' => [
891 'INNER JOIN',
893 'wl_namespace=rc_namespace',
894 'wl_title=rc_title'
897 'page' => [
898 'LEFT JOIN',
899 'rc_cur_id=page_id',
903 ->will( $this->returnValue( [] ) );
905 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
906 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
908 $items = $queryService->getWatchedItemsWithRecentChangeInfo(
909 $user,
910 [ 'usedInGenerator' => true ]
913 $this->assertEmpty( $items );
916 public function testGetWatchedItemsWithRecentChangeInfo_usedInGeneratorAllRevisionsOptions() {
917 $mockDb = $this->getMockDb();
918 $mockDb->expects( $this->once() )
919 ->method( 'select' )
920 ->with(
921 [ 'recentchanges', 'watchlist' ],
923 'rc_id',
924 'rc_namespace',
925 'rc_title',
926 'rc_timestamp',
927 'rc_type',
928 'rc_deleted',
929 'wl_notificationtimestamp',
930 'rc_this_oldid',
932 [ 'wl_user' => 1 ],
933 $this->isType( 'string' ),
936 'watchlist' => [
937 'INNER JOIN',
939 'wl_namespace=rc_namespace',
940 'wl_title=rc_title'
945 ->will( $this->returnValue( [] ) );
947 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
948 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
950 $items = $queryService->getWatchedItemsWithRecentChangeInfo(
951 $user,
952 [ 'usedInGenerator' => true, 'allRevisions' => true, ]
955 $this->assertEmpty( $items );
958 public function testGetWatchedItemsWithRecentChangeInfo_watchlistOwnerOptionAndEmptyResult() {
959 $mockDb = $this->getMockDb();
960 $mockDb->expects( $this->once() )
961 ->method( 'select' )
962 ->with(
963 $this->isType( 'array' ),
964 $this->isType( 'array' ),
966 'wl_user' => 2,
967 '(rc_this_oldid=page_latest) OR (rc_type=3)',
969 $this->isType( 'string' ),
970 $this->isType( 'array' ),
971 $this->isType( 'array' )
973 ->will( $this->returnValue( [] ) );
975 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
976 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
977 $otherUser = $this->getMockUnrestrictedNonAnonUserWithId( 2 );
978 $otherUser->expects( $this->once() )
979 ->method( 'getOption' )
980 ->with( 'watchlisttoken' )
981 ->willReturn( '0123456789abcdef' );
983 $items = $queryService->getWatchedItemsWithRecentChangeInfo(
984 $user,
985 [ 'watchlistOwner' => $otherUser, 'watchlistOwnerToken' => '0123456789abcdef' ]
988 $this->assertEmpty( $items );
991 public function invalidWatchlistTokenProvider() {
992 return [
993 [ 'wrongToken' ],
994 [ '' ],
999 * @dataProvider invalidWatchlistTokenProvider
1001 public function testGetWatchedItemsWithRecentChangeInfo_watchlistOwnerAndInvalidToken( $token ) {
1002 $mockDb = $this->getMockDb();
1003 $mockDb->expects( $this->never() )
1004 ->method( $this->anything() );
1006 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
1007 $user = $this->getMockUnrestrictedNonAnonUserWithId( 1 );
1008 $otherUser = $this->getMockUnrestrictedNonAnonUserWithId( 2 );
1009 $otherUser->expects( $this->once() )
1010 ->method( 'getOption' )
1011 ->with( 'watchlisttoken' )
1012 ->willReturn( '0123456789abcdef' );
1014 $this->setExpectedException( UsageException::class, 'Incorrect watchlist token provided' );
1015 $queryService->getWatchedItemsWithRecentChangeInfo(
1016 $user,
1017 [ 'watchlistOwner' => $otherUser, 'watchlistOwnerToken' => $token ]
1021 public function testGetWatchedItemsForUser() {
1022 $mockDb = $this->getMockDb();
1023 $mockDb->expects( $this->once() )
1024 ->method( 'select' )
1025 ->with(
1026 'watchlist',
1027 [ 'wl_namespace', 'wl_title', 'wl_notificationtimestamp' ],
1028 [ 'wl_user' => 1 ]
1030 ->will( $this->returnValue( [
1031 $this->getFakeRow( [
1032 'wl_namespace' => 0,
1033 'wl_title' => 'Foo1',
1034 'wl_notificationtimestamp' => '20151212010101',
1035 ] ),
1036 $this->getFakeRow( [
1037 'wl_namespace' => 1,
1038 'wl_title' => 'Foo2',
1039 'wl_notificationtimestamp' => null,
1040 ] ),
1041 ] ) );
1043 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
1044 $user = $this->getMockNonAnonUserWithId( 1 );
1046 $items = $queryService->getWatchedItemsForUser( $user );
1048 $this->assertInternalType( 'array', $items );
1049 $this->assertCount( 2, $items );
1050 $this->assertContainsOnlyInstancesOf( WatchedItem::class, $items );
1051 $this->assertEquals(
1052 new WatchedItem( $user, new TitleValue( 0, 'Foo1' ), '20151212010101' ),
1053 $items[0]
1055 $this->assertEquals(
1056 new WatchedItem( $user, new TitleValue( 1, 'Foo2' ), null ),
1057 $items[1]
1061 public function provideGetWatchedItemsForUserOptions() {
1062 return [
1064 [ 'namespaceIds' => [ 0, 1 ], ],
1065 [ 'wl_namespace' => [ 0, 1 ], ],
1069 [ 'sort' => WatchedItemQueryService::SORT_ASC, ],
1071 [ 'ORDER BY' => [ 'wl_namespace ASC', 'wl_title ASC' ] ]
1075 'namespaceIds' => [ 0 ],
1076 'sort' => WatchedItemQueryService::SORT_ASC,
1078 [ 'wl_namespace' => [ 0 ], ],
1079 [ 'ORDER BY' => 'wl_title ASC' ]
1082 [ 'limit' => 10 ],
1084 [ 'LIMIT' => 10 ]
1088 'namespaceIds' => [ 0, "1; DROP TABLE watchlist;\n--" ],
1089 'limit' => "10; DROP TABLE watchlist;\n--",
1091 [ 'wl_namespace' => [ 0, 1 ], ],
1092 [ 'LIMIT' => 10 ]
1095 [ 'filter' => WatchedItemQueryService::FILTER_CHANGED ],
1096 [ 'wl_notificationtimestamp IS NOT NULL' ],
1100 [ 'filter' => WatchedItemQueryService::FILTER_NOT_CHANGED ],
1101 [ 'wl_notificationtimestamp IS NULL' ],
1105 [ 'sort' => WatchedItemQueryService::SORT_DESC, ],
1107 [ 'ORDER BY' => [ 'wl_namespace DESC', 'wl_title DESC' ] ]
1111 'namespaceIds' => [ 0 ],
1112 'sort' => WatchedItemQueryService::SORT_DESC,
1114 [ 'wl_namespace' => [ 0 ], ],
1115 [ 'ORDER BY' => 'wl_title DESC' ]
1121 * @dataProvider provideGetWatchedItemsForUserOptions
1123 public function testGetWatchedItemsForUser_optionsAndEmptyResult(
1124 array $options,
1125 array $expectedConds,
1126 array $expectedDbOptions
1128 $mockDb = $this->getMockDb();
1129 $user = $this->getMockNonAnonUserWithId( 1 );
1131 $expectedConds = array_merge( [ 'wl_user' => 1 ], $expectedConds );
1132 $mockDb->expects( $this->once() )
1133 ->method( 'select' )
1134 ->with(
1135 'watchlist',
1136 [ 'wl_namespace', 'wl_title', 'wl_notificationtimestamp' ],
1137 $expectedConds,
1138 $this->isType( 'string' ),
1139 $expectedDbOptions
1141 ->will( $this->returnValue( [] ) );
1143 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
1145 $items = $queryService->getWatchedItemsForUser( $user, $options );
1146 $this->assertEmpty( $items );
1149 public function provideGetWatchedItemsForUser_fromUntilStartFromOptions() {
1150 return [
1153 'from' => new TitleValue( 0, 'SomeDbKey' ),
1154 'sort' => WatchedItemQueryService::SORT_ASC
1156 [ "(wl_namespace > 0) OR ((wl_namespace = 0) AND (wl_title >= 'SomeDbKey'))", ],
1157 [ 'ORDER BY' => [ 'wl_namespace ASC', 'wl_title ASC' ] ]
1161 'from' => new TitleValue( 0, 'SomeDbKey' ),
1162 'sort' => WatchedItemQueryService::SORT_DESC,
1164 [ "(wl_namespace < 0) OR ((wl_namespace = 0) AND (wl_title <= 'SomeDbKey'))", ],
1165 [ 'ORDER BY' => [ 'wl_namespace DESC', 'wl_title DESC' ] ]
1169 'until' => new TitleValue( 0, 'SomeDbKey' ),
1170 'sort' => WatchedItemQueryService::SORT_ASC
1172 [ "(wl_namespace < 0) OR ((wl_namespace = 0) AND (wl_title <= 'SomeDbKey'))", ],
1173 [ 'ORDER BY' => [ 'wl_namespace ASC', 'wl_title ASC' ] ]
1177 'until' => new TitleValue( 0, 'SomeDbKey' ),
1178 'sort' => WatchedItemQueryService::SORT_DESC
1180 [ "(wl_namespace > 0) OR ((wl_namespace = 0) AND (wl_title >= 'SomeDbKey'))", ],
1181 [ 'ORDER BY' => [ 'wl_namespace DESC', 'wl_title DESC' ] ]
1185 'from' => new TitleValue( 0, 'AnotherDbKey' ),
1186 'until' => new TitleValue( 0, 'SomeOtherDbKey' ),
1187 'startFrom' => new TitleValue( 0, 'SomeDbKey' ),
1188 'sort' => WatchedItemQueryService::SORT_ASC
1191 "(wl_namespace > 0) OR ((wl_namespace = 0) AND (wl_title >= 'AnotherDbKey'))",
1192 "(wl_namespace < 0) OR ((wl_namespace = 0) AND (wl_title <= 'SomeOtherDbKey'))",
1193 "(wl_namespace > 0) OR ((wl_namespace = 0) AND (wl_title >= 'SomeDbKey'))",
1195 [ 'ORDER BY' => [ 'wl_namespace ASC', 'wl_title ASC' ] ]
1199 'from' => new TitleValue( 0, 'SomeOtherDbKey' ),
1200 'until' => new TitleValue( 0, 'AnotherDbKey' ),
1201 'startFrom' => new TitleValue( 0, 'SomeDbKey' ),
1202 'sort' => WatchedItemQueryService::SORT_DESC
1205 "(wl_namespace < 0) OR ((wl_namespace = 0) AND (wl_title <= 'SomeOtherDbKey'))",
1206 "(wl_namespace > 0) OR ((wl_namespace = 0) AND (wl_title >= 'AnotherDbKey'))",
1207 "(wl_namespace < 0) OR ((wl_namespace = 0) AND (wl_title <= 'SomeDbKey'))",
1209 [ 'ORDER BY' => [ 'wl_namespace DESC', 'wl_title DESC' ] ]
1215 * @dataProvider provideGetWatchedItemsForUser_fromUntilStartFromOptions
1217 public function testGetWatchedItemsForUser_fromUntilStartFromOptions(
1218 array $options,
1219 array $expectedConds,
1220 array $expectedDbOptions
1222 $user = $this->getMockNonAnonUserWithId( 1 );
1224 $expectedConds = array_merge( [ 'wl_user' => 1 ], $expectedConds );
1226 $mockDb = $this->getMockDb();
1227 $mockDb->expects( $this->any() )
1228 ->method( 'addQuotes' )
1229 ->will( $this->returnCallback( function( $value ) {
1230 return "'$value'";
1231 } ) );
1232 $mockDb->expects( $this->any() )
1233 ->method( 'makeList' )
1234 ->with(
1235 $this->isType( 'array' ),
1236 $this->isType( 'int' )
1238 ->will( $this->returnCallback( function( $a, $conj ) {
1239 $sqlConj = $conj === LIST_AND ? ' AND ' : ' OR ';
1240 return join( $sqlConj, array_map( function( $s ) {
1241 return '(' . $s . ')';
1242 }, $a
1243 ) );
1244 } ) );
1245 $mockDb->expects( $this->once() )
1246 ->method( 'select' )
1247 ->with(
1248 'watchlist',
1249 [ 'wl_namespace', 'wl_title', 'wl_notificationtimestamp' ],
1250 $expectedConds,
1251 $this->isType( 'string' ),
1252 $expectedDbOptions
1254 ->will( $this->returnValue( [] ) );
1256 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
1258 $items = $queryService->getWatchedItemsForUser( $user, $options );
1259 $this->assertEmpty( $items );
1262 public function getWatchedItemsForUserInvalidOptionsProvider() {
1263 return [
1265 [ 'sort' => 'foo' ],
1266 'Bad value for parameter $options[\'sort\']'
1269 [ 'filter' => 'foo' ],
1270 'Bad value for parameter $options[\'filter\']'
1273 [ 'from' => new TitleValue( 0, 'SomeDbKey' ), ],
1274 'Bad value for parameter $options[\'sort\']: must be provided'
1277 [ 'until' => new TitleValue( 0, 'SomeDbKey' ), ],
1278 'Bad value for parameter $options[\'sort\']: must be provided'
1281 [ 'startFrom' => new TitleValue( 0, 'SomeDbKey' ), ],
1282 'Bad value for parameter $options[\'sort\']: must be provided'
1288 * @dataProvider getWatchedItemsForUserInvalidOptionsProvider
1290 public function testGetWatchedItemsForUser_invalidOptionThrowsException(
1291 array $options,
1292 $expectedInExceptionMessage
1294 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $this->getMockDb() ) );
1296 $this->setExpectedException( InvalidArgumentException::class, $expectedInExceptionMessage );
1297 $queryService->getWatchedItemsForUser( $this->getMockNonAnonUserWithId( 1 ), $options );
1300 public function testGetWatchedItemsForUser_userNotAllowedToViewWatchlist() {
1301 $mockDb = $this->getMockDb();
1303 $mockDb->expects( $this->never() )
1304 ->method( $this->anything() );
1306 $queryService = new WatchedItemQueryService( $this->getMockLoadBalancer( $mockDb ) );
1308 $items = $queryService->getWatchedItemsForUser( $this->getMockAnonUser() );
1309 $this->assertEmpty( $items );