wiki.techinc.nl/tests/phpunit/includes/page/ArticleViewTest.php

602 lines
20 KiB
PHP
Raw Normal View History

<?php
use MediaWiki\MediaWikiServices;
use MediaWiki\Revision\MutableRevisionRecord;
use MediaWiki\Revision\RevisionRecord;
use MediaWiki\Revision\SlotRecord;
use PHPUnit\Framework\MockObject\MockObject;
use Wikimedia\TestingAccessWrapper;
/**
* @covers \Article::view()
*/
class ArticleViewTest extends MediaWikiIntegrationTestCase {
protected function setUp() : void {
parent::setUp();
$this->setUserLang( 'qqx' );
}
private function getHtml( OutputPage $output ) {
return preg_replace( '/<!--.*?-->/s', '', $output->getHTML() );
}
/**
* @param string|Title $title
* @param Content[]|string[] $revisionContents Content of the revisions to create
* (as Content or string).
* @param RevisionRecord[] &$revisions will be filled with the RevisionRecord for $content.
*
* @return WikiPage
* @throws MWException
*/
private function getPage( $title, array $revisionContents = [], array &$revisions = [] ) {
if ( is_string( $title ) ) {
$title = Title::makeTitle( $this->getDefaultWikitextNS(), $title );
}
$page = WikiPage::factory( $title );
$user = $this->getTestUser()->getUser();
foreach ( $revisionContents as $key => $cont ) {
if ( is_string( $cont ) ) {
$cont = new WikitextContent( $cont );
}
$u = $page->newPageUpdater( $user );
$u->setContent( SlotRecord::MAIN, $cont );
$rev = $u->saveRevision( CommentStoreComment::newUnsavedComment( 'Rev ' . $key ) );
$revisions[ $key ] = $rev;
}
// Clear content model cache to support tests that mock the revision
$this->getServiceContainer()->getMainWANObjectCache()->clearProcessCache();
return $page;
}
/**
* @covers Article::getOldId()
* @covers Article::getRevIdFetched()
*/
public function testGetOldId() {
$this->hideDeprecated( 'Article::getRevisionFetched' );
$this->hideDeprecated( 'Revision::__construct' );
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$idB = $revisions[2]->getId();
// oldid in constructor
$article = new Article( $page->getTitle(), $idA );
$this->assertSame( $idA, $article->getOldID() );
$article->getRevisionFetched();
$this->assertSame( $idA, $article->getRevIdFetched() );
// oldid 0 in constructor
$article = new Article( $page->getTitle(), 0 );
$this->assertSame( 0, $article->getOldID() );
$article->getRevisionFetched();
$this->assertSame( $idB, $article->getRevIdFetched() );
// oldid in request
$article = new Article( $page->getTitle() );
$context = new RequestContext();
$context->setRequest( new FauxRequest( [ 'oldid' => $idA ] ) );
$article->setContext( $context );
$this->assertSame( $idA, $article->getOldID() );
$article->getRevisionFetched();
$this->assertSame( $idA, $article->getRevIdFetched() );
// no oldid
$article = new Article( $page->getTitle() );
$context = new RequestContext();
$context->setRequest( new FauxRequest( [] ) );
$article->setContext( $context );
$this->assertSame( 0, $article->getOldID() );
$article->getRevisionFetched();
$this->assertSame( $idB, $article->getRevIdFetched() );
}
public function testView() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ] );
$article = new Article( $page->getTitle(), 0 );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Test B', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'id="mw-revision-info"', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'id="mw-revision-nav"', $this->getHtml( $output ) );
}
public function testViewCached() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ] );
$po = new ParserOutput( 'Cached Text' );
$article = new Article( $page->getTitle(), 0 );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$cache = MediaWikiServices::getInstance()->getParserCache();
$cache->save( $po, $page, $article->getParserOptions() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Cached Text', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test B', $this->getHtml( $output ) );
}
/**
* @covers Article::getPage()
* @covers WikiPage::getRedirectTarget()
*/
public function testViewRedirect() {
$target = Title::makeTitle( $this->getDefaultWikitextNS(), 'Test_Target' );
$redirectText = '#REDIRECT [[' . $target->getPrefixedText() . ']]';
$page = $this->getPage( __METHOD__, [ $redirectText ] );
$article = new Article( $page->getTitle(), 0 );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$this->assertNotNull(
$article->getPage()->getRedirectTarget()->getPrefixedDBkey()
);
$this->assertSame(
$target->getPrefixedDBkey(),
$article->getPage()->getRedirectTarget()->getPrefixedDBkey()
);
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'class="redirectText"', $this->getHtml( $output ) );
$this->assertStringContainsString(
'>' . htmlspecialchars( $target->getPrefixedText() ) . '<',
$this->getHtml( $output )
);
}
public function testViewNonText() {
$dummy = $this->getPage( __METHOD__, [ 'Dummy' ] );
$dummyRev = $dummy->getRevisionRecord();
$title = $dummy->getTitle();
/** @var MockObject|ContentHandler $mockHandler */
$mockHandler = $this->getMockBuilder( ContentHandler::class )
->setMethods(
[
'isParserCacheSupported',
'serializeContent',
'unserializeContent',
'makeEmptyContent',
]
)
->setConstructorArgs( [ 'NotText', [ 'application/frobnitz' ] ] )
->getMock();
$mockHandler->method( 'isParserCacheSupported' )
->willReturn( false );
$this->setTemporaryHook(
'ContentHandlerForModelID',
function ( $id, &$handler ) use ( $mockHandler ) {
$handler = $mockHandler;
}
);
/** @var MockObject|Content $content */
$content = $this->createMock( Content::class );
$content->method( 'getParserOutput' )
->willReturn( new ParserOutput( 'Structured Output' ) );
$content->method( 'getModel' )
->willReturn( 'NotText' );
$content->expects( $this->never() )->method( 'getNativeData' );
$content->method( 'copy' )
->willReturn( $content );
$rev = new MutableRevisionRecord( $title );
$rev->setId( $dummyRev->getId() );
$rev->setPageId( $title->getArticleID() );
$rev->setUser( $dummyRev->getUser() );
$rev->setComment( $dummyRev->getComment() );
$rev->setTimestamp( $dummyRev->getTimestamp() );
$rev->setContent( SlotRecord::MAIN, $content );
/** @var MockObject|WikiPage $page */
$page = $this->getMockBuilder( WikiPage::class )
->setMethods( [ 'getRevisionRecord', 'getLatest' ] )
->setConstructorArgs( [ $title ] )
->getMock();
$page->method( 'getRevisionRecord' )
->willReturn( $rev );
$page->method( 'getLatest' )
->willReturn( $rev->getId() );
$article = Article::newFromWikiPage( $page, RequestContext::getMain() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Structured Output', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Dummy', $this->getHtml( $output ) );
}
public function testViewOfOldRevision() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$article = new Article( $page->getTitle(), $idA );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringContainsString( 'id="mw-revision-info"', $output->getSubtitle() );
$this->assertStringContainsString( 'id="mw-revision-nav"', $output->getSubtitle() );
$this->assertStringNotContainsString( 'id="revision-info-current"', $output->getSubtitle() );
$this->assertStringNotContainsString( 'Test B', $this->getHtml( $output ) );
}
public function testViewOfOldRevisionFromCache() {
$this->setMwGlobals( [
'wgOldRevisionParserCacheExpireTime' => 100500,
'wgMainWANCache' => 'main',
'wgWANObjectCaches' => [
'main' => [
'class' => WANObjectCache::class,
'cacheId' => 'hash',
],
],
] );
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$article = new Article( $page->getTitle(), $idA );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
// Force cache
MediaWikiServices::getInstance()
->getParserOutputAccess()
->getParserOutput(
$page,
ParserOptions::newCanonical( $article->getContext()->getUser() ),
$revisions[1]
);
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertSame( 1, substr_count( $output->getSubtitle(), 'class="mw-revision warningbox"' ) );
$this->assertSame( $idA, $output->getRevisionId() );
}
public function testViewOfCurrentRevision() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idB = $revisions[2]->getId();
$article = new Article( $page->getTitle(), $idB );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'Test B', $this->getHtml( $output ) );
$this->assertStringContainsString( 'id="mw-revision-info-current"', $output->getSubtitle() );
$this->assertStringContainsString( 'id="mw-revision-nav"', $output->getSubtitle() );
}
public function testViewOfMissingRevision() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A' ], $revisions );
$badId = $revisions[1]->getId() + 100;
$article = new Article( $page->getTitle(), $badId );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'missing-revision: ' . $badId, $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
}
public function testViewOfDeletedRevision() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$revDelList = $this->getRevDelRevisionList( $page->getTitle(), $idA );
$revDelList->setVisibility( [
'value' => [ RevisionRecord::DELETED_TEXT => 1 ],
'comment' => "Testing",
] );
$article = new Article( $page->getTitle(), $idA );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'rev-deleted-text-permission', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test B', $this->getHtml( $output ) );
}
public function testUnhiddenViewOfDeletedRevision() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$revDelList = $this->getRevDelRevisionList( $page->getTitle(), $idA );
$revDelList->setVisibility( [
'value' => [ RevisionRecord::DELETED_TEXT => 1 ],
'comment' => "Testing",
] );
$article = new Article( $page->getTitle(), $idA );
$context = new DerivativeContext( $article->getContext() );
$article->setContext( $context );
$context->getOutput()->setTitle( $page->getTitle() );
$context->getRequest()->setVal( 'unhide', 1 );
$context->setUser( $this->getTestUser( [ 'sysop' ] )->getUser() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'rev-deleted-text-view', $this->getHtml( $output ) );
$this->assertStringContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test B', $this->getHtml( $output ) );
}
public function testViewMissingPage() {
$page = $this->getPage( __METHOD__ );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
}
public function testViewDeletedPage() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ] );
$page->doDeleteArticleReal( 'Test', $this->getTestSysop()->getUser() );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( 'moveddeleted', $this->getHtml( $output ) );
$this->assertStringContainsString( 'logentry-delete-delete', $this->getHtml( $output ) );
$this->assertStringContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringNotContainsString( 'Test B', $this->getHtml( $output ) );
}
public function testViewMessagePage() {
$title = Title::makeTitle( NS_MEDIAWIKI, 'Mainpage' );
$page = $this->getPage( $title );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString(
wfMessage( 'mainpage' )->inContentLanguage()->parse(),
$this->getHtml( $output )
);
$this->assertStringNotContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
}
public function testViewMissingUserPage() {
$user = $this->getTestUser()->getUser();
$user->addToDatabase();
$title = Title::makeTitle( NS_USER, $user->getName() );
$page = $this->getPage( $title );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
$this->assertStringNotContainsString(
'(userpage-userdoesnotexist-view)',
$this->getHtml( $output )
);
}
public function testViewUserPageOfNonexistingUser() {
$user = User::newFromName( 'Testing ' . __METHOD__ );
$title = Title::makeTitle( NS_USER, $user->getName() );
$page = $this->getPage( $title );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
$this->assertStringContainsString(
'(userpage-userdoesnotexist-view:',
$this->getHtml( $output )
);
}
public function testArticleViewHeaderHook() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A' ] );
$article = new Article( $page->getTitle(), 0 );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$this->setTemporaryHook(
'ArticleViewHeader',
function ( Article $articlePage, &$outputDone, &$useParserCache ) use ( $article ) {
$this->assertSame( $article, $articlePage, '$articlePage' );
$outputDone = new ParserOutput( 'Hook Text' );
$outputDone->setTitleText( 'Hook Title' );
$articlePage->getContext()->getOutput()->addParserOutput( $outputDone );
}
);
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringContainsString( 'Hook Text', $this->getHtml( $output ) );
$this->assertSame( 'Hook Title', $output->getPageTitle() );
}
public function testArticleRevisionViewCustomHook() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A' ] );
$article = new Article( $page->getTitle(), 0 );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
// use ArticleViewHeader hook to bypass the parser cache
$this->setTemporaryHook(
'ArticleViewHeader',
function ( Article $articlePage, &$outputDone, &$useParserCache ) use ( $article ) {
$useParserCache = false;
}
);
$this->setTemporaryHook(
'ArticleRevisionViewCustom',
function ( RevisionRecord $rev, Title $title, $oldid, OutputPage $output ) use ( $page ) {
$content = $rev->getContent( SlotRecord::MAIN );
$this->assertSame( $page->getTitle(), $title, '$title' );
$this->assertSame( 'Test A', $content->getText(), '$content' );
$output->addHTML( 'Hook Text' );
return false;
}
);
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringNotContainsString( 'Test A', $this->getHtml( $output ) );
$this->assertStringContainsString( 'Hook Text', $this->getHtml( $output ) );
}
public function testShowMissingArticleHook() {
$page = $this->getPage( __METHOD__ );
$article = new Article( $page->getTitle() );
$article->getContext()->getOutput()->setTitle( $page->getTitle() );
$this->setTemporaryHook(
'ShowMissingArticle',
function ( Article $articlePage ) use ( $article ) {
$this->assertSame( $article, $articlePage, '$articlePage' );
$articlePage->getContext()->getOutput()->addHTML( 'Hook Text' );
}
);
$article->view();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString( '(noarticletextanon)', $this->getHtml( $output ) );
$this->assertStringContainsString( 'Hook Text', $this->getHtml( $output ) );
}
/**
* @covers \Article::showViewError()
*/
public function testViewLatestError() {
$page = $this->getPage( __METHOD__, [ 1 => 'Test A' ] );
$article = new Article( $page->getTitle(), 0 );
$output = $article->getContext()->getOutput();
$output->setTitle( $page->getTitle() );
// use ArticleViewHeader hook to bypass the parser cache
$this->setTemporaryHook(
'ArticleViewHeader',
function ( Article $articlePage, &$outputDone, &$useParserCache ) use ( $article ) {
$useParserCache = false;
}
);
$article = TestingAccessWrapper::newFromObject( $article );
$article->fetchResult = Status::newFatal(
'rev-deleted-text-permission',
$page->getTitle()->getPrefixedDBkey()
);
$article->view();
$this->assertStringContainsString(
'rev-deleted-text-permission: ArticleViewTest::testViewLatestError',
$this->getHtml( $output )
);
}
/**
* @covers \Article::showViewError()
*/
public function testViewOldError() {
$revisions = [];
$page = $this->getPage( __METHOD__, [ 1 => 'Test A', 2 => 'Test B' ], $revisions );
$idA = $revisions[1]->getId();
$article = new Article( $page->getTitle(), $idA );
$output = $article->getContext()->getOutput();
$output->setTitle( $page->getTitle() );
$article = TestingAccessWrapper::newFromObject( $article );
$article->fetchResult = Status::newFatal(
'rev-deleted-text-permission',
$page->getTitle()->getPrefixedDBkey()
);
$article->view();
$this->assertStringContainsString(
'rev-deleted-text-permission: ArticleViewTest::testViewOldError',
$this->getHtml( $output )
);
}
private function getRevDelRevisionList( $title, $revisionId ) {
$services = MediaWikiServices::getInstance();
return new RevDelRevisionList(
RequestContext::getMain(),
$title,
[ $revisionId ],
$services->getDBLoadBalancerFactory(),
$services->getHookContainer(),
$services->getHtmlCacheUpdater(),
$services->getRevisionStore(),
$services->getMainWANObjectCache()
);
}
}