MediaWiki  REL1_24
WebRequestTest.php
Go to the documentation of this file.
00001 <?php
00002 
00006 class WebRequestTest extends MediaWikiTestCase {
00007     protected $oldServer;
00008 
00009     protected function setUp() {
00010         parent::setUp();
00011 
00012         $this->oldServer = $_SERVER;
00013         IP::clearCaches();
00014     }
00015 
00016     protected function tearDown() {
00017         $_SERVER = $this->oldServer;
00018         IP::clearCaches();
00019 
00020         parent::tearDown();
00021     }
00022 
00027     public function testDetectServer( $expected, $input, $description ) {
00028         $_SERVER = $input;
00029         $result = WebRequest::detectServer();
00030         $this->assertEquals( $expected, $result, $description );
00031     }
00032 
00033     public static function provideDetectServer() {
00034         return array(
00035             array(
00036                 'http://x',
00037                 array(
00038                     'HTTP_HOST' => 'x'
00039                 ),
00040                 'Host header'
00041             ),
00042             array(
00043                 'https://x',
00044                 array(
00045                     'HTTP_HOST' => 'x',
00046                     'HTTPS' => 'on',
00047                 ),
00048                 'Host header with secure'
00049             ),
00050             array(
00051                 'http://x',
00052                 array(
00053                     'HTTP_HOST' => 'x',
00054                     'SERVER_PORT' => 80,
00055                 ),
00056                 'Default SERVER_PORT',
00057             ),
00058             array(
00059                 'http://x',
00060                 array(
00061                     'HTTP_HOST' => 'x',
00062                     'HTTPS' => 'off',
00063                 ),
00064                 'Secure off'
00065             ),
00066             array(
00067                 'http://y',
00068                 array(
00069                     'SERVER_NAME' => 'y',
00070                 ),
00071                 'Server name'
00072             ),
00073             array(
00074                 'http://x',
00075                 array(
00076                     'HTTP_HOST' => 'x',
00077                     'SERVER_NAME' => 'y',
00078                 ),
00079                 'Host server name precedence'
00080             ),
00081             array(
00082                 'http://[::1]:81',
00083                 array(
00084                     'HTTP_HOST' => '[::1]',
00085                     'SERVER_NAME' => '::1',
00086                     'SERVER_PORT' => '81',
00087                 ),
00088                 'Apache bug 26005'
00089             ),
00090             array(
00091                 'http://localhost',
00092                 array(
00093                     'SERVER_NAME' => '[2001'
00094                 ),
00095                 'Kind of like lighttpd per commit message in MW r83847',
00096             ),
00097             array(
00098                 'http://[2a01:e35:2eb4:1::2]:777',
00099                 array(
00100                     'SERVER_NAME' => '[2a01:e35:2eb4:1::2]:777'
00101                 ),
00102                 'Possible lighttpd environment per bug 14977 comment 13',
00103             ),
00104         );
00105     }
00106 
00111     public function testGetIP( $expected, $input, $squid, $xffList, $private, $description ) {
00112         $_SERVER = $input;
00113         $this->setMwGlobals( array(
00114             'wgSquidServersNoPurge' => $squid,
00115             'wgUsePrivateIPs' => $private,
00116             'wgHooks' => array(
00117                 'IsTrustedProxy' => array(
00118                     function ( &$ip, &$trusted ) use ( $xffList ) {
00119                         $trusted = $trusted || in_array( $ip, $xffList );
00120                         return true;
00121                     }
00122                 )
00123             )
00124         ) );
00125 
00126         $request = new WebRequest();
00127         $result = $request->getIP();
00128         $this->assertEquals( $expected, $result, $description );
00129     }
00130 
00131     public static function provideGetIP() {
00132         return array(
00133             array(
00134                 '127.0.0.1',
00135                 array(
00136                     'REMOTE_ADDR' => '127.0.0.1'
00137                 ),
00138                 array(),
00139                 array(),
00140                 false,
00141                 'Simple IPv4'
00142             ),
00143             array(
00144                 '::1',
00145                 array(
00146                     'REMOTE_ADDR' => '::1'
00147                 ),
00148                 array(),
00149                 array(),
00150                 false,
00151                 'Simple IPv6'
00152             ),
00153             array(
00154                 '12.0.0.1',
00155                 array(
00156                     'REMOTE_ADDR' => 'abcd:0001:002:03:4:555:6666:7777',
00157                     'HTTP_X_FORWARDED_FOR' => '12.0.0.1, abcd:0001:002:03:4:555:6666:7777',
00158                 ),
00159                 array( 'ABCD:1:2:3:4:555:6666:7777' ),
00160                 array(),
00161                 false,
00162                 'IPv6 normalisation'
00163             ),
00164             array(
00165                 '12.0.0.3',
00166                 array(
00167                     'REMOTE_ADDR' => '12.0.0.1',
00168                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00169                 ),
00170                 array( '12.0.0.1', '12.0.0.2' ),
00171                 array(),
00172                 false,
00173                 'With X-Forwaded-For'
00174             ),
00175             array(
00176                 '12.0.0.1',
00177                 array(
00178                     'REMOTE_ADDR' => '12.0.0.1',
00179                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00180                 ),
00181                 array(),
00182                 array(),
00183                 false,
00184                 'With X-Forwaded-For and disallowed server'
00185             ),
00186             array(
00187                 '12.0.0.2',
00188                 array(
00189                     'REMOTE_ADDR' => '12.0.0.1',
00190                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00191                 ),
00192                 array( '12.0.0.1' ),
00193                 array(),
00194                 false,
00195                 'With multiple X-Forwaded-For and only one allowed server'
00196             ),
00197             array(
00198                 '10.0.0.3',
00199                 array(
00200                     'REMOTE_ADDR' => '12.0.0.2',
00201                     'HTTP_X_FORWARDED_FOR' => '10.0.0.4, 10.0.0.3, 12.0.0.2'
00202                 ),
00203                 array( '12.0.0.1', '12.0.0.2' ),
00204                 array(),
00205                 false,
00206                 'With X-Forwaded-For and private IP (from cache proxy)'
00207             ),
00208             array(
00209                 '10.0.0.4',
00210                 array(
00211                     'REMOTE_ADDR' => '12.0.0.2',
00212                     'HTTP_X_FORWARDED_FOR' => '10.0.0.4, 10.0.0.3, 12.0.0.2'
00213                 ),
00214                 array( '12.0.0.1', '12.0.0.2', '10.0.0.3' ),
00215                 array(),
00216                 true,
00217                 'With X-Forwaded-For and private IP (allowed)'
00218             ),
00219             array(
00220                 '10.0.0.4',
00221                 array(
00222                     'REMOTE_ADDR' => '12.0.0.2',
00223                     'HTTP_X_FORWARDED_FOR' => '10.0.0.4, 10.0.0.3, 12.0.0.2'
00224                 ),
00225                 array( '12.0.0.1', '12.0.0.2' ),
00226                 array( '10.0.0.3' ),
00227                 true,
00228                 'With X-Forwaded-For and private IP (allowed)'
00229             ),
00230             array(
00231                 '10.0.0.3',
00232                 array(
00233                     'REMOTE_ADDR' => '12.0.0.2',
00234                     'HTTP_X_FORWARDED_FOR' => '10.0.0.4, 10.0.0.3, 12.0.0.2'
00235                 ),
00236                 array( '12.0.0.1', '12.0.0.2' ),
00237                 array( '10.0.0.3' ),
00238                 false,
00239                 'With X-Forwaded-For and private IP (disallowed)'
00240             ),
00241             array(
00242                 '12.0.0.3',
00243                 array(
00244                     'REMOTE_ADDR' => '12.0.0.1',
00245                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00246                 ),
00247                 array(),
00248                 array( '12.0.0.1', '12.0.0.2' ),
00249                 false,
00250                 'With X-Forwaded-For'
00251             ),
00252             array(
00253                 '12.0.0.2',
00254                 array(
00255                     'REMOTE_ADDR' => '12.0.0.1',
00256                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00257                 ),
00258                 array(),
00259                 array( '12.0.0.1' ),
00260                 false,
00261                 'With multiple X-Forwaded-For and only one allowed server'
00262             ),
00263             array(
00264                 '12.0.0.2',
00265                 array(
00266                     'REMOTE_ADDR' => '12.0.0.2',
00267                     'HTTP_X_FORWARDED_FOR' => '10.0.0.3, 12.0.0.2'
00268                 ),
00269                 array(),
00270                 array( '12.0.0.2' ),
00271                 false,
00272                 'With X-Forwaded-For and private IP and hook (disallowed)'
00273             ),
00274             array(
00275                 '12.0.0.1',
00276                 array(
00277                     'REMOTE_ADDR' => 'abcd:0001:002:03:4:555:6666:7777',
00278                     'HTTP_X_FORWARDED_FOR' => '12.0.0.1, abcd:0001:002:03:4:555:6666:7777',
00279                 ),
00280                 array( 'ABCD:1:2:3::/64' ),
00281                 array(),
00282                 false,
00283                 'IPv6 CIDR'
00284             ),
00285             array(
00286                 '12.0.0.3',
00287                 array(
00288                     'REMOTE_ADDR' => '12.0.0.1',
00289                     'HTTP_X_FORWARDED_FOR' => '12.0.0.3, 12.0.0.2'
00290                 ),
00291                 array( '12.0.0.0/24' ),
00292                 array(),
00293                 false,
00294                 'IPv4 CIDR'
00295             ),
00296         );
00297     }
00298 
00303     public function testGetIpLackOfRemoteAddrThrowAnException() {
00304         // ensure that local install state doesn't interfere with test
00305         $this->setMwGlobals( array(
00306             'wgSquidServersNoPurge' => array(),
00307             'wgSquidServers' => array(),
00308             'wgUsePrivateIPs' => false,
00309             'wgHooks' => array(),
00310         ) );
00311 
00312         $request = new WebRequest();
00313         # Next call throw an exception about lacking an IP
00314         $request->getIP();
00315     }
00316 
00317     public static function provideLanguageData() {
00318         return array(
00319             array( '', array(), 'Empty Accept-Language header' ),
00320             array( 'en', array( 'en' => 1 ), 'One language' ),
00321             array( 'en, ar', array( 'en' => 1, 'ar' => 1 ), 'Two languages listed in appearance order.' ),
00322             array(
00323                 'zh-cn,zh-tw',
00324                 array( 'zh-cn' => 1, 'zh-tw' => 1 ),
00325                 'Two equally prefered languages, listed in appearance order per rfc3282. Checks c9119'
00326             ),
00327             array(
00328                 'es, en; q=0.5',
00329                 array( 'es' => 1, 'en' => '0.5' ),
00330                 'Spanish as first language and English and second'
00331             ),
00332             array( 'en; q=0.5, es', array( 'es' => 1, 'en' => '0.5' ), 'Less prefered language first' ),
00333             array( 'fr, en; q=0.5, es', array( 'fr' => 1, 'es' => 1, 'en' => '0.5' ), 'Three languages' ),
00334             array( 'en; q=0.5, es', array( 'es' => 1, 'en' => '0.5' ), 'Two languages' ),
00335             array( 'en, zh;q=0', array( 'en' => 1 ), "It's Chinese to me" ),
00336             array(
00337                 'es; q=1, pt;q=0.7, it; q=0.6, de; q=0.1, ru;q=0',
00338                 array( 'es' => '1', 'pt' => '0.7', 'it' => '0.6', 'de' => '0.1' ),
00339                 'Preference for Romance languages'
00340             ),
00341             array(
00342                 'en-gb, en-us; q=1',
00343                 array( 'en-gb' => 1, 'en-us' => '1' ),
00344                 'Two equally prefered English variants'
00345             ),
00346         );
00347     }
00348 
00353     public function testAcceptLang( $acceptLanguageHeader, $expectedLanguages, $description ) {
00354         $_SERVER = array( 'HTTP_ACCEPT_LANGUAGE' => $acceptLanguageHeader );
00355         $request = new WebRequest();
00356         $this->assertSame( $request->getAcceptLang(), $expectedLanguages, $description );
00357     }
00358 }