41 private const QUERY_PROP_MODULES = [
43 'class' => ApiQueryCategories::class,
46 'class' => ApiQueryCategoryInfo::class,
49 'class' => ApiQueryContributors::class,
54 'GroupPermissionsLookup',
58 'deletedrevisions' => [
59 'class' => ApiQueryDeletedRevisions::class,
62 'ContentHandlerFactory',
76 'class' => ApiQueryDuplicateFiles::class,
82 'class' => ApiQueryExternalLinks::class,
88 'class' => ApiQueryBacklinksprop::class,
95 'class' => ApiQueryImages::class,
101 'class' => ApiQueryImageInfo::class,
110 'class' => ApiQueryInfo::class,
118 'LanguageConverterFactory',
123 'IntroMessageBuilder',
124 'PreloadedContentBuilder',
131 'class' => ApiQueryLinks::class,
139 'class' => ApiQueryBacklinksprop::class,
146 'class' => ApiQueryIWLinks::class,
152 'class' => ApiQueryLangLinks::class,
160 'class' => ApiQueryPageProps::class,
166 'class' => ApiQueryBacklinksprop::class,
173 'class' => ApiQueryRevisions::class,
176 'ContentHandlerFactory',
183 'ContentTransformer',
190 'stashimageinfo' => [
191 'class' => ApiQueryStashImageInfo::class,
200 'class' => ApiQueryLinks::class,
208 'class' => ApiQueryBacklinksprop::class,
219 private const QUERY_LIST_MODULES = [
221 'class' => ApiQueryAllCategories::class,
223 'alldeletedrevisions' => [
224 'class' => ApiQueryAllDeletedRevisions::class,
227 'ContentHandlerFactory',
234 'ContentTransformer',
241 'class' => ApiQueryAllLinks::class,
250 'class' => ApiQueryAllImages::class,
253 'GroupPermissionsLookup',
257 'class' => ApiQueryAllLinks::class,
266 'class' => ApiQueryAllPages::class,
274 'class' => ApiQueryAllLinks::class,
283 'class' => ApiQueryAllRevisions::class,
286 'ContentHandlerFactory',
293 'ContentTransformer',
299 'mystashedfiles' => [
300 'class' => ApiQueryMyStashedFiles::class,
302 'alltransclusions' => [
303 'class' => ApiQueryAllLinks::class,
312 'class' => ApiQueryAllUsers::class,
316 'GroupPermissionsLookup',
319 'RecentChangeLookup',
320 'TempUserDetailsLookup',
324 'class' => ApiQueryBacklinks::class,
330 'class' => ApiQueryBlocks::class,
332 'DatabaseBlockStore',
334 'BlockRestrictionStore',
340 'categorymembers' => [
341 'class' => ApiQueryCategoryMembers::class,
347 'class' => ApiQueryCodexIcons::class,
350 'class' => ApiQueryDeletedrevs::class,
353 'RowCommentFormatter',
361 'class' => ApiQueryBacklinks::class,
367 'class' => ApiQueryExtLinksUsage::class,
373 'class' => ApiQueryFilearchive::class,
380 'class' => ApiQueryBacklinks::class,
386 'class' => ApiQueryIWBacklinks::class,
389 'class' => ApiQueryLangBacklinks::class,
392 'class' => ApiQueryLogEvents::class,
395 'RowCommentFormatter',
399 'LogFormatterFactory',
403 'class' => ApiQueryPagesWithProp::class,
406 'class' => ApiQueryPagePropNames::class,
409 'class' => ApiQueryPrefixSearch::class,
411 'SearchEngineConfig',
412 'SearchEngineFactory',
415 'protectedtitles' => [
416 'class' => ApiQueryProtectedTitles::class,
419 'RowCommentFormatter'
423 'class' => ApiQueryQueryPage::class,
425 'SpecialPageFactory',
429 'class' => ApiQueryRandom::class,
431 'ContentHandlerFactory'
435 'class' => ApiQueryRecentChanges::class,
438 'RowCommentFormatter',
441 'LogFormatterFactory',
442 'ChangesListQueryFactory',
443 'RecentChangeLookup',
447 'class' => ApiQuerySearch::class,
449 'SearchEngineConfig',
450 'SearchEngineFactory',
455 'class' => ApiQueryTags::class,
460 'trackingcategories' => [
461 'class' => ApiQueryTrackingCategories::class,
463 'TrackingCategories',
467 'class' => ApiQueryUserContribs::class,
470 'UserIdentityLookup',
480 'class' => ApiQueryUsers::class,
488 'TempUserDetailsLookup',
492 'class' => ApiQueryWatchlist::class,
495 'ChangesListQueryFactory',
496 'RowCommentFormatter',
498 'LogFormatterFactory',
499 'RecentChangeLookup',
501 'WatchlistLabelStore',
505 'class' => ApiQueryWatchlistRaw::class,
507 'WatchedItemQueryService',
518 private const QUERY_META_MODULES = [
520 'class' => ApiQueryAllMessages::class,
529 'authmanagerinfo' => [
530 'class' => ApiQueryAuthManagerInfo::class,
536 'class' => ApiQuerySiteinfo::class,
541 'LanguageConverterFactory',
549 'SpecialPageFactory',
555 'GroupPermissionsLookup',
559 'class' => ApiQueryUserInfo::class,
561 'TalkPageNotificationManager',
566 'WatchlistLabelStore',
570 'class' => ApiQueryFileRepoInfo::class,
576 'class' => ApiQueryTokens::class,
579 'class' => ApiQueryLanguageinfo::class,
584 'LanguageConverterFactory',
602 ObjectFactory $objectFactory,
607 parent::__construct( $main, $action );
616 $this->mModuleMgr->addModules( self::QUERY_PROP_MODULES,
'prop' );
618 $this->mModuleMgr->addModules( self::QUERY_LIST_MODULES,
'list' );
620 $this->mModuleMgr->addModules( self::QUERY_META_MODULES,
'meta' );
623 $this->
getHookRunner()->onApiQuery__moduleManager( $this->mModuleMgr );
634 return $this->mModuleMgr;
642 return $this->mPageSet;
654 $this->
getMain()->createPrinterByName(
'xml' ) );
675 $this->instantiateModules( $allModules,
'prop' );
676 $propModules = array_keys( $allModules );
677 $this->instantiateModules( $allModules,
'list' );
678 $this->instantiateModules( $allModules,
'meta' );
684 $modules = $continuationManager->getRunModules();
685 '@phan-var ApiQueryBase[] $modules';
688 $message =
'hookaborted';
689 if ( !$this->
getHookRunner()->onApiQueryCheckCanExecute( $modules, $this->
getUser(), $message ) ) {
695 if ( !$continuationManager->isGeneratorDone() ) {
698 foreach ( $modules as $module ) {
700 $timer = $statsFactory->getTiming(
'api_query_extraDataTiming_seconds' )
701 ->setLabel(
'module', $module->getModuleName() )
703 $module->requestExtraData( $this->mPageSet );
707 $this->mPageSet->execute();
709 $this->outputGeneralPageInfo();
711 $this->mPageSet->executeDryRun();
714 $cacheMode = $this->mPageSet->getCacheMode();
717 foreach ( $modules as $module ) {
719 $timer = $statsFactory->getTiming(
'api_query_executeTiming_seconds' )
720 ->setLabel(
'module', $module->getModuleName() )
722 $t = microtime(
true );
724 $params = $module->extractRequestParams();
726 $cacheMode, $module->getCacheMode( $params ) );
727 $scope = LoggerFactory::getContext()->addScoped( [
728 'context.api_query_module_name' => $module->getModuleName(),
734 $module->recordUnifiedMetrics(
735 microtime(
true ) - $t
737 }
catch ( \Throwable $e ) {
739 $module->recordUnifiedMetrics(
740 microtime(
true ) - $t,
742 'status' =>
'error_' . $e->getCode(),
748 ScopedCallback::consume( $scope );
756 $this->
getMain()->setCacheMode( $cacheMode );
760 if ( $this->mParams[
'rawcontinue'] ) {
761 $data = $continuationManager->getRawNonContinuation();
763 $this->
getResult()->addValue(
null,
'query-noncontinue', $data,
766 $data = $continuationManager->getRawContinuation();
768 $this->
getResult()->addValue(
null,
'query-continue', $data,
772 $continuationManager->setContinuationIntoResult( $this->
getResult() );
786 if ( $modCacheMode ===
'anon-public-user-private' ) {
787 if ( $cacheMode !==
'private' ) {
788 $cacheMode =
'anon-public-user-private';
790 } elseif ( $modCacheMode ===
'public' ) {
793 $cacheMode =
'private';
804 private function instantiateModules( &$modules, $param ) {
805 $wasPosted = $this->
getRequest()->wasPosted();
806 if ( isset( $this->mParams[$param] ) ) {
807 foreach ( $this->mParams[$param] as $moduleName ) {
808 $instance = $this->mModuleMgr->getModule( $moduleName, $param );
809 if ( $instance ===
null ) {
812 if ( !$wasPosted && $instance->mustBePosted() ) {
816 if ( !array_key_exists( $moduleName, $modules ) ) {
817 $modules[$moduleName] = $instance;
828 private function outputGeneralPageInfo() {
836 $values = $pageSet->getNormalizedTitlesAsResult( $result );
839 $fit = $fit && $result->addValue(
'query',
'normalized', $values );
841 $values = $pageSet->getConvertedTitlesAsResult( $result );
843 $fit = $fit && $result->addValue(
'query',
'converted', $values );
845 $values = $pageSet->getInterwikiTitlesAsResult( $result, $this->mParams[
'iwurl'] );
847 $fit = $fit && $result->addValue(
'query',
'interwiki', $values );
849 $values = $pageSet->getRedirectTitlesAsResult( $result );
851 $fit = $fit && $result->addValue(
'query',
'redirects', $values );
853 $values = $pageSet->getMissingRevisionIDsAsResult( $result );
855 $fit = $fit && $result->addValue(
'query',
'badrevids', $values );
863 foreach ( $pageSet->getMissingPages() as $fakeId => $page ) {
865 'ns' => $page->getNamespace(),
866 'title' => $this->titleFormatter->getPrefixedText( $page ),
869 $title = $this->titleFactory->newFromPageIdentity( $page );
870 if ( $title->isKnown() ) {
871 $vals[
'known'] =
true;
873 $pages[$fakeId] = $vals;
876 foreach ( $pageSet->getInvalidTitlesAndReasons() as $fakeId => $data ) {
877 $pages[$fakeId] = $data + [
'invalid' => true ];
880 foreach ( $pageSet->getMissingPageIDs() as $pageid ) {
888 foreach ( $pageSet->getSpecialPages() as $fakeId => $page ) {
890 'ns' => $page->getNamespace(),
891 'title' => $this->titleFormatter->getPrefixedText( $page ),
894 $title = $this->titleFactory->newFromPageReference( $page );
895 if ( !$title->isKnown() ) {
896 $vals[
'missing'] =
true;
898 $pages[$fakeId] = $vals;
902 foreach ( $pageSet->getGoodPages() as $pageid => $page ) {
905 'ns' => $page->getNamespace(),
906 'title' => $this->titleFormatter->getPrefixedText( $page ),
910 if ( count( $pages ) ) {
911 $pageSet->populateGeneratorData( $pages );
914 if ( $this->mParams[
'indexpageids'] ) {
917 $pageIDs = array_map(
'strval', $pageIDs );
919 $fit = $fit && $result->addValue(
'query',
'pageids', $pageIDs );
923 $fit = $fit && $result->addValue(
'query',
'pages', $pages );
927 $this->
dieWithError(
'apierror-badconfig-resulttoosmall',
'badconfig' );
930 if ( $this->mParams[
'export'] ) {
931 $this->doExport( $pageSet, $result );
939 private function doExport( $pageSet, $result ) {
941 $titles = $pageSet->getGoodPages();
942 if ( count( $titles ) ) {
944 foreach ( $titles as $title ) {
945 if ( $this->
getAuthority()->authorizeRead(
'read', $title ) ) {
946 $exportTitles[] = $title;
951 $exporter = $this->wikiExporterFactory->getWikiExporter( $this->
getDB() );
952 $sink =
new DumpStringOutput;
953 $exporter->setOutputSink( $sink );
954 $exporter->setSchemaVersion( $this->mParams[
'exportschema'] );
955 $exporter->openStream();
956 foreach ( $exportTitles as $title ) {
957 $exporter->pageByTitle( $title );
959 $exporter->closeStream();
964 if ( $this->mParams[
'exportnowrap'] ) {
980 ParamValidator::PARAM_ISMULTI =>
true,
981 ParamValidator::PARAM_TYPE =>
'submodule',
984 ParamValidator::PARAM_ISMULTI =>
true,
985 ParamValidator::PARAM_TYPE =>
'submodule',
988 ParamValidator::PARAM_ISMULTI =>
true,
989 ParamValidator::PARAM_TYPE =>
'submodule',
991 'indexpageids' =>
false,
993 'exportnowrap' =>
false,
1002 'rawcontinue' =>
false,
1005 $result += $this->
getPageSet()->getFinalParams( $flags );
1019 $allowedParams = [
'rawcontinue' => 1,
'indexpageids' => 1 ];
1023 $needed = $param ===
'meta';
1024 if ( !isset( $allowedParams[$param] ) && $request->getCheck( $param ) !== $needed ) {
1032 $this->instantiateModules( $modules,
'meta' );
1033 foreach ( $modules as $module ) {
1034 if ( $module->isReadMode() ) {
1047 $this->instantiateModules( $modules,
'list' );
1048 $this->instantiateModules( $modules,
'meta' );
1049 $this->instantiateModules( $modules,
'prop' );
1050 foreach ( $modules as $module ) {
1051 if ( $module->isWriteMode() ) {
1061 $title = Title::newMainPage()->getPrefixedText();
1062 $mp = rawurlencode( $title );
1065 'action=query&prop=revisions&meta=siteinfo&' .
1066 "titles={$mp}&rvprop=user|comment&continue="
1067 =>
'apihelp-query-example-revisions',
1068 'action=query&generator=allpages&gapprefix=API/&prop=revisions&continue='
1069 =>
'apihelp-query-example-allpages',
1076 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Query',
1077 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Meta',
1078 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Properties',
1079 'https://www.mediawiki.org/wiki/Special:MyLanguage/API:Lists',