5 * @author Matthew Flaschen
9 * @todo factor tests in this class into providers and test methods
11 class OutputPageTest
extends MediaWikiTestCase
{
12 const SCREEN_MEDIA_QUERY
= 'screen and (min-width: 982px)';
13 const SCREEN_ONLY_MEDIA_QUERY
= 'only screen and (min-width: 982px)';
16 * Tests a particular case of transformCssMedia, using the given input, globals,
17 * expected return, and message
19 * Asserts that $expectedReturn is returned.
21 * options['printableQuery'] - value of query string for printable, or omitted for none
22 * options['handheldQuery'] - value of query string for handheld, or omitted for none
23 * options['media'] - passed into the method under the same name
24 * options['expectedReturn'] - expected return value
25 * options['message'] - PHPUnit message for assertion
27 * @param array $args Key-value array of arguments as shown above
29 protected function assertTransformCssMediaCase( $args ) {
31 if ( isset( $args['printableQuery'] ) ) {
32 $queryData['printable'] = $args['printableQuery'];
35 if ( isset( $args['handheldQuery'] ) ) {
36 $queryData['handheld'] = $args['handheldQuery'];
39 $fauxRequest = new FauxRequest( $queryData, false );
40 $this->setMwGlobals( [
41 'wgRequest' => $fauxRequest,
44 $actualReturn = OutputPage
::transformCssMedia( $args['media'] );
45 $this->assertSame( $args['expectedReturn'], $actualReturn, $args['message'] );
49 * Tests print requests
50 * @covers OutputPage::transformCssMedia
52 public function testPrintRequests() {
53 $this->assertTransformCssMediaCase( [
54 'printableQuery' => '1',
56 'expectedReturn' => null,
57 'message' => 'On printable request, screen returns null'
60 $this->assertTransformCssMediaCase( [
61 'printableQuery' => '1',
62 'media' => self
::SCREEN_MEDIA_QUERY
,
63 'expectedReturn' => null,
64 'message' => 'On printable request, screen media query returns null'
67 $this->assertTransformCssMediaCase( [
68 'printableQuery' => '1',
69 'media' => self
::SCREEN_ONLY_MEDIA_QUERY
,
70 'expectedReturn' => null,
71 'message' => 'On printable request, screen media query with only returns null'
74 $this->assertTransformCssMediaCase( [
75 'printableQuery' => '1',
77 'expectedReturn' => '',
78 'message' => 'On printable request, media print returns empty string'
83 * Tests screen requests, without either query parameter set
84 * @covers OutputPage::transformCssMedia
86 public function testScreenRequests() {
87 $this->assertTransformCssMediaCase( [
89 'expectedReturn' => 'screen',
90 'message' => 'On screen request, screen media type is preserved'
93 $this->assertTransformCssMediaCase( [
94 'media' => 'handheld',
95 'expectedReturn' => 'handheld',
96 'message' => 'On screen request, handheld media type is preserved'
99 $this->assertTransformCssMediaCase( [
100 'media' => self
::SCREEN_MEDIA_QUERY
,
101 'expectedReturn' => self
::SCREEN_MEDIA_QUERY
,
102 'message' => 'On screen request, screen media query is preserved.'
105 $this->assertTransformCssMediaCase( [
106 'media' => self
::SCREEN_ONLY_MEDIA_QUERY
,
107 'expectedReturn' => self
::SCREEN_ONLY_MEDIA_QUERY
,
108 'message' => 'On screen request, screen media query with only is preserved.'
111 $this->assertTransformCssMediaCase( [
113 'expectedReturn' => 'print',
114 'message' => 'On screen request, print media type is preserved'
119 * Tests handheld behavior
120 * @covers OutputPage::transformCssMedia
122 public function testHandheld() {
123 $this->assertTransformCssMediaCase( [
124 'handheldQuery' => '1',
125 'media' => 'handheld',
126 'expectedReturn' => '',
127 'message' => 'On request with handheld querystring and media is handheld, returns empty string'
130 $this->assertTransformCssMediaCase( [
131 'handheldQuery' => '1',
133 'expectedReturn' => null,
134 'message' => 'On request with handheld querystring and media is screen, returns null'
138 public static function provideMakeResourceLoaderLink() {
139 // @codingStandardsIgnoreStart Generic.Files.LineLength
141 // Single only=scripts load
143 [ 'test.foo', ResourceLoaderModule
::TYPE_SCRIPTS
],
144 "<script>(window.RLQ=window.RLQ||[]).push(function(){"
145 . 'mw.loader.load("http://127.0.0.1:8080/w/load.php?debug=false\u0026lang=en\u0026modules=test.foo\u0026only=scripts\u0026skin=fallback");'
148 // Multiple only=styles load
150 [ [ 'test.baz', 'test.foo', 'test.bar' ], ResourceLoaderModule
::TYPE_STYLES
],
152 '<link rel="stylesheet" href="http://127.0.0.1:8080/w/load.php?debug=false&lang=en&modules=test.bar%2Cbaz%2Cfoo&only=styles&skin=fallback"/>'
154 // Private embed (only=scripts)
156 [ 'test.quux', ResourceLoaderModule
::TYPE_SCRIPTS
],
157 "<script>(window.RLQ=window.RLQ||[]).push(function(){"
158 . "mw.test.baz({token:123});mw.loader.state({\"test.quux\":\"ready\"});"
162 // @codingStandardsIgnoreEnd
166 * See ResourceLoaderClientHtmlTest for full coverage.
168 * @dataProvider provideMakeResourceLoaderLink
169 * @covers OutputPage::makeResourceLoaderLink
171 public function testMakeResourceLoaderLink( $args, $expectedHtml ) {
172 $this->setMwGlobals( [
173 'wgResourceLoaderDebug' => false,
174 'wgLoadScript' => 'http://127.0.0.1:8080/w/load.php',
176 $class = new ReflectionClass( 'OutputPage' );
177 $method = $class->getMethod( 'makeResourceLoaderLink' );
178 $method->setAccessible( true );
179 $ctx = new RequestContext();
180 $ctx->setSkin( SkinFactory
::getDefaultInstance()->makeSkin( 'fallback' ) );
181 $ctx->setLanguage( 'en' );
182 $out = new OutputPage( $ctx );
183 $rl = $out->getResourceLoader();
184 $rl->setMessageBlobStore( new NullMessageBlobStore() );
186 'test.foo' => new ResourceLoaderTestModule( [
187 'script' => 'mw.test.foo( { a: true } );',
188 'styles' => '.mw-test-foo { content: "style"; }',
190 'test.bar' => new ResourceLoaderTestModule( [
191 'script' => 'mw.test.bar( { a: true } );',
192 'styles' => '.mw-test-bar { content: "style"; }',
194 'test.baz' => new ResourceLoaderTestModule( [
195 'script' => 'mw.test.baz( { a: true } );',
196 'styles' => '.mw-test-baz { content: "style"; }',
198 'test.quux' => new ResourceLoaderTestModule( [
199 'script' => 'mw.test.baz( { token: 123 } );',
200 'styles' => '/* pref-animate=off */ .mw-icon { transition: none; }',
201 'group' => 'private',
204 $links = $method->invokeArgs( $out, $args );
205 $actualHtml = strval( $links );
206 $this->assertEquals( $expectedHtml, $actualHtml );
210 * @dataProvider provideVaryHeaders
211 * @covers OutputPage::addVaryHeader
212 * @covers OutputPage::getVaryHeader
213 * @covers OutputPage::getKeyHeader
215 public function testVaryHeaders( $calls, $vary, $key ) {
216 // get rid of default Vary fields
217 $outputPage = $this->getMockBuilder( 'OutputPage' )
218 ->setConstructorArgs( [ new RequestContext() ] )
219 ->setMethods( [ 'getCacheVaryCookies' ] )
221 $outputPage->expects( $this->any() )
222 ->method( 'getCacheVaryCookies' )
223 ->will( $this->returnValue( [] ) );
224 TestingAccessWrapper
::newFromObject( $outputPage )->mVaryHeader
= [];
226 foreach ( $calls as $call ) {
227 call_user_func_array( [ $outputPage, 'addVaryHeader' ], $call );
229 $this->assertEquals( $vary, $outputPage->getVaryHeader(), 'Vary:' );
230 $this->assertEquals( $key, $outputPage->getKeyHeader(), 'Key:' );
233 public function provideVaryHeaders() {
234 // note: getKeyHeader() automatically adds Vary: Cookie
243 [ // non-unique headers
246 [ 'Accept-Language' ],
249 'Vary: Cookie, Accept-Language',
250 'Key: Cookie,Accept-Language',
252 [ // two headers with single options
254 [ 'Cookie', [ 'param=phpsessid' ] ],
255 [ 'Accept-Language', [ 'substr=en' ] ],
257 'Vary: Cookie, Accept-Language',
258 'Key: Cookie;param=phpsessid,Accept-Language;substr=en',
260 [ // one header with multiple options
262 [ 'Cookie', [ 'param=phpsessid', 'param=userId' ] ],
265 'Key: Cookie;param=phpsessid;param=userId',
267 [ // Duplicate option
269 [ 'Cookie', [ 'param=phpsessid' ] ],
270 [ 'Cookie', [ 'param=phpsessid' ] ],
271 [ 'Accept-Language', [ 'substr=en', 'substr=en' ] ],
273 'Vary: Cookie, Accept-Language',
274 'Key: Cookie;param=phpsessid,Accept-Language;substr=en',
276 [ // Same header, different options
278 [ 'Cookie', [ 'param=phpsessid' ] ],
279 [ 'Cookie', [ 'param=userId' ] ],
282 'Key: Cookie;param=phpsessid;param=userId',
288 * @covers OutputPage::haveCacheVaryCookies
290 function testHaveCacheVaryCookies() {
291 $request = new FauxRequest();
292 $context = new RequestContext();
293 $context->setRequest( $request );
294 $outputPage = new OutputPage( $context );
296 // No cookies are set.
297 $this->assertFalse( $outputPage->haveCacheVaryCookies() );
299 // 'Token' is present but empty, so it shouldn't count.
300 $request->setCookie( 'Token', '' );
301 $this->assertFalse( $outputPage->haveCacheVaryCookies() );
303 // 'Token' present and nonempty.
304 $request->setCookie( 'Token', '123' );
305 $this->assertTrue( $outputPage->haveCacheVaryCookies() );
309 * @covers OutputPage::addCategoryLinks
310 * @covers OutputPage::getCategories
312 function testGetCategories() {
313 $fakeResultWrapper = new FakeResultWrapper( [
316 'page_title' => 'Test'
319 'page_title' => 'Test2'
322 $outputPage = $this->getMockBuilder( 'OutputPage' )
323 ->setConstructorArgs( [ new RequestContext() ] )
324 ->setMethods( [ 'addCategoryLinksToLBAndGetResult' ] )
326 $outputPage->expects( $this->any() )
327 ->method( 'addCategoryLinksToLBAndGetResult' )
328 ->will( $this->returnValue( $fakeResultWrapper ) );
330 $outputPage->addCategoryLinks( [
334 $this->assertEquals( [ 0 => 'Test', '1' => 'Test2' ], $outputPage->getCategories() );
335 $this->assertEquals( [ 0 => 'Test2' ], $outputPage->getCategories( 'normal' ) );
336 $this->assertEquals( [ 0 => 'Test' ], $outputPage->getCategories( 'hidden' ) );
341 * MessageBlobStore that doesn't do anything
343 class NullMessageBlobStore
extends MessageBlobStore
{
344 public function get( ResourceLoader
$resourceLoader, $modules, $lang ) {
348 public function insertMessageBlob( $name, ResourceLoaderModule
$module, $lang ) {
352 public function updateModule( $name, ResourceLoaderModule
$module, $lang ) {
355 public function updateMessage( $key ) {
358 public function clear() {