5 * @group GlobalFunctions
7 class GlobalTest
extends MediaWikiTestCase
{
8 protected function setUp() {
11 $readOnlyFile = $this->getNewTempFile();
12 unlink( $readOnlyFile );
14 $this->setMwGlobals( [
15 'wgReadOnlyFile' => $readOnlyFile,
21 'file://', # Non-default
27 * @dataProvider provideForWfArrayDiff2
28 * @covers ::wfArrayDiff2
30 public function testWfArrayDiff2( $a, $b, $expected ) {
32 wfArrayDiff2( $a, $b ), $expected
36 // @todo Provide more tests
37 public static function provideForWfArrayDiff2() {
46 [ [ 'a' ], [ 'a', 'b', 'c' ] ],
47 [ [ 'a' ], [ 'a', 'b' ] ],
48 [ 1 => [ 'a', 'b', 'c' ] ],
54 * Test cases for random functions could hypothetically fail,
55 * even though they shouldn't.
61 public function testRandom() {
63 wfRandom() == wfRandom()
68 * @covers ::wfRandomString
70 public function testRandomString() {
72 wfRandomString() == wfRandomString()
75 strlen( wfRandomString( 10 ) ), 10
78 preg_match( '/^[0-9a-f]+$/i', wfRandomString() ) === 1
83 * @covers ::wfUrlencode
85 public function testUrlencode() {
87 "%E7%89%B9%E5%88%A5:Contributions/Foobar",
88 wfUrlencode( "\xE7\x89\xB9\xE5\x88\xA5:Contributions/Foobar" ) );
92 * @covers ::wfExpandIRI
94 public function testExpandIRI() {
96 "https://te.wikibooks.org/wiki/ఉబుంటు_వాడుకరి_మార్గదర్శని",
97 wfExpandIRI( "https://te.wikibooks.org/wiki/"
98 . "%E0%B0%89%E0%B0%AC%E0%B1%81%E0%B0%82%E0%B0%9F%E0%B1%81_"
99 . "%E0%B0%B5%E0%B0%BE%E0%B0%A1%E0%B1%81%E0%B0%95%E0%B0%B0%E0%B0%BF_"
100 . "%E0%B0%AE%E0%B0%BE%E0%B0%B0%E0%B1%8D%E0%B0%97%E0%B0%A6%E0%B0%B0"
101 . "%E0%B1%8D%E0%B0%B6%E0%B0%A8%E0%B0%BF" ) );
105 * Intended to cover the relevant bits of ServiceWiring.php, as well as GlobalFunctions.php
106 * @covers ::wfReadOnly
108 public function testReadOnlyEmpty() {
112 MediaWiki\MediaWikiServices
::getInstance()->getReadOnlyMode()->clearCache();
113 $this->assertFalse( wfReadOnly() );
114 $this->assertFalse( wfReadOnly() );
118 * Intended to cover the relevant bits of ServiceWiring.php, as well as GlobalFunctions.php
119 * @covers ::wfReadOnly
121 public function testReadOnlySet() {
122 global $wgReadOnly, $wgReadOnlyFile;
124 $readOnlyMode = MediaWiki\MediaWikiServices
::getInstance()->getReadOnlyMode();
125 $readOnlyMode->clearCache();
127 $f = fopen( $wgReadOnlyFile, "wt" );
128 fwrite( $f, 'Message' );
130 $wgReadOnly = null; # Check on $wgReadOnlyFile
132 $this->assertTrue( wfReadOnly() );
133 $this->assertTrue( wfReadOnly() ); # Check cached
135 unlink( $wgReadOnlyFile );
136 $readOnlyMode->clearCache();
137 $this->assertFalse( wfReadOnly() );
138 $this->assertFalse( wfReadOnly() );
142 * This behaviour could probably be deprecated. Several extensions rely on it as of 1.29.
143 * @covers ::wfReadOnlyReason
145 public function testReadOnlyGlobalChange() {
146 $this->assertFalse( wfReadOnlyReason() );
147 $this->setMwGlobals( [
148 'wgReadOnly' => 'reason'
150 $this->assertSame( 'reason', wfReadOnlyReason() );
153 public static function provideArrayToCGI() {
156 [ [ 'foo' => 'bar' ], 'foo=bar' ], // string test
157 [ [ 'foo' => '' ], 'foo=' ], // empty string test
158 [ [ 'foo' => 1 ], 'foo=1' ], // number test
159 [ [ 'foo' => true ], 'foo=1' ], // true test
160 [ [ 'foo' => false ], '' ], // false test
161 [ [ 'foo' => null ], '' ], // null test
162 [ [ 'foo' => 'A&B=5+6@!"\'' ], 'foo=A%26B%3D5%2B6%40%21%22%27' ], // urlencoding test
164 [ 'foo' => 'bar', 'baz' => 'is', 'asdf' => 'qwerty' ],
165 'foo=bar&baz=is&asdf=qwerty'
166 ], // multi-item test
167 [ [ 'foo' => [ 'bar' => 'baz' ] ], 'foo%5Bbar%5D=baz' ],
169 [ 'foo' => [ 'bar' => 'baz', 'qwerty' => 'asdf' ] ],
170 'foo%5Bbar%5D=baz&foo%5Bqwerty%5D=asdf'
172 [ [ 'foo' => [ 'bar', 'baz' ] ], 'foo%5B0%5D=bar&foo%5B1%5D=baz' ],
174 [ 'foo' => [ 'bar' => [ 'bar' => 'baz' ] ] ],
175 'foo%5Bbar%5D%5Bbar%5D=baz'
181 * @dataProvider provideArrayToCGI
182 * @covers ::wfArrayToCgi
184 public function testArrayToCGI( $array, $result ) {
185 $this->assertEquals( $result, wfArrayToCgi( $array ) );
189 * @covers ::wfArrayToCgi
191 public function testArrayToCGI2() {
196 [ 'foo' => 'bar', 'baz' => 'overridden value' ] ) );
199 public static function provideCgiToArray() {
202 [ 'foo=bar', [ 'foo' => 'bar' ] ], // string
203 [ 'foo=', [ 'foo' => '' ] ], // empty string
204 [ 'foo', [ 'foo' => '' ] ], // missing =
205 [ 'foo=bar&qwerty=asdf', [ 'foo' => 'bar', 'qwerty' => 'asdf' ] ], // multiple value
206 [ 'foo=A%26B%3D5%2B6%40%21%22%27', [ 'foo' => 'A&B=5+6@!"\'' ] ], // urldecoding test
207 [ 'foo%5Bbar%5D=baz', [ 'foo' => [ 'bar' => 'baz' ] ] ],
209 'foo%5Bbar%5D=baz&foo%5Bqwerty%5D=asdf',
210 [ 'foo' => [ 'bar' => 'baz', 'qwerty' => 'asdf' ] ]
212 [ 'foo%5B0%5D=bar&foo%5B1%5D=baz', [ 'foo' => [ 0 => 'bar', 1 => 'baz' ] ] ],
214 'foo%5Bbar%5D%5Bbar%5D=baz',
215 [ 'foo' => [ 'bar' => [ 'bar' => 'baz' ] ] ]
221 * @dataProvider provideCgiToArray
222 * @covers ::wfCgiToArray
224 public function testCgiToArray( $cgi, $result ) {
225 $this->assertEquals( $result, wfCgiToArray( $cgi ) );
228 public static function provideCgiRoundTrip() {
233 [ 'foo=bar&baz=biz' ],
234 [ 'foo=A%26B%3D5%2B6%40%21%22%27' ],
235 [ 'foo%5Bbar%5D=baz' ],
236 [ 'foo%5B0%5D=bar&foo%5B1%5D=baz' ],
237 [ 'foo%5Bbar%5D%5Bbar%5D=baz' ],
242 * @dataProvider provideCgiRoundTrip
243 * @covers ::wfArrayToCgi
245 public function testCgiRoundTrip( $cgi ) {
246 $this->assertEquals( $cgi, wfArrayToCgi( wfCgiToArray( $cgi ) ) );
250 * @covers ::mimeTypeMatch
252 public function testMimeTypeMatch() {
255 mimeTypeMatch( 'text/html',
256 [ 'application/xhtml+xml' => 1.0,
258 'text/plain' => 0.3 ] ) );
261 mimeTypeMatch( 'text/html',
263 'text/*' => 0.5 ] ) );
266 mimeTypeMatch( 'text/html',
267 [ '*/*' => 1.0 ] ) );
269 mimeTypeMatch( 'text/html',
270 [ 'image/png' => 1.0,
271 'image/svg+xml' => 0.5 ] ) );
275 * @covers ::wfNegotiateType
277 public function testNegotiateType() {
281 [ 'application/xhtml+xml' => 1.0,
285 [ 'text/html' => 1.0 ] ) );
287 'application/xhtml+xml',
289 [ 'application/xhtml+xml' => 1.0,
293 [ 'application/xhtml+xml' => 1.0,
294 'text/html' => 0.5 ] ) );
298 [ 'text/html' => 1.0,
301 'application/xhtml+xml' => 0.2 ],
302 [ 'application/xhtml+xml' => 1.0,
303 'text/html' => 0.5 ] ) );
310 [ 'application/xhtml+xml' => 1.0,
311 'text/html' => 0.5 ] ) );
315 [ 'application/xhtml+xml' => 1.0 ] ) );
320 * @covers ::wfDebugMem
322 public function testDebugFunctionTest() {
323 $debugLogFile = $this->getNewTempFile();
325 $this->setMwGlobals( [
326 'wgDebugLogFile' => $debugLogFile,
327 # @todo FIXME: $wgDebugTimestamps should be tested
328 'wgDebugTimestamps' => false
331 wfDebug( "This is a normal string" );
332 $this->assertEquals( "This is a normal string\n", file_get_contents( $debugLogFile ) );
333 unlink( $debugLogFile );
335 wfDebug( "This is nöt an ASCII string" );
336 $this->assertEquals( "This is nöt an ASCII string\n", file_get_contents( $debugLogFile ) );
337 unlink( $debugLogFile );
339 wfDebug( "\00305This has böth UTF and control chars\003" );
341 " 05This has böth UTF and control chars \n",
342 file_get_contents( $debugLogFile )
344 unlink( $debugLogFile );
347 $this->assertGreaterThan(
349 preg_replace( '/\D/', '', file_get_contents( $debugLogFile ) )
351 unlink( $debugLogFile );
354 $this->assertGreaterThan(
356 preg_replace( '/\D/', '', file_get_contents( $debugLogFile ) )
358 unlink( $debugLogFile );
362 * @covers ::wfClientAcceptsGzip
364 public function testClientAcceptsGzipTest() {
370 'compress, gzip' => true,
371 'gzip;q=1.0' => true,
374 'gzip;q=abcde' => true, // is this REALLY valid?
375 'gzip;q=12345678.9' => true,
379 if ( isset( $_SERVER['HTTP_ACCEPT_ENCODING'] ) ) {
380 $old_server_setting = $_SERVER['HTTP_ACCEPT_ENCODING'];
383 foreach ( $settings as $encoding => $expect ) {
384 $_SERVER['HTTP_ACCEPT_ENCODING'] = $encoding;
386 $this->assertEquals( $expect, wfClientAcceptsGzip( true ),
387 "'$encoding' => " . wfBoolToStr( $expect ) );
390 if ( isset( $old_server_setting ) ) {
391 $_SERVER['HTTP_ACCEPT_ENCODING'] = $old_server_setting;
396 * @covers ::wfPercent
398 public function testWfPercentTest() {
401 [ 6 / 7, '0.86%', 2, false ],
403 [ 22 / 7, '3.14286%', 5 ],
406 [ 10 / 3, '0%', -1 ],
407 [ 3 / 4 / 5, '0.1%', 1 ],
408 [ 6 / 7 * 8, '6.8571428571%', 10 ],
411 foreach ( $pcts as $pct ) {
412 if ( !isset( $pct[2] ) ) {
415 if ( !isset( $pct[3] ) ) {
419 $this->assertEquals( wfPercent( $pct[0], $pct[2], $pct[3] ), $pct[1], $pct[1] );
424 * test @see wfShorthandToInteger()
425 * @dataProvider provideShorthand
426 * @covers ::wfShorthandToInteger
428 public function testWfShorthandToInteger( $shorthand, $expected ) {
429 $this->assertEquals( $expected,
430 wfShorthandToInteger( $shorthand )
434 public static function provideShorthand() {
435 // Syntax: [ shorthand, expected integer ]
442 # Failures returns 0 :(
446 # Int, strings with spaces
455 [ '1m', 1024 * 1024 ],
456 [ '1M', 1024 * 1024 ],
457 [ '1g', 1024 * 1024 * 1024 ],
458 [ '1G', 1024 * 1024 * 1024 ],
479 * @param string $old Text as it was in the database
480 * @param string $mine Text submitted while user was editing
481 * @param string $yours Text submitted by the user
482 * @param bool $expectedMergeResult Whether the merge should be a success
483 * @param string $expectedText Text after merge has been completed
485 * @dataProvider provideMerge()
489 public function testMerge( $old, $mine, $yours, $expectedMergeResult, $expectedText ) {
490 $this->markTestSkippedIfNoDiff3();
493 $isMerged = wfMerge( $old, $mine, $yours, $mergedText );
495 $msg = 'Merge should be a ';
496 $msg .= $expectedMergeResult ?
'success' : 'failure';
497 $this->assertEquals( $expectedMergeResult, $isMerged, $msg );
500 // Verify the merged text
501 $this->assertEquals( $expectedText, $mergedText,
502 'is merged text as expected?' );
506 public static function provideMerge() {
507 $EXPECT_MERGE_SUCCESS = true;
508 $EXPECT_MERGE_FAILURE = false;
514 "one one one\n" . // trimmed
519 "one one one ONE ONE\n" .
521 "two two two\n", // with tailing whitespace
526 "two two TWO TWO", // trimmed
529 $EXPECT_MERGE_SUCCESS,
532 "one one one ONE ONE\n" .
534 "two two TWO TWO\n", // note: will always end in a newline
537 // #1: conflict, fail
540 "one one one", // trimmed
543 "one one one ONE ONE\n" .
546 "\n", // with tailing whitespace
551 "two two", // trimmed
553 $EXPECT_MERGE_FAILURE,
562 * @dataProvider provideMakeUrlIndexes()
563 * @covers ::wfMakeUrlIndexes
565 public function testMakeUrlIndexes( $url, $expected ) {
566 $index = wfMakeUrlIndexes( $url );
567 $this->assertEquals( $expected, $index, "wfMakeUrlIndexes(\"$url\")" );
570 public static function provideMakeUrlIndexes() {
572 // Testcase for T30627
574 'https://example.org/test.cgi?id=12345',
575 [ 'https://org.example./test.cgi?id=12345' ]
578 // mailtos are handled special
579 // is this really right though? that final . probably belongs earlier?
580 'mailto:wiki@wikimedia.org',
581 [ 'mailto:org.wikimedia@wiki.' ]
584 // file URL cases per T30627...
586 // three slashes: local filesystem path Unix-style
587 'file:///whatever/you/like.txt',
588 [ 'file://./whatever/you/like.txt' ]
591 // three slashes: local filesystem path Windows-style
592 'file:///c:/whatever/you/like.txt',
593 [ 'file://./c:/whatever/you/like.txt' ]
596 // two slashes: UNC filesystem path Windows-style
597 'file://intranet/whatever/you/like.txt',
598 [ 'file://intranet./whatever/you/like.txt' ]
600 // Multiple-slash cases that can sorta work on Mozilla
601 // if you hack it just right are kinda pathological,
602 // and unreliable cross-platform or on IE which means they're
603 // unlikely to appear on intranets.
604 // Those will survive the algorithm but with results that
605 // are less consistent.
607 // protocol-relative URL cases per T31854...
609 '//example.org/test.cgi?id=12345',
611 'http://org.example./test.cgi?id=12345',
612 'https://org.example./test.cgi?id=12345'
619 * @dataProvider provideWfMatchesDomainList
620 * @covers ::wfMatchesDomainList
622 public function testWfMatchesDomainList( $url, $domains, $expected, $description ) {
623 $actual = wfMatchesDomainList( $url, $domains );
624 $this->assertEquals( $expected, $actual, $description );
627 public static function provideWfMatchesDomainList() {
629 $protocols = [ 'HTTP' => 'http:', 'HTTPS' => 'https:', 'protocol-relative' => '' ];
630 foreach ( $protocols as $pDesc => $p ) {
631 $a = array_merge( $a, [
633 "$p//www.example.com",
636 "No matches for empty domains array, $pDesc URL"
639 "$p//www.example.com",
640 [ 'www.example.com' ],
642 "Exact match in domains array, $pDesc URL"
645 "$p//www.example.com",
648 "Match without subdomain in domains array, $pDesc URL"
651 "$p//www.example2.com",
652 [ 'www.example.com', 'www.example2.com', 'www.example3.com' ],
654 "Exact match with other domains in array, $pDesc URL"
657 "$p//www.example2.com",
658 [ 'example.com', 'example2.com', 'example3,com' ],
660 "Match without subdomain with other domains in array, $pDesc URL"
663 "$p//www.example4.com",
664 [ 'example.com', 'example2.com', 'example3,com' ],
666 "Domain not in array, $pDesc URL"
669 "$p//nds-nl.wikipedia.org",
670 [ 'nl.wikipedia.org' ],
672 "Non-matching substring of domain, $pDesc URL"
681 * @covers ::wfMkdirParents
683 public function testWfMkdirParents() {
684 // Should not return true if file exists instead of directory
685 $fname = $this->getNewTempFile();
686 MediaWiki\
suppressWarnings();
687 $ok = wfMkdirParents( $fname );
688 MediaWiki\restoreWarnings
();
689 $this->assertFalse( $ok );
693 * @dataProvider provideWfShellWikiCmdList
694 * @covers ::wfShellWikiCmd
696 public function testWfShellWikiCmd( $script, $parameters, $options,
697 $expected, $description
699 if ( wfIsWindows() ) {
700 // Approximation that's good enough for our purposes just now
701 $expected = str_replace( "'", '"', $expected );
703 $actual = wfShellWikiCmd( $script, $parameters, $options );
704 $this->assertEquals( $expected, $actual, $description );
707 public function wfWikiID() {
708 $this->setMwGlobals( [
709 'wgDBname' => 'example',
717 $this->setMwGlobals( [
718 'wgDBname' => 'example',
719 'wgDBprefix' => 'mw_',
727 public function testWfMemcKey() {
728 $cache = ObjectCache
::getLocalClusterInstance();
730 $cache->makeKey( 'foo', 123, 'bar' ),
731 wfMemcKey( 'foo', 123, 'bar' )
735 public function testWfForeignMemcKey() {
736 $cache = ObjectCache
::getLocalClusterInstance();
737 $keyspace = $this->readAttribute( $cache, 'keyspace' );
739 wfForeignMemcKey( $keyspace, '', 'foo', 'bar' ),
740 $cache->makeKey( 'foo', 'bar' )
744 public function testWfGlobalCacheKey() {
745 $cache = ObjectCache
::getLocalClusterInstance();
747 $cache->makeGlobalKey( 'foo', 123, 'bar' ),
748 wfGlobalCacheKey( 'foo', 123, 'bar' )
752 public static function provideWfShellWikiCmdList() {
756 [ 'eval.php', [ '--help', '--test' ], [],
757 "'$wgPhpCli' 'eval.php' '--help' '--test'",
758 "Called eval.php --help --test" ],
759 [ 'eval.php', [ '--help', '--test space' ], [ 'php' => 'php5' ],
760 "'php5' 'eval.php' '--help' '--test space'",
761 "Called eval.php --help --test with php option" ],
762 [ 'eval.php', [ '--help', '--test', 'X' ], [ 'wrapper' => 'MWScript.php' ],
763 "'$wgPhpCli' 'MWScript.php' 'eval.php' '--help' '--test' 'X'",
764 "Called eval.php --help --test with wrapper option" ],
767 [ '--help', '--test', 'y' ],
768 [ 'php' => 'php5', 'wrapper' => 'MWScript.php' ],
769 "'php5' 'MWScript.php' 'eval.php' '--help' '--test' 'y'",
770 "Called eval.php --help --test with wrapper and php option"
774 /* @todo many more! */