Special:Version
When extensions specify a list of authors they expect them to appear
in Special:Version as they're specified. Sorting lists on
Special:Version as introduced in r30117 makes sense for things like
the names of parser functions, but not authors.
Some extensions give arrays like array("foo", "bar", "others") and
expect output like "foo, bar and others". Sometimes (like in the case
of Maps.php) this would only incidentally work.
Furthermore in some cases the first author in the array indicates the
primary author, this is the case with CheckUser.php and other similar
extensions with multiple authors.
509 lines
15 KiB
PHP
509 lines
15 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Give information about the version of MediaWiki, PHP, the DB and extensions
|
|
*
|
|
* @ingroup SpecialPage
|
|
*
|
|
* @author Ævar Arnfjörð Bjarmason <avarab@gmail.com>
|
|
* @copyright Copyright © 2005, Ævar Arnfjörð Bjarmason
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU General Public License 2.0 or later
|
|
*/
|
|
class SpecialVersion extends SpecialPage {
|
|
private $firstExtOpened = true;
|
|
|
|
static $viewvcUrls = array(
|
|
'svn+ssh://svn.wikimedia.org/svnroot/mediawiki' => 'http://svn.wikimedia.org/viewvc/mediawiki',
|
|
'http://svn.wikimedia.org/svnroot/mediawiki' => 'http://svn.wikimedia.org/viewvc/mediawiki',
|
|
# Doesn't work at the time of writing but maybe some day:
|
|
'https://svn.wikimedia.org/viewvc/mediawiki' => 'http://svn.wikimedia.org/viewvc/mediawiki',
|
|
);
|
|
|
|
function __construct(){
|
|
parent::__construct( 'Version' );
|
|
}
|
|
|
|
/**
|
|
* main()
|
|
*/
|
|
function execute( $par ) {
|
|
global $wgOut, $wgMessageCache, $wgSpecialVersionShowHooks, $wgContLang;
|
|
$wgMessageCache->loadAllMessages();
|
|
|
|
$this->setHeaders();
|
|
$this->outputHeader();
|
|
|
|
$wgOut->addHTML( Xml::openElement( 'div',
|
|
array( 'dir' => $wgContLang->getDir() ) ) );
|
|
$text =
|
|
$this->MediaWikiCredits() .
|
|
$this->softwareInformation() .
|
|
$this->extensionCredits();
|
|
if ( $wgSpecialVersionShowHooks ) {
|
|
$text .= $this->wgHooks();
|
|
}
|
|
$wgOut->addWikiText( $text );
|
|
$wgOut->addHTML( $this->IPInfo() );
|
|
$wgOut->addHTML( '</div>' );
|
|
}
|
|
|
|
/**#@+
|
|
* @private
|
|
*/
|
|
|
|
/**
|
|
* @return wiki text showing the license information
|
|
*/
|
|
static function MediaWikiCredits() {
|
|
global $wgContLang;
|
|
|
|
$ret = Xml::element( 'h2', array( 'id' => 'mw-version-license' ), wfMsg( 'version-license' ) );
|
|
|
|
// This text is always left-to-right.
|
|
$ret .= '<div dir="ltr">';
|
|
$ret .= "__NOTOC__
|
|
This wiki is powered by '''[http://www.mediawiki.org/ MediaWiki]''',
|
|
copyright © 2001-2010 Magnus Manske, Brion Vibber, Lee Daniel Crocker,
|
|
Tim Starling, Erik Möller, Gabriel Wicke, Ævar Arnfjörð Bjarmason,
|
|
Niklas Laxström, Domas Mituzas, Rob Church, Yuri Astrakhan, Aryeh Gregor,
|
|
Aaron Schulz, Andrew Garrett, Raimond Spekking, Alexandre Emsenhuber,
|
|
Siebrand Mazeland, Chad Horohoe and others.
|
|
|
|
MediaWiki 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.
|
|
|
|
MediaWiki 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 [{{SERVER}}{{SCRIPTPATH}}/COPYING 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
|
|
or [http://www.gnu.org/licenses/old-licenses/gpl-2.0.html read it online].
|
|
";
|
|
$ret .= '</div>';
|
|
|
|
return str_replace( "\t\t", '', $ret ) . "\n";
|
|
}
|
|
|
|
/**
|
|
* @return wiki text showing the third party software versions (apache, php, mysql).
|
|
*/
|
|
static function softwareInformation() {
|
|
$dbr = wfGetDB( DB_SLAVE );
|
|
|
|
// Put the software in an array of form 'name' => 'version'. All messages should
|
|
// be loaded here, so feel free to use wfMsg*() in the 'name'. Raw HTML or wikimarkup
|
|
// can be used
|
|
$software = array();
|
|
$software['[http://www.mediawiki.org/ MediaWiki]'] = self::getVersionLinked();
|
|
$software['[http://www.php.net/ PHP]'] = phpversion() . " (" . php_sapi_name() . ")";
|
|
$software[$dbr->getSoftwareLink()] = $dbr->getServerVersion();
|
|
|
|
// Allow a hook to add/remove items
|
|
wfRunHooks( 'SoftwareInfo', array( &$software ) );
|
|
|
|
$out = Xml::element( 'h2', array( 'id' => 'mw-version-software' ), wfMsg( 'version-software' ) ) .
|
|
Xml::openElement( 'table', array( 'class' => 'wikitable', 'id' => 'sv-software' ) ) .
|
|
"<tr>
|
|
<th>" . wfMsg( 'version-software-product' ) . "</th>
|
|
<th>" . wfMsg( 'version-software-version' ) . "</th>
|
|
</tr>\n";
|
|
foreach( $software as $name => $version ) {
|
|
$out .= "<tr>
|
|
<td>" . $name . "</td>
|
|
<td>" . $version . "</td>
|
|
</tr>\n";
|
|
}
|
|
return $out . Xml::closeElement( 'table' );
|
|
}
|
|
|
|
/**
|
|
* Return a string of the MediaWiki version with SVN revision if available
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public static function getVersion( $flags = '' ) {
|
|
global $wgVersion, $IP;
|
|
wfProfileIn( __METHOD__ );
|
|
|
|
$info = self::getSvnInfo( $IP );
|
|
if ( !$info ) {
|
|
$version = $wgVersion;
|
|
} elseif( $flags === 'nodb' ) {
|
|
$version = "$wgVersion (r{$info['checkout-rev']})";
|
|
} else {
|
|
$version = $wgVersion . ' ' .
|
|
wfMsg(
|
|
'version-svn-revision',
|
|
isset( $info['directory-rev'] ) ? $info['directory-rev'] : '',
|
|
$info['checkout-rev']
|
|
);
|
|
}
|
|
|
|
wfProfileOut( __METHOD__ );
|
|
return $version;
|
|
}
|
|
|
|
/**
|
|
* Return a wikitext-formatted string of the MediaWiki version with a link to
|
|
* the SVN revision if available
|
|
*
|
|
* @return mixed
|
|
*/
|
|
public static function getVersionLinked() {
|
|
global $wgVersion, $IP;
|
|
wfProfileIn( __METHOD__ );
|
|
$info = self::getSvnInfo( $IP );
|
|
if ( isset( $info['checkout-rev'] ) ) {
|
|
$linkText = wfMsg(
|
|
'version-svn-revision',
|
|
isset( $info['directory-rev'] ) ? $info['directory-rev'] : '',
|
|
$info['checkout-rev']
|
|
);
|
|
if ( isset( $info['viewvc-url'] ) ) {
|
|
$version = "$wgVersion [{$info['viewvc-url']} $linkText]";
|
|
} else {
|
|
$version = "$wgVersion $linkText";
|
|
}
|
|
} else {
|
|
$version = $wgVersion;
|
|
}
|
|
wfProfileOut( __METHOD__ );
|
|
return $version;
|
|
}
|
|
|
|
/** Generate wikitext showing extensions name, URL, author and description */
|
|
function extensionCredits() {
|
|
global $wgExtensionCredits, $wgExtensionFunctions, $wgParser, $wgSkinExtensionFunctions;
|
|
|
|
if ( ! count( $wgExtensionCredits ) && ! count( $wgExtensionFunctions ) && ! count( $wgSkinExtensionFunctions ) )
|
|
return '';
|
|
|
|
$extensionTypes = array(
|
|
'specialpage' => wfMsg( 'version-specialpages' ),
|
|
'parserhook' => wfMsg( 'version-parserhooks' ),
|
|
'variable' => wfMsg( 'version-variables' ),
|
|
'media' => wfMsg( 'version-mediahandlers' ),
|
|
'other' => wfMsg( 'version-other' ),
|
|
);
|
|
wfRunHooks( 'SpecialVersionExtensionTypes', array( &$this, &$extensionTypes ) );
|
|
|
|
$out = Xml::element( 'h2', array( 'id' => 'mw-version-ext' ), wfMsg( 'version-extensions' ) ) .
|
|
Xml::openElement( 'table', array( 'class' => 'wikitable', 'id' => 'sv-ext' ) );
|
|
|
|
foreach ( $extensionTypes as $type => $text ) {
|
|
if ( isset ( $wgExtensionCredits[$type] ) && count ( $wgExtensionCredits[$type] ) ) {
|
|
$out .= $this->openExtType( $text, 'credits-' . $type );
|
|
|
|
usort( $wgExtensionCredits[$type], array( $this, 'compare' ) );
|
|
|
|
foreach ( $wgExtensionCredits[$type] as $extension ) {
|
|
$out .= $this->formatCredits( $extension );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( count( $wgExtensionFunctions ) ) {
|
|
$out .= $this->openExtType( wfMsg( 'version-extension-functions' ), 'extension-functions' );
|
|
$out .= '<tr><td colspan="4">' . $this->listToText( $wgExtensionFunctions ) . "</td></tr>\n";
|
|
}
|
|
|
|
if ( $cnt = count( $tags = $wgParser->getTags() ) ) {
|
|
for ( $i = 0; $i < $cnt; ++$i )
|
|
$tags[$i] = "<{$tags[$i]}>";
|
|
$out .= $this->openExtType( wfMsg( 'version-parser-extensiontags' ), 'parser-tags' );
|
|
$out .= '<tr><td colspan="4">' . $this->listToText( $tags ). "</td></tr>\n";
|
|
}
|
|
|
|
if( $cnt = count( $fhooks = $wgParser->getFunctionHooks() ) ) {
|
|
$out .= $this->openExtType( wfMsg( 'version-parser-function-hooks' ), 'parser-function-hooks' );
|
|
$out .= '<tr><td colspan="4">' . $this->listToText( $fhooks ) . "</td></tr>\n";
|
|
}
|
|
|
|
if ( count( $wgSkinExtensionFunctions ) ) {
|
|
$out .= $this->openExtType( wfMsg( 'version-skin-extension-functions' ), 'skin-extension-functions' );
|
|
$out .= '<tr><td colspan="4">' . $this->listToText( $wgSkinExtensionFunctions ) . "</td></tr>\n";
|
|
}
|
|
$out .= Xml::closeElement( 'table' );
|
|
return $out;
|
|
}
|
|
|
|
/** Callback to sort extensions by type */
|
|
function compare( $a, $b ) {
|
|
global $wgLang;
|
|
if( $a['name'] === $b['name'] ) {
|
|
return 0;
|
|
} else {
|
|
return $wgLang->lc( $a['name'] ) > $wgLang->lc( $b['name'] )
|
|
? 1
|
|
: -1;
|
|
}
|
|
}
|
|
|
|
function formatCredits( $extension ) {
|
|
$name = isset( $extension['name'] ) ? $extension['name'] : '[no name]';
|
|
if ( isset( $extension['path'] ) ) {
|
|
$svnInfo = self::getSvnInfo( dirname($extension['path']) );
|
|
$directoryRev = isset( $svnInfo['directory-rev'] ) ? $svnInfo['directory-rev'] : null;
|
|
$checkoutRev = isset( $svnInfo['checkout-rev'] ) ? $svnInfo['checkout-rev'] : null;
|
|
$viewvcUrl = isset( $svnInfo['viewvc-url'] ) ? $svnInfo['viewvc-url'] : null;
|
|
} else {
|
|
$directoryRev = null;
|
|
$checkoutRev = null;
|
|
$viewvcUrl = null;
|
|
}
|
|
|
|
# Make main link (or just the name if there is no URL)
|
|
if ( isset( $extension['url'] ) ) {
|
|
$mainLink = "[{$extension['url']} $name]";
|
|
} else {
|
|
$mainLink = $name;
|
|
}
|
|
if ( isset( $extension['version'] ) ) {
|
|
$versionText = '<span class="mw-version-ext-version">' .
|
|
wfMsg( 'version-version', $extension['version'] ) .
|
|
'</span>';
|
|
} else {
|
|
$versionText = '';
|
|
}
|
|
|
|
# Make subversion text/link
|
|
if ( $checkoutRev ) {
|
|
$svnText = wfMsg( 'version-svn-revision', $directoryRev, $checkoutRev );
|
|
$svnText = isset( $viewvcUrl ) ? "[$viewvcUrl $svnText]" : $svnText;
|
|
} else {
|
|
$svnText = false;
|
|
}
|
|
|
|
# Make description text
|
|
$description = isset ( $extension['description'] ) ? $extension['description'] : '';
|
|
if( isset ( $extension['descriptionmsg'] ) ) {
|
|
# Look for a localized description
|
|
$descriptionMsg = $extension['descriptionmsg'];
|
|
if( is_array( $descriptionMsg ) ) {
|
|
$descriptionMsgKey = $descriptionMsg[0]; // Get the message key
|
|
array_shift( $descriptionMsg ); // Shift out the message key to get the parameters only
|
|
array_map( "htmlspecialchars", $descriptionMsg ); // For sanity
|
|
$msg = wfMsg( $descriptionMsgKey, $descriptionMsg );
|
|
} else {
|
|
$msg = wfMsg( $descriptionMsg );
|
|
}
|
|
if ( !wfEmptyMsg( $descriptionMsg, $msg ) && $msg != '' ) {
|
|
$description = $msg;
|
|
}
|
|
}
|
|
|
|
if ( $svnText !== false ) {
|
|
$extNameVer = "<tr>
|
|
<td><em>$mainLink $versionText</em></td>
|
|
<td><em>$svnText</em></td>";
|
|
} else {
|
|
$extNameVer = "<tr>
|
|
<td colspan=\"2\"><em>$mainLink $versionText</em></td>";
|
|
}
|
|
$author = isset ( $extension['author'] ) ? $extension['author'] : array();
|
|
$extDescAuthor = "<td>$description</td>
|
|
<td>" . $this->listToText( (array)$author, false ) . "</td>
|
|
</tr>\n";
|
|
return $extNameVer . $extDescAuthor;
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
function wgHooks() {
|
|
global $wgHooks;
|
|
|
|
if ( count( $wgHooks ) ) {
|
|
$myWgHooks = $wgHooks;
|
|
ksort( $myWgHooks );
|
|
|
|
$ret = Xml::element( 'h2', array( 'id' => 'mw-version-hooks' ), wfMsg( 'version-hooks' ) ) .
|
|
Xml::openElement( 'table', array( 'class' => 'wikitable', 'id' => 'sv-hooks' ) ) .
|
|
"<tr>
|
|
<th>" . wfMsg( 'version-hook-name' ) . "</th>
|
|
<th>" . wfMsg( 'version-hook-subscribedby' ) . "</th>
|
|
</tr>\n";
|
|
|
|
foreach ( $myWgHooks as $hook => $hooks )
|
|
$ret .= "<tr>
|
|
<td>$hook</td>
|
|
<td>" . $this->listToText( $hooks ) . "</td>
|
|
</tr>\n";
|
|
|
|
$ret .= Xml::closeElement( 'table' );
|
|
return $ret;
|
|
} else
|
|
return '';
|
|
}
|
|
|
|
private function openExtType( $text, $name = null ) {
|
|
$opt = array( 'colspan' => 4 );
|
|
$out = '';
|
|
|
|
if( !$this->firstExtOpened ) {
|
|
// Insert a spacing line
|
|
$out .= '<tr class="sv-space">' . Xml::element( 'td', $opt ) . "</tr>\n";
|
|
}
|
|
$this->firstExtOpened = false;
|
|
|
|
if( $name )
|
|
$opt['id'] = "sv-$name";
|
|
|
|
$out .= "<tr>" . Xml::element( 'th', $opt, $text ) . "</tr>\n";
|
|
return $out;
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
function IPInfo() {
|
|
$ip = str_replace( '--', ' - ', htmlspecialchars( wfGetIP() ) );
|
|
return "<!-- visited from $ip -->\n" .
|
|
"<span style='display:none'>visited from $ip</span>";
|
|
}
|
|
|
|
/**
|
|
* @param array $list
|
|
* @param bool $sort
|
|
* @return string
|
|
*/
|
|
function listToText( $list, $sort = true ) {
|
|
$cnt = count( $list );
|
|
|
|
if ( $cnt == 1 ) {
|
|
// Enforce always returning a string
|
|
return (string)self::arrayToString( $list[0] );
|
|
} elseif ( $cnt == 0 ) {
|
|
return '';
|
|
} else {
|
|
global $wgLang;
|
|
if ( $sort ) {
|
|
sort( $list );
|
|
}
|
|
return $wgLang->listToText( array_map( array( __CLASS__, 'arrayToString' ), $list ) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param mixed $list Will convert an array to string if given and return
|
|
* the paramater unaltered otherwise
|
|
* @return mixed
|
|
*/
|
|
static function arrayToString( $list ) {
|
|
if( is_array( $list ) && count( $list ) == 1 )
|
|
$list = $list[0];
|
|
if( is_object( $list ) ) {
|
|
$class = get_class( $list );
|
|
return "($class)";
|
|
} elseif ( !is_array( $list ) ) {
|
|
return $list;
|
|
} else {
|
|
if( is_object( $list[0] ) )
|
|
$class = get_class( $list[0] );
|
|
else
|
|
$class = $list[0];
|
|
return "($class, {$list[1]})";
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get an associative array of information about a given path, from its .svn
|
|
* subdirectory. Returns false on error, such as if the directory was not
|
|
* checked out with subversion.
|
|
*
|
|
* Returned keys are:
|
|
* Required:
|
|
* checkout-rev The revision which was checked out
|
|
* Optional:
|
|
* directory-rev The revision when the directory was last modified
|
|
* url The subversion URL of the directory
|
|
* repo-url The base URL of the repository
|
|
* viewvc-url A ViewVC URL pointing to the checked-out revision
|
|
*/
|
|
public static function getSvnInfo( $dir ) {
|
|
// http://svnbook.red-bean.com/nightly/en/svn.developer.insidewc.html
|
|
$entries = $dir . '/.svn/entries';
|
|
|
|
if( !file_exists( $entries ) ) {
|
|
return false;
|
|
}
|
|
|
|
$lines = file( $entries );
|
|
if ( !count( $lines ) ) {
|
|
return false;
|
|
}
|
|
|
|
// check if file is xml (subversion release <= 1.3) or not (subversion release = 1.4)
|
|
if( preg_match( '/^<\?xml/', $lines[0] ) ) {
|
|
// subversion is release <= 1.3
|
|
if( !function_exists( 'simplexml_load_file' ) ) {
|
|
// We could fall back to expat... YUCK
|
|
return false;
|
|
}
|
|
|
|
// SimpleXml whines about the xmlns...
|
|
wfSuppressWarnings();
|
|
$xml = simplexml_load_file( $entries );
|
|
wfRestoreWarnings();
|
|
|
|
if( $xml ) {
|
|
foreach( $xml->entry as $entry ) {
|
|
if( $xml->entry[0]['name'] == '' ) {
|
|
// The directory entry should always have a revision marker.
|
|
if( $entry['revision'] ) {
|
|
return array( 'checkout-rev' => intval( $entry['revision'] ) );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// subversion is release 1.4 or above
|
|
if ( count( $lines ) < 11 ) {
|
|
return false;
|
|
}
|
|
$info = array(
|
|
'checkout-rev' => intval( trim( $lines[3] ) ),
|
|
'url' => trim( $lines[4] ),
|
|
'repo-url' => trim( $lines[5] ),
|
|
'directory-rev' => intval( trim( $lines[10] ) )
|
|
);
|
|
if ( isset( self::$viewvcUrls[$info['repo-url']] ) ) {
|
|
$viewvc = str_replace(
|
|
$info['repo-url'],
|
|
self::$viewvcUrls[$info['repo-url']],
|
|
$info['url']
|
|
);
|
|
$pathRelativeToRepo = substr( $info['url'], strlen( $info['repo-url'] ) );
|
|
$viewvc .= '/?pathrev=';
|
|
$viewvc .= urlencode( $info['checkout-rev'] );
|
|
$info['viewvc-url'] = $viewvc;
|
|
}
|
|
return $info;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the revision number of a Subversion working directory.
|
|
*
|
|
* @param String $dir Directory of the svn checkout
|
|
* @return int revision number as int
|
|
*/
|
|
public static function getSvnRevision( $dir ) {
|
|
$info = self::getSvnInfo( $dir );
|
|
if ( $info === false ) {
|
|
return false;
|
|
} elseif ( isset( $info['checkout-rev'] ) ) {
|
|
return $info['checkout-rev'];
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**#@-*/
|
|
}
|