3 use Wikimedia\TestingAccessWrapper
;
6 * @group ResourceLoader
8 class ResourceLoaderClientHtmlTest
extends PHPUnit\Framework\TestCase
{
10 use MediaWikiCoversValidator
;
12 protected static function expandVariables( $text ) {
13 return strtr( $text, [
14 '{blankVer}' => ResourceLoaderTestCase
::BLANK_VERSION
18 protected static function makeContext( $extraQuery = [] ) {
19 $conf = new HashConfig( [
20 'ResourceLoaderSources' => [],
21 'ResourceModuleSkinStyles' => [],
22 'ResourceModules' => [],
23 'EnableJavaScriptTest' => false,
24 'ResourceLoaderDebug' => false,
25 'LoadScript' => '/w/load.php',
27 return new ResourceLoaderContext(
28 new ResourceLoader( $conf ),
29 new FauxRequest( array_merge( [
33 'target' => 'phpunit',
38 protected static function makeModule( array $options = [] ) {
39 return new ResourceLoaderTestModule( $options );
42 protected static function makeSampleModules() {
45 'test.private' => [ 'group' => 'private' ],
46 'test.shouldembed.empty' => [ 'shouldEmbed' => true, 'isKnownEmpty' => true ],
47 'test.shouldembed' => [ 'shouldEmbed' => true ],
49 'test.styles.pure' => [ 'type' => ResourceLoaderModule
::LOAD_STYLES
],
50 'test.styles.mixed' => [],
51 'test.styles.noscript' => [
52 'type' => ResourceLoaderModule
::LOAD_STYLES
,
53 'group' => 'noscript',
55 'test.styles.user' => [
56 'type' => ResourceLoaderModule
::LOAD_STYLES
,
59 'test.styles.user.empty' => [
60 'type' => ResourceLoaderModule
::LOAD_STYLES
,
62 'isKnownEmpty' => true,
64 'test.styles.private' => [
65 'type' => ResourceLoaderModule
::LOAD_STYLES
,
67 'styles' => '.private{}',
69 'test.styles.shouldembed' => [
70 'type' => ResourceLoaderModule
::LOAD_STYLES
,
71 'shouldEmbed' => true,
72 'styles' => '.shouldembed{}',
76 'test.scripts.user' => [ 'group' => 'user' ],
77 'test.scripts.user.empty' => [ 'group' => 'user', 'isKnownEmpty' => true ],
78 'test.scripts.raw' => [ 'isRaw' => true ],
79 'test.scripts.shouldembed' => [ 'shouldEmbed' => true ],
81 'test.ordering.a' => [ 'shouldEmbed' => false ],
82 'test.ordering.b' => [ 'shouldEmbed' => false ],
83 'test.ordering.c' => [ 'shouldEmbed' => true, 'styles' => '.orderingC{}' ],
84 'test.ordering.d' => [ 'shouldEmbed' => true, 'styles' => '.orderingD{}' ],
85 'test.ordering.e' => [ 'shouldEmbed' => false ],
87 return array_map( function ( $options ) {
88 return self
::makeModule( $options );
93 * @covers ResourceLoaderClientHtml::getDocumentAttributes
95 public function testGetDocumentAttributes() {
96 $client = new ResourceLoaderClientHtml( self
::makeContext() );
97 $this->assertInternalType( 'array', $client->getDocumentAttributes() );
101 * @covers ResourceLoaderClientHtml::__construct
102 * @covers ResourceLoaderClientHtml::setModules
103 * @covers ResourceLoaderClientHtml::setModuleStyles
104 * @covers ResourceLoaderClientHtml::setModuleScripts
105 * @covers ResourceLoaderClientHtml::getData
106 * @covers ResourceLoaderClientHtml::getContext
108 public function testGetData() {
109 $context = self
::makeContext();
110 $context->getResourceLoader()->register( self
::makeSampleModules() );
112 $client = new ResourceLoaderClientHtml( $context );
113 $client->setModules( [
116 'test.shouldembed.empty',
120 $client->setModuleStyles( [
122 'test.styles.user.empty',
123 'test.styles.private',
125 'test.styles.shouldembed',
126 'test.unregistered.styles',
128 $client->setModuleScripts( [
131 'test.scripts.user.empty',
132 'test.scripts.shouldembed',
133 'test.unregistered.scripts',
138 'test.private' => 'loading',
139 'test.shouldembed.empty' => 'ready',
140 'test.shouldembed' => 'loading',
141 'test.styles.pure' => 'ready',
142 'test.styles.user.empty' => 'ready',
143 'test.styles.private' => 'ready',
144 'test.styles.shouldembed' => 'ready',
145 'test.scripts' => 'loading',
146 'test.scripts.user' => 'loading',
147 'test.scripts.user.empty' => 'ready',
148 'test.scripts.shouldembed' => 'loading',
159 'test.scripts.shouldembed',
162 'styles' => [ 'test.styles.private', 'test.styles.shouldembed' ],
170 $access = TestingAccessWrapper
::newFromObject( $client );
171 $this->assertEquals( $expected, $access->getData() );
175 * @covers ResourceLoaderClientHtml::setConfig
176 * @covers ResourceLoaderClientHtml::setExemptStates
177 * @covers ResourceLoaderClientHtml::getHeadHtml
178 * @covers ResourceLoaderClientHtml::getLoad
179 * @covers ResourceLoader::makeLoaderStateScript
181 public function testGetHeadHtml() {
182 $context = self
::makeContext();
183 $context->getResourceLoader()->register( self
::makeSampleModules() );
185 $client = new ResourceLoaderClientHtml( $context );
186 $client->setConfig( [ 'key' => 'value' ] );
187 $client->setModules( [
191 $client->setModuleStyles( [
193 'test.styles.private',
195 $client->setModuleScripts( [
198 $client->setExemptStates( [
199 'test.exempt' => 'ready',
202 // phpcs:disable Generic.Files.LineLength
203 $expected = '<script>document.documentElement.className = document.documentElement.className.replace( /(^|\s)client-nojs(\s|$)/, "$1client-js$2" );</script>' . "\n"
204 . '<script>(window.RLQ=window.RLQ||[]).push(function(){'
205 . 'mw.config.set({"key":"value"});'
206 . 'mw.loader.state({"test.exempt":"ready","test.private":"loading","test.styles.pure":"ready","test.styles.private":"ready","test.scripts":"loading"});'
207 . 'mw.loader.implement("test.private@{blankVer}",function($,jQuery,require,module){},{"css":[]});'
208 . 'mw.loader.load(["test"]);'
209 . 'mw.loader.load("/w/load.php?debug=false\u0026lang=nl\u0026modules=test.scripts\u0026only=scripts\u0026skin=fallback");'
210 . '});</script>' . "\n"
211 . '<link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.styles.pure&only=styles&skin=fallback"/>' . "\n"
212 . '<style>.private{}</style>' . "\n"
213 . '<script async="" src="/w/load.php?debug=false&lang=nl&modules=startup&only=scripts&skin=fallback"></script>';
215 $expected = self
::expandVariables( $expected );
217 $this->assertEquals( $expected, $client->getHeadHtml() );
221 * Confirm that 'target' is passed down to the startup module's load url.
223 * @covers ResourceLoaderClientHtml::getHeadHtml
225 public function testGetHeadHtmlWithTarget() {
226 $client = new ResourceLoaderClientHtml(
228 [ 'target' => 'example' ]
231 // phpcs:disable Generic.Files.LineLength
232 $expected = '<script>document.documentElement.className = document.documentElement.className.replace( /(^|\s)client-nojs(\s|$)/, "$1client-js$2" );</script>' . "\n"
233 . '<script async="" src="/w/load.php?debug=false&lang=nl&modules=startup&only=scripts&skin=fallback&target=example"></script>';
236 $this->assertEquals( $expected, $client->getHeadHtml() );
240 * Confirm that a null 'target' is the same as no target.
242 * @covers ResourceLoaderClientHtml::getHeadHtml
244 public function testGetHeadHtmlWithNullTarget() {
245 $client = new ResourceLoaderClientHtml(
250 // phpcs:disable Generic.Files.LineLength
251 $expected = '<script>document.documentElement.className = document.documentElement.className.replace( /(^|\s)client-nojs(\s|$)/, "$1client-js$2" );</script>' . "\n"
252 . '<script async="" src="/w/load.php?debug=false&lang=nl&modules=startup&only=scripts&skin=fallback"></script>';
255 $this->assertEquals( $expected, $client->getHeadHtml() );
259 * @covers ResourceLoaderClientHtml::getBodyHtml
260 * @covers ResourceLoaderClientHtml::getLoad
262 public function testGetBodyHtml() {
263 $context = self
::makeContext();
264 $context->getResourceLoader()->register( self
::makeSampleModules() );
266 $client = new ResourceLoaderClientHtml( $context );
267 $client->setConfig( [ 'key' => 'value' ] );
268 $client->setModules( [
270 'test.private.bottom',
272 $client->setModuleScripts( [
277 $expected = self
::expandVariables( $expected );
279 $this->assertEquals( $expected, $client->getBodyHtml() );
282 public static function provideMakeLoad() {
283 // phpcs:disable Generic.Files.LineLength
287 'modules' => [ 'test.unknown' ],
288 'only' => ResourceLoaderModule
::TYPE_STYLES
,
293 'modules' => [ 'test.styles.private' ],
294 'only' => ResourceLoaderModule
::TYPE_STYLES
,
295 'output' => '<style>.private{}</style>',
299 'modules' => [ 'test.private' ],
300 'only' => ResourceLoaderModule
::TYPE_COMBINED
,
301 'output' => '<script>(window.RLQ=window.RLQ||[]).push(function(){mw.loader.implement("test.private@{blankVer}",function($,jQuery,require,module){},{"css":[]});});</script>',
305 // Eg. startup module
306 'modules' => [ 'test.scripts.raw' ],
307 'only' => ResourceLoaderModule
::TYPE_SCRIPTS
,
308 'output' => '<script async="" src="/w/load.php?debug=false&lang=nl&modules=test.scripts.raw&only=scripts&skin=fallback"></script>',
311 'context' => [ 'sync' => true ],
312 'modules' => [ 'test.scripts.raw' ],
313 'only' => ResourceLoaderModule
::TYPE_SCRIPTS
,
314 'output' => '<script src="/w/load.php?debug=false&lang=nl&modules=test.scripts.raw&only=scripts&skin=fallback&sync=1"></script>',
318 'modules' => [ 'test.scripts.user' ],
319 'only' => ResourceLoaderModule
::TYPE_SCRIPTS
,
320 'output' => '<script>(window.RLQ=window.RLQ||[]).push(function(){mw.loader.load("/w/load.php?debug=false\u0026lang=nl\u0026modules=test.scripts.user\u0026only=scripts\u0026skin=fallback\u0026user=Example\u0026version=0a56zyi");});</script>',
323 'context' => [ 'debug' => true ],
324 'modules' => [ 'test.styles.pure', 'test.styles.mixed' ],
325 'only' => ResourceLoaderModule
::TYPE_STYLES
,
326 'output' => '<link rel="stylesheet" href="/w/load.php?debug=true&lang=nl&modules=test.styles.mixed&only=styles&skin=fallback"/>' . "\n"
327 . '<link rel="stylesheet" href="/w/load.php?debug=true&lang=nl&modules=test.styles.pure&only=styles&skin=fallback"/>',
330 'context' => [ 'debug' => false ],
331 'modules' => [ 'test.styles.pure', 'test.styles.mixed' ],
332 'only' => ResourceLoaderModule
::TYPE_STYLES
,
333 'output' => '<link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.styles.mixed%2Cpure&only=styles&skin=fallback"/>',
337 'modules' => [ 'test.styles.noscript' ],
338 'only' => ResourceLoaderModule
::TYPE_STYLES
,
339 'output' => '<noscript><link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.styles.noscript&only=styles&skin=fallback"/></noscript>',
343 'modules' => [ 'test.shouldembed' ],
344 'only' => ResourceLoaderModule
::TYPE_COMBINED
,
345 'output' => '<script>(window.RLQ=window.RLQ||[]).push(function(){mw.loader.implement("test.shouldembed@09p30q0",function($,jQuery,require,module){},{"css":[]});});</script>',
349 'modules' => [ 'test.styles.shouldembed' ],
350 'only' => ResourceLoaderModule
::TYPE_STYLES
,
351 'output' => '<style>.shouldembed{}</style>',
355 'modules' => [ 'test.scripts.shouldembed' ],
356 'only' => ResourceLoaderModule
::TYPE_SCRIPTS
,
357 'output' => '<script>(window.RLQ=window.RLQ||[]).push(function(){mw.loader.state({"test.scripts.shouldembed":"ready"});});</script>',
361 'modules' => [ 'test', 'test.shouldembed' ],
362 'only' => ResourceLoaderModule
::TYPE_COMBINED
,
363 'output' => '<script>(window.RLQ=window.RLQ||[]).push(function(){mw.loader.load("/w/load.php?debug=false\u0026lang=nl\u0026modules=test\u0026skin=fallback");mw.loader.implement("test.shouldembed@09p30q0",function($,jQuery,require,module){},{"css":[]});});</script>',
367 'modules' => [ 'test.styles.pure', 'test.styles.shouldembed' ],
368 'only' => ResourceLoaderModule
::TYPE_STYLES
,
370 '<link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.styles.pure&only=styles&skin=fallback"/>' . "\n"
371 . '<style>.shouldembed{}</style>'
375 'modules' => [ 'test.ordering.a', 'test.ordering.e', 'test.ordering.b', 'test.ordering.d', 'test.ordering.c' ],
376 'only' => ResourceLoaderModule
::TYPE_STYLES
,
378 '<link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.ordering.a%2Cb&only=styles&skin=fallback"/>' . "\n"
379 . '<style>.orderingC{}.orderingD{}</style>' . "\n"
380 . '<link rel="stylesheet" href="/w/load.php?debug=false&lang=nl&modules=test.ordering.e&only=styles&skin=fallback"/>'
387 * @dataProvider provideMakeLoad
388 * @covers ResourceLoaderClientHtml::makeLoad
389 * @covers ResourceLoaderClientHtml::makeContext
390 * @covers ResourceLoader::makeModuleResponse
391 * @covers ResourceLoaderModule::getModuleContent
392 * @covers ResourceLoader::getCombinedVersion
393 * @covers ResourceLoader::createLoaderURL
394 * @covers ResourceLoader::createLoaderQuery
395 * @covers ResourceLoader::makeLoaderQuery
396 * @covers ResourceLoader::makeInlineScript
398 public function testMakeLoad( array $extraQuery, array $modules, $type, $expected ) {
399 $context = self
::makeContext( $extraQuery );
400 $context->getResourceLoader()->register( self
::makeSampleModules() );
401 $actual = ResourceLoaderClientHtml
::makeLoad( $context, $modules, $type, $extraQuery );
402 $expected = self
::expandVariables( $expected );
403 $this->assertEquals( $expected, (string)$actual );