2 use MediaWiki\Revision\MutableRevisionRecord
;
3 use MediaWiki\Revision\RevisionStore
;
4 use MediaWiki\Revision\SlotRecord
;
5 use MediaWiki\User\UserIdentityValue
;
10 * @covers BlockLevelPass
12 class ParserMethodsTest
extends MediaWikiLangTestCase
{
14 public static function providePreSaveTransform() {
16 [ 'hello this is ~~~',
17 "hello this is [[Special:Contributions/127.0.0.1|127.0.0.1]]",
19 [ 'hello \'\'this\'\' is <nowiki>~~~</nowiki>',
20 'hello \'\'this\'\' is <nowiki>~~~</nowiki>',
26 * @dataProvider providePreSaveTransform
28 public function testPreSaveTransform( $text, $expected ) {
31 $title = Title
::newFromText( str_replace( '::', '__', __METHOD__
) );
33 $user->setName( "127.0.0.1" );
34 $popts = ParserOptions
::newFromUser( $user );
35 $text = $wgParser->preSaveTransform( $text, $title, $user, $popts );
37 $this->assertEquals( $expected, $text );
40 public static function provideStripOuterParagraph() {
41 // This mimics the most common use case (stripping paragraphs generated by the parser).
42 $message = new RawMessage( "Message text." );
50 "<p class='foo'>Text.</p>",
51 "<p class='foo'>Text.</p>",
58 "<p>Text.</p><p>More text.</p>",
59 "<p>Text.</p><p>More text.</p>",
69 * @dataProvider provideStripOuterParagraph
71 public function testStripOuterParagraph( $text, $expected ) {
72 $this->assertEquals( $expected, Parser
::stripOuterParagraph( $text ) );
76 * @expectedException MWException
77 * @expectedExceptionMessage Parser state cleared while parsing.
78 * Did you call Parser::parse recursively?
80 public function testRecursiveParse() {
82 $title = Title
::newFromText( 'foo' );
83 $po = new ParserOptions
;
84 $wgParser->setHook( 'recursivecallparser', [ $this, 'helperParserFunc' ] );
85 $wgParser->parse( '<recursivecallparser>baz</recursivecallparser>', $title, $po );
88 public function helperParserFunc( $input, $args, $parser ) {
89 $title = Title
::newFromText( 'foo' );
90 $po = new ParserOptions
;
91 $parser->parse( $input, $title, $po );
95 public function testCallParserFunction() {
98 // Normal parses test passing PPNodes. Test passing an array.
99 $title = Title
::newFromText( str_replace( '::', '__', __METHOD__
) );
100 $wgParser->startExternalParse( $title, new ParserOptions(), Parser
::OT_HTML
);
101 $frame = $wgParser->getPreprocessor()->newFrame();
102 $ret = $wgParser->callParserFunction( $frame, '#tag',
103 [ 'pre', 'foo', 'style' => 'margin-left: 1.6em' ]
105 $ret['text'] = $wgParser->mStripState
->unstripBoth( $ret['text'] );
108 'text' => '<pre style="margin-left: 1.6em">foo</pre>',
109 ], $ret, 'callParserFunction works for {{#tag:pre|foo|style=margin-left: 1.6em}}' );
114 * @covers ParserOutput::getSections
116 public function testGetSections() {
119 $title = Title
::newFromText( str_replace( '::', '__', __METHOD__
) );
120 $out = $wgParser->parse( "==foo==\n<h2>bar</h2>\n==baz==\n", $title, new ParserOptions() );
128 'fromtitle' => $title->getPrefixedDBkey(),
138 'fromtitle' => false,
139 'byteoffset' => null,
148 'fromtitle' => $title->getPrefixedDBkey(),
152 ], $out->getSections(), 'getSections() with proper value when <h2> is used' );
156 * @dataProvider provideNormalizeLinkUrl
158 public function testNormalizeLinkUrl( $explanation, $url, $expected ) {
159 $this->assertEquals( $expected, Parser
::normalizeLinkUrl( $url ), $explanation );
162 public static function provideNormalizeLinkUrl() {
165 'Escaping of unsafe characters',
166 'http://example.org/foo bar?param[]="value"¶m[]=valüe',
167 'http://example.org/foo%20bar?param%5B%5D=%22value%22¶m%5B%5D=val%C3%BCe',
170 'Case normalization of percent-encoded characters',
171 'http://example.org/%ab%cD%Ef%FF',
172 'http://example.org/%AB%CD%EF%FF',
175 'Unescaping of safe characters',
176 'http://example.org/%3C%66%6f%6F%3E?%3C%66%6f%6F%3E#%3C%66%6f%6F%3E',
177 'http://example.org/%3Cfoo%3E?%3Cfoo%3E#%3Cfoo%3E',
180 'Context-sensitive replacement of sometimes-safe characters',
181 'http://example.org/%23%2F%3F%26%3D%2B%3B?%23%2F%3F%26%3D%2B%3B#%23%2F%3F%26%3D%2B%3B',
182 'http://example.org/%23%2F%3F&=+;?%23/?%26%3D%2B%3B#%23/?&=+;',
185 'IPv6 links aren\'t escaped',
186 'http://[::1]/foobar',
187 'http://[::1]/foobar',
190 'non-IPv6 links aren\'t unescaped',
191 'http://%5B::1%5D/foobar',
192 'http://%5B::1%5D/foobar',
197 public function testWrapOutput() {
199 $title = Title
::newFromText( 'foo' );
200 $po = new ParserOptions();
201 $wgParser->parse( 'Hello World', $title, $po );
202 $text = $wgParser->getOutput()->getText();
204 $this->assertContains( 'Hello World', $text );
205 $this->assertContains( '<div', $text );
206 $this->assertContains( 'class="mw-parser-output"', $text );
210 * @param string $name
213 private function getMockTitle( $name ) {
214 $title = $this->getMock( Title
::class );
215 $title->method( 'getPrefixedDBkey' )->willReturn( $name );
216 $title->method( 'getPrefixedText' )->willReturn( $name );
217 $title->method( 'getDBkey' )->willReturn( $name );
218 $title->method( 'getText' )->willReturn( $name );
219 $title->method( 'getNamespace' )->willReturn( 0 );
220 $title->method( 'getPageLanguage' )->willReturn( Language
::factory( 'en' ) );
225 public function provideRevisionAccess() {
226 $title = $this->getMockTitle( 'ParserRevisionAccessTest' );
228 $frank = $this->getMockBuilder( User
::class )
229 ->disableOriginalConstructor()
232 $frank->method( 'getName' )->willReturn( 'Frank' );
234 $text = '* user:{{REVISIONUSER}};id:{{REVISIONID}};time:{{REVISIONTIMESTAMP}};';
235 $po = new ParserOptions( $frank );
237 yield
'current' => [ $text, $po, 0, 'user:CurrentAuthor;id:200;time:20160606000000;' ];
238 yield
'current with ID' => [ $text, $po, 200, 'user:CurrentAuthor;id:200;time:20160606000000;' ];
240 $text = '* user:{{REVISIONUSER}};id:{{REVISIONID}};time:{{REVISIONTIMESTAMP}};';
241 $po = new ParserOptions( $frank );
243 yield
'old' => [ $text, $po, 100, 'user:OldAuthor;id:100;time:20140404000000;' ];
245 $oldRevision = new MutableRevisionRecord( $title );
246 $oldRevision->setId( 100 );
247 $oldRevision->setUser( new UserIdentityValue( 7, 'FauxAuthor', 0 ) );
248 $oldRevision->setTimestamp( '20141111111111' );
249 $oldRevision->setContent( SlotRecord
::MAIN
, new WikitextContent( 'FAUX' ) );
251 $po = new ParserOptions( $frank );
252 $po->setCurrentRevisionCallback( function () use ( $oldRevision ) {
253 return new Revision( $oldRevision );
256 yield
'old with override' => [ $text, $po, 100, 'user:FauxAuthor;id:100;time:20141111111111;' ];
258 $text = '* user:{{REVISIONUSER}};user-subst:{{subst:REVISIONUSER}};';
260 $po = new ParserOptions( $frank );
261 $po->setIsPreview( true );
263 yield
'preview without override, using context' => [
271 $text = '* user:{{REVISIONUSER}};time:{{REVISIONTIMESTAMP}};'
272 . 'user-subst:{{subst:REVISIONUSER}};time-subst:{{subst:REVISIONTIMESTAMP}};';
274 $newRevision = new MutableRevisionRecord( $title );
275 $newRevision->setUser( new UserIdentityValue( 9, 'NewAuthor', 0 ) );
276 $newRevision->setTimestamp( '20180808000000' );
277 $newRevision->setContent( SlotRecord
::MAIN
, new WikitextContent( 'NEW' ) );
279 $po = new ParserOptions( $frank );
280 $po->setIsPreview( true );
281 $po->setCurrentRevisionCallback( function () use ( $newRevision ) {
282 return new Revision( $newRevision );
289 'user:NewAuthor;time:20180808000000;',
290 'user-subst:NewAuthor;time-subst:20180808000000;',
293 $po = new ParserOptions( $frank );
294 $po->setCurrentRevisionCallback( function () use ( $newRevision ) {
295 return new Revision( $newRevision );
298 yield
'pre-save' => [
302 'user:NewAuthor;time:20180808000000;',
303 'user-subst:NewAuthor;time-subst:20180808000000;',
306 $text = "(ONE)<includeonly>(TWO)</includeonly>"
307 . "<noinclude>#{{:ParserRevisionAccessTest}}#</noinclude>";
309 $newRevision = new MutableRevisionRecord( $title );
310 $newRevision->setUser( new UserIdentityValue( 9, 'NewAuthor', 0 ) );
311 $newRevision->setTimestamp( '20180808000000' );
312 $newRevision->setContent( SlotRecord
::MAIN
, new WikitextContent( $text ) );
314 $po = new ParserOptions( $frank );
315 $po->setIsPreview( true );
316 $po->setCurrentRevisionCallback( function () use ( $newRevision ) {
317 return new Revision( $newRevision );
320 yield
'preview with self-transclude' => [ $text, $po, null, '(ONE)#(ONE)(TWO)#' ];
324 * @dataProvider provideRevisionAccess
326 public function testRevisionAccess(
331 $expectedInPst = null
335 $title = $this->getMockTitle( 'ParserRevisionAccessTest' );
337 $po->enableLimitReport( false );
339 $oldRevision = new MutableRevisionRecord( $title );
340 $oldRevision->setId( 100 );
341 $oldRevision->setUser( new UserIdentityValue( 7, 'OldAuthor', 0 ) );
342 $oldRevision->setTimestamp( '20140404000000' );
343 $oldRevision->setContent( SlotRecord
::MAIN
, new WikitextContent( 'OLD' ) );
345 $currentRevision = new MutableRevisionRecord( $title );
346 $currentRevision->setId( 200 );
347 $currentRevision->setUser( new UserIdentityValue( 9, 'CurrentAuthor', 0 ) );
348 $currentRevision->setTimestamp( '20160606000000' );
349 $currentRevision->setContent( SlotRecord
::MAIN
, new WikitextContent( 'CURRENT' ) );
351 $revisionStore = $this->getMockBuilder( RevisionStore
::class )
352 ->disableOriginalConstructor()
356 ->method( 'getKnownCurrentRevision' )
358 [ $title, 100, $oldRevision ],
359 [ $title, 200, $currentRevision ],
360 [ $title, 0, $currentRevision ],
364 ->method( 'getRevisionById' )
366 [ 100, 0, $oldRevision ],
367 [ 200, 0, $currentRevision ],
370 $this->setService( 'RevisionStore', $revisionStore );
372 $wgParser->parse( $text, $title, $po, true, true, $revId );
373 $html = $wgParser->getOutput()->getText();
375 $this->assertContains( $expectedInHtml, $html, 'In HTML' );
377 if ( $expectedInPst !== null ) {
378 $pst = $wgParser->preSaveTransform( $text, $title, $po->getUser(), $po );
379 $this->assertContains( $expectedInPst, $pst, 'After Pre-Safe Transform' );
383 public static function provideGuessSectionNameFromWikiText() {
385 [ '1/2', 'html5', '#1/2' ],
386 [ '1/2', 'legacy', '#1.2F2' ],
390 /** @dataProvider provideGuessSectionNameFromWikiText */
391 public function testGuessSectionNameFromWikiText( $input, $mode, $expected ) {
392 $this->setMwGlobals( [ 'wgFragmentMode' => [ $mode ] ] );
394 $result = $wgParser->guessSectionNameFromWikiText( $input );
395 $this->assertEquals( $result, $expected );
398 // @todo Add tests for cleanSig() / cleanSigInSig(), getSection(),
399 // replaceSection(), getPreloadText()