Also for all sub-classes Remove simple doc-blocks without further information Change-Id: I981934efe32d44f52e5ab865a9b887be5bd0f41e
1150 lines
38 KiB
PHP
1150 lines
38 KiB
PHP
<?php
|
|
/**
|
|
* Copyright © 2006 Yuri Astrakhan "<Firstname><Lastname>@gmail.com"
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
namespace MediaWiki\Api;
|
|
|
|
use MediaWiki\Interwiki\InterwikiLookup;
|
|
use MediaWiki\Language\Language;
|
|
use MediaWiki\Language\LanguageCode;
|
|
use MediaWiki\Language\LanguageConverter;
|
|
use MediaWiki\Languages\LanguageConverterFactory;
|
|
use MediaWiki\Languages\LanguageFactory;
|
|
use MediaWiki\Languages\LanguageNameUtils;
|
|
use MediaWiki\MainConfigNames;
|
|
use MediaWiki\MediaWikiServices;
|
|
use MediaWiki\Parser\MagicWordFactory;
|
|
use MediaWiki\Parser\ParserFactory;
|
|
use MediaWiki\Registration\ExtensionRegistry;
|
|
use MediaWiki\ResourceLoader\SkinModule;
|
|
use MediaWiki\SiteStats\SiteStats;
|
|
use MediaWiki\SpecialPage\SpecialPage;
|
|
use MediaWiki\SpecialPage\SpecialPageFactory;
|
|
use MediaWiki\Specials\SpecialVersion;
|
|
use MediaWiki\Title\NamespaceInfo;
|
|
use MediaWiki\Title\Title;
|
|
use MediaWiki\User\Options\UserOptionsLookup;
|
|
use MediaWiki\User\TempUser\TempUserConfig;
|
|
use MediaWiki\User\UserGroupManager;
|
|
use MediaWiki\Utils\ExtensionInfo;
|
|
use MediaWiki\Utils\GitInfo;
|
|
use MediaWiki\Utils\UrlUtils;
|
|
use MediaWiki\WikiMap\WikiMap;
|
|
use Skin;
|
|
use SkinFactory;
|
|
use UploadBase;
|
|
use Wikimedia\Composer\ComposerInstalled;
|
|
use Wikimedia\ParamValidator\ParamValidator;
|
|
use Wikimedia\Rdbms\ILoadBalancer;
|
|
use Wikimedia\Rdbms\ReadOnlyMode;
|
|
|
|
/**
|
|
* A query action to return meta information about the wiki site.
|
|
*
|
|
* @ingroup API
|
|
*/
|
|
class ApiQuerySiteinfo extends ApiQueryBase {
|
|
|
|
private UserOptionsLookup $userOptionsLookup;
|
|
private UserGroupManager $userGroupManager;
|
|
private LanguageConverterFactory $languageConverterFactory;
|
|
private LanguageFactory $languageFactory;
|
|
private LanguageNameUtils $languageNameUtils;
|
|
private Language $contentLanguage;
|
|
private NamespaceInfo $namespaceInfo;
|
|
private InterwikiLookup $interwikiLookup;
|
|
private ParserFactory $parserFactory;
|
|
private MagicWordFactory $magicWordFactory;
|
|
private SpecialPageFactory $specialPageFactory;
|
|
private SkinFactory $skinFactory;
|
|
private ILoadBalancer $loadBalancer;
|
|
private ReadOnlyMode $readOnlyMode;
|
|
private UrlUtils $urlUtils;
|
|
private TempUserConfig $tempUserConfig;
|
|
|
|
public function __construct(
|
|
ApiQuery $query,
|
|
string $moduleName,
|
|
UserOptionsLookup $userOptionsLookup,
|
|
UserGroupManager $userGroupManager,
|
|
LanguageConverterFactory $languageConverterFactory,
|
|
LanguageFactory $languageFactory,
|
|
LanguageNameUtils $languageNameUtils,
|
|
Language $contentLanguage,
|
|
NamespaceInfo $namespaceInfo,
|
|
InterwikiLookup $interwikiLookup,
|
|
ParserFactory $parserFactory,
|
|
MagicWordFactory $magicWordFactory,
|
|
SpecialPageFactory $specialPageFactory,
|
|
SkinFactory $skinFactory,
|
|
ILoadBalancer $loadBalancer,
|
|
ReadOnlyMode $readOnlyMode,
|
|
UrlUtils $urlUtils,
|
|
TempUserConfig $tempUserConfig
|
|
) {
|
|
parent::__construct( $query, $moduleName, 'si' );
|
|
$this->userOptionsLookup = $userOptionsLookup;
|
|
$this->userGroupManager = $userGroupManager;
|
|
$this->languageConverterFactory = $languageConverterFactory;
|
|
$this->languageFactory = $languageFactory;
|
|
$this->languageNameUtils = $languageNameUtils;
|
|
$this->contentLanguage = $contentLanguage;
|
|
$this->namespaceInfo = $namespaceInfo;
|
|
$this->interwikiLookup = $interwikiLookup;
|
|
$this->parserFactory = $parserFactory;
|
|
$this->magicWordFactory = $magicWordFactory;
|
|
$this->specialPageFactory = $specialPageFactory;
|
|
$this->skinFactory = $skinFactory;
|
|
$this->loadBalancer = $loadBalancer;
|
|
$this->readOnlyMode = $readOnlyMode;
|
|
$this->urlUtils = $urlUtils;
|
|
$this->tempUserConfig = $tempUserConfig;
|
|
}
|
|
|
|
public function execute() {
|
|
$params = $this->extractRequestParams();
|
|
$done = [];
|
|
foreach ( $params['prop'] as $p ) {
|
|
switch ( $p ) {
|
|
case 'general':
|
|
$fit = $this->appendGeneralInfo( $p );
|
|
break;
|
|
case 'namespaces':
|
|
$fit = $this->appendNamespaces( $p );
|
|
break;
|
|
case 'namespacealiases':
|
|
$fit = $this->appendNamespaceAliases( $p );
|
|
break;
|
|
case 'specialpagealiases':
|
|
$fit = $this->appendSpecialPageAliases( $p );
|
|
break;
|
|
case 'magicwords':
|
|
$fit = $this->appendMagicWords( $p );
|
|
break;
|
|
case 'interwikimap':
|
|
$fit = $this->appendInterwikiMap( $p, $params['filteriw'] );
|
|
break;
|
|
case 'dbrepllag':
|
|
$fit = $this->appendDbReplLagInfo( $p, $params['showalldb'] );
|
|
break;
|
|
case 'statistics':
|
|
$fit = $this->appendStatistics( $p );
|
|
break;
|
|
case 'usergroups':
|
|
$fit = $this->appendUserGroups( $p, $params['numberingroup'] );
|
|
break;
|
|
case 'autocreatetempuser':
|
|
$fit = $this->appendAutoCreateTempUser( $p );
|
|
break;
|
|
case 'clientlibraries':
|
|
$fit = $this->appendInstalledClientLibraries( $p );
|
|
break;
|
|
case 'libraries':
|
|
$fit = $this->appendInstalledLibraries( $p );
|
|
break;
|
|
case 'extensions':
|
|
$fit = $this->appendExtensions( $p );
|
|
break;
|
|
case 'fileextensions':
|
|
$fit = $this->appendFileExtensions( $p );
|
|
break;
|
|
case 'rightsinfo':
|
|
$fit = $this->appendRightsInfo( $p );
|
|
break;
|
|
case 'restrictions':
|
|
$fit = $this->appendRestrictions( $p );
|
|
break;
|
|
case 'languages':
|
|
$fit = $this->appendLanguages( $p );
|
|
break;
|
|
case 'languagevariants':
|
|
$fit = $this->appendLanguageVariants( $p );
|
|
break;
|
|
case 'skins':
|
|
$fit = $this->appendSkins( $p );
|
|
break;
|
|
case 'extensiontags':
|
|
$fit = $this->appendExtensionTags( $p );
|
|
break;
|
|
case 'functionhooks':
|
|
$fit = $this->appendFunctionHooks( $p );
|
|
break;
|
|
case 'showhooks':
|
|
$fit = $this->appendSubscribedHooks( $p );
|
|
break;
|
|
case 'variables':
|
|
$fit = $this->appendVariables( $p );
|
|
break;
|
|
case 'protocols':
|
|
$fit = $this->appendProtocols( $p );
|
|
break;
|
|
case 'defaultoptions':
|
|
$fit = $this->appendDefaultOptions( $p );
|
|
break;
|
|
case 'uploaddialog':
|
|
$fit = $this->appendUploadDialog( $p );
|
|
break;
|
|
case 'autopromote':
|
|
$fit = $this->appendAutoPromote( $p );
|
|
break;
|
|
case 'autopromoteonce':
|
|
$fit = $this->appendAutoPromoteOnce( $p );
|
|
break;
|
|
default:
|
|
ApiBase::dieDebug( __METHOD__, "Unknown prop=$p" ); // @codeCoverageIgnore
|
|
}
|
|
if ( !$fit ) {
|
|
// Abuse siprop as a query-continue parameter
|
|
// and set it to all unprocessed props
|
|
$this->setContinueEnumParameter( 'prop', implode( '|',
|
|
array_diff( $params['prop'], $done ) ) );
|
|
break;
|
|
}
|
|
$done[] = $p;
|
|
}
|
|
}
|
|
|
|
protected function appendGeneralInfo( $property ) {
|
|
$config = $this->getConfig();
|
|
$mainPage = Title::newMainPage();
|
|
$logo = SkinModule::getAvailableLogos( $config, $this->getLanguage()->getCode() );
|
|
|
|
$data = [
|
|
'mainpage' => $mainPage->getPrefixedText(),
|
|
'base' => (string)$this->urlUtils->expand( $mainPage->getFullURL(), PROTO_CURRENT ),
|
|
'sitename' => $config->get( MainConfigNames::Sitename ),
|
|
'mainpageisdomainroot' => (bool)$config->get( MainConfigNames::MainPageIsDomainRoot ),
|
|
|
|
// A logo can either be a relative or an absolute path
|
|
// make sure we always return an absolute path
|
|
'logo' => (string)$this->urlUtils->expand( $logo['1x'], PROTO_RELATIVE ),
|
|
|
|
'generator' => 'MediaWiki ' . MW_VERSION,
|
|
|
|
'phpversion' => PHP_VERSION,
|
|
'phpsapi' => PHP_SAPI,
|
|
'dbtype' => $config->get( MainConfigNames::DBtype ),
|
|
'dbversion' => $this->getDB()->getServerVersion(),
|
|
];
|
|
|
|
$allowFrom = [ '' ];
|
|
$allowException = true;
|
|
if ( !$config->get( MainConfigNames::AllowExternalImages ) ) {
|
|
$data['imagewhitelistenabled'] =
|
|
(bool)$config->get( MainConfigNames::EnableImageWhitelist );
|
|
$allowFrom = $config->get( MainConfigNames::AllowExternalImagesFrom );
|
|
$allowException = (bool)$allowFrom;
|
|
}
|
|
if ( $allowException ) {
|
|
$data['externalimages'] = (array)$allowFrom;
|
|
ApiResult::setIndexedTagName( $data['externalimages'], 'prefix' );
|
|
}
|
|
|
|
$data['langconversion'] = !$this->languageConverterFactory->isConversionDisabled();
|
|
$data['linkconversion'] = !$this->languageConverterFactory->isLinkConversionDisabled();
|
|
// For backwards compatibility (soft deprecated since MW 1.36)
|
|
$data['titleconversion'] = $data['linkconversion'];
|
|
|
|
$contLangConverter = $this->languageConverterFactory->getLanguageConverter( $this->contentLanguage );
|
|
if ( $this->contentLanguage->linkPrefixExtension() ) {
|
|
$linkPrefixCharset = $this->contentLanguage->linkPrefixCharset();
|
|
$data['linkprefixcharset'] = $linkPrefixCharset;
|
|
// For backwards compatibility
|
|
$data['linkprefix'] = "/^((?>.*[^$linkPrefixCharset]|))(.+)$/sDu";
|
|
} else {
|
|
$data['linkprefixcharset'] = '';
|
|
$data['linkprefix'] = '';
|
|
}
|
|
|
|
$data['linktrail'] = $this->contentLanguage->linkTrail() ?: '';
|
|
|
|
$data['legaltitlechars'] = Title::legalChars();
|
|
$data['invalidusernamechars'] = $config->get( MainConfigNames::InvalidUsernameCharacters );
|
|
|
|
$data['allunicodefixes'] = (bool)$config->get( MainConfigNames::AllUnicodeFixes );
|
|
$data['fixarabicunicode'] = true; // Config removed in 1.35, always true
|
|
$data['fixmalayalamunicode'] = true; // Config removed in 1.35, always true
|
|
|
|
$git = GitInfo::repo()->getHeadSHA1();
|
|
if ( $git ) {
|
|
$data['git-hash'] = $git;
|
|
$data['git-branch'] = GitInfo::repo()->getCurrentBranch();
|
|
}
|
|
|
|
// 'case-insensitive' option is reserved for future
|
|
$data['case'] =
|
|
$config->get( MainConfigNames::CapitalLinks ) ? 'first-letter' : 'case-sensitive';
|
|
$data['lang'] = $config->get( MainConfigNames::LanguageCode );
|
|
|
|
$data['fallback'] = [];
|
|
foreach ( $this->contentLanguage->getFallbackLanguages() as $code ) {
|
|
$data['fallback'][] = [ 'code' => $code ];
|
|
}
|
|
ApiResult::setIndexedTagName( $data['fallback'], 'lang' );
|
|
|
|
if ( $contLangConverter->hasVariants() ) {
|
|
$data['variants'] = [];
|
|
foreach ( $contLangConverter->getVariants() as $code ) {
|
|
$data['variants'][] = [
|
|
'code' => $code,
|
|
'name' => $this->contentLanguage->getVariantname( $code ),
|
|
];
|
|
}
|
|
ApiResult::setIndexedTagName( $data['variants'], 'lang' );
|
|
}
|
|
|
|
$data['rtl'] = $this->contentLanguage->isRTL();
|
|
$data['fallback8bitEncoding'] = $this->contentLanguage->fallback8bitEncoding();
|
|
|
|
$data['readonly'] = $this->readOnlyMode->isReadOnly();
|
|
if ( $data['readonly'] ) {
|
|
$data['readonlyreason'] = $this->readOnlyMode->getReason();
|
|
}
|
|
$data['writeapi'] = true; // Deprecated since MW 1.32
|
|
|
|
$data['maxarticlesize'] = $config->get( MainConfigNames::MaxArticleSize ) * 1024;
|
|
|
|
$data['timezone'] = $config->get( MainConfigNames::Localtimezone );
|
|
$data['timeoffset'] = (int)( $config->get( MainConfigNames::LocalTZoffset ) );
|
|
$data['articlepath'] = $config->get( MainConfigNames::ArticlePath );
|
|
$data['scriptpath'] = $config->get( MainConfigNames::ScriptPath );
|
|
$data['script'] = $config->get( MainConfigNames::Script );
|
|
$data['variantarticlepath'] = $config->get( MainConfigNames::VariantArticlePath );
|
|
$data[ApiResult::META_BC_BOOLS][] = 'variantarticlepath';
|
|
$data['server'] = $config->get( MainConfigNames::Server );
|
|
$data['servername'] = $config->get( MainConfigNames::ServerName );
|
|
$data['wikiid'] = WikiMap::getCurrentWikiId();
|
|
$data['time'] = wfTimestamp( TS_ISO_8601, time() );
|
|
|
|
$data['misermode'] = (bool)$config->get( MainConfigNames::MiserMode );
|
|
|
|
$data['uploadsenabled'] = UploadBase::isEnabled();
|
|
$data['maxuploadsize'] = UploadBase::getMaxUploadSize();
|
|
$data['minuploadchunksize'] = ApiUpload::getMinUploadChunkSize( $config );
|
|
|
|
$data['galleryoptions'] = $config->get( MainConfigNames::GalleryOptions );
|
|
|
|
$data['thumblimits'] = $config->get( MainConfigNames::ThumbLimits );
|
|
ApiResult::setArrayType( $data['thumblimits'], 'BCassoc' );
|
|
ApiResult::setIndexedTagName( $data['thumblimits'], 'limit' );
|
|
$data['imagelimits'] = [];
|
|
foreach ( $config->get( MainConfigNames::ImageLimits ) as $k => $limit ) {
|
|
$data['imagelimits'][$k] = [ 'width' => $limit[0], 'height' => $limit[1] ];
|
|
}
|
|
ApiResult::setArrayType( $data['imagelimits'], 'BCassoc' );
|
|
ApiResult::setIndexedTagName( $data['imagelimits'], 'limit' );
|
|
|
|
$favicon = $config->get( MainConfigNames::Favicon );
|
|
if ( $favicon ) {
|
|
// Expand any local path to full URL to improve API usability (T77093).
|
|
$data['favicon'] = (string)$this->urlUtils->expand( $favicon );
|
|
}
|
|
|
|
$data['centralidlookupprovider'] = $config->get( MainConfigNames::CentralIdLookupProvider );
|
|
$providerIds = array_keys( $config->get( MainConfigNames::CentralIdLookupProviders ) );
|
|
$data['allcentralidlookupproviders'] = $providerIds;
|
|
|
|
$data['interwikimagic'] = (bool)$config->get( MainConfigNames::InterwikiMagic );
|
|
$data['magiclinks'] = $config->get( MainConfigNames::EnableMagicLinks );
|
|
|
|
$data['categorycollation'] = $config->get( MainConfigNames::CategoryCollation );
|
|
|
|
$data['nofollowlinks'] = $config->get( MainConfigNames::NoFollowLinks );
|
|
$data['nofollownsexceptions'] = $config->get( MainConfigNames::NoFollowNsExceptions );
|
|
$data['nofollowdomainexceptions'] = $config->get( MainConfigNames::NoFollowDomainExceptions );
|
|
$data['externallinktarget'] = $config->get( MainConfigNames::ExternalLinkTarget );
|
|
|
|
$this->getHookRunner()->onAPIQuerySiteInfoGeneralInfo( $this, $data );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendNamespaces( $property ) {
|
|
$nsProtection = $this->getConfig()->get( MainConfigNames::NamespaceProtection );
|
|
|
|
$data = [ ApiResult::META_TYPE => 'assoc' ];
|
|
foreach ( $this->contentLanguage->getFormattedNamespaces() as $ns => $title ) {
|
|
$data[$ns] = [
|
|
'id' => (int)$ns,
|
|
'case' => $this->namespaceInfo->isCapitalized( $ns ) ? 'first-letter' : 'case-sensitive',
|
|
];
|
|
ApiResult::setContentValue( $data[$ns], 'name', $title );
|
|
$canonical = $this->namespaceInfo->getCanonicalName( $ns );
|
|
|
|
$data[$ns]['subpages'] = $this->namespaceInfo->hasSubpages( $ns );
|
|
|
|
if ( $canonical ) {
|
|
$data[$ns]['canonical'] = strtr( $canonical, '_', ' ' );
|
|
}
|
|
|
|
$data[$ns]['content'] = $this->namespaceInfo->isContent( $ns );
|
|
$data[$ns]['nonincludable'] = $this->namespaceInfo->isNonincludable( $ns );
|
|
|
|
$specificNs = $nsProtection[$ns] ?? '';
|
|
if ( is_array( $specificNs ) ) {
|
|
$specificNs = implode( "|", array_filter( $specificNs ) );
|
|
}
|
|
if ( $specificNs !== '' ) {
|
|
$data[$ns]['namespaceprotection'] = $specificNs;
|
|
}
|
|
|
|
$contentmodel = $this->namespaceInfo->getNamespaceContentModel( $ns );
|
|
if ( $contentmodel ) {
|
|
$data[$ns]['defaultcontentmodel'] = $contentmodel;
|
|
}
|
|
}
|
|
|
|
ApiResult::setArrayType( $data, 'assoc' );
|
|
ApiResult::setIndexedTagName( $data, 'ns' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendNamespaceAliases( $property ) {
|
|
$aliases = $this->contentLanguage->getNamespaceAliases();
|
|
$namespaces = $this->contentLanguage->getNamespaces();
|
|
$data = [];
|
|
foreach ( $aliases as $title => $ns ) {
|
|
if ( $namespaces[$ns] == $title ) {
|
|
// Don't list duplicates
|
|
continue;
|
|
}
|
|
$item = [ 'id' => (int)$ns ];
|
|
ApiResult::setContentValue( $item, 'alias', strtr( $title, '_', ' ' ) );
|
|
$data[] = $item;
|
|
}
|
|
|
|
sort( $data );
|
|
|
|
ApiResult::setIndexedTagName( $data, 'ns' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendSpecialPageAliases( $property ) {
|
|
$data = [];
|
|
$aliases = $this->contentLanguage->getSpecialPageAliases();
|
|
foreach ( $this->specialPageFactory->getNames() as $specialpage ) {
|
|
if ( isset( $aliases[$specialpage] ) ) {
|
|
$arr = [ 'realname' => $specialpage, 'aliases' => $aliases[$specialpage] ];
|
|
ApiResult::setIndexedTagName( $arr['aliases'], 'alias' );
|
|
$data[] = $arr;
|
|
}
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'specialpage' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendMagicWords( $property ) {
|
|
$data = [];
|
|
foreach ( $this->contentLanguage->getMagicWords() as $name => $aliases ) {
|
|
$caseSensitive = (bool)array_shift( $aliases );
|
|
$arr = [
|
|
'name' => $name,
|
|
'aliases' => $aliases,
|
|
'case-sensitive' => $caseSensitive,
|
|
];
|
|
ApiResult::setIndexedTagName( $arr['aliases'], 'alias' );
|
|
$data[] = $arr;
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'magicword' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendInterwikiMap( $property, $filter ) {
|
|
$local = $filter ? $filter === 'local' : null;
|
|
|
|
$params = $this->extractRequestParams();
|
|
$langCode = $params['inlanguagecode'] ?? '';
|
|
$interwikiMagic = $this->getConfig()->get( MainConfigNames::InterwikiMagic );
|
|
|
|
if ( $interwikiMagic ) {
|
|
$langNames = $this->languageNameUtils->getLanguageNames( $langCode );
|
|
}
|
|
|
|
$extraLangPrefixes = $this->getConfig()->get( MainConfigNames::ExtraInterlanguageLinkPrefixes );
|
|
$extraLangCodeMap = $this->getConfig()->get( MainConfigNames::InterlanguageLinkCodeMap );
|
|
$localInterwikis = $this->getConfig()->get( MainConfigNames::LocalInterwikis );
|
|
$data = [];
|
|
|
|
foreach ( $this->interwikiLookup->getAllPrefixes( $local ) as $row ) {
|
|
$prefix = $row['iw_prefix'];
|
|
$val = [];
|
|
$val['prefix'] = $prefix;
|
|
if ( $row['iw_local'] ?? false ) {
|
|
$val['local'] = true;
|
|
}
|
|
if ( $row['iw_trans'] ?? false ) {
|
|
$val['trans'] = true;
|
|
}
|
|
|
|
if ( $interwikiMagic && isset( $langNames[$prefix] ) ) {
|
|
$val['language'] = $langNames[$prefix];
|
|
$standard = LanguageCode::replaceDeprecatedCodes( $prefix );
|
|
if ( $standard !== $prefix ) {
|
|
# Note that even if this code is deprecated, it should
|
|
# only be remapped if extralanglink (set below) is false.
|
|
$val['deprecated'] = $standard;
|
|
}
|
|
$val['bcp47'] = LanguageCode::bcp47( $standard );
|
|
}
|
|
if ( in_array( $prefix, $localInterwikis ) ) {
|
|
$val['localinterwiki'] = true;
|
|
}
|
|
if ( $interwikiMagic && in_array( $prefix, $extraLangPrefixes ) ) {
|
|
$val['extralanglink'] = true;
|
|
$val['code'] = $extraLangCodeMap[$prefix] ?? $prefix;
|
|
$val['bcp47'] = LanguageCode::bcp47( $val['code'] );
|
|
|
|
$linktext = $this->msg( "interlanguage-link-$prefix" );
|
|
if ( !$linktext->isDisabled() ) {
|
|
$val['linktext'] = $linktext->text();
|
|
}
|
|
|
|
$sitename = $this->msg( "interlanguage-link-sitename-$prefix" );
|
|
if ( !$sitename->isDisabled() ) {
|
|
$val['sitename'] = $sitename->text();
|
|
}
|
|
}
|
|
|
|
$val['url'] = (string)$this->urlUtils->expand( $row['iw_url'], PROTO_CURRENT );
|
|
$val['protorel'] = str_starts_with( $row['iw_url'], '//' );
|
|
if ( ( $row['iw_wikiid'] ?? '' ) !== '' ) {
|
|
$val['wikiid'] = $row['iw_wikiid'];
|
|
}
|
|
if ( ( $row['iw_api'] ?? '' ) !== '' ) {
|
|
$val['api'] = $row['iw_api'];
|
|
}
|
|
|
|
$data[] = $val;
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $data, 'iw' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendDbReplLagInfo( $property, $includeAll ) {
|
|
$data = [];
|
|
$showHostnames = $this->getConfig()->get( MainConfigNames::ShowHostnames );
|
|
if ( $includeAll ) {
|
|
if ( !$showHostnames ) {
|
|
$this->dieWithError( 'apierror-siteinfo-includealldenied', 'includeAllDenied' );
|
|
}
|
|
|
|
foreach ( $this->loadBalancer->getLagTimes() as $i => $lag ) {
|
|
$data[] = [
|
|
'host' => $this->loadBalancer->getServerName( $i ),
|
|
'lag' => $lag
|
|
];
|
|
}
|
|
} else {
|
|
[ , $lag, $index ] = $this->loadBalancer->getMaxLag();
|
|
$data[] = [
|
|
'host' => $showHostnames ? $this->loadBalancer->getServerName( $index ) : '',
|
|
'lag' => $lag
|
|
];
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $data, 'db' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendStatistics( $property ) {
|
|
$data = [
|
|
'pages' => SiteStats::pages(),
|
|
'articles' => SiteStats::articles(),
|
|
'edits' => SiteStats::edits(),
|
|
'images' => SiteStats::images(),
|
|
'users' => SiteStats::users(),
|
|
'activeusers' => SiteStats::activeUsers(),
|
|
'admins' => SiteStats::numberingroup( 'sysop' ),
|
|
'jobs' => SiteStats::jobs(),
|
|
];
|
|
|
|
$this->getHookRunner()->onAPIQuerySiteInfoStatisticsInfo( $data );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendUserGroups( $property, $numberInGroup ) {
|
|
$config = $this->getConfig();
|
|
|
|
$data = [];
|
|
$result = $this->getResult();
|
|
$allGroups = array_values( $this->userGroupManager->listAllGroups() );
|
|
foreach ( $config->get( MainConfigNames::GroupPermissions ) as $group => $permissions ) {
|
|
$arr = [
|
|
'name' => $group,
|
|
'rights' => array_keys( $permissions, true ),
|
|
];
|
|
|
|
if ( $numberInGroup ) {
|
|
$autopromote = $config->get( MainConfigNames::Autopromote );
|
|
|
|
if ( $group == 'user' ) {
|
|
$arr['number'] = SiteStats::users();
|
|
// '*' and autopromote groups have no size
|
|
} elseif ( $group !== '*' && !isset( $autopromote[$group] ) ) {
|
|
$arr['number'] = SiteStats::numberingroup( $group );
|
|
}
|
|
}
|
|
|
|
$groupArr = [
|
|
'add' => $config->get( MainConfigNames::AddGroups ),
|
|
'remove' => $config->get( MainConfigNames::RemoveGroups ),
|
|
'add-self' => $config->get( MainConfigNames::GroupsAddToSelf ),
|
|
'remove-self' => $config->get( MainConfigNames::GroupsRemoveFromSelf )
|
|
];
|
|
|
|
foreach ( $groupArr as $type => $rights ) {
|
|
if ( isset( $rights[$group] ) ) {
|
|
if ( $rights[$group] === true ) {
|
|
$groups = $allGroups;
|
|
} else {
|
|
$groups = array_intersect( $rights[$group], $allGroups );
|
|
}
|
|
if ( $groups ) {
|
|
$arr[$type] = $groups;
|
|
ApiResult::setArrayType( $arr[$type], 'BCarray' );
|
|
ApiResult::setIndexedTagName( $arr[$type], 'group' );
|
|
}
|
|
}
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $arr['rights'], 'permission' );
|
|
$data[] = $arr;
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $data, 'group' );
|
|
|
|
return $result->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendAutoCreateTempUser( $property ) {
|
|
$data = [ 'enabled' => $this->tempUserConfig->isEnabled() ];
|
|
if ( $this->tempUserConfig->isKnown() ) {
|
|
$data['matchPatterns'] = $this->tempUserConfig->getMatchPatterns();
|
|
}
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendFileExtensions( $property ) {
|
|
$data = [];
|
|
foreach (
|
|
array_unique( $this->getConfig()->get( MainConfigNames::FileExtensions ) ) as $ext
|
|
) {
|
|
$data[] = [ 'ext' => $ext ];
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'fe' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendInstalledClientLibraries( $property ) {
|
|
$data = [];
|
|
foreach ( SpecialVersion::parseForeignResources() as $name => $info ) {
|
|
$data[] = [
|
|
// Can't use $name as it is version suffixed (as multiple versions
|
|
// of a library may exist, provided by different skins/extensions)
|
|
'name' => $info['name'],
|
|
'version' => $info['version'],
|
|
];
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'library' );
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendInstalledLibraries( $property ) {
|
|
$path = MW_INSTALL_PATH . '/vendor/composer/installed.json';
|
|
if ( !file_exists( $path ) ) {
|
|
return true;
|
|
}
|
|
|
|
$data = [];
|
|
$installed = new ComposerInstalled( $path );
|
|
foreach ( $installed->getInstalledDependencies() as $name => $info ) {
|
|
if ( str_starts_with( $info['type'], 'mediawiki-' ) ) {
|
|
// Skip any extensions or skins since they'll be listed
|
|
// in their proper section
|
|
continue;
|
|
}
|
|
$data[] = [
|
|
'name' => $name,
|
|
'version' => $info['version'],
|
|
];
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'library' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendExtensions( $property ) {
|
|
$data = [];
|
|
$credits = SpecialVersion::getCredits(
|
|
ExtensionRegistry::getInstance(),
|
|
$this->getConfig()
|
|
);
|
|
foreach ( $credits as $type => $extensions ) {
|
|
foreach ( $extensions as $ext ) {
|
|
$ret = [ 'type' => $type ];
|
|
if ( isset( $ext['name'] ) ) {
|
|
$ret['name'] = $ext['name'];
|
|
}
|
|
if ( isset( $ext['namemsg'] ) ) {
|
|
$ret['namemsg'] = $ext['namemsg'];
|
|
}
|
|
if ( isset( $ext['description'] ) ) {
|
|
$ret['description'] = $ext['description'];
|
|
}
|
|
if ( isset( $ext['descriptionmsg'] ) ) {
|
|
// Can be a string or [ key, param1, param2, ... ]
|
|
if ( is_array( $ext['descriptionmsg'] ) ) {
|
|
$ret['descriptionmsg'] = $ext['descriptionmsg'][0];
|
|
$ret['descriptionmsgparams'] = array_slice( $ext['descriptionmsg'], 1 );
|
|
ApiResult::setIndexedTagName( $ret['descriptionmsgparams'], 'param' );
|
|
} else {
|
|
$ret['descriptionmsg'] = $ext['descriptionmsg'];
|
|
}
|
|
}
|
|
if ( isset( $ext['author'] ) ) {
|
|
$ret['author'] = is_array( $ext['author'] ) ?
|
|
implode( ', ', $ext['author'] ) : $ext['author'];
|
|
}
|
|
if ( isset( $ext['url'] ) ) {
|
|
$ret['url'] = $ext['url'];
|
|
}
|
|
if ( isset( $ext['version'] ) ) {
|
|
$ret['version'] = $ext['version'];
|
|
}
|
|
if ( isset( $ext['path'] ) ) {
|
|
$extensionPath = dirname( $ext['path'] );
|
|
$gitInfo = new GitInfo( $extensionPath );
|
|
$vcsVersion = $gitInfo->getHeadSHA1();
|
|
if ( $vcsVersion !== false ) {
|
|
$ret['vcs-system'] = 'git';
|
|
$ret['vcs-version'] = $vcsVersion;
|
|
$ret['vcs-url'] = $gitInfo->getHeadViewUrl();
|
|
$vcsDate = $gitInfo->getHeadCommitDate();
|
|
if ( $vcsDate !== false ) {
|
|
$ret['vcs-date'] = wfTimestamp( TS_ISO_8601, $vcsDate );
|
|
}
|
|
}
|
|
|
|
if ( ExtensionInfo::getLicenseFileNames( $extensionPath ) ) {
|
|
$ret['license-name'] = $ext['license-name'] ?? '';
|
|
$ret['license'] = SpecialPage::getTitleFor(
|
|
'Version',
|
|
"License/{$ext['name']}"
|
|
)->getLinkURL();
|
|
}
|
|
|
|
if ( ExtensionInfo::getAuthorsFileName( $extensionPath ) ) {
|
|
$ret['credits'] = SpecialPage::getTitleFor(
|
|
'Version',
|
|
"Credits/{$ext['name']}"
|
|
)->getLinkURL();
|
|
}
|
|
}
|
|
$data[] = $ret;
|
|
}
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $data, 'ext' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendRightsInfo( $property ) {
|
|
$config = $this->getConfig();
|
|
$title = Title::newFromText( $config->get( MainConfigNames::RightsPage ) );
|
|
if ( $title ) {
|
|
$url = $this->urlUtils->expand( $title->getLinkURL(), PROTO_CURRENT );
|
|
} else {
|
|
$url = $config->get( MainConfigNames::RightsUrl );
|
|
}
|
|
$text = $config->get( MainConfigNames::RightsText ) ?? '';
|
|
if ( $text === '' && $title ) {
|
|
$text = $title->getPrefixedText();
|
|
}
|
|
|
|
$data = [
|
|
'url' => (string)$url,
|
|
'text' => (string)$text,
|
|
];
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
protected function appendRestrictions( $property ) {
|
|
$config = $this->getConfig();
|
|
$data = [
|
|
'types' => $config->get( MainConfigNames::RestrictionTypes ),
|
|
'levels' => $config->get( MainConfigNames::RestrictionLevels ),
|
|
'cascadinglevels' => $config->get( MainConfigNames::CascadingRestrictionLevels ),
|
|
'semiprotectedlevels' => $config->get( MainConfigNames::SemiprotectedRestrictionLevels ),
|
|
];
|
|
|
|
ApiResult::setArrayType( $data['types'], 'BCarray' );
|
|
ApiResult::setArrayType( $data['levels'], 'BCarray' );
|
|
ApiResult::setArrayType( $data['cascadinglevels'], 'BCarray' );
|
|
ApiResult::setArrayType( $data['semiprotectedlevels'], 'BCarray' );
|
|
|
|
ApiResult::setIndexedTagName( $data['types'], 'type' );
|
|
ApiResult::setIndexedTagName( $data['levels'], 'level' );
|
|
ApiResult::setIndexedTagName( $data['cascadinglevels'], 'level' );
|
|
ApiResult::setIndexedTagName( $data['semiprotectedlevels'], 'level' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
public function appendLanguages( $property ) {
|
|
$params = $this->extractRequestParams();
|
|
$langCode = $params['inlanguagecode'] ?? '';
|
|
$langNames = $this->languageNameUtils->getLanguageNames( $langCode );
|
|
|
|
$data = [];
|
|
|
|
foreach ( $langNames as $code => $name ) {
|
|
$lang = [
|
|
'code' => $code,
|
|
'bcp47' => LanguageCode::bcp47( $code ),
|
|
];
|
|
ApiResult::setContentValue( $lang, 'name', $name );
|
|
$data[] = $lang;
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'lang' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
// Export information about which page languages will trigger
|
|
// language conversion. (T153341)
|
|
public function appendLanguageVariants( $property ) {
|
|
$langNames = $this->languageConverterFactory->isConversionDisabled() ? [] :
|
|
LanguageConverter::$languagesWithVariants;
|
|
sort( $langNames );
|
|
|
|
$data = [];
|
|
foreach ( $langNames as $langCode ) {
|
|
$lang = $this->languageFactory->getLanguage( $langCode );
|
|
$langConverter = $this->languageConverterFactory->getLanguageConverter( $lang );
|
|
if ( !$langConverter->hasVariants() ) {
|
|
// Only languages which have variants should be listed
|
|
continue;
|
|
}
|
|
$data[$langCode] = [];
|
|
ApiResult::setIndexedTagName( $data[$langCode], 'variant' );
|
|
ApiResult::setArrayType( $data[$langCode], 'kvp', 'code' );
|
|
|
|
$variants = $langConverter->getVariants();
|
|
sort( $variants );
|
|
foreach ( $variants as $v ) {
|
|
$data[$langCode][$v] = [
|
|
'fallbacks' => (array)$langConverter->getVariantFallbacks( $v ),
|
|
];
|
|
ApiResult::setIndexedTagName(
|
|
$data[$langCode][$v]['fallbacks'], 'variant'
|
|
);
|
|
}
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'lang' );
|
|
ApiResult::setArrayType( $data, 'kvp', 'code' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
public function appendSkins( $property ) {
|
|
$data = [];
|
|
$allowed = $this->skinFactory->getAllowedSkins();
|
|
$default = Skin::normalizeKey( 'default' );
|
|
|
|
foreach ( $this->skinFactory->getInstalledSkins() as $name => $displayName ) {
|
|
$msg = $this->msg( "skinname-{$name}" );
|
|
$code = $this->getParameter( 'inlanguagecode' );
|
|
if ( $code && $this->languageNameUtils->isValidCode( $code ) ) {
|
|
$msg->inLanguage( $code );
|
|
} else {
|
|
$msg->inContentLanguage();
|
|
}
|
|
if ( $msg->exists() ) {
|
|
$displayName = $msg->text();
|
|
}
|
|
$skin = [ 'code' => $name ];
|
|
ApiResult::setContentValue( $skin, 'name', $displayName );
|
|
if ( !isset( $allowed[$name] ) ) {
|
|
$skin['unusable'] = true;
|
|
}
|
|
if ( $name === $default ) {
|
|
$skin['default'] = true;
|
|
}
|
|
$data[] = $skin;
|
|
}
|
|
ApiResult::setIndexedTagName( $data, 'skin' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
public function appendExtensionTags( $property ) {
|
|
$tags = array_map(
|
|
static function ( $item ) {
|
|
return "<$item>";
|
|
},
|
|
$this->parserFactory->getMainInstance()->getTags()
|
|
);
|
|
ApiResult::setArrayType( $tags, 'BCarray' );
|
|
ApiResult::setIndexedTagName( $tags, 't' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $tags );
|
|
}
|
|
|
|
public function appendFunctionHooks( $property ) {
|
|
$hooks = $this->parserFactory->getMainInstance()->getFunctionHooks();
|
|
ApiResult::setArrayType( $hooks, 'BCarray' );
|
|
ApiResult::setIndexedTagName( $hooks, 'h' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $hooks );
|
|
}
|
|
|
|
public function appendVariables( $property ) {
|
|
$variables = $this->magicWordFactory->getVariableIDs();
|
|
ApiResult::setArrayType( $variables, 'BCarray' );
|
|
ApiResult::setIndexedTagName( $variables, 'v' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $variables );
|
|
}
|
|
|
|
public function appendProtocols( $property ) {
|
|
// Make a copy of the global so we don't try to set the _element key of it - T47130
|
|
$protocols = array_values( $this->getConfig()->get( MainConfigNames::UrlProtocols ) );
|
|
ApiResult::setArrayType( $protocols, 'BCarray' );
|
|
ApiResult::setIndexedTagName( $protocols, 'p' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $protocols );
|
|
}
|
|
|
|
public function appendDefaultOptions( $property ) {
|
|
$options = $this->userOptionsLookup->getDefaultOptions( null );
|
|
$options[ApiResult::META_BC_BOOLS] = array_keys( $options );
|
|
return $this->getResult()->addValue( 'query', $property, $options );
|
|
}
|
|
|
|
public function appendUploadDialog( $property ) {
|
|
$config = $this->getConfig()->get( MainConfigNames::UploadDialog );
|
|
return $this->getResult()->addValue( 'query', $property, $config );
|
|
}
|
|
|
|
private function getAutoPromoteConds() {
|
|
$allowedConditions = [];
|
|
foreach ( get_defined_constants() as $constantName => $constantValue ) {
|
|
if ( strpos( $constantName, 'APCOND_' ) !== false ) {
|
|
$allowedConditions[$constantName] = $constantValue;
|
|
}
|
|
}
|
|
return $allowedConditions;
|
|
}
|
|
|
|
private function processAutoPromote( $input, $allowedConditions ) {
|
|
$data = [];
|
|
foreach ( $input as $groupName => $conditions ) {
|
|
$row = $this->recAutopromote( $conditions, $allowedConditions );
|
|
if ( !isset( $row[0] ) || is_string( $row ) ) {
|
|
$row = [ $row ];
|
|
}
|
|
$data[$groupName] = $row;
|
|
}
|
|
return $data;
|
|
}
|
|
|
|
private function appendAutoPromote( $property ) {
|
|
return $this->getResult()->addValue(
|
|
'query',
|
|
$property,
|
|
$this->processAutoPromote(
|
|
$this->getConfig()->get( MainConfigNames::Autopromote ),
|
|
$this->getAutoPromoteConds()
|
|
)
|
|
);
|
|
}
|
|
|
|
private function appendAutoPromoteOnce( $property ) {
|
|
$allowedConditions = $this->getAutoPromoteConds();
|
|
$data = [];
|
|
foreach ( $this->getConfig()->get( MainConfigNames::AutopromoteOnce ) as $key => $value ) {
|
|
$data[$key] = $this->processAutoPromote( $value, $allowedConditions );
|
|
}
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
/**
|
|
* @param array|int|string $cond
|
|
* @param array $allowedConditions
|
|
* @return array|string
|
|
*/
|
|
private function recAutopromote( $cond, $allowedConditions ) {
|
|
$config = [];
|
|
// First, checks if $cond is an array
|
|
if ( is_array( $cond ) ) {
|
|
// Checks if $cond[0] is a valid operand
|
|
if ( in_array( $cond[0], UserGroupManager::VALID_OPS, true ) ) {
|
|
$config['operand'] = $cond[0];
|
|
// Traversal checks conditions
|
|
foreach ( array_slice( $cond, 1 ) as $value ) {
|
|
$config[] = $this->recAutopromote( $value, $allowedConditions );
|
|
}
|
|
} elseif ( is_string( $cond[0] ) ) {
|
|
// Returns $cond directly, if $cond[0] is a string
|
|
$config = $cond;
|
|
} else {
|
|
// When $cond is equal to an APCOND_ constant value
|
|
$params = array_slice( $cond, 1 );
|
|
if ( $params === [ null ] ) {
|
|
// Special casing for these conditions and their default of null,
|
|
// to replace their values with $wgAutoConfirmCount/$wgAutoConfirmAge as appropriate
|
|
if ( $cond[0] === APCOND_EDITCOUNT ) {
|
|
$params = [ $this->getConfig()->get( MainConfigNames::AutoConfirmCount ) ];
|
|
} elseif ( $cond[0] === APCOND_AGE ) {
|
|
$params = [ $this->getConfig()->get( MainConfigNames::AutoConfirmAge ) ];
|
|
}
|
|
}
|
|
$config = [
|
|
'condname' => array_search( $cond[0], $allowedConditions ),
|
|
'params' => $params
|
|
];
|
|
ApiResult::setIndexedTagName( $config, 'params' );
|
|
}
|
|
} elseif ( is_string( $cond ) ) {
|
|
$config = $cond;
|
|
} else {
|
|
// When $cond is equal to an APCOND_ constant value
|
|
$config = [
|
|
'condname' => array_search( $cond, $allowedConditions ),
|
|
'params' => []
|
|
];
|
|
ApiResult::setIndexedTagName( $config, 'params' );
|
|
}
|
|
|
|
return $config;
|
|
}
|
|
|
|
public function appendSubscribedHooks( $property ) {
|
|
$hookContainer = MediaWikiServices::getInstance()->getHookContainer();
|
|
$hookNames = $hookContainer->getHookNames();
|
|
sort( $hookNames );
|
|
|
|
$data = [];
|
|
foreach ( $hookNames as $name ) {
|
|
$arr = [
|
|
'name' => $name,
|
|
'subscribers' => $hookContainer->getHandlerDescriptions( $name ),
|
|
];
|
|
|
|
ApiResult::setArrayType( $arr['subscribers'], 'array' );
|
|
ApiResult::setIndexedTagName( $arr['subscribers'], 's' );
|
|
$data[] = $arr;
|
|
}
|
|
|
|
ApiResult::setIndexedTagName( $data, 'hook' );
|
|
|
|
return $this->getResult()->addValue( 'query', $property, $data );
|
|
}
|
|
|
|
public function getCacheMode( $params ) {
|
|
// Messages for $wgExtraInterlanguageLinkPrefixes depend on user language
|
|
if ( $this->getConfig()->get( MainConfigNames::ExtraInterlanguageLinkPrefixes ) &&
|
|
in_array( 'interwikimap', $params['prop'] ?? [] )
|
|
) {
|
|
return 'anon-public-user-private';
|
|
}
|
|
|
|
return 'public';
|
|
}
|
|
|
|
public function getAllowedParams() {
|
|
return [
|
|
'prop' => [
|
|
ParamValidator::PARAM_DEFAULT => 'general',
|
|
ParamValidator::PARAM_ISMULTI => true,
|
|
ParamValidator::PARAM_TYPE => [
|
|
'general',
|
|
'namespaces',
|
|
'namespacealiases',
|
|
'specialpagealiases',
|
|
'magicwords',
|
|
'interwikimap',
|
|
'dbrepllag',
|
|
'statistics',
|
|
'usergroups',
|
|
'autocreatetempuser',
|
|
'clientlibraries',
|
|
'libraries',
|
|
'extensions',
|
|
'fileextensions',
|
|
'rightsinfo',
|
|
'restrictions',
|
|
'languages',
|
|
'languagevariants',
|
|
'skins',
|
|
'extensiontags',
|
|
'functionhooks',
|
|
'showhooks',
|
|
'variables',
|
|
'protocols',
|
|
'defaultoptions',
|
|
'uploaddialog',
|
|
'autopromote',
|
|
'autopromoteonce',
|
|
],
|
|
ApiBase::PARAM_HELP_MSG_PER_VALUE => [],
|
|
],
|
|
'filteriw' => [
|
|
ParamValidator::PARAM_TYPE => [
|
|
'local',
|
|
'!local',
|
|
]
|
|
],
|
|
'showalldb' => false,
|
|
'numberingroup' => false,
|
|
'inlanguagecode' => null,
|
|
];
|
|
}
|
|
|
|
protected function getExamplesMessages() {
|
|
return [
|
|
'action=query&meta=siteinfo&siprop=general|namespaces|namespacealiases|statistics'
|
|
=> 'apihelp-query+siteinfo-example-simple',
|
|
'action=query&meta=siteinfo&siprop=interwikimap&sifilteriw=local'
|
|
=> 'apihelp-query+siteinfo-example-interwiki',
|
|
'action=query&meta=siteinfo&siprop=dbrepllag&sishowalldb='
|
|
=> 'apihelp-query+siteinfo-example-replag',
|
|
];
|
|
}
|
|
|
|
public function getHelpUrls() {
|
|
return 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Siteinfo';
|
|
}
|
|
}
|
|
|
|
/** @deprecated class alias since 1.43 */
|
|
class_alias( ApiQuerySiteinfo::class, 'ApiQuerySiteinfo' );
|