wiki.techinc.nl/tests/phpunit/includes/page/ArticleTest.php
C. Scott Ananian 1a1ac6a82d [HtmlOutputRendererHelper] Remove use of ParsoidOutputAccess::parseUncacheable
This removes the last use of ParsoidOutputAccess in core, allowing it
to be deprecated and eventually removed.

Bug: T367074
Bug: T317018
Change-Id: Ica2c880e2e7c2b126aaea66a3e4be460b3f2234f
2024-07-19 03:09:32 -04:00

275 lines
8.3 KiB
PHP

<?php
use MediaWiki\Context\RequestContext;
use MediaWiki\MainConfigNames;
use MediaWiki\MainConfigSchema;
use MediaWiki\Message\Message;
use MediaWiki\Page\ParserOutputAccess;
use MediaWiki\Parser\ParserOutput;
use MediaWiki\Status\Status;
use MediaWiki\Title\Title;
use MediaWiki\User\User;
/**
* @group Database
*/
class ArticleTest extends \MediaWikiIntegrationTestCase {
/**
* @param Title $title
* @param User|null $user
*
* @return Article
*/
private function newArticle( Title $title, User $user = null ): Article {
if ( !$user ) {
$user = $this->getTestUser()->getUser();
}
$context = new RequestContext();
$article = new Article( $title );
$context->setUser( $user );
$context->setTitle( $title );
$article->setContext( $context );
return $article;
}
/**
* @covers \Article::__sleep
*/
public function testSerialization_fails() {
$article = new Article( Title::newMainPage() );
$this->expectException( LogicException::class );
serialize( $article );
}
/**
* Tests that missing article page shows parser contents
* of the well-known system message for NS_MEDIAWIKI pages
* @covers \Article::showMissingArticle
*/
public function testMissingArticleMessage() {
// Use a well-known system message
$title = Title::makeTitle( NS_MEDIAWIKI, 'Uploadedimage' );
$article = $this->newArticle( $title );
$article->showMissingArticle();
$output = $article->getContext()->getOutput();
$this->assertStringContainsString(
Message::newFromKey( 'uploadedimage' )->parse(),
$output->getHTML()
);
}
/**
* Test if patrol footer is possible to show
* @covers \Article::showPatrolFooter
* @dataProvider provideShowPatrolFooter
*/
public function testShowPatrolFooter( $group, $title, $editPageText, $isEditedBySameUser, $expectedResult ) {
$testPage = $this->getNonexistingTestPage( $title );
$user1 = $this->getTestUser( $group )->getUser();
$user2 = $this->getTestUser()->getUser();
if ( $editPageText !== null ) {
$editedUser = $isEditedBySameUser ? $user1 : $user2;
$editIsGood = $this->editPage( $testPage, $editPageText, '', NS_MAIN, $editedUser )->isGood();
$this->assertTrue( $editIsGood, 'edited a page' );
}
$article = $this->newArticle( $title, $user1 );
$this->assertSame( $expectedResult, $article->showPatrolFooter() );
}
public static function provideShowPatrolFooter() {
yield 'UserAllowedRevExist' => [
'sysop',
Title::makeTitle( NS_MAIN, 'Page1' ),
'EditPage1',
false,
true
];
yield 'UserNotAllowedRevExist' => [
null,
Title::makeTitle( NS_MAIN, 'Page2' ),
'EditPage2',
false,
false
];
yield 'UserAllowedNoRev' => [
'sysop',
Title::makeTitle( NS_MAIN, 'Page3' ),
null,
false,
false
];
yield 'UserAllowedRevExistBySameUser' => [
'sysop',
Title::makeTitle( NS_MAIN, 'Page4' ),
'EditPage4',
true,
false
];
}
/**
* Show patrol footer even if the page was moved (T162871).
*
* @covers \Article::showPatrolFooter
*/
public function testShowPatrolFooterMovedPage() {
$oldTitle = Title::makeTitle( NS_USER, 'NewDraft' );
$newTitle = Title::makeTitle( NS_MAIN, 'NewDraft' );
$editor = $this->getTestUser()->getUser();
$editIsGood = $this->editPage( $oldTitle, 'Content', '', NS_USER, $editor )->isGood();
$this->assertTrue( $editIsGood, 'edited a page' );
$status = $this->getServiceContainer()
->getMovePageFactory()
->newMovePage( $oldTitle, $newTitle )
->move( $this->getTestUser()->getUser() );
$this->assertTrue( $status->isOK() );
$sysop = $this->getTestUser( 'sysop' )->getUser();
$article = $this->newArticle( $newTitle, $sysop );
$this->assertTrue( $article->showPatrolFooter() );
}
/**
* Ensure that content that is present in the parser cache will be used.
*
* @covers \Article::generateContentOutput
*/
public function testUsesCachedOutput() {
$title = $this->getExistingTestPage()->getTitle();
$parserOutputAccess = $this->createNoOpMock( ParserOutputAccess::class, [ 'getCachedParserOutput' ] );
$parserOutputAccess->method( 'getCachedParserOutput' )
->willReturn( new ParserOutput( 'Kittens' ) );
$this->setService( 'ParserOutputAccess', $parserOutputAccess );
$article = $this->newArticle( $title );
$article->view();
$this->assertStringContainsString( 'Kittens', $article->getContext()->getOutput()->getHTML() );
}
/**
* Ensure that content that is present in the parser cache will be used.
*
* @covers \Article::generateContentOutput
*/
public function testOutputIsCached() {
$this->overrideConfigValue(
MainConfigNames::ParsoidCacheConfig,
[ 'WarmParsoidParserCache' => true ]
+ MainConfigSchema::getDefaultValue( MainConfigNames::ParsoidCacheConfig )
);
$title = $this->getExistingTestPage()->getTitle();
// Run any jobs enqueued by the creation of the test page
$this->runJobs( [ 'minJobs' => 0 ] );
$beforePreWarm = true;
$parserOutputAccess = $this->createNoOpMock(
ParserOutputAccess::class,
[ 'getCachedParserOutput', 'getParserOutput', ]
);
$parserOutputAccess->method( 'getCachedParserOutput' )
->willReturn( null );
$parserOutputAccess
->expects( $this->exactly( 2 ) ) // This is the key assertion in this test case.
->method( 'getParserOutput' )
->with(
$this->anything(),
$this->callback( function ( ParserOptions $parserOptions ) use ( &$beforePreWarm ) {
$expectedReason = $beforePreWarm ? 'page-view' : 'view';
$this->assertSame( $expectedReason, $parserOptions->getRenderReason() );
return true;
} ),
$this->anything(),
$this->callback( function ( $options ) use ( &$beforePreWarm ) {
if ( $beforePreWarm ) {
$this->assertTrue( (bool)( $options & ParserOutputAccess::OPT_NO_CHECK_CACHE ),
"The cache is not checked again" );
$this->assertTrue( (bool)( $options & ParserOutputAccess::OPT_LINKS_UPDATE ),
"WikiPage::triggerOpportunisticLinksUpdate is attempted" );
}
return true;
} )
)
->willReturnCallback( static function ( $page, $parserOptions, $revision, $options ) use ( &$beforePreWarm ) {
$content = $beforePreWarm ? 'Old Kittens' : 'New Kittens';
return Status::newGood( new ParserOutput( $content ) );
} );
$this->setService( 'ParserOutputAccess', $parserOutputAccess );
$article = $this->newArticle( $title );
$article->view();
$beforePreWarm = false;
$this->runJobs( [ 'minJobs' => 1, 'maxJobs' => 1 ], [ 'type' => 'parsoidCachePrewarm' ] );
// This is just a sanity check, not the key assertion.
$this->assertStringContainsString( 'Old Kittens', $article->getContext()->getOutput()->getHTML() );
}
/**
* Ensure that protection indicators are shown when the page is protected.
* @covers \Article::showProtectionIndicator
*/
public function testShowProtectionIndicator() {
$this->overrideConfigValue(
MainConfigNames::EnableProtectionIndicators,
true
);
$title = $this->getExistingTestPage()->getTitle();
$article = $this->newArticle( $title );
$wikiPage = new WikiPage( $title );
$cascade = false;
$wikiPage->doUpdateRestrictions( [
'edit' => 'autoconfirmed',
],
[ 'edit' => 'infinity' ],
$cascade,
'Test reason',
$this->getTestSysop()->getUser()
);
$article->showProtectionIndicator();
$output = $article->getContext()->getOutput();
$this->assertArrayHasKey( 'protection-autoconfirmed', $output->getIndicators(), 'Protection indicators are shown when a page is protected' );
$templateTitle = Title::newFromText( 'CascadeProtectionTest', NS_TEMPLATE );
$this->editPage( $templateTitle, 'Some text here', 'Test', NS_TEMPLATE, $this->getTestSysop()->getUser() );
$articleTitle = $this->getExistingTestPage()->getTitle();
$this->editPage( $articleTitle, '{{CascadeProtectionTest}}', 'Test', NS_MAIN, $this->getTestSysop()->getUser() );
$wikiPage = new WikiPage( $articleTitle );
$cascade = true;
$wikiPage->doUpdateRestrictions( [
'edit' => 'sysop',
],
[ 'edit' => 'infinity' ],
$cascade,
'Test reason',
$this->getTestSysop()->getUser()
);
$template = $this->newArticle( $templateTitle );
$template->showProtectionIndicator();
$output = $template->getContext()->getOutput();
$this->assertArrayHasKey(
'protection-sysop-cascade',
$output->getIndicators(),
'Protection indicators are shown when a page protected using cascade protection'
);
}
}