Update docs/hooks.txt for ShowSearchHitTitle
[mediawiki.git] / tests / phpunit / includes / LinkerTest.php
blob3edf99f2e249ef4ac75eaf25cc2e4949d7619791
1 <?php
3 use MediaWiki\MediaWikiServices;
5 /**
6 * @group Database
7 */
9 class LinkerTest extends MediaWikiLangTestCase {
11 /**
12 * @dataProvider provideCasesForUserLink
13 * @covers Linker::userLink
15 public function testUserLink( $expected, $userId, $userName, $altUserName = false, $msg = '' ) {
16 $this->setMwGlobals( [
17 'wgArticlePath' => '/wiki/$1',
18 ] );
20 $this->assertEquals(
21 $expected,
22 Linker::userLink( $userId, $userName, $altUserName ),
23 $msg
27 public static function provideCasesForUserLink() {
28 # Format:
29 # - expected
30 # - userid
31 # - username
32 # - optional altUserName
33 # - optional message
34 return [
36 # ## ANONYMOUS USER ########################################
38 '<a href="/wiki/Special:Contributions/JohnDoe" '
39 . 'class="mw-userlink mw-anonuserlink" '
40 . 'title="Special:Contributions/JohnDoe"><bdi>JohnDoe</bdi></a>',
41 0, 'JohnDoe', false,
44 '<a href="/wiki/Special:Contributions/::1" '
45 . 'class="mw-userlink mw-anonuserlink" '
46 . 'title="Special:Contributions/::1"><bdi>::1</bdi></a>',
47 0, '::1', false,
48 'Anonymous with pretty IPv6'
51 '<a href="/wiki/Special:Contributions/0:0:0:0:0:0:0:1" '
52 . 'class="mw-userlink mw-anonuserlink" '
53 . 'title="Special:Contributions/0:0:0:0:0:0:0:1"><bdi>::1</bdi></a>',
54 0, '0:0:0:0:0:0:0:1', false,
55 'Anonymous with almost pretty IPv6'
58 '<a href="/wiki/Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001" '
59 . 'class="mw-userlink mw-anonuserlink" '
60 . 'title="Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001"><bdi>::1</bdi></a>',
61 0, '0000:0000:0000:0000:0000:0000:0000:0001', false,
62 'Anonymous with full IPv6'
65 '<a href="/wiki/Special:Contributions/::1" '
66 . 'class="mw-userlink mw-anonuserlink" '
67 . 'title="Special:Contributions/::1"><bdi>AlternativeUsername</bdi></a>',
68 0, '::1', 'AlternativeUsername',
69 'Anonymous with pretty IPv6 and an alternative username'
72 # IPV4
74 '<a href="/wiki/Special:Contributions/127.0.0.1" '
75 . 'class="mw-userlink mw-anonuserlink" '
76 . 'title="Special:Contributions/127.0.0.1"><bdi>127.0.0.1</bdi></a>',
77 0, '127.0.0.1', false,
78 'Anonymous with IPv4'
81 '<a href="/wiki/Special:Contributions/127.0.0.1" '
82 . 'class="mw-userlink mw-anonuserlink" '
83 . 'title="Special:Contributions/127.0.0.1"><bdi>AlternativeUsername</bdi></a>',
84 0, '127.0.0.1', 'AlternativeUsername',
85 'Anonymous with IPv4 and an alternative username'
88 # ## Regular user ##########################################
89 # TODO!
93 /**
94 * @dataProvider provideCasesForFormatComment
95 * @covers Linker::formatComment
96 * @covers Linker::formatAutocomments
97 * @covers Linker::formatLinksInComment
99 public function testFormatComment(
100 $expected, $comment, $title = false, $local = false, $wikiId = null
102 $conf = new SiteConfiguration();
103 $conf->settings = [
104 'wgServer' => [
105 'enwiki' => '//en.example.org',
106 'dewiki' => '//de.example.org',
108 'wgArticlePath' => [
109 'enwiki' => '/w/$1',
110 'dewiki' => '/w/$1',
113 $conf->suffixes = [ 'wiki' ];
115 $this->setMwGlobals( [
116 'wgScript' => '/wiki/index.php',
117 'wgArticlePath' => '/wiki/$1',
118 'wgCapitalLinks' => true,
119 'wgConf' => $conf,
120 ] );
122 if ( $title === false ) {
123 // We need a page title that exists
124 $title = Title::newFromText( 'Special:BlankPage' );
127 $this->assertEquals(
128 $expected,
129 Linker::formatComment( $comment, $title, $local, $wikiId )
133 public function provideCasesForFormatComment() {
134 $wikiId = 'enwiki'; // $wgConf has a fake entry for this
136 // @codingStandardsIgnoreStart Generic.Files.LineLength
137 return [
138 // Linker::formatComment
140 'a&lt;script&gt;b',
141 'a<script>b',
144 'a—b',
145 'a&mdash;b',
148 "&#039;&#039;&#039;not bolded&#039;&#039;&#039;",
149 "'''not bolded'''",
152 "try &lt;script&gt;evil&lt;/scipt&gt; things",
153 "try <script>evil</scipt> things",
155 // Linker::formatAutocomments
157 '<a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
158 "/* autocomment */",
161 '<a href="/wiki/Special:BlankPage#linkie.3F" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment"><a href="/wiki/index.php?title=Linkie%3F&amp;action=edit&amp;redlink=1" class="new" title="Linkie? (page does not exist)">linkie?</a></span></span>',
162 "/* [[linkie?]] */",
165 '<a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment: </span> post</span>',
166 "/* autocomment */ post",
169 'pre <a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
170 "pre /* autocomment */",
173 'pre <a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment: </span> post</span>',
174 "pre /* autocomment */ post",
177 '<a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment: </span> multiple? <a href="/wiki/Special:BlankPage#autocomment2" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment2: </span> </span></span>',
178 "/* autocomment */ multiple? /* autocomment2 */ ",
181 '<a href="/wiki/Special:BlankPage#autocomment_containing_.2F.2A" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment containing /*: </span> T70361</span>',
182 "/* autocomment containing /* */ T70361"
185 '<a href="/wiki/Special:BlankPage#autocomment_containing_.22quotes.22" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment containing &quot;quotes&quot;</span></span>',
186 "/* autocomment containing \"quotes\" */"
189 '<a href="/wiki/Special:BlankPage#autocomment_containing_.3Cscript.3Etags.3C.2Fscript.3E" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment containing &lt;script&gt;tags&lt;/script&gt;</span></span>',
190 "/* autocomment containing <script>tags</script> */"
193 '<a href="#autocomment">→</a>‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
194 "/* autocomment */",
195 false, true
198 '‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
199 "/* autocomment */",
200 null
203 '<a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→</a>‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
204 "/* autocomment */",
205 false, false
208 '<a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage#autocomment">→</a>‎<span dir="auto"><span class="autocomment">autocomment</span></span>',
209 "/* autocomment */",
210 false, false, $wikiId
212 // Linker::formatLinksInComment
214 'abc <a href="/wiki/index.php?title=Link&amp;action=edit&amp;redlink=1" class="new" title="Link (page does not exist)">link</a> def',
215 "abc [[link]] def",
218 'abc <a href="/wiki/index.php?title=Link&amp;action=edit&amp;redlink=1" class="new" title="Link (page does not exist)">text</a> def',
219 "abc [[link|text]] def",
222 'abc <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a> def',
223 "abc [[Special:BlankPage|]] def",
226 'abc <a href="/wiki/index.php?title=%C4%84%C5%9B%C5%BC&amp;action=edit&amp;redlink=1" class="new" title="Ąśż (page does not exist)">ąśż</a> def',
227 "abc [[%C4%85%C5%9B%C5%BC]] def",
230 'abc <a href="/wiki/Special:BlankPage#section" title="Special:BlankPage">#section</a> def',
231 "abc [[#section]] def",
234 'abc <a href="/wiki/index.php?title=/subpage&amp;action=edit&amp;redlink=1" class="new" title="/subpage (page does not exist)">/subpage</a> def',
235 "abc [[/subpage]] def",
238 'abc <a href="/wiki/index.php?title=%22evil!%22&amp;action=edit&amp;redlink=1" class="new" title="&quot;evil!&quot; (page does not exist)">&quot;evil!&quot;</a> def',
239 "abc [[\"evil!\"]] def",
242 'abc [[&lt;script&gt;very evil&lt;/script&gt;]] def',
243 "abc [[<script>very evil</script>]] def",
246 'abc [[|]] def',
247 "abc [[|]] def",
250 'abc <a href="/wiki/index.php?title=Link&amp;action=edit&amp;redlink=1" class="new" title="Link (page does not exist)">link</a> def',
251 "abc [[link]] def",
252 false, false
255 'abc <a class="external" rel="nofollow" href="//en.example.org/w/Link">link</a> def',
256 "abc [[link]] def",
257 false, false, $wikiId
260 // @codingStandardsIgnoreEnd
264 * @covers Linker::formatLinksInComment
265 * @dataProvider provideCasesForFormatLinksInComment
267 public function testFormatLinksInComment( $expected, $input, $wiki ) {
269 $conf = new SiteConfiguration();
270 $conf->settings = [
271 'wgServer' => [
272 'enwiki' => '//en.example.org'
274 'wgArticlePath' => [
275 'enwiki' => '/w/$1',
278 $conf->suffixes = [ 'wiki' ];
279 $this->setMwGlobals( [
280 'wgScript' => '/wiki/index.php',
281 'wgArticlePath' => '/wiki/$1',
282 'wgCapitalLinks' => true,
283 'wgConf' => $conf,
284 ] );
286 $this->assertEquals(
287 $expected,
288 Linker::formatLinksInComment( $input, Title::newFromText( 'Special:BlankPage' ), false, $wiki )
292 public static function provideCasesForFormatLinksInComment() {
293 // @codingStandardsIgnoreStart Generic.Files.LineLength
294 return [
296 'foo bar <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
297 'foo bar [[Special:BlankPage]]',
298 null,
301 '<a class="external" rel="nofollow" href="//en.example.org/w/Foo%27bar">Foo\'bar</a>',
302 "[[Foo'bar]]",
303 'enwiki',
306 'foo bar <a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage">Special:BlankPage</a>',
307 'foo bar [[Special:BlankPage]]',
308 'enwiki',
311 // @codingStandardsIgnoreEnd
314 public static function provideLinkBeginHook() {
315 // @codingStandardsIgnoreStart Generic.Files.LineLength
316 return [
317 // Modify $html
319 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
320 $html = 'foobar';
322 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage">foobar</a>'
324 // Modify $attribs
326 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
327 $attribs['bar'] = 'baz';
329 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage" bar="baz">Special:BlankPage</a>'
331 // Modify $query
333 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
334 $query['bar'] = 'baz';
336 '<a href="/w/index.php?title=Special:BlankPage&amp;bar=baz" title="Special:BlankPage">Special:BlankPage</a>'
338 // Force HTTP $options
340 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
341 $options = [ 'http' ];
343 '<a href="http://example.org/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>'
345 // Force 'forcearticlepath' in $options
347 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
348 $options = [ 'forcearticlepath' ];
349 $query['foo'] = 'bar';
351 '<a href="/wiki/Special:BlankPage?foo=bar" title="Special:BlankPage">Special:BlankPage</a>'
353 // Abort early
355 function( $dummy, $title, &$html, &$attribs, &$query, &$options, &$ret ) {
356 $ret = 'foobar';
357 return false;
359 'foobar'
362 // @codingStandardsIgnoreEnd
366 * @covers MediaWiki\Linker\LinkRenderer::runLegacyBeginHook
367 * @dataProvider provideLinkBeginHook
369 public function testLinkBeginHook( $callback, $expected ) {
370 $this->setMwGlobals( [
371 'wgArticlePath' => '/wiki/$1',
372 'wgServer' => '//example.org',
373 'wgCanonicalServer' => 'http://example.org',
374 'wgScriptPath' => '/w',
375 'wgScript' => '/w/index.php',
376 ] );
378 $this->setMwGlobals( 'wgHooks', [ 'LinkBegin' => [ $callback ] ] );
379 $title = SpecialPage::getTitleFor( 'Blankpage' );
380 $out = Linker::link( $title );
381 $this->assertEquals( $expected, $out );
384 public static function provideLinkEndHook() {
385 return [
386 // Override $html
388 function( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
389 $html = 'foobar';
391 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage">foobar</a>'
393 // Modify $attribs
395 function( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
396 $attribs['bar'] = 'baz';
398 '<a href="/wiki/Special:BlankPage" title="Special:BlankPage" bar="baz">Special:BlankPage</a>'
400 // Fully override return value and abort hook
402 function( $dummy, $title, $options, &$html, &$attribs, &$ret ) {
403 $ret = 'blahblahblah';
404 return false;
406 'blahblahblah'
413 * @covers MediaWiki\Linker\LinkRenderer::buildAElement
414 * @dataProvider provideLinkEndHook
416 public function testLinkEndHook( $callback, $expected ) {
417 $this->setMwGlobals( [
418 'wgArticlePath' => '/wiki/$1',
419 ] );
421 $this->setMwGlobals( 'wgHooks', [ 'LinkEnd' => [ $callback ] ] );
423 $title = SpecialPage::getTitleFor( 'Blankpage' );
424 $out = Linker::link( $title );
425 $this->assertEquals( $expected, $out );
429 * @covers Linker::getLinkColour
431 public function testGetLinkColour() {
432 $this->hideDeprecated( 'Linker::getLinkColour' );
433 $linkCache = MediaWikiServices::getInstance()->getLinkCache();
434 $foobarTitle = Title::makeTitle( NS_MAIN, 'FooBar' );
435 $redirectTitle = Title::makeTitle( NS_MAIN, 'Redirect' );
436 $userTitle = Title::makeTitle( NS_USER, 'Someuser' );
437 $linkCache->addGoodLinkObj(
438 1, // id
439 $foobarTitle,
440 10, // len
441 0 // redir
443 $linkCache->addGoodLinkObj(
444 2, // id
445 $redirectTitle,
446 10, // len
447 1 // redir
450 $linkCache->addGoodLinkObj(
451 3, // id
452 $userTitle,
453 10, // len
454 0 // redir
457 $this->assertEquals(
459 Linker::getLinkColour( $foobarTitle, 0 )
462 $this->assertEquals(
463 'stub',
464 Linker::getLinkColour( $foobarTitle, 20 )
467 $this->assertEquals(
468 'mw-redirect',
469 Linker::getLinkColour( $redirectTitle, 0 )
472 $this->assertEquals(
474 Linker::getLinkColour( $userTitle, 20 )