Implement a number of namespace related equals functions:
* MWNamespace::equals to test equivalence of two namespaces (forward compatible with any changes we may make like introducing namespace keys like 'USER') * MWNamespace::subjectEquals to test equivalence of the subject of two namespaces e.g.: MWNamespace::subjectEquals( NS_USER, $ns ); instead of testing for equivalence to both NS_USER and NS_USER_TALK * Title::inNamespace to use instead of $title->getNamespace() == NS_??? * Title::inNamespaces for use like $title->inNamespaces( NS_USER, NS_PROJECT ) when you only care if it's in one of a number of namespaces (also accepts an array) * Title::hasSubjectNamespace for use instead of testing for equivalence to both the subject and talk such as NS_USER and NS_USER_TALK. Include phpunit tests for all this new code, and also add some tests for some existing code.
This commit is contained in:
parent
3e35e3bb8e
commit
3414e91bae
4 changed files with 237 additions and 24 deletions
|
|
@ -58,9 +58,18 @@ class MWNamespace {
|
|||
*
|
||||
* @param $index Int: namespace index
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public static function isSubject( $index ) {
|
||||
return !self::isTalk( $index );
|
||||
}
|
||||
|
||||
/**
|
||||
* @see self::isSubject
|
||||
* @deprecated Please use the more consistently named isSubject (since 1.19)
|
||||
*/
|
||||
public static function isMain( $index ) {
|
||||
return !self::isTalk( $index );
|
||||
return self::isSubject( $index );
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -131,12 +140,46 @@ class MWNamespace {
|
|||
* @param $index
|
||||
*
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public static function exists( $index ) {
|
||||
$nslist = self::getCanonicalNamespaces();
|
||||
return isset( $nslist[$index] );
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether the specified namespaces are the same namespace
|
||||
*
|
||||
* @note It's possible that in the future we may start using something
|
||||
* other than just namespace indexes. Under that circumstance making use
|
||||
* of this function rather than directly doing comparison will make
|
||||
* sure that code will not potentially break.
|
||||
*
|
||||
* @param $ns1 The first namespace index
|
||||
* @param $ns2 The second namespae index
|
||||
*
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public static function equals( $ns1, $ns2 ) {
|
||||
return $ns1 == $ns2;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns whether the specified namespaces share the same subject.
|
||||
* eg: NS_USER and NS_USER wil return true, as well
|
||||
* NS_USER and NS_USER_TALK will return true.
|
||||
*
|
||||
* @param $ns1 The first namespace index
|
||||
* @param $ns2 The second namespae index
|
||||
*
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public static function subjectEquals( $ns1, $ns2 ) {
|
||||
return self::getSubject( $ns1 ) == self::getSubject( $ns2 );
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns array of all defined namespaces with their canonical
|
||||
* (English) names.
|
||||
|
|
|
|||
|
|
@ -1945,6 +1945,57 @@ class Title {
|
|||
: false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns true if the title is inside the specified namespace.
|
||||
*
|
||||
* Please make use of this instead of comparing to getNamespace()
|
||||
* This function is much more resistant to changes we may make
|
||||
* to namespaces than code that makes direct comparisons.
|
||||
* @param $ns The namespace
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public function inNamespace( $ns ) {
|
||||
return MWNamespace::equals( $this->getNamespace(), $ns );
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns true if the title is inside one of the specified namespaces.
|
||||
*
|
||||
* @param ...$namespaces The namespaces to check for
|
||||
* @return bool
|
||||
* @since 1.19
|
||||
*/
|
||||
public function inNamespaces( /* ... */ ) {
|
||||
$namespaces = func_get_args();
|
||||
if ( count( $namespaces ) > 0 && is_array( $namespaces[0] ) ) {
|
||||
$namespaces = $namespaces[0];
|
||||
}
|
||||
|
||||
foreach ( $namespaces as $ns ) {
|
||||
if ( $this->inNamespace( $ns ) ) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns true if the title has the same subject namespace as the
|
||||
* namespace specified.
|
||||
* For example this method will take NS_USER and return true if namespace
|
||||
* is either NS_USER or NS_USER_TALK since both of them have NS_USER
|
||||
* as their subject namespace.
|
||||
*
|
||||
* This is MUCH simpler than individually testing for equivilance
|
||||
* against both NS_USER and NS_USER_TALK, and is also forward compatible.
|
||||
* @since 1.19
|
||||
*/
|
||||
public function hasSubjectNamespace( $ns ) {
|
||||
return MWNamespace::subjectEquals( $this->getNamespace(), $ns );
|
||||
}
|
||||
|
||||
/**
|
||||
* Does this have subpages? (Warning, usually requires an extra DB query.)
|
||||
*
|
||||
|
|
|
|||
|
|
@ -39,25 +39,29 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
/**
|
||||
* Please make sure to change testIsTalk() if you change the assertions below
|
||||
*/
|
||||
public function testIsMain() {
|
||||
public function testIsSubject() {
|
||||
// Special namespaces
|
||||
$this->assertTrue( MWNamespace::isMain( NS_MEDIA ) );
|
||||
$this->assertTrue( MWNamespace::isMain( NS_SPECIAL ) );
|
||||
$this->assertTrue( MWNamespace::isSubject( NS_MEDIA ) );
|
||||
$this->assertTrue( MWNamespace::isSubject( NS_SPECIAL ) );
|
||||
|
||||
// Subject pages
|
||||
$this->assertTrue( MWNamespace::isMain( NS_MAIN ) );
|
||||
$this->assertTrue( MWNamespace::isMain( NS_USER ) );
|
||||
$this->assertTrue( MWNamespace::isMain( 100 ) ); # user defined
|
||||
$this->assertTrue( MWNamespace::isSubject( NS_MAIN ) );
|
||||
$this->assertTrue( MWNamespace::isSubject( NS_USER ) );
|
||||
$this->assertTrue( MWNamespace::isSubject( 100 ) ); # user defined
|
||||
|
||||
// Talk pages
|
||||
$this->assertFalse( MWNamespace::isMain( NS_TALK ) );
|
||||
$this->assertFalse( MWNamespace::isMain( NS_USER_TALK ) );
|
||||
$this->assertFalse( MWNamespace::isMain( 101 ) ); # user defined
|
||||
$this->assertFalse( MWNamespace::isSubject( NS_TALK ) );
|
||||
$this->assertFalse( MWNamespace::isSubject( NS_USER_TALK ) );
|
||||
$this->assertFalse( MWNamespace::isSubject( 101 ) ); # user defined
|
||||
|
||||
// Back compat
|
||||
$this->assertTrue( MWNamespace::isMain( NS_MAIN ) == MWNamespace::isSubject( NS_MAIN ) );
|
||||
$this->assertTrue( MWNamespace::isMain( NS_USER_TALK ) == MWNamespace::isSubject( NS_USER_TALK ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Reverse of testIsMain().
|
||||
* Please update testIsMain() if you change assertions below
|
||||
* Reverse of testIsSubject().
|
||||
* Please update testIsSubject() if you change assertions below
|
||||
*/
|
||||
public function testIsTalk() {
|
||||
// Special namespaces
|
||||
|
|
@ -75,6 +79,17 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
$this->assertTrue( MWNamespace::isTalk( 101 ) ); # user defined
|
||||
}
|
||||
|
||||
/**
|
||||
*/
|
||||
public function testGetSubject() {
|
||||
// Special namespaces are their own subjects
|
||||
$this->assertEquals( NS_MEDIA, MWNamespace::getSubject( NS_MEDIA ) );
|
||||
$this->assertEquals( NS_SPECIAL, MWNamespace::getSubject( NS_SPECIAL ) );
|
||||
|
||||
$this->assertEquals( NS_MAIN, MWNamespace::getSubject( NS_TALK ) );
|
||||
$this->assertEquals( NS_USER, MWNamespace::getSubject( NS_USER_TALK ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Regular getTalk() calls
|
||||
* Namespaces without a talk page (NS_MEDIA, NS_SPECIAL) are tested in
|
||||
|
|
@ -82,6 +97,9 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
*/
|
||||
public function testGetTalk() {
|
||||
$this->assertEquals( NS_TALK, MWNamespace::getTalk( NS_MAIN ) );
|
||||
$this->assertEquals( NS_TALK, MWNamespace::getTalk( NS_TALK ) );
|
||||
$this->assertEquals( NS_USER_TALK, MWNamespace::getTalk( NS_USER ) );
|
||||
$this->assertEquals( NS_USER_TALK, MWNamespace::getTalk( NS_USER_TALK ) );
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
@ -108,7 +126,7 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
* the function testGetAssociatedExceptions()
|
||||
*/
|
||||
public function testGetAssociated() {
|
||||
$this->assertEquals( NS_TALK, MWNamespace::getAssociated( NS_MAIN ) );
|
||||
$this->assertEquals( NS_TALK, MWNamespace::getAssociated( NS_MAIN ) );
|
||||
$this->assertEquals( NS_MAIN, MWNamespace::getAssociated( NS_TALK ) );
|
||||
|
||||
}
|
||||
|
|
@ -130,17 +148,6 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
$this->assertNull( MWNamespace::getAssociated( NS_SPECIAL ) );
|
||||
}
|
||||
|
||||
/**
|
||||
*/
|
||||
public function testGetSubject() {
|
||||
// Special namespaces are their own subjects
|
||||
$this->assertEquals( NS_MEDIA, MWNamespace::getSubject( NS_MEDIA ) );
|
||||
$this->assertEquals( NS_SPECIAL, MWNamespace::getSubject( NS_SPECIAL ) );
|
||||
|
||||
$this->assertEquals( NS_MAIN, MWNamespace::getSubject( NS_TALK ) );
|
||||
$this->assertEquals( NS_USER, MWNamespace::getSubject( NS_USER_TALK ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo Implement testExists().
|
||||
*/
|
||||
|
|
@ -152,6 +159,40 @@ class MWNamespaceTest extends MediaWikiTestCase {
|
|||
);
|
||||
}
|
||||
*/
|
||||
|
||||
/**
|
||||
* Test MWNamespace::equals
|
||||
* Note if we add a namespace registration system with keys like 'MAIN'
|
||||
* we should add tests here for equivilance on things like 'MAIN' == 0
|
||||
* and 'MAIN' == NS_MAIN.
|
||||
*/
|
||||
public function testEquals() {
|
||||
$this->assertTrue( MWNamespace::equals( NS_MAIN, NS_MAIN ) );
|
||||
$this->assertTrue( MWNamespace::equals( NS_MAIN, 0 ) ); // In case we make NS_MAIN 'MAIN'
|
||||
$this->assertTrue( MWNamespace::equals( NS_USER, NS_USER ) );
|
||||
$this->assertTrue( MWNamespace::equals( NS_USER, 2 ) );
|
||||
$this->assertTrue( MWNamespace::equals( NS_USER_TALK, NS_USER_TALK ) );
|
||||
$this->assertTrue( MWNamespace::equals( NS_SPECIAL, NS_SPECIAL ) );
|
||||
$this->assertFalse( MWNamespace::equals( NS_MAIN, NS_TALK ) );
|
||||
$this->assertFalse( MWNamespace::equals( NS_USER, NS_USER_TALK ) );
|
||||
$this->assertFalse( MWNamespace::equals( NS_PROJECT, NS_TEMPLATE ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test MWNamespace::subjectEquals
|
||||
*/
|
||||
public function testSubjectEquals() {
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_MAIN, NS_MAIN ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_MAIN, 0 ) ); // In case we make NS_MAIN 'MAIN'
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_USER, NS_USER ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_USER, 2 ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_USER_TALK, NS_USER_TALK ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_SPECIAL, NS_SPECIAL ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_MAIN, NS_TALK ) );
|
||||
$this->assertTrue( MWNamespace::subjectEquals( NS_USER, NS_USER_TALK ) );
|
||||
$this->assertFalse( MWNamespace::subjectEquals( NS_PROJECT, NS_TEMPLATE ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* @todo Implement testGetCanonicalNamespaces().
|
||||
*/
|
||||
|
|
|
|||
78
tests/phpunit/includes/TitleMethodsTest.php
Normal file
78
tests/phpunit/includes/TitleMethodsTest.php
Normal file
|
|
@ -0,0 +1,78 @@
|
|||
<?php
|
||||
|
||||
class TitleMethodsTest extends MediaWikiTestCase {
|
||||
|
||||
public function dataEquals() {
|
||||
return array(
|
||||
array( 'Main Page', 'Main Page', true ),
|
||||
array( 'Main Page', 'Not The Main Page', false ),
|
||||
array( 'Main Page', 'Project:Main Page', false ),
|
||||
array( 'File:Example.png', 'Image:Example.png', true ),
|
||||
array( 'Special:Version', 'Special:Version', true ),
|
||||
array( 'Special:Version', 'Special:Recentchanges', false ),
|
||||
array( 'Special:Version', 'Main Page', false ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider dataEquals
|
||||
*/
|
||||
public function testEquals( $titleA, $titleB, $expectedBool ) {
|
||||
$titleA = Title::newFromText( $titleA );
|
||||
$titleB = Title::newFromText( $titleB );
|
||||
|
||||
$this->assertEquals( $titleA->equals( $titleB ), $expectedBool );
|
||||
$this->assertEquals( $titleB->equals( $titleA ), $expectedBool );
|
||||
}
|
||||
|
||||
public function dataInNamespace() {
|
||||
return array(
|
||||
array( 'Main Page', NS_MAIN, true ),
|
||||
array( 'Main Page', NS_TALK, false ),
|
||||
array( 'Main Page', NS_USER, false ),
|
||||
array( 'User:Foo', NS_USER, true ),
|
||||
array( 'User:Foo', NS_USER_TALK, false ),
|
||||
array( 'User:Foo', NS_TEMPLATE, false ),
|
||||
array( 'User_talk:Foo', NS_USER_TALK, true ),
|
||||
array( 'User_talk:Foo', NS_USER, false ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider dataInNamespace
|
||||
*/
|
||||
public function testInNamespace( $title, $ns, $expectedBool ) {
|
||||
$title = Title::newFromText( $title );
|
||||
$this->assertEquals( $title->inNamespace( $ns ), $expectedBool );
|
||||
}
|
||||
|
||||
public function testInNamespaces() {
|
||||
$mainpage = Title::newFromText( 'Main Page' );
|
||||
$this->assertTrue( $mainpage->inNamespaces( NS_MAIN, NS_USER ) );
|
||||
$this->assertTrue( $mainpage->inNamespaces( array( NS_MAIN, NS_USER ) ) );
|
||||
$this->assertTrue( $mainpage->inNamespaces( array( NS_USER, NS_MAIN ) ) );
|
||||
$this->assertFalse( $mainpage->inNamespaces( array( NS_PROJECT, NS_TEMPLATE ) ) );
|
||||
}
|
||||
|
||||
public function dataHasSubjectNamespace() {
|
||||
return array(
|
||||
array( 'Main Page', NS_MAIN, true ),
|
||||
array( 'Main Page', NS_TALK, true ),
|
||||
array( 'Main Page', NS_USER, false ),
|
||||
array( 'User:Foo', NS_USER, true ),
|
||||
array( 'User:Foo', NS_USER_TALK, true ),
|
||||
array( 'User:Foo', NS_TEMPLATE, false ),
|
||||
array( 'User_talk:Foo', NS_USER_TALK, true ),
|
||||
array( 'User_talk:Foo', NS_USER, true ),
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider dataHasSubjectNamespace
|
||||
*/
|
||||
public function testHasSubjectNamespace( $title, $ns, $expectedBool ) {
|
||||
$title = Title::newFromText( $title );
|
||||
$this->assertEquals( $title->hasSubjectNamespace( $ns ), $expectedBool );
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Reference in a new issue