linkRenderer = $linkRenderer; $this->cache = $cache; $this->jobQueueGroup = $jobQueueGroup; } /** * Hook: BeforePageDisplay * * If there is a lintid parameter, look up that error in the database * and setup and output our client-side helpers * * @param OutputPage $out * @param Skin $skin */ public function onBeforePageDisplay( $out, $skin ): void { $request = $out->getRequest(); $lintId = $request->getInt( 'lintid' ); if ( !$lintId ) { return; } $title = $out->getTitle(); if ( !$title ) { return; } $lintError = ( new Database( $title->getArticleID() ) )->getFromId( $lintId ); if ( !$lintError ) { // Already fixed or bogus URL parameter? return; } $out->addJsConfigVars( [ 'wgLinterErrorCategory' => $lintError->category, 'wgLinterErrorLocation' => $lintError->location, ] ); $out->addModules( 'ext.linter.edit' ); } /** * Hook: WikiPageDeletionUpdates * * Remove entries from the linter table upon page deletion * * @param WikiPage $wikiPage * @param Content $content * @param array &$updates */ public function onWikiPageDeletionUpdates( $wikiPage, $content, &$updates ) { $id = $wikiPage->getId(); $cache = $this->cache; $updates[] = new MWCallableUpdate( static function () use ( $id, $cache ) { $database = new Database( $id ); $totalsLookup = new TotalsLookup( new CategoryManager(), $cache, $database ); $totalsLookup->updateStats( $database->setForPage( [] ) ); }, __METHOD__ ); } /** * This should match Parsoid's PageConfig::hasLintableContentModel() */ public const LINTABLE_CONTENT_MODELS = [ CONTENT_MODEL_WIKITEXT, 'proofread-page' ]; /** * Hook: RevisionFromEditComplete * * Remove entries from the linter table upon page content model change away from wikitext * * @param WikiPage $wikiPage * @param RevisionRecord $newRevisionRecord * @param bool|int $originalRevId * @param UserIdentity $user * @param string[] &$tags */ public function onRevisionFromEditComplete( $wikiPage, $newRevisionRecord, $originalRevId, $user, &$tags ) { // This is just a stop-gap to deal with callers that aren't complying // with the advertised hook signature. if ( !is_array( $tags ) ) { return; } if ( in_array( "mw-blank", $tags ) || ( in_array( "mw-contentmodelchange", $tags ) && !in_array( $wikiPage->getContentModel(), self::LINTABLE_CONTENT_MODELS ) ) ) { $database = new Database( $wikiPage->getId() ); $totalsLookup = new TotalsLookup( new CategoryManager(), $this->cache, $database ); $totalsLookup->updateStats( $database->setForPage( [] ) ); } } /** * Hook: APIQuerySiteInfoGeneralInfo * * Expose categories via action=query&meta=siteinfo * * @param ApiQuerySiteInfo $api * @param array &$data */ public function onAPIQuerySiteInfoGeneralInfo( $api, &$data ) { $catManager = new CategoryManager(); $data['linter'] = [ 'high' => $catManager->getHighPriority(), 'medium' => $catManager->getMediumPriority(), 'low' => $catManager->getLowPriority(), ]; } /** * Hook: InfoAction * * Display quick summary of errors for this page on ?action=info * * @param IContextSource $context * @param array &$pageInfo */ public function onInfoAction( $context, &$pageInfo ) { $title = $context->getTitle(); $pageId = $title->getArticleID(); if ( !$pageId ) { return; } $database = new Database( $pageId ); $totals = array_filter( $database->getTotalsForPage() ); if ( !$totals ) { // No errors, yay! return; } foreach ( $totals as $name => $count ) { $pageInfo['linter'][] = [ $context->msg( "linter-category-$name" ), htmlspecialchars( (string)$count ) ]; } $pageInfo['linter'][] = [ 'below', $this->linkRenderer->makeKnownLink( SpecialPage::getTitleFor( 'LintErrors' ), $context->msg( 'pageinfo-linter-moreinfo' )->text(), [], [ 'wpNamespaceRestrictions' => $title->getNamespace(), 'titlesearch' => $title->getText(), 'exactmatch' => 1 ] ), ]; } /** * Hook: ParserLogLinterData * * To record a lint errors. * * @param string $page * @param int $revision * @param array[] $data * @return bool */ public function onParserLogLinterData( string $page, int $revision, array $data ): bool { $errors = []; $title = Title::newFromText( $page ); if ( !$title || !$title->getArticleID() || $title->getLatestRevID() != $revision ) { return false; } $categoryMgr = new CategoryManager(); $catCounts = []; foreach ( $data as $info ) { if ( $categoryMgr->isKnownCategory( $info['type'] ) ) { $info[ 'dbid' ] = null; } elseif ( !isset( $info[ 'dbid' ] ) ) { continue; } $count = $catCounts[$info['type']] ?? 0; if ( $count > Database::MAX_PER_CAT ) { // Drop continue; } $catCounts[$info['type']] = $count + 1; if ( !isset( $info['dsr'] ) ) { LoggerFactory::getInstance( 'Linter' )->warning( 'dsr for {page} @ rev {revid}, for lint: {lint} is missing', [ 'page' => $page, 'revid' => $revision, 'lint' => $info['type'], ] ); continue; } $info['location'] = array_slice( $info['dsr'], 0, 2 ); if ( !isset( $info['params'] ) ) { $info['params'] = []; } if ( isset( $info['templateInfo'] ) && $info['templateInfo'] ) { $info['params']['templateInfo'] = $info['templateInfo']; } $errors[] = $info; } $job = new RecordLintJob( $title, [ 'errors' => $errors, 'revision' => $revision, ], $this->cache ); $this->jobQueueGroup->push( $job ); return true; } }