6 class LinkerTest
extends MediaWikiLangTestCase
{
9 * @dataProvider provideCasesForUserLink
10 * @covers Linker::userLink
12 public function testUserLink( $expected, $userId, $userName, $altUserName = false, $msg = '' ) {
13 $this->setMwGlobals( [
14 'wgArticlePath' => '/wiki/$1',
17 // We'd also test the warning, but injecting a mock logger into a static method is tricky.
18 if ( $userName === '' ) {
19 Wikimedia\
suppressWarnings();
21 $actual = Linker
::userLink( $userId, $userName, $altUserName );
22 if ( $userName === '' ) {
23 Wikimedia\restoreWarnings
();
26 $this->assertEquals( $expected, $actual, $msg );
29 public static function provideCasesForUserLink() {
34 # - optional altUserName
37 # Empty name (T222529)
38 'Empty username, userid 0' => [ '(no username available)', 0, '' ],
39 'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
41 # ## ANONYMOUS USER ########################################
43 '<a href="/wiki/Special:Contributions/JohnDoe" '
44 . 'class="mw-userlink mw-anonuserlink" '
45 . 'title="Special:Contributions/JohnDoe"><bdi>JohnDoe</bdi></a>',
49 '<a href="/wiki/Special:Contributions/::1" '
50 . 'class="mw-userlink mw-anonuserlink" '
51 . 'title="Special:Contributions/::1"><bdi>::1</bdi></a>',
53 'Anonymous with pretty IPv6'
56 '<a href="/wiki/Special:Contributions/0:0:0:0:0:0:0:1" '
57 . 'class="mw-userlink mw-anonuserlink" '
58 . 'title="Special:Contributions/0:0:0:0:0:0:0:1"><bdi>::1</bdi></a>',
59 0, '0:0:0:0:0:0:0:1', false,
60 'Anonymous with almost pretty IPv6'
63 '<a href="/wiki/Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001" '
64 . 'class="mw-userlink mw-anonuserlink" '
65 . 'title="Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001"><bdi>::1</bdi></a>',
66 0, '0000:0000:0000:0000:0000:0000:0000:0001', false,
67 'Anonymous with full IPv6'
70 '<a href="/wiki/Special:Contributions/::1" '
71 . 'class="mw-userlink mw-anonuserlink" '
72 . 'title="Special:Contributions/::1"><bdi>AlternativeUsername</bdi></a>',
73 0, '::1', 'AlternativeUsername',
74 'Anonymous with pretty IPv6 and an alternative username'
79 '<a href="/wiki/Special:Contributions/127.0.0.1" '
80 . 'class="mw-userlink mw-anonuserlink" '
81 . 'title="Special:Contributions/127.0.0.1"><bdi>127.0.0.1</bdi></a>',
82 0, '127.0.0.1', false,
86 '<a href="/wiki/Special:Contributions/127.0.0.1" '
87 . 'class="mw-userlink mw-anonuserlink" '
88 . 'title="Special:Contributions/127.0.0.1"><bdi>AlternativeUsername</bdi></a>',
89 0, '127.0.0.1', 'AlternativeUsername',
90 'Anonymous with IPv4 and an alternative username'
93 # ## Regular user ##########################################
99 * @dataProvider provideUserToolLinks
100 * @covers Linker::userToolLinks
101 * @param string $expected
103 * @param string $userText
105 public function testUserToolLinks( $expected, $userId, $userText ) {
106 // We'd also test the warning, but injecting a mock logger into a static method is tricky.
107 if ( $userText === '' ) {
108 Wikimedia\
suppressWarnings();
110 $actual = Linker
::userToolLinks( $userId, $userText );
111 if ( $userText === '' ) {
112 Wikimedia\restoreWarnings
();
115 $this->assertSame( $expected, $actual );
118 public static function provideUserToolLinks() {
120 // Empty name (T222529)
121 'Empty username, userid 0' => [ ' (no username available)', 0, '' ],
122 'Empty username, userid > 0' => [ ' (no username available)', 73, '' ],
127 * @dataProvider provideUserTalkLink
128 * @covers Linker::userTalkLink
129 * @param string $expected
131 * @param string $userText
133 public function testUserTalkLink( $expected, $userId, $userText ) {
134 // We'd also test the warning, but injecting a mock logger into a static method is tricky.
135 if ( $userText === '' ) {
136 Wikimedia\
suppressWarnings();
138 $actual = Linker
::userTalkLink( $userId, $userText );
139 if ( $userText === '' ) {
140 Wikimedia\restoreWarnings
();
143 $this->assertSame( $expected, $actual );
146 public static function provideUserTalkLink() {
148 // Empty name (T222529)
149 'Empty username, userid 0' => [ '(no username available)', 0, '' ],
150 'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
155 * @dataProvider provideBlockLink
156 * @covers Linker::blockLink
157 * @param string $expected
159 * @param string $userText
161 public function testBlockLink( $expected, $userId, $userText ) {
162 // We'd also test the warning, but injecting a mock logger into a static method is tricky.
163 if ( $userText === '' ) {
164 Wikimedia\
suppressWarnings();
166 $actual = Linker
::blockLink( $userId, $userText );
167 if ( $userText === '' ) {
168 Wikimedia\restoreWarnings
();
171 $this->assertSame( $expected, $actual );
174 public static function provideBlockLink() {
176 // Empty name (T222529)
177 'Empty username, userid 0' => [ '(no username available)', 0, '' ],
178 'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
183 * @dataProvider provideEmailLink
184 * @covers Linker::emailLink
185 * @param string $expected
187 * @param string $userText
189 public function testEmailLink( $expected, $userId, $userText ) {
190 // We'd also test the warning, but injecting a mock logger into a static method is tricky.
191 if ( $userText === '' ) {
192 Wikimedia\
suppressWarnings();
194 $actual = Linker
::emailLink( $userId, $userText );
195 if ( $userText === '' ) {
196 Wikimedia\restoreWarnings
();
199 $this->assertSame( $expected, $actual );
202 public static function provideEmailLink() {
204 // Empty name (T222529)
205 'Empty username, userid 0' => [ '(no username available)', 0, '' ],
206 'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
211 * @dataProvider provideCasesForFormatComment
212 * @covers Linker::formatComment
213 * @covers Linker::formatAutocomments
214 * @covers Linker::formatLinksInComment
216 public function testFormatComment(
217 $expected, $comment, $title = false, $local = false, $wikiId = null
219 $conf = new SiteConfiguration();
222 'enwiki' => '//en.example.org',
223 'dewiki' => '//de.example.org',
230 $conf->suffixes
= [ 'wiki' ];
232 $this->setMwGlobals( [
233 'wgScript' => '/wiki/index.php',
234 'wgArticlePath' => '/wiki/$1',
235 'wgCapitalLinks' => true,
239 if ( $title === false ) {
240 // We need a page title that exists
241 $title = Title
::newFromText( 'Special:BlankPage' );
246 Linker
::formatComment( $comment, $title, $local, $wikiId )
250 public function provideCasesForFormatComment() {
251 $wikiId = 'enwiki'; // $wgConf has a fake entry for this
253 // phpcs:disable Generic.Files.LineLength
255 // Linker::formatComment
265 "'''not bolded'''",
269 "try <script>evil</scipt> things",
270 "try <script>evil</scipt> things",
272 // Linker::formatAutocomments
274 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a></span></span>',
278 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#linkie.3F" title="Special:BlankPage">→‎[[linkie?]]</a></span></span>',
282 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage" title="Special:BlankPage">→‎</a>: </span> // Edit via via</span>',
283 // Regression test for T222857
284 "/* */ // Edit via via",
287 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a>: </span> post</span>',
288 "/* autocomment */ post",
291 'pre <span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a></span></span>',
292 "pre /* autocomment */",
295 'pre <span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a>: </span> post</span>',
296 "pre /* autocomment */ post",
299 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a>: </span> multiple? <span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment2" title="Special:BlankPage">→‎autocomment2</a></span></span></span>',
300 "/* autocomment */ multiple? /* autocomment2 */",
303 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment_containing_.2F.2A" title="Special:BlankPage">→‎autocomment containing /*</a>: </span> T70361</span>',
304 "/* autocomment containing /* */ T70361"
307 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment_containing_.22quotes.22" title="Special:BlankPage">→‎autocomment containing "quotes"</a></span></span>',
308 "/* autocomment containing \"quotes\" */"
311 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment_containing_.3Cscript.3Etags.3C.2Fscript.3E" title="Special:BlankPage">→‎autocomment containing <script>tags</script></a></span></span>',
312 "/* autocomment containing <script>tags</script> */"
315 '<span dir="auto"><span class="autocomment"><a href="#autocomment">→‎autocomment</a></span></span>',
320 '<span dir="auto"><span class="autocomment">autocomment</span></span>',
325 '<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→‎autocomment</a></span></span>',
330 '<span dir="auto"><span class="autocomment"><a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage#autocomment">→‎autocomment</a></span></span>',
332 false, false, $wikiId
334 // Linker::formatLinksInComment
336 'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">link</a> def',
340 'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">text</a> def',
341 "abc [[link|text]] def",
344 'abc <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a> def',
345 "abc [[Special:BlankPage|]] def",
348 'abc <a href="/wiki/index.php?title=%C4%84%C5%9B%C5%BC&action=edit&redlink=1" class="new" title="Ąśż (page does not exist)">ąśż</a> def',
349 "abc [[%C4%85%C5%9B%C5%BC]] def",
352 'abc <a href="/wiki/Special:BlankPage#section" title="Special:BlankPage">#section</a> def',
353 "abc [[#section]] def",
356 'abc <a href="/wiki/index.php?title=/subpage&action=edit&redlink=1" class="new" title="/subpage (page does not exist)">/subpage</a> def',
357 "abc [[/subpage]] def",
360 'abc <a href="/wiki/index.php?title=%22evil!%22&action=edit&redlink=1" class="new" title=""evil!" (page does not exist)">"evil!"</a> def',
361 "abc [[\"evil!\"]] def",
364 'abc [[<script>very evil</script>]] def',
365 "abc [[<script>very evil</script>]] def",
372 'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">link</a> def',
377 'abc <a class="external" rel="nofollow" href="//en.example.org/w/Link">link</a> def',
379 false, false, $wikiId
386 * @covers Linker::formatLinksInComment
387 * @dataProvider provideCasesForFormatLinksInComment
389 public function testFormatLinksInComment( $expected, $input, $wiki ) {
390 $conf = new SiteConfiguration();
393 'enwiki' => '//en.example.org'
399 $conf->suffixes
= [ 'wiki' ];
400 $this->setMwGlobals( [
401 'wgScript' => '/wiki/index.php',
402 'wgArticlePath' => '/wiki/$1',
403 'wgCapitalLinks' => true,
409 Linker
::formatLinksInComment( $input, Title
::newFromText( 'Special:BlankPage' ), false, $wiki )
414 * @covers Linker::generateRollback
415 * @dataProvider provideCasesForRollbackGeneration
417 public function testGenerateRollback( $rollbackEnabled, $expectedModules ) {
418 $this->markTestSkippedIfDbType( 'postgres' );
420 $context = RequestContext
::getMain();
421 $user = $context->getUser();
422 $user->setOption( 'showrollbackconfirmation', $rollbackEnabled );
424 $pageData = $this->insertPage( 'Rollback_Test_Page' );
425 $page = WikiPage
::factory( $pageData['title'] );
427 $updater = $page->newPageUpdater( $user );
428 $updater->setContent( \MediaWiki\Revision\SlotRecord
::MAIN
,
429 new TextContent( 'Technical Wishes 123!' )
431 $summary = CommentStoreComment
::newUnsavedComment( 'Some comment!' );
432 $updater->saveRevision( $summary );
434 $rollbackOutput = Linker
::generateRollback( $page->getRevision(), $context );
435 $modules = $context->getOutput()->getModules();
437 $this->assertEquals( $expectedModules, $modules );
438 $this->assertContains( 'rollback 1 edit', $rollbackOutput );
441 public static function provideCasesForRollbackGeneration() {
445 [ 'mediawiki.page.rollback.confirmation' ]
455 public static function provideCasesForFormatLinksInComment() {
456 // phpcs:disable Generic.Files.LineLength
459 'foo bar <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
460 'foo bar [[Special:BlankPage]]',
464 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
465 '[[ :Special:BlankPage]]',
469 '<a class="external" rel="nofollow" href="//en.example.org/w/Foo%27bar">Foo\'bar</a>',
474 'foo bar <a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage">Special:BlankPage</a>',
475 'foo bar [[Special:BlankPage]]',
479 'foo bar <a class="external" rel="nofollow" href="//en.example.org/w/File:Example">Image:Example</a>',
480 'foo bar [[Image:Example]]',
487 public static function provideLinkBeginHook() {
488 // phpcs:disable Generic.Files.LineLength
492 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
495 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage">foobar</a>'
499 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
500 $attribs['bar'] = 'baz';
502 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage" bar="baz">Special:BlankPage</a>'
506 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
507 $query['bar'] = 'baz';
509 '<a href="/w/index.php?title=Special:BlankPage&bar=baz" title="Special:BlankPage">Special:BlankPage</a>'
511 // Force HTTP $options
513 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
514 $options = [ 'http' ];
516 '<a href="http://example.org/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>'
518 // Force 'forcearticlepath' in $options
520 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
521 $options = [ 'forcearticlepath' ];
522 $query['foo'] = 'bar';
524 '<a href="/wiki/Special:BlankPage?foo=bar" title="Special:BlankPage">Special:BlankPage</a>'
528 function ( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
539 * @covers MediaWiki\Linker\LinkRenderer::runLegacyBeginHook
540 * @dataProvider provideLinkBeginHook
542 public function testLinkBeginHook( $callback, $expected ) {
543 $this->hideDeprecated( 'LinkBegin hook (used in hook-LinkBegin-closure)' );
544 $this->setMwGlobals( [
545 'wgArticlePath' => '/wiki/$1',
546 'wgServer' => '//example.org',
547 'wgCanonicalServer' => 'http://example.org',
548 'wgScriptPath' => '/w',
549 'wgScript' => '/w/index.php',
552 $this->setMwGlobals( 'wgHooks', [ 'LinkBegin' => [ $callback ] ] );
553 $title = SpecialPage
::getTitleFor( 'Blankpage' );
554 $out = Linker
::link( $title );
555 $this->assertEquals( $expected, $out );
558 public static function provideLinkEndHook() {
562 function ( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
565 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage">foobar</a>'
569 function ( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
570 $attribs['bar'] = 'baz';
572 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage" bar="baz">Special:BlankPage</a>'
574 // Fully override return value and abort hook
576 function ( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
577 $ret = 'blahblahblah';
587 * @covers MediaWiki\Linker\LinkRenderer::buildAElement
588 * @dataProvider provideLinkEndHook
590 public function testLinkEndHook( $callback, $expected ) {
591 $this->hideDeprecated( 'LinkEnd hook (used in hook-LinkEnd-closure)' );
592 $this->setMwGlobals( [
593 'wgArticlePath' => '/wiki/$1',
596 $this->setMwGlobals( 'wgHooks', [ 'LinkEnd' => [ $callback ] ] );
598 $title = SpecialPage
::getTitleFor( 'Blankpage' );
599 $out = Linker
::link( $title );
600 $this->assertEquals( $expected, $out );