parsoidClientFactory = $parsoidClientFactory; $this->commentParser = $commentParser; $this->revisionLookup = $revisionLookup; } /** * @inheritDoc */ public function execute() { $params = $this->extractRequestParams(); $this->requireOnlyOneParameter( $params, 'fromtitle', 'fromrev' ); $this->requireOnlyOneParameter( $params, 'totitle', 'torev' ); if ( isset( $params['torev'] ) ) { $toRev = $this->revisionLookup->getRevisionById( $params['torev'] ); if ( !$toRev ) { $this->dieWithError( [ 'apierror-nosuchrevid', $params['torev'] ] ); } } else { $toTitle = Title::newFromText( $params['totitle'] ); if ( !$toTitle ) { $this->dieWithError( [ 'apierror-invalidtitle', wfEscapeWikiText( $params['totitle'] ) ] ); } $toRev = $this->revisionLookup->getRevisionByTitle( $toTitle ); if ( !$toRev ) { $this->dieWithError( [ 'apierror-missingrev-title', wfEscapeWikiText( $toTitle->getPrefixedText() ) ], 'nosuchrevid' ); } } // When polling for new comments this is an important optimisation, // as usually there is no new revision. if ( $toRev->getId() === $params['fromrev'] ) { $this->addResult( $toRev, $toRev ); return; } if ( isset( $params['fromrev'] ) ) { $fromRev = $this->revisionLookup->getRevisionById( $params['fromrev'] ); if ( !$fromRev ) { $this->dieWithError( [ 'apierror-nosuchrevid', $params['fromrev'] ] ); } } else { $fromTitle = Title::newFromText( $params['fromtitle'] ); if ( !$fromTitle ) { $this->dieWithError( [ 'apierror-invalidtitle', wfEscapeWikiText( $params['fromtitle'] ) ] ); } $fromRev = $this->revisionLookup->getRevisionByTitle( $fromTitle ); if ( !$fromRev ) { $this->dieWithError( [ 'apierror-missingrev-title', wfEscapeWikiText( $fromTitle->getPrefixedText() ) ], 'nosuchrevid' ); } } if ( $fromRev->hasSameContent( $toRev ) ) { $this->addResult( $fromRev, $toRev ); return; } $fromItemSet = HookUtils::parseRevisionParsoidHtml( $fromRev ); $toItemSet = HookUtils::parseRevisionParsoidHtml( $toRev ); $removedComments = []; foreach ( $fromItemSet->getCommentItems() as $fromComment ) { if ( !$toItemSet->findCommentById( $fromComment->getId() ) ) { $removedComments[] = $fromComment->jsonSerializeForDiff(); } } $addedComments = []; foreach ( $toItemSet->getCommentItems() as $toComment ) { if ( !$fromItemSet->findCommentById( $toComment->getId() ) ) { $addedComments[] = $toComment->jsonSerializeForDiff(); } } $this->addResult( $fromRev, $toRev, $removedComments, $addedComments ); } /** * Add the result object from revisions and comment lists * * @param RevisionRecord $fromRev From revision * @param RevisionRecord $toRev To revision * @param array $removedComments Removed comments * @param array $addedComments Added comments */ protected function addResult( RevisionRecord $fromRev, RevisionRecord $toRev, array $removedComments = [], array $addedComments = [] ) { $fromTitle = Title::newFromLinkTarget( $fromRev->getPageAsLinkTarget() ); $toTitle = Title::newFromLinkTarget( $toRev->getPageAsLinkTarget() ); $result = [ 'fromrevid' => $fromRev->getId(), 'fromtitle' => $fromTitle->getPrefixedText(), 'torevid' => $toRev->getId(), 'totitle' => $toTitle->getPrefixedText(), 'removedcomments' => $removedComments, 'addedcomments' => $addedComments, ]; $this->getResult()->addValue( null, $this->getModuleName(), $result ); } /** * @inheritDoc */ public function getAllowedParams() { return [ 'fromtitle' => [ ApiBase::PARAM_HELP_MSG => 'apihelp-compare-param-fromtitle', ], 'fromrev' => [ ParamValidator::PARAM_TYPE => 'integer', ApiBase::PARAM_HELP_MSG => 'apihelp-compare-param-fromrev', ], 'totitle' => [ ApiBase::PARAM_HELP_MSG => 'apihelp-compare-param-totitle', ], 'torev' => [ ParamValidator::PARAM_TYPE => 'integer', ApiBase::PARAM_HELP_MSG => 'apihelp-compare-param-torev', ], ]; } /** * @inheritDoc */ public function needsToken() { return false; } /** * @inheritDoc */ public function isInternal() { return true; } /** * @inheritDoc */ public function isWriteMode() { return false; } }