594 lines
18 KiB
PHP
594 lines
18 KiB
PHP
<?php
|
||
|
||
use MediaWiki\MediaWikiServices;
|
||
use MediaWiki\Revision\RevisionRecord;
|
||
|
||
/**
|
||
* @group Database
|
||
*/
|
||
class LinkerTest extends MediaWikiLangTestCase {
|
||
/**
|
||
* @dataProvider provideCasesForUserLink
|
||
* @covers Linker::userLink
|
||
*/
|
||
public function testUserLink( $expected, $userId, $userName, $altUserName = false, $msg = '' ) {
|
||
$this->setMwGlobals( [
|
||
'wgArticlePath' => '/wiki/$1',
|
||
] );
|
||
|
||
// We'd also test the warning, but injecting a mock logger into a static method is tricky.
|
||
if ( !$userName ) {
|
||
Wikimedia\suppressWarnings();
|
||
}
|
||
$actual = Linker::userLink( $userId, $userName, $altUserName );
|
||
if ( !$userName ) {
|
||
Wikimedia\restoreWarnings();
|
||
}
|
||
|
||
$this->assertEquals( $expected, $actual, $msg );
|
||
}
|
||
|
||
public static function provideCasesForUserLink() {
|
||
# Format:
|
||
# - expected
|
||
# - userid
|
||
# - username
|
||
# - optional altUserName
|
||
# - optional message
|
||
return [
|
||
# Empty name (T222529)
|
||
'Empty username, userid 0' => [ '(no username available)', 0, '' ],
|
||
'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
|
||
|
||
'false instead of username' => [ '(no username available)', 73, false ],
|
||
'null instead of username' => [ '(no username available)', 0, null ],
|
||
|
||
# ## ANONYMOUS USER ########################################
|
||
[
|
||
'<a href="/wiki/Special:Contributions/JohnDoe" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/JohnDoe"><bdi>JohnDoe</bdi></a>',
|
||
0, 'JohnDoe', false,
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:Contributions/::1" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/::1"><bdi>::1</bdi></a>',
|
||
0, '::1', false,
|
||
'Anonymous with pretty IPv6'
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:Contributions/0:0:0:0:0:0:0:1" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/0:0:0:0:0:0:0:1"><bdi>::1</bdi></a>',
|
||
0, '0:0:0:0:0:0:0:1', false,
|
||
'Anonymous with almost pretty IPv6'
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/0000:0000:0000:0000:0000:0000:0000:0001"><bdi>::1</bdi></a>',
|
||
0, '0000:0000:0000:0000:0000:0000:0000:0001', false,
|
||
'Anonymous with full IPv6'
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:Contributions/::1" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/::1"><bdi>AlternativeUsername</bdi></a>',
|
||
0, '::1', 'AlternativeUsername',
|
||
'Anonymous with pretty IPv6 and an alternative username'
|
||
],
|
||
|
||
# IPV4
|
||
[
|
||
'<a href="/wiki/Special:Contributions/127.0.0.1" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/127.0.0.1"><bdi>127.0.0.1</bdi></a>',
|
||
0, '127.0.0.1', false,
|
||
'Anonymous with IPv4'
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:Contributions/127.0.0.1" '
|
||
. 'class="mw-userlink mw-anonuserlink" '
|
||
. 'title="Special:Contributions/127.0.0.1"><bdi>AlternativeUsername</bdi></a>',
|
||
0, '127.0.0.1', 'AlternativeUsername',
|
||
'Anonymous with IPv4 and an alternative username'
|
||
],
|
||
|
||
# ## Regular user ##########################################
|
||
# TODO!
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideUserToolLinks
|
||
* @covers Linker::userToolLinks
|
||
* @param string $expected
|
||
* @param int $userId
|
||
* @param string $userText
|
||
*/
|
||
public function testUserToolLinks( $expected, $userId, $userText ) {
|
||
// We'd also test the warning, but injecting a mock logger into a static method is tricky.
|
||
if ( $userText === '' ) {
|
||
Wikimedia\suppressWarnings();
|
||
}
|
||
$actual = Linker::userToolLinks( $userId, $userText );
|
||
if ( $userText === '' ) {
|
||
Wikimedia\restoreWarnings();
|
||
}
|
||
|
||
$this->assertSame( $expected, $actual );
|
||
}
|
||
|
||
public static function provideUserToolLinks() {
|
||
return [
|
||
// Empty name (T222529)
|
||
'Empty username, userid 0' => [ ' (no username available)', 0, '' ],
|
||
'Empty username, userid > 0' => [ ' (no username available)', 73, '' ],
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideUserTalkLink
|
||
* @covers Linker::userTalkLink
|
||
* @param string $expected
|
||
* @param int $userId
|
||
* @param string $userText
|
||
*/
|
||
public function testUserTalkLink( $expected, $userId, $userText ) {
|
||
// We'd also test the warning, but injecting a mock logger into a static method is tricky.
|
||
if ( $userText === '' ) {
|
||
Wikimedia\suppressWarnings();
|
||
}
|
||
$actual = Linker::userTalkLink( $userId, $userText );
|
||
if ( $userText === '' ) {
|
||
Wikimedia\restoreWarnings();
|
||
}
|
||
|
||
$this->assertSame( $expected, $actual );
|
||
}
|
||
|
||
public static function provideUserTalkLink() {
|
||
return [
|
||
// Empty name (T222529)
|
||
'Empty username, userid 0' => [ '(no username available)', 0, '' ],
|
||
'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideBlockLink
|
||
* @covers Linker::blockLink
|
||
* @param string $expected
|
||
* @param int $userId
|
||
* @param string $userText
|
||
*/
|
||
public function testBlockLink( $expected, $userId, $userText ) {
|
||
// We'd also test the warning, but injecting a mock logger into a static method is tricky.
|
||
if ( $userText === '' ) {
|
||
Wikimedia\suppressWarnings();
|
||
}
|
||
$actual = Linker::blockLink( $userId, $userText );
|
||
if ( $userText === '' ) {
|
||
Wikimedia\restoreWarnings();
|
||
}
|
||
|
||
$this->assertSame( $expected, $actual );
|
||
}
|
||
|
||
public static function provideBlockLink() {
|
||
return [
|
||
// Empty name (T222529)
|
||
'Empty username, userid 0' => [ '(no username available)', 0, '' ],
|
||
'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideEmailLink
|
||
* @covers Linker::emailLink
|
||
* @param string $expected
|
||
* @param int $userId
|
||
* @param string $userText
|
||
*/
|
||
public function testEmailLink( $expected, $userId, $userText ) {
|
||
// We'd also test the warning, but injecting a mock logger into a static method is tricky.
|
||
if ( $userText === '' ) {
|
||
Wikimedia\suppressWarnings();
|
||
}
|
||
$actual = Linker::emailLink( $userId, $userText );
|
||
if ( $userText === '' ) {
|
||
Wikimedia\restoreWarnings();
|
||
}
|
||
|
||
$this->assertSame( $expected, $actual );
|
||
}
|
||
|
||
public static function provideEmailLink() {
|
||
return [
|
||
// Empty name (T222529)
|
||
'Empty username, userid 0' => [ '(no username available)', 0, '' ],
|
||
'Empty username, userid > 0' => [ '(no username available)', 73, '' ],
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @dataProvider provideCasesForFormatComment
|
||
* @covers Linker::formatComment
|
||
* @covers Linker::formatAutocomments
|
||
* @covers Linker::formatLinksInComment
|
||
*/
|
||
public function testFormatComment(
|
||
$expected, $comment, $title = false, $local = false, $wikiId = null
|
||
) {
|
||
$conf = new SiteConfiguration();
|
||
$conf->settings = [
|
||
'wgServer' => [
|
||
'enwiki' => '//en.example.org',
|
||
'dewiki' => '//de.example.org',
|
||
],
|
||
'wgArticlePath' => [
|
||
'enwiki' => '/w/$1',
|
||
'dewiki' => '/w/$1',
|
||
],
|
||
];
|
||
$conf->suffixes = [ 'wiki' ];
|
||
|
||
$this->setMwGlobals( [
|
||
'wgScript' => '/wiki/index.php',
|
||
'wgArticlePath' => '/wiki/$1',
|
||
'wgCapitalLinks' => true,
|
||
'wgConf' => $conf,
|
||
// TODO: update tests when the default changes
|
||
'wgFragmentMode' => [ 'legacy' ],
|
||
] );
|
||
|
||
if ( $title === false ) {
|
||
// We need a page title that exists
|
||
$title = Title::newFromText( 'Special:BlankPage' );
|
||
}
|
||
|
||
$this->assertEquals(
|
||
$expected,
|
||
Linker::formatComment( $comment, $title, $local, $wikiId )
|
||
);
|
||
}
|
||
|
||
public function provideCasesForFormatComment() {
|
||
$wikiId = 'enwiki'; // $wgConf has a fake entry for this
|
||
|
||
// phpcs:disable Generic.Files.LineLength
|
||
return [
|
||
// Linker::formatComment
|
||
[
|
||
'a<script>b',
|
||
'a<script>b',
|
||
],
|
||
[
|
||
'a—b',
|
||
'a—b',
|
||
],
|
||
[
|
||
"'''not bolded'''",
|
||
"'''not bolded'''",
|
||
],
|
||
[
|
||
"try <script>evil</scipt> things",
|
||
"try <script>evil</scipt> things",
|
||
],
|
||
// Linker::formatAutocomments
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→autocomment</a></span></span>',
|
||
"/* autocomment */",
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#linkie.3F" title="Special:BlankPage">→[[linkie?]]</a></span></span>',
|
||
"/* [[linkie?]] */",
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment">: </span> // Edit via via</span>',
|
||
// Regression test for T222857
|
||
"/* */ // Edit via via",
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment">: </span> foobar</span>',
|
||
// Regression test for T222857
|
||
"/**/ foobar",
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→autocomment</a>: </span> post</span>',
|
||
"/* autocomment */ post",
|
||
],
|
||
[
|
||
'pre <span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→autocomment</a></span></span>',
|
||
"pre /* autocomment */",
|
||
],
|
||
[
|
||
'pre <span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→autocomment</a>: </span> post</span>',
|
||
"pre /* autocomment */ post",
|
||
],
|
||
[
|
||
'<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>',
|
||
"/* autocomment */ multiple? /* autocomment2 */",
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment_containing_.2F.2A" title="Special:BlankPage">→autocomment containing /*</a>: </span> T70361</span>',
|
||
"/* autocomment containing /* */ T70361"
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment_containing_.22quotes.22" title="Special:BlankPage">→autocomment containing "quotes"</a></span></span>',
|
||
"/* autocomment containing \"quotes\" */"
|
||
],
|
||
[
|
||
'<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>',
|
||
"/* autocomment containing <script>tags</script> */"
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="#autocomment">→autocomment</a></span></span>',
|
||
"/* autocomment */",
|
||
false, true
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment">autocomment</span></span>',
|
||
"/* autocomment */",
|
||
null
|
||
],
|
||
[
|
||
'',
|
||
"/* */",
|
||
false, true
|
||
],
|
||
[
|
||
'',
|
||
"/* */",
|
||
null
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment">[[</span></span>',
|
||
"/* [[ */",
|
||
false, true
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment">[[</span></span>',
|
||
"/* [[ */",
|
||
null
|
||
],
|
||
[
|
||
"foo <span dir=\"auto\"><span class=\"autocomment\"><a href=\"#.23\">→[[#_\t_]]</a></span></span>",
|
||
"foo /* [[#_\t_]] */",
|
||
false, true
|
||
],
|
||
[
|
||
"foo <span dir=\"auto\"><span class=\"autocomment\"><a href=\"#_.09\">#_\t_</a></span></span>",
|
||
"foo /* [[#_\t_]] */",
|
||
null
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a href="/wiki/Special:BlankPage#autocomment" title="Special:BlankPage">→autocomment</a></span></span>',
|
||
"/* autocomment */",
|
||
false, false
|
||
],
|
||
[
|
||
'<span dir="auto"><span class="autocomment"><a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage#autocomment">→autocomment</a></span></span>',
|
||
"/* autocomment */",
|
||
false, false, $wikiId
|
||
],
|
||
// Linker::formatLinksInComment
|
||
[
|
||
'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">link</a> def',
|
||
"abc [[link]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">text</a> def',
|
||
"abc [[link|text]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a> def',
|
||
"abc [[Special:BlankPage|]] def",
|
||
],
|
||
[
|
||
'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',
|
||
"abc [[%C4%85%C5%9B%C5%BC]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/Special:BlankPage#section" title="Special:BlankPage">#section</a> def',
|
||
"abc [[#section]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/index.php?title=/subpage&action=edit&redlink=1" class="new" title="/subpage (page does not exist)">/subpage</a> def',
|
||
"abc [[/subpage]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/index.php?title=%22evil!%22&action=edit&redlink=1" class="new" title=""evil!" (page does not exist)">"evil!"</a> def',
|
||
"abc [[\"evil!\"]] def",
|
||
],
|
||
[
|
||
'abc [[<script>very evil</script>]] def',
|
||
"abc [[<script>very evil</script>]] def",
|
||
],
|
||
[
|
||
'abc [[|]] def',
|
||
"abc [[|]] def",
|
||
],
|
||
[
|
||
'abc <a href="/wiki/index.php?title=Link&action=edit&redlink=1" class="new" title="Link (page does not exist)">link</a> def',
|
||
"abc [[link]] def",
|
||
false, false
|
||
],
|
||
[
|
||
'abc <a class="external" rel="nofollow" href="//en.example.org/w/Link">link</a> def',
|
||
"abc [[link]] def",
|
||
false, false, $wikiId
|
||
],
|
||
];
|
||
// phpcs:enable
|
||
}
|
||
|
||
/**
|
||
* @covers Linker::formatLinksInComment
|
||
* @dataProvider provideCasesForFormatLinksInComment
|
||
*/
|
||
public function testFormatLinksInComment( $expected, $input, $wiki ) {
|
||
$conf = new SiteConfiguration();
|
||
$conf->settings = [
|
||
'wgServer' => [
|
||
'enwiki' => '//en.example.org'
|
||
],
|
||
'wgArticlePath' => [
|
||
'enwiki' => '/w/$1',
|
||
],
|
||
];
|
||
$conf->suffixes = [ 'wiki' ];
|
||
$this->setMwGlobals( [
|
||
'wgScript' => '/wiki/index.php',
|
||
'wgArticlePath' => '/wiki/$1',
|
||
'wgCapitalLinks' => true,
|
||
'wgConf' => $conf,
|
||
] );
|
||
|
||
$this->assertEquals(
|
||
$expected,
|
||
Linker::formatLinksInComment( $input, Title::newFromText( 'Special:BlankPage' ), false, $wiki )
|
||
);
|
||
}
|
||
|
||
/**
|
||
* @covers Linker::generateRollback
|
||
* @dataProvider provideCasesForRollbackGeneration
|
||
*/
|
||
public function testGenerateRollback( $rollbackEnabled, $expectedModules, $title ) {
|
||
$this->markTestSkippedIfDbType( 'postgres' );
|
||
|
||
$context = RequestContext::getMain();
|
||
$user = $context->getUser();
|
||
$user->setOption( 'showrollbackconfirmation', $rollbackEnabled );
|
||
|
||
$this->assertSame( 0, Title::newFromText( $title )->getArticleID() );
|
||
$pageData = $this->insertPage( $title );
|
||
$page = WikiPage::factory( $pageData['title'] );
|
||
|
||
$updater = $page->newPageUpdater( $user );
|
||
$updater->setContent( \MediaWiki\Revision\SlotRecord::MAIN,
|
||
new TextContent( 'Technical Wishes 123!' )
|
||
);
|
||
$summary = CommentStoreComment::newUnsavedComment( 'Some comment!' );
|
||
$updater->saveRevision( $summary );
|
||
|
||
$rollbackOutput = Linker::generateRollback( $page->getRevisionRecord(), $context );
|
||
$modules = $context->getOutput()->getModules();
|
||
$currentRev = $page->getRevisionRecord();
|
||
$revisionLookup = MediaWikiServices::getInstance()->getRevisionLookup();
|
||
$oldestRev = $revisionLookup->getFirstRevision( $page->getTitle() );
|
||
|
||
$this->assertEquals( $expectedModules, $modules );
|
||
$this->assertInstanceOf( RevisionRecord::class, $currentRev );
|
||
$this->assertInstanceOf( User::class, $currentRev->getUser() );
|
||
$this->assertEquals( $user->getName(), $currentRev->getUser()->getName() );
|
||
$this->assertEquals(
|
||
static::getTestSysop()->getUser(),
|
||
$oldestRev->getUser()->getName()
|
||
);
|
||
|
||
$ids = [];
|
||
$r = $oldestRev;
|
||
while ( $r ) {
|
||
$ids[] = $r->getId();
|
||
$r = $revisionLookup->getNextRevision( $r );
|
||
}
|
||
$this->assertEquals( [ $oldestRev->getId(), $currentRev->getId() ], $ids );
|
||
|
||
$this->assertStringContainsString( 'rollback 1 edit', $rollbackOutput );
|
||
}
|
||
|
||
public static function provideCasesForRollbackGeneration() {
|
||
return [
|
||
[
|
||
true,
|
||
[ 'mediawiki.misc-authed-curate' ],
|
||
'Rollback_Test_Page'
|
||
],
|
||
[
|
||
false,
|
||
[],
|
||
'Rollback_Test_Page2'
|
||
]
|
||
];
|
||
}
|
||
|
||
public static function provideCasesForFormatLinksInComment() {
|
||
// phpcs:disable Generic.Files.LineLength
|
||
return [
|
||
[
|
||
'foo bar <a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
|
||
'foo bar [[Special:BlankPage]]',
|
||
null,
|
||
],
|
||
[
|
||
'<a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
|
||
'[[ :Special:BlankPage]]',
|
||
null,
|
||
],
|
||
[
|
||
'[[Foo<a href="/wiki/Special:BlankPage" title="Special:BlankPage">Special:BlankPage</a>',
|
||
'[[Foo[[Special:BlankPage]]',
|
||
null,
|
||
],
|
||
[
|
||
'<a class="external" rel="nofollow" href="//en.example.org/w/Foo%27bar">Foo\'bar</a>',
|
||
"[[Foo'bar]]",
|
||
'enwiki',
|
||
],
|
||
[
|
||
'foo bar <a class="external" rel="nofollow" href="//en.example.org/w/Special:BlankPage">Special:BlankPage</a>',
|
||
'foo bar [[Special:BlankPage]]',
|
||
'enwiki',
|
||
],
|
||
[
|
||
'foo bar <a class="external" rel="nofollow" href="//en.example.org/w/File:Example">Image:Example</a>',
|
||
'foo bar [[Image:Example]]',
|
||
'enwiki',
|
||
],
|
||
];
|
||
// phpcs:enable
|
||
}
|
||
|
||
public static function provideTooltipAndAccesskeyAttribs() {
|
||
return [
|
||
'Watch no expiry' => [
|
||
'ca-watch', [], null, [ 'title' => 'Add this page to your watchlist [w]', 'accesskey' => 'w' ]
|
||
],
|
||
'Key does not exist' => [
|
||
'key-does-not-exist', [], null, []
|
||
],
|
||
'Unwatch no expiry' => [
|
||
'ca-unwatch', [], null, [ 'title' => 'Remove this page from your watchlist [w]',
|
||
'accesskey' => 'w' ]
|
||
],
|
||
];
|
||
}
|
||
|
||
/**
|
||
* @covers Linker::tooltipAndAccesskeyAttribs
|
||
* @dataProvider provideTooltipAndAccesskeyAttribs
|
||
*/
|
||
public function testTooltipAndAccesskeyAttribs( $name, $msgParams, $options, $expected ) {
|
||
$this->setMwGlobals( [
|
||
'wgWatchlistExpiry' => true,
|
||
] );
|
||
$user = $this->createMock( User::class );
|
||
$user->method( 'isRegistered' )->willReturn( true );
|
||
$user->method( 'isLoggedIn' )->willReturn( true );
|
||
|
||
$title = SpecialPage::getTitleFor( 'Blankpage' );
|
||
|
||
$context = RequestContext::getMain();
|
||
$context->setTitle( $title );
|
||
$context->setUser( $user );
|
||
|
||
$watchedItemWithoutExpiry = new WatchedItem( $user, $title, null, null );
|
||
|
||
$result = Linker::tooltipAndAccesskeyAttribs( $name, $msgParams, $options );
|
||
|
||
$this->assertEquals( $expected, $result );
|
||
}
|
||
}
|