3 use MediaWiki\Context\RequestContext
;
4 use MediaWiki\MainConfigNames
;
5 use MediaWiki\Request\FauxRequest
;
6 use MediaWiki\Specials\SpecialRecentChanges
;
7 use MediaWiki\Tests\SpecialPage\AbstractChangesListSpecialPageTestCase
;
8 use MediaWiki\Tests\Unit\Permissions\MockAuthorityTrait
;
9 use MediaWiki\Tests\User\TempUser\TempUserTestTrait
;
10 use MediaWiki\Title\Title
;
11 use MediaWiki\Watchlist\WatchedItemStoreInterface
;
12 use Wikimedia\TestingAccessWrapper
;
15 * Test class for SpecialRecentchanges class
19 * @covers \MediaWiki\Specials\SpecialRecentChanges
20 * @covers \MediaWiki\SpecialPage\ChangesListSpecialPage
22 class SpecialRecentChangesTest
extends AbstractChangesListSpecialPageTestCase
{
23 use MockAuthorityTrait
;
24 use TempUserTestTrait
;
26 protected function getPage(): SpecialRecentChanges
{
27 return new SpecialRecentChanges(
28 $this->getServiceContainer()->getWatchedItemStore(),
29 $this->getServiceContainer()->getMessageCache(),
30 $this->getServiceContainer()->getUserOptionsLookup(),
31 $this->getServiceContainer()->getChangeTagsStore(),
32 $this->getServiceContainer()->getUserIdentityUtils(),
33 $this->getServiceContainer()->getTempUserConfig()
38 * @return TestingAccessWrapper
40 protected function getPageAccessWrapper() {
41 return TestingAccessWrapper
::newFromObject( $this->getPage() );
44 // Below providers should only be for features specific to
45 // RecentChanges. Otherwise, it should go in ChangesListSpecialPageTest
47 public function provideParseParameters() {
49 [ 'limit=123', [ 'limit' => '123' ] ],
51 [ '234', [ 'limit' => '234' ] ],
53 [ 'days=3', [ 'days' => '3' ] ],
55 [ 'days=0.25', [ 'days' => '0.25' ] ],
57 [ 'namespace=5', [ 'namespace' => '5' ] ],
59 [ 'namespace=5|3', [ 'namespace' => '5|3' ] ],
61 [ 'tagfilter=foo', [ 'tagfilter' => 'foo' ] ],
63 [ 'tagfilter=foo;bar', [ 'tagfilter' => 'foo;bar' ] ],
67 public function validateOptionsProvider() {
70 // hidebots=1 is default for Special:RecentChanges
71 [ 'hideanons' => 1, 'hideliu' => 1 ],
79 public function testAddWatchlistJoins() {
80 // Edit a test page so that it shows up in RC.
81 $testPage = $this->getExistingTestPage( 'Test page' );
82 $this->editPage( $testPage, 'Test content', '' );
85 $context = new RequestContext
;
86 $context->setTitle( Title
::newFromText( __METHOD__
) );
87 $context->setUser( $this->getTestUser()->getUser() );
88 $context->setRequest( new FauxRequest
);
90 // Confirm that the test page is in RC.
91 $rc1 = $this->getPage();
92 $rc1->setContext( $context );
93 $rc1->execute( null );
94 $this->assertStringContainsString( 'Test page', $rc1->getOutput()->getHTML() );
95 $this->assertStringContainsString( 'mw-changeslist-line-not-watched', $rc1->getOutput()->getHTML() );
97 // Watch the page, and check that it's now watched in RC.
98 $watchedItemStore = $this->getServiceContainer()->getWatchedItemStore();
99 $watchedItemStore->addWatch( $context->getUser(), $testPage );
100 $rc2 = $this->getPage();
101 $rc2->setContext( $context );
102 $rc2->execute( null );
103 $this->assertStringContainsString( 'Test page', $rc2->getOutput()->getHTML() );
104 $this->assertStringContainsString( 'mw-changeslist-line-watched', $rc2->getOutput()->getHTML() );
106 // Force a past expiry date on the watchlist item.
107 $db = $this->getDb();
108 $watchedItemId = $db->newSelectQueryBuilder()
110 ->from( 'watchlist' )
111 ->where( [ 'wl_namespace' => $testPage->getNamespace(), 'wl_title' => $testPage->getDBkey() ] )
112 ->caller( __METHOD__
)->fetchField();
113 $db->newUpdateQueryBuilder()
114 ->update( 'watchlist_expiry' )
115 ->set( [ 'we_expiry' => $db->timestamp( '20200101000000' ) ] )
116 ->where( [ 'we_item' => $watchedItemId ] )
117 ->caller( __METHOD__
)->execute();
119 // Check that the page is still in RC, but that it's no longer watched.
120 $rc3 = $this->getPage();
121 $rc3->setContext( $context );
122 $rc3->execute( null );
123 $this->assertStringContainsString( 'Test page', $rc3->getOutput()->getHTML() );
124 $this->assertStringContainsString( 'mw-changeslist-line-not-watched', $rc3->getOutput()->getHTML() );
127 public function testExperienceLevelFilter() {
128 $this->disableAutoCreateTempUser();
130 // Edit a test page so that it shows up in RC.
131 $testPage = $this->getExistingTestPage( 'Experience page' );
132 $this->editPage( $testPage, 'Registered content',
133 'registered summary', NS_MAIN
, $this->getTestUser()->getUser() );
134 $this->editPage( $testPage, 'Anon content',
135 'anon summary', NS_MAIN
, $this->mockAnonUltimateAuthority() );
138 $context = new RequestContext
;
139 $context->setTitle( Title
::newFromText( __METHOD__
) );
140 $context->setUser( $this->getTestUser()->getUser() );
141 $context->setRequest( new FauxRequest
);
143 // Confirm that the test page is in RC.
144 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
149 $this->assertStringContainsString( 'Experience page', $html );
152 $req = new FauxRequest();
153 $req->setVal( 'userExpLevel', 'newcomer' );
154 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
159 $this->assertStringContainsString( 'registered summary', $html );
162 $req = new FauxRequest();
163 $req->setVal( 'userExpLevel', 'unregistered' );
164 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
169 $this->assertStringContainsString( 'anon summary', $html );
170 $this->assertStringNotContainsString( 'registered summary', $html );
173 $req = new FauxRequest();
174 $req->setVal( 'userExpLevel', 'registered' );
175 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
180 $this->assertStringContainsString( 'registered summary', $html );
181 $this->assertStringNotContainsString( 'anon summary', $html );
185 * This integration test just tries to run the isDenseFilter() queries, to
186 * check for syntax errors etc. It doesn't verify the logic.
188 public function testIsDenseTagFilter() {
189 $this->getServiceContainer()->getChangeTagsStore()->defineTag( 'rc-test-tag' );
190 $req = new FauxRequest();
191 $req->setVal( 'tagfilter', 'rc-test-tag' );
192 $page = $this->getPage();
194 // Make sure thresholds are passed
195 $page->denseRcSizeThreshold
= 0;
196 $this->overrideConfigValue( MainConfigNames
::MiserMode
, true );
198 ( new SpecialPageExecutor() )->executeSpecialPage( $page, '', $req );
199 $this->assertTrue( true );
202 public static function provideDenseTagFilter() {
210 * This integration test injects the return value of isDenseFilter(),
211 * verifying the correctness of the resulting STRAIGHT_JOIN.
213 * @dataProvider provideDenseTagFilter
215 public function testDenseTagFilter( $dense ) {
216 $this->getServiceContainer()->getChangeTagsStore()->defineTag( 'rc-test-tag' );
217 $req = new FauxRequest();
218 $req->setVal( 'tagfilter', 'rc-test-tag' );
222 $this->getServiceContainer()->getWatchedItemStore(),
223 $this->getServiceContainer()->getMessageCache(),
224 $this->getServiceContainer()->getUserOptionsLookup()
225 ) extends SpecialRecentChanges
{
228 public function __construct(
230 ?WatchedItemStoreInterface
$watchedItemStore = null,
231 ?MessageCache
$messageCache = null,
232 ?\MediaWiki\User\Options\UserOptionsLookup
$userOptionsLookup = null
234 parent
::__construct( $watchedItemStore, $messageCache, $userOptionsLookup );
235 $this->dense
= $dense;
238 protected function isDenseTagFilter( $tagIds, $limit ) {
243 ( new SpecialPageExecutor() )->executeSpecialPage( $page, '', $req );
244 $this->assertTrue( true );