Revert "Temporarily remove tests added in I8eef5a165"
authorAude <aude.wiki@gmail.com>
Sun, 6 Apr 2014 13:18:07 +0000 (13:18 +0000)
committerAude <aude.wiki@gmail.com>
Sun, 6 Apr 2014 13:18:07 +0000 (13:18 +0000)
this was to be temporary, right? we need the tests in.

If there are issues with jenkins, this is not the *best*
way to fix them.

This reverts commit 44a19685317de16f5afe42b710d8d412352d7fe9.

Change-Id: I5dd60a9818c28155d96a6642c588781667e682c3

tests/phpunit/includes/TitleTest.php
tests/phpunit/includes/title/MediaWikiPageLinkRendererTest.php [new file with mode: 0644]
tests/phpunit/includes/title/MediaWikiTitleCodecTest.php [new file with mode: 0644]
tests/phpunit/includes/title/TitleValueTest.php [new file with mode: 0644]

index a332368..8093824 100644 (file)
@@ -517,4 +517,74 @@ class TitleTest extends MediaWikiTestCase {
                        array( 'User:John_Doe/subOne', 'subOne' ),
                );
        }
+
+       public function provideNewFromTitleValue() {
+               return array(
+                       array( new TitleValue( NS_MAIN, 'Foo' ) ),
+                       array( new TitleValue( NS_MAIN, 'Foo', 'bar' ) ),
+                       array( new TitleValue( NS_USER, 'Hansi_Maier' ) ),
+               );
+       }
+
+       /**
+        * @dataProvider provideNewFromTitleValue
+        */
+       public function testNewFromTitleValue( TitleValue $value ) {
+               $title = Title::newFromTitleValue( $value );
+
+               $dbkey = str_replace( ' ', '_', $value->getText() );
+               $this->assertEquals( $dbkey, $title->getDBkey() );
+               $this->assertEquals( $value->getNamespace(), $title->getNamespace() );
+               $this->assertEquals( $value->getFragment(), $title->getFragment() );
+       }
+
+       public function provideGetTitleValue() {
+               return array(
+                       array( 'Foo' ),
+                       array( 'Foo#bar' ),
+                       array( 'User:Hansi_Maier' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetTitleValue
+        */
+       public function testGetTitleValue( $text ) {
+               $title = Title::newFromText( $text );
+               $value = $title->getTitleValue();
+
+               $dbkey = str_replace( ' ', '_', $value->getText() );
+               $this->assertEquals( $title->getDBkey(), $dbkey );
+               $this->assertEquals( $title->getNamespace(), $value->getNamespace() );
+               $this->assertEquals( $title->getFragment(), $value->getFragment() );
+       }
+
+       public function provideGetFragment() {
+               return array(
+                       array( 'Foo', '' ),
+                       array( 'Foo#bar', 'bar' ),
+                       array( 'Foo#bär', 'bär' ),
+
+                       // Inner whitespace is normalized
+                       array( 'Foo#bar_bar', 'bar bar' ),
+                       array( 'Foo#bar bar', 'bar bar' ),
+                       array( 'Foo#bar   bar', 'bar bar' ),
+
+                       // Leading whitespace is kept, trailing whitespace is trimmed.
+                       // XXX: Is this really want we want?
+                       array( 'Foo#_bar_bar_', ' bar bar' ),
+                       array( 'Foo# bar bar ', ' bar bar' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetFragment
+        *
+        * @param $full
+        * @param $fragment
+        */
+       public function testGetFragment( $full, $fragment ) {
+               $title = Title::newFromText( $full );
+               $this->assertEquals( $fragment, $title->getFragment() );
+       }
 }
diff --git a/tests/phpunit/includes/title/MediaWikiPageLinkRendererTest.php b/tests/phpunit/includes/title/MediaWikiPageLinkRendererTest.php
new file mode 100644 (file)
index 0000000..73d7ff9
--- /dev/null
@@ -0,0 +1,160 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ * @license GPL 2+
+ * @author Daniel Kinzler
+ */
+
+/**
+ * @covers MediaWikiPageLinkRenderer
+ *
+ * @group Title
+ */
+class MediaWikiPageLinkRendererTest extends MediaWikiTestCase {
+
+       /**
+        * Returns a mock GenderCache that will return "female" always.
+        *
+        * @return GenderCache
+        */
+       private function getGenderCache() {
+               $genderCache = $this->getMockBuilder( 'GenderCache' )
+                       ->disableOriginalConstructor()
+                       ->getMock();
+
+               $genderCache->expects( $this->any() )
+                       ->method( 'getGenderOf' )
+                       ->will( $this->returnValue( 'female' ) );
+
+               return $genderCache;
+       }
+
+       public function provideGetPageUrl() {
+               return array(
+                       array(
+                               new TitleValue( NS_MAIN, 'Foo_Bar' ),
+                               array(),
+                               '/Foo_Bar'
+                       ),
+                       array(
+                               new TitleValue( NS_USER, 'Hansi_Maier', 'stuff' ),
+                               array( 'foo' => 'bar' ),
+                               '/User:Hansi_Maier?foo=bar#stuff'
+                       ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetPageUrl
+        */
+       public function testGetPageUrl( TitleValue $title, $params, $url ) {
+               // NOTE: was of Feb 2014, MediaWikiPageLinkRenderer *ignores* the
+               // WikitextTitleFormatter we pass here, and relies on the Linker
+               // class for generating the link! This may break the test e.g.
+               // of Linker uses a different language for the namespace names.
+
+               $lang = Language::factory( 'en' );
+
+               $formatter = new MediaWikiTitleCodec( $lang, $this->getGenderCache() );
+               $renderer = new MediaWikiPageLinkRenderer( $formatter, '/' );
+               $actual = $renderer->getPageUrl( $title, $params );
+
+               $this->assertEquals( $url, $actual );
+       }
+
+       public function provideRenderHtmlLink() {
+               return array(
+                       array(
+                               new TitleValue( NS_MAIN, 'Foo_Bar' ),
+                               'Foo Bar',
+                               '!<a .*href=".*?Foo_Bar.*?".*?>Foo Bar</a>!'
+                       ),
+                       array(
+                               //NOTE: Linker doesn't include fragments in "broken" links
+                               //NOTE: once this no longer uses Linker, we will get "2" instead of "User" for the namespace.
+                               new TitleValue( NS_USER, 'Hansi_Maier', 'stuff' ),
+                               'Hansi Maier\'s Stuff',
+                               '!<a .*href=".*?User:Hansi_Maier.*?>Hansi Maier\'s Stuff</a>!'
+                       ),
+                       array(
+                               //NOTE: Linker doesn't include fragments in "broken" links
+                               //NOTE: once this no longer uses Linker, we will get "2" instead of "User" for the namespace.
+                               new TitleValue( NS_USER, 'Hansi_Maier', 'stuff' ),
+                               null,
+                               '!<a .*href=".*?User:Hansi_Maier.*?>User:Hansi Maier#stuff</a>!'
+                       ),
+               );
+       }
+
+       /**
+        * @dataProvider provideRenderHtmlLink
+        */
+       public function testRenderHtmlLink( TitleValue $title, $text, $pattern ) {
+               // NOTE: was of Feb 2014, MediaWikiPageLinkRenderer *ignores* the
+               // WikitextTitleFormatter we pass here, and relies on the Linker
+               // class for generating the link! This may break the test e.g.
+               // of Linker uses a different language for the namespace names.
+
+               $lang = Language::factory( 'en' );
+
+               $formatter = new MediaWikiTitleCodec( $lang, $this->getGenderCache() );
+               $renderer = new MediaWikiPageLinkRenderer( $formatter );
+               $actual = $renderer->renderHtmlLink( $title, $text );
+
+               $this->assertRegExp( $pattern, $actual );
+       }
+
+       public function provideRenderWikitextLink() {
+               return array(
+                       array(
+                               new TitleValue( NS_MAIN, 'Foo_Bar' ),
+                               'Foo Bar',
+                               '[[:0:Foo Bar|Foo Bar]]'
+                       ),
+                       array(
+                               new TitleValue( NS_USER, 'Hansi_Maier', 'stuff' ),
+                               'Hansi Maier\'s Stuff',
+                               '[[:2:Hansi Maier#stuff|Hansi Maier&#39;s Stuff]]'
+                       ),
+                       array(
+                               new TitleValue( NS_USER, 'Hansi_Maier', 'stuff' ),
+                               null,
+                               '[[:2:Hansi Maier#stuff|2:Hansi Maier#stuff]]'
+                       ),
+               );
+       }
+
+       /**
+        * @dataProvider provideRenderWikitextLink
+        */
+       public function testRenderWikitextLink( TitleValue $title, $text, $expected ) {
+               $formatter = $this->getMock( 'TitleFormatter' );
+               $formatter->expects( $this->any() )
+                       ->method( 'getFullText' )
+                       ->will( $this->returnCallback(
+                               function( TitleValue $title ) {
+                                       return str_replace( '_', ' ', "$title" );
+                               }
+                       ));
+
+               $renderer = new MediaWikiPageLinkRenderer( $formatter, '/' );
+               $actual = $renderer->renderWikitextLink( $title, $text );
+
+               $this->assertEquals( $expected, $actual );
+       }
+}
diff --git a/tests/phpunit/includes/title/MediaWikiTitleCodecTest.php b/tests/phpunit/includes/title/MediaWikiTitleCodecTest.php
new file mode 100644 (file)
index 0000000..e4f9396
--- /dev/null
@@ -0,0 +1,372 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ * @license GPL 2+
+ * @author Daniel Kinzler
+ */
+
+/**
+ * @covers MediaWikiTitleCodec
+ *
+ * @group Title
+ * @group Database
+ *        ^--- needed because of global state in
+ */
+class MediaWikiTitleCodecTest extends MediaWikiTestCase {
+
+       public function setUp() {
+               parent::setUp();
+
+               $this->setMwGlobals( array(
+                       'wgLanguageCode' => 'en',
+                       'wgContLang' => Language::factory( 'en' ),
+                       // User language
+                       'wgLang' => Language::factory( 'en' ),
+                       'wgAllowUserJs' => false,
+                       'wgDefaultLanguageVariant' => false,
+                       'wgLocalInterwikis' => array( 'localtestiw' ),
+
+                       // NOTE: this is why global state is evil.
+                       // TODO: refactor access to the interwiki codes so it can be injected.
+                       'wgHooks' => array(
+                               'InterwikiLoadPrefix' => array(
+                                       function ( $prefix, &$data ) {
+                                               if ( $prefix === 'localtestiw' ) {
+                                                       $data = array( 'iw_url' => 'localtestiw' );
+                                               } elseif ( $prefix === 'remotetestiw' ) {
+                                                       $data = array( 'iw_url' => 'remotetestiw' );
+                                               }
+                                               return false;
+                                       }
+                               )
+                       )
+               ) );
+       }
+
+       /**
+        * Returns a mock GenderCache that will consider a user "female" if the
+        * first part of the user name ends with "a".
+        *
+        * @return GenderCache
+        */
+       private function getGenderCache() {
+               $genderCache = $this->getMockBuilder( 'GenderCache' )
+                       ->disableOriginalConstructor()
+                       ->getMock();
+
+               $genderCache->expects( $this->any() )
+                       ->method( 'getGenderOf' )
+                       ->will( $this->returnCallback( function( $userName ) {
+                               return preg_match( '/^[^- _]+a( |_|$)/u', $userName ) ? 'female' : 'male';
+                       } ) );
+
+               return $genderCache;
+       }
+
+       protected function makeCodec( $lang ) {
+               $gender = $this->getGenderCache();
+               $lang = Language::factory( $lang );
+               return new MediaWikiTitleCodec( $lang, $gender );
+       }
+
+       public function provideFormat() {
+               return array(
+                       array( NS_MAIN, 'Foo_Bar', '', 'en', 'Foo Bar' ),
+                       array( NS_USER, 'Hansi_Maier', 'stuff_and_so_on', 'en', 'User:Hansi Maier#stuff and so on' ),
+                       array( false, 'Hansi_Maier', '', 'en', 'Hansi Maier' ),
+                       array( NS_USER_TALK, 'hansi__maier', '', 'en', 'User talk:hansi  maier', 'User talk:Hansi maier' ),
+
+                       // getGenderCache() provides a mock that considers first
+                       // names ending in "a" to be female.
+                       array( NS_USER, 'Lisa_Müller', '', 'de', 'Benutzerin:Lisa Müller' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideFormat
+        */
+       public function testFormat( $namespace, $text, $fragment, $lang, $expected, $normalized = null ) {
+               if ( $normalized === null ) {
+                       $normalized = $expected;
+               }
+
+               $codec = $this->makeCodec( $lang );
+               $actual = $codec->formatTitle( $namespace, $text, $fragment );
+
+               $this->assertEquals( $expected, $actual, 'formatted' );
+
+               // test round trip
+               $parsed = $codec->parseTitle( $actual, NS_MAIN );
+               $actual2 = $codec->formatTitle( $parsed->getNamespace(), $parsed->getText(), $parsed->getFragment() );
+
+               $this->assertEquals( $normalized, $actual2, 'normalized after round trip' );
+       }
+
+       public function provideGetText() {
+               return array(
+                       array( NS_MAIN, 'Foo_Bar', '', 'en', 'Foo Bar' ),
+                       array( NS_USER, 'Hansi_Maier', 'stuff_and_so_on', 'en', 'Hansi Maier' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetText
+        */
+       public function testGetText( $namespace, $dbkey, $fragment, $lang, $expected ) {
+               $codec = $this->makeCodec( $lang );
+               $title = new TitleValue( $namespace, $dbkey, $fragment );
+
+               $actual = $codec->getText( $title );
+
+               $this->assertEquals( $expected, $actual );
+       }
+
+       public function provideGetPrefixedText() {
+               return array(
+                       array( NS_MAIN, 'Foo_Bar', '', 'en', 'Foo Bar' ),
+                       array( NS_USER, 'Hansi_Maier', 'stuff_and_so_on', 'en', 'User:Hansi Maier' ),
+
+                       // No capitalization or normalization is applied while formatting!
+                       array( NS_USER_TALK, 'hansi__maier', '', 'en', 'User talk:hansi  maier' ),
+
+                       // getGenderCache() provides a mock that considers first
+                       // names ending in "a" to be female.
+                       array( NS_USER, 'Lisa_Müller', '', 'de', 'Benutzerin:Lisa Müller' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetPrefixedText
+        */
+       public function testGetPrefixedText( $namespace, $dbkey, $fragment, $lang, $expected ) {
+               $codec = $this->makeCodec( $lang );
+               $title = new TitleValue( $namespace, $dbkey, $fragment );
+
+               $actual = $codec->getPrefixedText( $title );
+
+               $this->assertEquals( $expected, $actual );
+       }
+
+       public function provideGetFullText() {
+               return array(
+                       array( NS_MAIN, 'Foo_Bar', '', 'en', 'Foo Bar' ),
+                       array( NS_USER, 'Hansi_Maier', 'stuff_and_so_on', 'en', 'User:Hansi Maier#stuff and so on' ),
+
+                       // No capitalization or normalization is applied while formatting!
+                       array( NS_USER_TALK, 'hansi__maier', '', 'en', 'User talk:hansi  maier' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetFullText
+        */
+       public function testGetFullText( $namespace, $dbkey, $fragment, $lang, $expected ) {
+               $codec = $this->makeCodec( $lang );
+               $title = new TitleValue( $namespace, $dbkey, $fragment );
+
+               $actual = $codec->getFullText( $title );
+
+               $this->assertEquals( $expected, $actual );
+       }
+
+       public function provideParseTitle() {
+               //TODO: test capitalization and trimming
+               //TODO: test unicode normalization
+
+               return array(
+                       array( '  : Hansi_Maier _ ', NS_MAIN, 'en',
+                               new TitleValue( NS_MAIN, 'Hansi_Maier', '' ) ),
+                       array( 'User:::1', NS_MAIN, 'de',
+                               new TitleValue( NS_USER, '0:0:0:0:0:0:0:1', '' ) ),
+                       array( ' lisa Müller', NS_USER, 'de',
+                               new TitleValue( NS_USER, 'Lisa_Müller', '' ) ),
+                       array( 'benutzerin:lisa Müller#stuff', NS_MAIN, 'de',
+                               new TitleValue( NS_USER, 'Lisa_Müller', 'stuff' ) ),
+
+                       array( ':Category:Quux', NS_MAIN, 'en',
+                               new TitleValue( NS_CATEGORY, 'Quux', '' ) ),
+                       array( 'Category:Quux', NS_MAIN, 'en',
+                               new TitleValue( NS_CATEGORY, 'Quux', '' ) ),
+                       array( 'Category:Quux', NS_CATEGORY, 'en',
+                               new TitleValue( NS_CATEGORY, 'Quux', '' ) ),
+                       array( 'Quux', NS_CATEGORY, 'en',
+                               new TitleValue( NS_CATEGORY, 'Quux', '' ) ),
+                       array( ':Quux', NS_CATEGORY, 'en',
+                               new TitleValue( NS_MAIN, 'Quux', '' ) ),
+
+                       // getGenderCache() provides a mock that considers first
+                       // names ending in "a" to be female.
+
+                       array( 'a b c', NS_MAIN, 'en',
+                               new TitleValue( NS_MAIN, 'A_b_c' ) ),
+                       array( ' a  b  c ', NS_MAIN, 'en',
+                               new TitleValue( NS_MAIN, 'A_b_c' ) ),
+                       array( ' _ Foo __ Bar_ _', NS_MAIN, 'en',
+                               new TitleValue( NS_MAIN, 'Foo_Bar' ) ),
+
+                       //NOTE: cases copied from TitleTest::testSecureAndSplit. Keep in sync.
+                       array( 'Sandbox', NS_MAIN, 'en', ),
+                       array( 'A "B"', NS_MAIN, 'en', ),
+                       array( 'A \'B\'', NS_MAIN, 'en', ),
+                       array( '.com', NS_MAIN, 'en', ),
+                       array( '~', NS_MAIN, 'en', ),
+                       array( '"', NS_MAIN, 'en', ),
+                       array( '\'', NS_MAIN, 'en', ),
+
+                       array( 'Talk:Sandbox', NS_MAIN, 'en',
+                               new TitleValue( NS_TALK, 'Sandbox' ) ),
+                       array( 'Talk:Foo:Sandbox', NS_MAIN, 'en',
+                               new TitleValue( NS_TALK, 'Foo:Sandbox' ) ),
+                       array( 'File:Example.svg', NS_MAIN, 'en',
+                               new TitleValue( NS_FILE, 'Example.svg' ) ),
+                       array( 'File_talk:Example.svg', NS_MAIN, 'en',
+                               new TitleValue( NS_FILE_TALK, 'Example.svg' ) ),
+                       array( 'Foo/.../Sandbox', NS_MAIN, 'en',
+                               'Foo/.../Sandbox' ),
+                       array( 'Sandbox/...', NS_MAIN, 'en',
+                               'Sandbox/...' ),
+                       array( 'A~~', NS_MAIN, 'en',
+                               'A~~' ),
+                       // Length is 256 total, but only title part matters
+                       array( 'Category:' . str_repeat( 'x', 248 ), NS_MAIN, 'en',
+                               new TitleValue( NS_CATEGORY,
+                                       'X' . str_repeat( 'x', 247 ) ) ),
+                       array( str_repeat( 'x', 252 ), NS_MAIN, 'en',
+                               'X' . str_repeat( 'x', 251 ) )
+               );
+       }
+
+       /**
+        * @dataProvider provideParseTitle
+        */
+       public function testParseTitle( $text, $ns, $lang, $title = null ) {
+               if ( $title === null ) {
+                       $title = str_replace( ' ', '_', trim( $text ) );
+               }
+
+               if ( is_string( $title ) ) {
+                       $title = new TitleValue( NS_MAIN, $title, '' );
+               }
+
+               $codec = $this->makeCodec( $lang );
+               $actual = $codec->parseTitle( $text, $ns );
+
+               $this->assertEquals( $title, $actual );
+       }
+
+       public function provideParseTitle_invalid() {
+               //TODO: test unicode errors
+
+               return array(
+                       array( '#' ),
+                       array( '::' ),
+                       array( '::xx' ),
+                       array( '::##' ),
+                       array( ' :: x' ),
+
+                       array( 'Talk:File:Foo.jpg' ),
+                       array( 'Talk:localtestiw:Foo' ),
+                       array( 'remotetestiw:Foo' ),
+                       array( '::1' ), // only valid in user namespace
+                       array( 'User::x' ), // leading ":" in a user name is only valid of IPv6 addresses
+
+                       //NOTE: cases copied from TitleTest::testSecureAndSplit. Keep in sync.
+                       array( '' ),
+                       array( ':' ),
+                       array( '__  __' ),
+                       array( '  __  ' ),
+                       // Bad characters forbidden regardless of wgLegalTitleChars
+                       array( 'A [ B' ),
+                       array( 'A ] B' ),
+                       array( 'A { B' ),
+                       array( 'A } B' ),
+                       array( 'A < B' ),
+                       array( 'A > B' ),
+                       array( 'A | B' ),
+                       // URL encoding
+                       array( 'A%20B' ),
+                       array( 'A%23B' ),
+                       array( 'A%2523B' ),
+                       // XML/HTML character entity references
+                       // Note: Commented out because they are not marked invalid by the PHP test as
+                       // Title::newFromText runs Sanitizer::decodeCharReferencesAndNormalize first.
+                       //array( 'A &eacute; B' ),
+                       //array( 'A &#233; B' ),
+                       //array( 'A &#x00E9; B' ),
+                       // Subject of NS_TALK does not roundtrip to NS_MAIN
+                       array( 'Talk:File:Example.svg' ),
+                       // Directory navigation
+                       array( '.' ),
+                       array( '..' ),
+                       array( './Sandbox' ),
+                       array( '../Sandbox' ),
+                       array( 'Foo/./Sandbox' ),
+                       array( 'Foo/../Sandbox' ),
+                       array( 'Sandbox/.' ),
+                       array( 'Sandbox/..' ),
+                       // Tilde
+                       array( 'A ~~~ Name' ),
+                       array( 'A ~~~~ Signature' ),
+                       array( 'A ~~~~~ Timestamp' ),
+                       array( str_repeat( 'x', 256 ) ),
+                       // Namespace prefix without actual title
+                       array( 'Talk:' ),
+                       array( 'Category: ' ),
+                       array( 'Category: #bar' )
+               );
+       }
+
+       /**
+        * @dataProvider provideParseTitle_invalid
+        */
+       public function testParseTitle_invalid( $text ) {
+               $this->setExpectedException( 'MalformedTitleException' );
+
+               $codec = $this->makeCodec( 'en' );
+               $codec->parseTitle( $text, NS_MAIN );
+       }
+
+       public function provideGetNamespaceName() {
+               return array(
+                       array( NS_MAIN, 'Foo', 'en', '' ),
+                       array( NS_USER, 'Foo', 'en', 'User' ),
+                       array( NS_USER, 'Hansi Maier', 'de', 'Benutzer' ),
+
+                       // getGenderCache() provides a mock that considers first
+                       // names ending in "a" to be female.
+                       array( NS_USER, 'Lisa Müller', 'de', 'Benutzerin' ),
+               );
+       }
+
+       /**
+        * @dataProvider provideGetNamespaceName
+        *
+        * @param $namespace
+        * @param $text
+        * @param $lang
+        * @param $expected
+        *
+        * @internal param \TitleValue $title
+        */
+       public function testGetNamespaceName( $namespace, $text, $lang, $expected ) {
+               $codec = $this->makeCodec( $lang );
+               $name = $codec->getNamespaceName( $namespace, $text );
+
+               $this->assertEquals( $expected, $name );
+       }
+}
diff --git a/tests/phpunit/includes/title/TitleValueTest.php b/tests/phpunit/includes/title/TitleValueTest.php
new file mode 100644 (file)
index 0000000..3ba008d
--- /dev/null
@@ -0,0 +1,100 @@
+<?php
+/**
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ * @license GPL 2+
+ * @author Daniel Kinzler
+ */
+
+/**
+ * @covers TitleValue
+ *
+ * @group Title
+ */
+class TitleValueTest extends MediaWikiTestCase {
+
+       public function testConstruction() {
+               $title = new TitleValue( NS_USER, 'TestThis', 'stuff' );
+
+               $this->assertEquals( NS_USER, $title->getNamespace() );
+               $this->assertEquals( 'TestThis', $title->getText() );
+               $this->assertEquals( 'stuff', $title->getFragment() );
+       }
+
+       public function badConstructorProvider() {
+               return array(
+                       array( 'foo', 'title', 'fragment' ),
+                       array( null, 'title', 'fragment' ),
+                       array( 2.3, 'title', 'fragment' ),
+
+                       array( NS_MAIN, 5, 'fragment' ),
+                       array( NS_MAIN, null, 'fragment' ),
+                       array( NS_MAIN, '', 'fragment' ),
+                       array( NS_MAIN, 'foo bar', '' ),
+                       array( NS_MAIN, 'bar_', '' ),
+                       array( NS_MAIN, '_foo', '' ),
+                       array( NS_MAIN, ' eek ', '' ),
+
+                       array( NS_MAIN, 'title', 5 ),
+                       array( NS_MAIN, 'title', null ),
+                       array( NS_MAIN, 'title', array() ),
+               );
+       }
+
+       /**
+        * @dataProvider badConstructorProvider
+        */
+       public function testConstructionErrors( $ns, $text, $fragment ) {
+               $this->setExpectedException( 'InvalidArgumentException' );
+               new TitleValue( $ns, $text, $fragment );
+       }
+
+       public function fragmentTitleProvider() {
+               return array(
+                       array( new TitleValue( NS_MAIN, 'Test' ), 'foo' ),
+                       array( new TitleValue( NS_TALK, 'Test', 'foo' ), '' ),
+                       array( new TitleValue( NS_CATEGORY, 'Test', 'foo' ), 'bar' ),
+               );
+       }
+
+       /**
+        * @dataProvider fragmentTitleProvider
+        */
+       public function testCreateFragmentTitle( TitleValue $title, $fragment ) {
+               $fragmentTitle = $title->createFragmentTitle( $fragment );
+
+               $this->assertEquals( $title->getNamespace(), $fragmentTitle->getNamespace() );
+               $this->assertEquals( $title->getText(), $fragmentTitle->getText() );
+               $this->assertEquals( $fragment, $fragmentTitle->getFragment() );
+       }
+
+       public function getTextProvider() {
+               return array(
+                       array( 'Foo', 'Foo' ),
+                       array( 'Foo_Bar', 'Foo Bar' ),
+               );
+       }
+
+       /**
+        * @dataProvider getTextProvider
+        */
+       public function testGetText( $dbkey, $text ) {
+               $title = new TitleValue( NS_MAIN, $dbkey );
+
+               $this->assertEquals( $text, $title->getText() );
+       }
+}