2 var util
= require( 'mediawiki.util' ),
3 // Based on IPTest.php > testisIPv4
5 [ false, false, 'Boolean false is not an IP' ],
6 [ false, true, 'Boolean true is not an IP' ],
7 [ false, '', 'Empty string is not an IP' ],
8 [ false, 'abc', '"abc" is not an IP' ],
9 [ false, ':', 'Colon is not an IP' ],
10 [ false, '124.24.52', 'IPv4 not enough quads' ],
11 [ false, '24.324.52.13', 'IPv4 out of range' ],
12 [ false, '.24.52.13', 'IPv4 starts with period' ],
14 [ true, '124.24.52.13', '124.24.52.134 is a valid IP' ],
15 [ true, '1.24.52.13', '1.24.52.13 is a valid IP' ],
16 [ false, '74.24.52.13/20', 'IPv4 ranges are not recognized as valid IPs' ]
19 // Based on IPTest.php > testisIPv6
21 [ false, ':fc:100::', 'IPv6 starting with lone ":"' ],
22 [ false, 'fc:100:::', 'IPv6 ending with a ":::"' ],
23 [ false, 'fc:300', 'IPv6 with only 2 words' ],
24 [ false, 'fc:100:300', 'IPv6 with only 3 words' ],
26 [ false, 'fc:100:a:d:1:e:ac:0::', 'IPv6 with 8 words ending with "::"' ],
27 [ false, 'fc:100:a:d:1:e:ac:0:1::', 'IPv6 with 9 words ending with "::"' ],
30 [ false, '::0:', 'IPv6 ending in a lone ":"' ],
32 [ true, '::', 'IPv6 zero address' ],
34 [ false, '::fc:100:a:d:1:e:ac:0', 'IPv6 with "::" and 8 words' ],
35 [ false, '::fc:100:a:d:1:e:ac:0:1', 'IPv6 with 9 words' ],
37 [ false, ':fc::100', 'IPv6 starting with lone ":"' ],
38 [ false, 'fc::100:', 'IPv6 ending with lone ":"' ],
39 [ false, 'fc:::100', 'IPv6 with ":::" in the middle' ],
41 [ true, 'fc::100', 'IPv6 with "::" and 2 words' ],
42 [ true, 'fc::100:a', 'IPv6 with "::" and 3 words' ],
43 [ true, 'fc::100:a:d', 'IPv6 with "::" and 4 words' ],
44 [ true, 'fc::100:a:d:1', 'IPv6 with "::" and 5 words' ],
45 [ true, 'fc::100:a:d:1:e', 'IPv6 with "::" and 6 words' ],
46 [ true, 'fc::100:a:d:1:e:ac', 'IPv6 with "::" and 7 words' ],
47 [ true, '2001::df', 'IPv6 with "::" and 2 words' ],
48 [ true, '2001:5c0:1400:a::df', 'IPv6 with "::" and 5 words' ],
49 [ true, '2001:5c0:1400:a::df:2', 'IPv6 with "::" and 6 words' ],
51 [ false, 'fc::100:a:d:1:e:ac:0', 'IPv6 with "::" and 8 words' ],
52 [ false, 'fc::100:a:d:1:e:ac:0:1', 'IPv6 with 9 words' ]
55 Array
.prototype.push
.apply( IPV6_CASES
,
62 'fc:100:a:d:1:e:ac::',
70 '::fc:100:a:d:1:e:ac',
72 ].map( function ( el
) {
73 return [ true, el
, el
+ ' is a valid IP' ];
77 QUnit
.module( 'mediawiki.util', QUnit
.newMwEnvironment( {
79 $.fn
.updateTooltipAccessKeys
.setTestMode( true );
81 teardown: function () {
82 $.fn
.updateTooltipAccessKeys
.setTestMode( false );
85 // Used by accessKeyLabel in test for addPortletLink
91 QUnit
.test( 'rawurlencode', function ( assert
) {
92 assert
.equal( util
.rawurlencode( 'Test:A & B/Here' ), 'Test%3AA%20%26%20B%2FHere' );
95 QUnit
.test( 'escapeId', function ( assert
) {
96 mw
.config
.set( 'wgFragmentMode', [ 'legacy' ] );
113 'Test:A & B/Here': 'Test:A_.26_B.2FHere',
114 'A&B&C&D&E': 'A.26B.26amp.3BC.26amp.3Bamp.3BD.26amp.3Bamp.3Bamp.3BE'
115 }, function ( input
, output
) {
116 assert
.equal( util
.escapeId( input
), output
);
120 QUnit
.test( 'escapeIdForAttribute', function ( assert
) {
121 // Test cases are kept in sync with SanitizerTest.php
122 var text
= 'foo тест_#%!\'()[]:<>',
123 legacyEncoded
= 'foo_.D1.82.D0.B5.D1.81.D1.82_.23.25.21.27.28.29.5B.5D:.3C.3E',
124 html5Encoded
= 'foo_тест_#%!\'()[]:<>',
125 // Settings: this is $wgFragmentMode
126 legacy
= [ 'legacy' ],
127 legacyNew
= [ 'legacy', 'html5' ],
128 newLegacy
= [ 'html5', 'legacy' ],
129 allNew
= [ 'html5' ];
131 // Test cases are kept in sync with SanitizerTest.php
133 // Pure legacy: how MW worked before 2017
134 [ legacy
, text
, legacyEncoded
],
135 // Transition to a new world: legacy links with HTML5 fallback
136 [ legacyNew
, text
, legacyEncoded
],
137 // New world: HTML5 links, legacy fallbacks
138 [ newLegacy
, text
, html5Encoded
],
139 // Distant future: no legacy fallbacks
140 [ allNew
, text
, html5Encoded
]
141 ].forEach( function ( testCase
) {
142 mw
.config
.set( 'wgFragmentMode', testCase
[ 0 ] );
144 assert
.equal( util
.escapeIdForAttribute( testCase
[ 1 ] ), testCase
[ 2 ] );
148 QUnit
.test( 'escapeIdForLink', function ( assert
) {
149 // Test cases are kept in sync with SanitizerTest.php
150 var text
= 'foo тест_#%!\'()[]:<>',
151 legacyEncoded
= 'foo_.D1.82.D0.B5.D1.81.D1.82_.23.25.21.27.28.29.5B.5D:.3C.3E',
152 html5Encoded
= 'foo_тест_#%!\'()[]:<>',
153 // Settings: this is wgFragmentMode
154 legacy
= [ 'legacy' ],
155 legacyNew
= [ 'legacy', 'html5' ],
156 newLegacy
= [ 'html5', 'legacy' ],
157 allNew
= [ 'html5' ];
160 // Pure legacy: how MW worked before 2017
161 [ legacy
, text
, legacyEncoded
],
162 // Transition to a new world: legacy links with HTML5 fallback
163 [ legacyNew
, text
, legacyEncoded
],
164 // New world: HTML5 links, legacy fallbacks
165 [ newLegacy
, text
, html5Encoded
],
166 // Distant future: no legacy fallbacks
167 [ allNew
, text
, html5Encoded
]
168 ].forEach( function ( testCase
) {
169 mw
.config
.set( 'wgFragmentMode', testCase
[ 0 ] );
171 assert
.equal( util
.escapeIdForLink( testCase
[ 1 ] ), testCase
[ 2 ] );
175 QUnit
.test( 'wikiUrlencode', function ( assert
) {
176 assert
.equal( util
.wikiUrlencode( 'Test:A & B/Here' ), 'Test:A_%26_B/Here' );
177 // See also wfUrlencodeTest.php#provideURLS
183 ';@$-_.!*': ';@$-_.!*',
189 }, function ( input
, output
) {
190 assert
.equal( util
.wikiUrlencode( input
), output
);
194 QUnit
.test( 'getUrl', function ( assert
) {
197 wgScript
: '/w/index.php',
198 wgArticlePath
: '/wiki/$1',
202 href
= util
.getUrl( 'Sandbox' );
203 assert
.equal( href
, '/wiki/Sandbox', 'simple title' );
205 href
= util
.getUrl( 'Foo:Sandbox? 5+5=10! (test)/sub ' );
206 assert
.equal( href
, '/wiki/Foo:Sandbox%3F_5%2B5%3D10!_(test)/sub_', 'complex title' );
209 href
= util
.getUrl( 'My$$test$$$$$title' );
210 assert
.equal( href
, '/wiki/My$$test$$$$$title', 'title with multiple consecutive dollar signs' );
212 href
= util
.getUrl();
213 assert
.equal( href
, '/wiki/Foobar', 'default title' );
215 href
= util
.getUrl( null, { action
: 'edit' } );
216 assert
.equal( href
, '/w/index.php?title=Foobar&action=edit', 'default title with query string' );
218 href
= util
.getUrl( 'Sandbox', { action
: 'edit' } );
219 assert
.equal( href
, '/w/index.php?title=Sandbox&action=edit', 'simple title with query string' );
222 href
= util
.getUrl( 'Foo:Sandbox#Fragment', { action
: 'edit' } );
223 assert
.equal( href
, '/w/index.php?title=Foo:Sandbox&action=edit#Fragment', 'namespaced title with query string and fragment' );
225 href
= util
.getUrl( 'Sandbox#', { action
: 'edit' } );
226 assert
.equal( href
, '/w/index.php?title=Sandbox&action=edit', 'title with query string and empty fragment' );
228 href
= util
.getUrl( 'Sandbox', {} );
229 assert
.equal( href
, '/wiki/Sandbox', 'title with empty query string' );
231 href
= util
.getUrl( '#Fragment' );
232 assert
.equal( href
, '/wiki/#Fragment', 'empty title with fragment' );
234 href
= util
.getUrl( '#Fragment', { action
: 'edit' } );
235 assert
.equal( href
, '/w/index.php?action=edit#Fragment', 'empty title with query string and fragment' );
237 mw
.config
.set( 'wgFragmentMode', [ 'legacy' ] );
238 href
= util
.getUrl( 'Foo:Sandbox \xC4#Fragment \xC4', { action
: 'edit' } );
239 assert
.equal( href
, '/w/index.php?title=Foo:Sandbox_%C3%84&action=edit#Fragment_.C3.84', 'title with query string, fragment, and special characters' );
241 mw
.config
.set( 'wgFragmentMode', [ 'html5' ] );
242 href
= util
.getUrl( 'Foo:Sandbox \xC4#Fragment \xC4', { action
: 'edit' } );
243 assert
.equal( href
, '/w/index.php?title=Foo:Sandbox_%C3%84&action=edit#Fragment_Ä', 'title with query string, fragment, and special characters' );
245 href
= util
.getUrl( 'Foo:%23#Fragment', { action
: 'edit' } );
246 assert
.equal( href
, '/w/index.php?title=Foo:%2523&action=edit#Fragment', 'title containing %23 (#), fragment, and a query string' );
248 mw
.config
.set( 'wgFragmentMode', [ 'legacy' ] );
249 href
= util
.getUrl( '#+&=:;@$-_.!*/[]<>\'§', { action
: 'edit' } );
250 assert
.equal( href
, '/w/index.php?action=edit#.2B.26.3D:.3B.40.24-_..21.2A.2F.5B.5D.3C.3E.27.C2.A7', 'fragment with various characters' );
252 mw
.config
.set( 'wgFragmentMode', [ 'html5' ] );
253 href
= util
.getUrl( '#+&=:;@$-_.!*/[]<>\'§', { action
: 'edit' } );
254 assert
.equal( href
, '/w/index.php?action=edit#+&=:;@$-_.!*/[]<>\'§', 'fragment with various characters' );
257 QUnit
.test( 'wikiScript', function ( assert
) {
259 // customized wgScript for T41103
260 wgScript
: '/w/i.php',
261 // customized wgLoadScript for T41103
262 wgLoadScript
: '/w/l.php',
266 assert
.equal( util
.wikiScript(), mw
.config
.get( 'wgScript' ),
267 'wikiScript() returns wgScript'
269 assert
.equal( util
.wikiScript( 'index' ), mw
.config
.get( 'wgScript' ),
270 'wikiScript( index ) returns wgScript'
272 assert
.equal( util
.wikiScript( 'load' ), mw
.config
.get( 'wgLoadScript' ),
273 'wikiScript( load ) returns wgLoadScript'
275 assert
.equal( util
.wikiScript( 'api' ), '/w/api.php', 'API path' );
278 QUnit
.test( 'addCSS', function ( assert
) {
280 $el
= $( '<div>' ).attr( 'id', 'mw-addcsstest' ).appendTo( '#qunit-fixture' );
282 style
= util
.addCSS( '#mw-addcsstest { visibility: hidden; }' );
283 assert
.equal( typeof style
, 'object', 'addCSS returned an object' );
284 assert
.strictEqual( style
.disabled
, false, 'property "disabled" is available and set to false' );
286 assert
.equal( $el
.css( 'visibility' ), 'hidden', 'Added style properties are in effect' );
289 $( style
.ownerNode
).remove();
292 QUnit
.test( 'getParamValue', function ( assert
) {
295 url
= 'http://example.org/?foo=wrong&foo=right#&foo=bad';
296 assert
.equal( util
.getParamValue( 'foo', url
), 'right', 'Use latest one, ignore hash' );
297 assert
.strictEqual( util
.getParamValue( 'bar', url
), null, 'Return null when not found' );
299 url
= 'http://example.org/#&foo=bad';
300 assert
.strictEqual( util
.getParamValue( 'foo', url
), null, 'Ignore hash if param is not in querystring but in hash (T29427)' );
302 url
= 'example.org?' + $.param( { TEST
: 'a b+c' } );
303 assert
.strictEqual( util
.getParamValue( 'TEST', url
), 'a b+c', 'T32441: getParamValue must understand "+" encoding of space' );
305 url
= 'example.org?' + $.param( { TEST
: 'a b+c d' } ); // check for sloppy code from r95332 :)
306 assert
.strictEqual( util
.getParamValue( 'TEST', url
), 'a b+c d', 'T32441: getParamValue must understand "+" encoding of space (multiple spaces)' );
309 QUnit
.test( '$content', function ( assert
) {
310 assert
.ok( util
.$content
instanceof jQuery
, 'mw.util.$content instance of jQuery' );
311 assert
.strictEqual( util
.$content
.length
, 1, 'mw.util.$content must have length of 1' );
315 * Portlet names are prefixed with 'p-test' to avoid conflict with core
316 * when running the test suite under a wiki page.
317 * Previously, test elements where invisible to the selector since only
318 * one element can have a given id.
320 QUnit
.test( 'addPortletLink', function ( assert
) {
321 var pTestTb
, pCustom
, vectorTabs
, tbRL
, cuQuux
, $cuQuux
, tbMW
, $tbMW
, tbRLDM
, caFoo
,
322 addedAfter
, tbRLDMnonexistentid
, tbRLDMemptyjquery
;
325 '<div class="portlet" id="p-test-tb">' +
327 '<ul class="body"></ul>' +
330 '<div class="portlet" id="p-test-custom">' +
332 '<ul class="body">' +
333 '<li id="c-foo"><a href="#">Foo</a></li>' +
334 '<li id="c-barmenu">' +
336 '<li id="c-bar-baz"><a href="#">Baz</a></a>' +
342 '<div id="p-test-views" class="vectorTabs">' +
347 $( '#qunit-fixture' ).append( pTestTb
, pCustom
, vectorTabs
);
349 tbRL
= util
.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/ResourceLoader',
350 'ResourceLoader', 't-rl', 'More info about ResourceLoader on MediaWiki.org ', 'l'
353 assert
.ok( tbRL
&& tbRL
.nodeType
, 'addPortletLink returns a DOM Node' );
355 tbMW
= util
.addPortletLink( 'p-test-tb', '//mediawiki.org/',
356 'MediaWiki.org', 't-mworg', 'Go to MediaWiki.org', 'm', tbRL
);
364 'Validate attributes of created element'
368 $tbMW
.find( 'a' ).getAttrs(),
370 href
: '//mediawiki.org/',
371 title
: 'Go to MediaWiki.org [test-m]',
374 'Validate attributes of anchor tag in created element'
377 assert
.equal( $tbMW
.closest( '.portlet' ).attr( 'id' ), 'p-test-tb', 'Link was inserted within correct portlet' );
378 assert
.strictEqual( $tbMW
.next()[ 0 ], tbRL
, 'Link is in the correct position (nextnode as Node object)' );
380 cuQuux
= util
.addPortletLink( 'p-test-custom', '#', 'Quux', null, 'Example [shift-x]', 'q' );
381 $cuQuux
= $( cuQuux
);
383 assert
.equal( $cuQuux
.find( 'a' ).attr( 'title' ), 'Example [test-q]', 'Existing accesskey is stripped and updated' );
386 $( '#p-test-custom #c-barmenu ul li' ).length
,
388 'addPortletLink did not add the item to all <ul> elements in the portlet (T37082)'
391 tbRLDM
= util
.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
392 'Default modules', 't-rldm', 'List of all default modules ', 'd', '#t-rl' );
394 assert
.strictEqual( $( tbRLDM
).next()[ 0 ], tbRL
, 'Link is in the correct position (CSS selector as nextnode)' );
396 caFoo
= util
.addPortletLink( 'p-test-views', '#', 'Foo' );
398 assert
.strictEqual( $tbMW
.find( 'span' ).length
, 0, 'No <span> element should be added for porlets without vectorTabs class.' );
399 assert
.strictEqual( $( caFoo
).find( 'span' ).length
, 1, 'A <span> element should be added for porlets with vectorTabs class.' );
401 addedAfter
= util
.addPortletLink( 'p-test-tb', '#', 'After foo', 'post-foo', 'After foo', null, $( tbRL
) );
402 assert
.strictEqual( $( addedAfter
).next()[ 0 ], tbRL
, 'Link is in the correct position (jQuery object as nextnode)' );
404 // test case - nonexistent id as next node
405 tbRLDMnonexistentid
= util
.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
406 'Default modules', 't-rldm-nonexistent', 'List of all default modules ', 'd', '#t-rl-nonexistent' );
408 assert
.equal( tbRLDMnonexistentid
, $( '#p-test-tb li:last' )[ 0 ], 'Fallback to adding at the end (nextnode non-matching CSS selector)' );
410 // test case - empty jquery object as next node
411 tbRLDMemptyjquery
= util
.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
412 'Default modules', 't-rldm-empty-jquery', 'List of all default modules ', 'd', $( '#t-rl-nonexistent' ) );
414 assert
.equal( tbRLDMemptyjquery
, $( '#p-test-tb li:last' )[ 0 ], 'Fallback to adding at the end (nextnode as empty jQuery object)' );
417 QUnit
.test( 'validateEmail', function ( assert
) {
418 assert
.strictEqual( util
.validateEmail( '' ), null, 'Should return null for empty string ' );
419 assert
.strictEqual( util
.validateEmail( 'user@localhost' ), true, 'Return true for a valid e-mail address' );
421 // testEmailWithCommasAreInvalids
422 assert
.strictEqual( util
.validateEmail( 'user,foo@example.org' ), false, 'Emails with commas are invalid' );
423 assert
.strictEqual( util
.validateEmail( 'userfoo@ex,ample.org' ), false, 'Emails with commas are invalid' );
425 // testEmailWithHyphens
426 assert
.strictEqual( util
.validateEmail( 'user-foo@example.org' ), true, 'Emails may contain a hyphen' );
427 assert
.strictEqual( util
.validateEmail( 'userfoo@ex-ample.org' ), true, 'Emails may contain a hyphen' );
430 QUnit
.test( 'isIPv6Address', function ( assert
) {
431 IPV6_CASES
.forEach( function ( ipCase
) {
432 assert
.strictEqual( util
.isIPv6Address( ipCase
[ 1 ] ), ipCase
[ 0 ], ipCase
[ 2 ] );
436 QUnit
.test( 'isIPv4Address', function ( assert
) {
437 IPV4_CASES
.forEach( function ( ipCase
) {
438 assert
.strictEqual( util
.isIPv4Address( ipCase
[ 1 ] ), ipCase
[ 0 ], ipCase
[ 2 ] );
442 QUnit
.test( 'isIPAddress', function ( assert
) {
443 IPV4_CASES
.forEach( function ( ipCase
) {
444 assert
.strictEqual( util
.isIPv4Address( ipCase
[ 1 ] ), ipCase
[ 0 ], ipCase
[ 2 ] );
447 IPV6_CASES
.forEach( function ( ipCase
) {
448 assert
.strictEqual( util
.isIPv6Address( ipCase
[ 1 ] ), ipCase
[ 0 ], ipCase
[ 2 ] );
451 }( mediaWiki
, jQuery
) );