b4f460a75670c0b20edc582afd9b5691f2098f39
3 * Give information about the version of MediaWiki, PHP, the DB and extensions
5 * @addtogroup SpecialPage
7 * @author Ævar Arnfjörð Bjarmason <avarab@gmail.com>
8 * @copyright Copyright © 2005, Ævar Arnfjörð Bjarmason
9 * @license http://www.gnu.org/copyleft/gpl.html GNU General Public License 2.0 or later
15 function wfSpecialVersion() {
16 $version = new SpecialVersion
;
20 class SpecialVersion
{
21 private $firstExtOpened = true;
27 global $wgOut, $wgMessageCache;
28 $wgMessageCache->loadAllMessages();
30 $wgOut->addHTML( '<div dir="ltr">' );
32 $this->MediaWikiCredits() .
33 $this->softwareInformation() .
34 $this->extensionCredits() .
37 $wgOut->addHTML( $this->IPInfo() );
38 $wgOut->addHTML( '</div>' );
46 * @return wiki text showing the license information
48 static function MediaWikiCredits() {
49 $ret = Xml
::element( 'h2', array( 'id' => 'mw-version-license' ), wfMsg( 'version-license' ) ) .
51 This wiki is powered by '''[http://www.mediawiki.org/ MediaWiki]''',
52 copyright (C) 2001-2008 Magnus Manske, Brion Vibber, Lee Daniel Crocker,
53 Tim Starling, Erik Möller, Gabriel Wicke, Ævar Arnfjörð Bjarmason,
54 Niklas Laxström, Domas Mituzas, Rob Church, Yuri Astrakhan and others.
56 MediaWiki is free software; you can redistribute it and/or modify
57 it under the terms of the GNU General Public License as published by
58 the Free Software Foundation; either version 2 of the License, or
59 (at your option) any later version.
61 MediaWiki is distributed in the hope that it will be useful,
62 but WITHOUT ANY WARRANTY; without even the implied warranty of
63 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
64 GNU General Public License for more details.
66 You should have received [{{SERVER}}{{SCRIPTPATH}}/COPYING a copy of the GNU General Public License]
67 along with this program; if not, write to the Free Software
68 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
69 or [http://www.gnu.org/licenses/old-licenses/gpl-2.0.html read it online].
72 return str_replace( "\t\t", '', $ret ) . "\n";
76 * @return wiki text showing the third party software versions (apache, php, mysql).
78 static function softwareInformation() {
79 $dbr = wfGetDB( DB_SLAVE
);
81 return Xml
::element( 'h2', array( 'id' => 'mw-version-software' ), wfMsg( 'version-software' ) ) .
82 Xml
::openElement( 'table', array( 'id' => 'sv-software' ) ) .
84 <th>" . wfMsg( 'version-software-product' ) . "</th>
85 <th>" . wfMsg( 'version-software-version' ) . "</th>
88 <td>[http://www.mediawiki.org/ MediaWiki]</td>
89 <td>" . self
::getVersionLinked() . "</td>
92 <td>[http://www.php.net/ PHP]</td>
93 <td>" . phpversion() . " (" . php_sapi_name() . ")</td>
96 <td>" . $dbr->getSoftwareLink() . "</td>
97 <td>" . $dbr->getServerVersion() . "</td>
99 Xml
::closeElement( 'table' );
103 * Return a string of the MediaWiki version with SVN revision if available
107 public static function getVersion() {
108 global $wgVersion, $IP;
109 wfProfileIn( __METHOD__
);
110 $svn = self
::getSvnRevision( $IP );
111 $version = $svn ?
"$wgVersion (r$svn)" : $wgVersion;
112 wfProfileOut( __METHOD__
);
117 * Return a string of the MediaWiki version with a link to SVN revision if
122 public static function getVersionLinked() {
123 global $wgVersion, $IP;
124 wfProfileIn( __METHOD__
);
125 $svn = self
::getSvnRevision( $IP );
126 $viewvc = 'http://svn.wikimedia.org/viewvc/mediawiki/trunk/phase3/?pathrev=';
127 $version = $svn ?
"$wgVersion ([{$viewvc}{$svn} r$svn])" : $wgVersion;
128 wfProfileOut( __METHOD__
);
132 /** Generate wikitext showing extensions name, URL, author and description */
133 function extensionCredits() {
134 global $wgExtensionCredits, $wgExtensionFunctions, $wgParser, $wgSkinExtensionFunctions;
136 if ( ! count( $wgExtensionCredits ) && ! count( $wgExtensionFunctions ) && ! count( $wgSkinExtensionFunctions ) )
139 $extensionTypes = array(
140 'specialpage' => wfMsg( 'version-specialpages' ),
141 'parserhook' => wfMsg( 'version-parserhooks' ),
142 'variable' => wfMsg( 'version-variables' ),
143 'media' => wfMsg( 'version-mediahandlers' ),
144 'other' => wfMsg( 'version-other' ),
146 wfRunHooks( 'SpecialVersionExtensionTypes', array( &$this, &$extensionTypes ) );
148 $out = Xml
::element( 'h2', array( 'id' => 'mw-version-ext' ), wfMsg( 'version-extensions' ) ) .
149 Xml
::openElement( 'table', array( 'id' => 'sv-ext' ) );
151 foreach ( $extensionTypes as $type => $text ) {
152 if ( isset ( $wgExtensionCredits[$type] ) && count ( $wgExtensionCredits[$type] ) ) {
153 $out .= $this->openExtType( $text );
155 usort( $wgExtensionCredits[$type], array( $this, 'compare' ) );
157 foreach ( $wgExtensionCredits[$type] as $extension ) {
158 if ( isset( $extension['version'] ) ) {
159 $version = $extension['version'];
160 } elseif ( isset( $extension['svn-revision'] ) &&
161 preg_match( '/\$(?:Rev|LastChangedRevision|Revision): *(\d+)/',
162 $extension['svn-revision'], $m ) )
164 $version = 'r' . $m[1];
169 $out .= $this->formatCredits(
170 isset ( $extension['name'] ) ?
$extension['name'] : '',
172 isset ( $extension['author'] ) ?
$extension['author'] : '',
173 isset ( $extension['url'] ) ?
$extension['url'] : null,
174 isset ( $extension['description'] ) ?
$extension['description'] : '',
175 isset ( $extension['descriptionmsg'] ) ?
$extension['descriptionmsg'] : ''
181 if ( count( $wgExtensionFunctions ) ) {
182 $out .= $this->openExtType( wfMsg( 'version-extension-functions' ) );
183 $out .= '<tr><td colspan="3">' . $this->listToText( $wgExtensionFunctions ) . "</td></tr>\n";
186 if ( $cnt = count( $tags = $wgParser->getTags() ) ) {
187 for ( $i = 0; $i < $cnt; ++
$i )
188 $tags[$i] = "<{$tags[$i]}>";
189 $out .= $this->openExtType( wfMsg( 'version-parser-extensiontags' ) );
190 $out .= '<tr><td colspan="3">' . $this->listToText( $tags ). "</td></tr>\n";
193 if( $cnt = count( $fhooks = $wgParser->getFunctionHooks() ) ) {
194 $out .= $this->openExtType( wfMsg( 'version-parser-function-hooks' ) );
195 $out .= '<tr><td colspan="3">' . $this->listToText( $fhooks ) . "</td></tr>\n";
198 if ( count( $wgSkinExtensionFunctions ) ) {
199 $out .= $this->openExtType( wfMsg( 'version-skin-extension-functions' ) );
200 $out .= '<tr><td colspan="3">' . $this->listToText( $wgSkinExtensionFunctions ) . "</td></tr>\n";
202 $out .= Xml
::closeElement( 'table' );
206 /** Callback to sort extensions by type */
207 function compare( $a, $b ) {
209 if( $a['name'] === $b['name'] ) {
212 return $wgLang->lc( $a['name'] ) > $wgLang->lc( $b['name'] )
218 function formatCredits( $name, $version = null, $author = null, $url = null, $description = null, $descriptionMsg = null ) {
219 $extension = isset( $url ) ?
"[$url $name]" : $name;
220 $version = isset( $version ) ?
"(" . wfMsg( 'version-version' ) . " $version)" : '';
222 # Look for a localized description
223 if( isset( $descriptionMsg ) ) {
224 $msg = wfMsg( $descriptionMsg );
225 if ( !wfEmptyMsg( $descriptionMsg, $msg ) && $msg != '' ) {
231 <td><em>$extension $version</em></td>
232 <td>$description</td>
233 <td>" . $this->listToText( (array)$author ) . "</td>
243 if ( count( $wgHooks ) ) {
244 $myWgHooks = $wgHooks;
247 $ret = Xml
::element( 'h2', array( 'id' => 'mw-version-hooks' ), wfMsg( 'version-hooks' ) ) .
248 Xml
::openElement( 'table', array( 'id' => 'sv-hooks' ) ) .
250 <th>" . wfMsg( 'version-hook-name' ) . "</th>
251 <th>" . wfMsg( 'version-hook-subscribedby' ) . "</th>
254 foreach ( $myWgHooks as $hook => $hooks )
257 <td>" . $this->listToText( $hooks ) . "</td>
260 $ret .= Xml
::closeElement( 'table' );
266 private function openExtType($text, $name = null) {
267 $opt = array( 'colspan' => 3 );
270 if(!$this->firstExtOpened
) {
271 // Insert a spacing line
272 $out .= '<tr class="sv-space">' . Xml
::element( 'td', $opt ) . "</tr>\n";
274 $this->firstExtOpened
= false;
276 if($name) { $opt['id'] = "sv-$name"; }
278 $out .= "<tr>" . Xml
::element( 'th', $opt, $text) . "</tr>\n";
288 $ip = str_replace( '--', ' - ', htmlspecialchars( wfGetIP() ) );
289 return "<!-- visited from $ip -->\n" .
290 "<span style='display:none'>visited from $ip</span>";
297 function listToText( $list ) {
298 $cnt = count( $list );
301 // Enforce always returning a string
302 return (string)$this->arrayToString( $list[0] );
303 } elseif ( $cnt == 0 ) {
307 $t = array_slice( $list, 0, $cnt - 1 );
308 $one = array_map( array( &$this, 'arrayToString' ), $t );
309 $two = $this->arrayToString( $list[$cnt - 1] );
310 $and = wfMsg( 'and' );
312 return implode( ', ', $one ) . " $and $two";
319 * @param mixed $list Will convert an array to string if given and return
320 * the paramater unaltered otherwise
323 function arrayToString( $list ) {
324 if( is_object( $list ) ) {
325 $class = get_class( $list );
327 } elseif ( ! is_array( $list ) ) {
330 $class = get_class( $list[0] );
331 return "($class, {$list[1]})";
336 * Retrieve the revision number of a Subversion working directory.
339 * @return mixed revision number as int, or false if not a SVN checkout
341 public static function getSvnRevision( $dir ) {
342 // http://svnbook.red-bean.com/nightly/en/svn.developer.insidewc.html
343 $entries = $dir . '/.svn/entries';
345 if( !file_exists( $entries ) ) {
349 $content = file( $entries );
351 // check if file is xml (subversion release <= 1.3) or not (subversion release = 1.4)
352 if( preg_match( '/^<\?xml/', $content[0] ) ) {
353 // subversion is release <= 1.3
354 if( !function_exists( 'simplexml_load_file' ) ) {
355 // We could fall back to expat... YUCK
359 // SimpleXml whines about the xmlns...
360 wfSuppressWarnings();
361 $xml = simplexml_load_file( $entries );
365 foreach( $xml->entry
as $entry ) {
366 if( $xml->entry
[0]['name'] == '' ) {
367 // The directory entry should always have a revision marker.
368 if( $entry['revision'] ) {
369 return intval( $entry['revision'] );
376 // subversion is release 1.4
377 return intval( $content[3] );