Merge "Drop cache interwiki"
[mediawiki.git] / tests / phpunit / includes / specials / SpecialRecentChangesTest.php
blob899b502ae735ab9ced0694af4ebd2c471ab3b725
1 <?php
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;
14 /**
15 * Test class for SpecialRecentchanges class
17 * @group Database
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()
37 /**
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() {
48 return [
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() {
68 return [
70 // hidebots=1 is default for Special:RecentChanges
71 [ 'hideanons' => 1, 'hideliu' => 1 ],
72 true,
73 [ 'hideliu' => 1 ],
74 false,
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', '' );
84 // Set up RC.
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()
109 ->select( 'wl_id' )
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() );
137 // Set up RC.
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(
145 $this->getPage(),
147 new FauxRequest()
149 $this->assertStringContainsString( 'Experience page', $html );
151 // newcomer
152 $req = new FauxRequest();
153 $req->setVal( 'userExpLevel', 'newcomer' );
154 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
155 $this->getPage(),
157 $req
159 $this->assertStringContainsString( 'registered summary', $html );
161 // anon
162 $req = new FauxRequest();
163 $req->setVal( 'userExpLevel', 'unregistered' );
164 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
165 $this->getPage(),
167 $req
169 $this->assertStringContainsString( 'anon summary', $html );
170 $this->assertStringNotContainsString( 'registered summary', $html );
172 // registered
173 $req = new FauxRequest();
174 $req->setVal( 'userExpLevel', 'registered' );
175 [ $html ] = ( new SpecialPageExecutor() )->executeSpecialPage(
176 $this->getPage(),
178 $req
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() {
203 return [
204 [ false ],
205 [ true ]
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' );
220 $page = new class (
221 $dense,
222 $this->getServiceContainer()->getWatchedItemStore(),
223 $this->getServiceContainer()->getMessageCache(),
224 $this->getServiceContainer()->getUserOptionsLookup()
225 ) extends SpecialRecentChanges {
226 private $dense;
228 public function __construct(
229 $dense,
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 ) {
239 return $this->dense;
243 ( new SpecialPageExecutor() )->executeSpecialPage( $page, '', $req );
244 $this->assertTrue( true );