Use mw.util.wikiScript( 'load' ) instead of mw.config.get( 'wgLoadScript' ) The module 'mediawiki.util' now gets the value from a private config variable. Change-Id: Iae4f4754f40999ba9d7c7161e9a37820c4b5931c
439 lines
16 KiB
JavaScript
439 lines
16 KiB
JavaScript
( function () {
|
||
var util = require( 'mediawiki.util' ),
|
||
// Based on IPTest.php > testisIPv4
|
||
IPV4_CASES = [
|
||
[ false, false, 'Boolean false is not an IP' ],
|
||
[ false, true, 'Boolean true is not an IP' ],
|
||
[ false, '', 'Empty string is not an IP' ],
|
||
[ false, 'abc', '"abc" is not an IP' ],
|
||
[ false, ':', 'Colon is not an IP' ],
|
||
[ false, '124.24.52', 'IPv4 not enough quads' ],
|
||
[ false, '24.324.52.13', 'IPv4 out of range' ],
|
||
[ false, '.24.52.13', 'IPv4 starts with period' ],
|
||
|
||
[ true, '124.24.52.13', '124.24.52.134 is a valid IP' ],
|
||
[ true, '1.24.52.13', '1.24.52.13 is a valid IP' ],
|
||
[ false, '74.24.52.13/20', 'IPv4 ranges are not recognized as valid IPs' ]
|
||
],
|
||
|
||
// Based on IPTest.php > testisIPv6
|
||
IPV6_CASES = [
|
||
[ false, ':fc:100::', 'IPv6 starting with lone ":"' ],
|
||
[ false, 'fc:100:::', 'IPv6 ending with a ":::"' ],
|
||
[ false, 'fc:300', 'IPv6 with only 2 words' ],
|
||
[ false, 'fc:100:300', 'IPv6 with only 3 words' ],
|
||
|
||
[ false, 'fc:100:a:d:1:e:ac:0::', 'IPv6 with 8 words ending with "::"' ],
|
||
[ false, 'fc:100:a:d:1:e:ac:0:1::', 'IPv6 with 9 words ending with "::"' ],
|
||
|
||
[ false, ':::' ],
|
||
[ false, '::0:', 'IPv6 ending in a lone ":"' ],
|
||
|
||
[ true, '::', 'IPv6 zero address' ],
|
||
|
||
[ false, '::fc:100:a:d:1:e:ac:0', 'IPv6 with "::" and 8 words' ],
|
||
[ false, '::fc:100:a:d:1:e:ac:0:1', 'IPv6 with 9 words' ],
|
||
|
||
[ false, ':fc::100', 'IPv6 starting with lone ":"' ],
|
||
[ false, 'fc::100:', 'IPv6 ending with lone ":"' ],
|
||
[ false, 'fc:::100', 'IPv6 with ":::" in the middle' ],
|
||
|
||
[ true, 'fc::100', 'IPv6 with "::" and 2 words' ],
|
||
[ true, 'fc::100:a', 'IPv6 with "::" and 3 words' ],
|
||
[ true, 'fc::100:a:d', 'IPv6 with "::" and 4 words' ],
|
||
[ true, 'fc::100:a:d:1', 'IPv6 with "::" and 5 words' ],
|
||
[ true, 'fc::100:a:d:1:e', 'IPv6 with "::" and 6 words' ],
|
||
[ true, 'fc::100:a:d:1:e:ac', 'IPv6 with "::" and 7 words' ],
|
||
[ true, '2001::df', 'IPv6 with "::" and 2 words' ],
|
||
[ true, '2001:5c0:1400:a::df', 'IPv6 with "::" and 5 words' ],
|
||
[ true, '2001:5c0:1400:a::df:2', 'IPv6 with "::" and 6 words' ],
|
||
|
||
[ false, 'fc::100:a:d:1:e:ac:0', 'IPv6 with "::" and 8 words' ],
|
||
[ false, 'fc::100:a:d:1:e:ac:0:1', 'IPv6 with 9 words' ]
|
||
];
|
||
|
||
Array.prototype.push.apply( IPV6_CASES,
|
||
[
|
||
'fc:100::',
|
||
'fc:100:a::',
|
||
'fc:100:a:d::',
|
||
'fc:100:a:d:1::',
|
||
'fc:100:a:d:1:e::',
|
||
'fc:100:a:d:1:e:ac::',
|
||
'::0',
|
||
'::fc',
|
||
'::fc:100',
|
||
'::fc:100:a',
|
||
'::fc:100:a:d',
|
||
'::fc:100:a:d:1',
|
||
'::fc:100:a:d:1:e',
|
||
'::fc:100:a:d:1:e:ac',
|
||
'fc:100:a:d:1:e:ac:0'
|
||
].map( function ( el ) {
|
||
return [ true, el, el + ' is a valid IP' ];
|
||
} )
|
||
);
|
||
|
||
QUnit.module( 'mediawiki.util', QUnit.newMwEnvironment( {
|
||
setup: function () {
|
||
$.fn.updateTooltipAccessKeys.setTestMode( true );
|
||
},
|
||
teardown: function () {
|
||
$.fn.updateTooltipAccessKeys.setTestMode( false );
|
||
mw.util.resetOptionsForTest();
|
||
},
|
||
messages: {
|
||
// Used by accessKeyLabel in test for addPortletLink
|
||
brackets: '[$1]',
|
||
'word-separator': ' '
|
||
}
|
||
} ) );
|
||
|
||
QUnit.test( 'rawurlencode', function ( assert ) {
|
||
assert.strictEqual( util.rawurlencode( 'Test:A & B/Here' ), 'Test%3AA%20%26%20B%2FHere' );
|
||
} );
|
||
|
||
QUnit.test( 'escapeIdForAttribute', function ( assert ) {
|
||
// Test cases are kept in sync with SanitizerTest.php
|
||
var text = 'foo тест_#%!\'()[]:<>',
|
||
legacyEncoded = 'foo_.D1.82.D0.B5.D1.81.D1.82_.23.25.21.27.28.29.5B.5D:.3C.3E',
|
||
html5Encoded = 'foo_тест_#%!\'()[]:<>',
|
||
// Settings: this is $wgFragmentMode
|
||
legacy = [ 'legacy' ],
|
||
legacyNew = [ 'legacy', 'html5' ],
|
||
newLegacy = [ 'html5', 'legacy' ],
|
||
allNew = [ 'html5' ];
|
||
|
||
// Test cases are kept in sync with SanitizerTest.php
|
||
[
|
||
// Pure legacy: how MW worked before 2017
|
||
[ legacy, text, legacyEncoded ],
|
||
// Transition to a new world: legacy links with HTML5 fallback
|
||
[ legacyNew, text, legacyEncoded ],
|
||
// New world: HTML5 links, legacy fallbacks
|
||
[ newLegacy, text, html5Encoded ],
|
||
// Distant future: no legacy fallbacks
|
||
[ allNew, text, html5Encoded ]
|
||
].forEach( function ( testCase ) {
|
||
mw.util.setOptionsForTest( { FragmentMode: testCase[ 0 ] } );
|
||
|
||
assert.strictEqual( util.escapeIdForAttribute( testCase[ 1 ] ), testCase[ 2 ] );
|
||
} );
|
||
} );
|
||
|
||
QUnit.test( 'escapeIdForLink', function ( assert ) {
|
||
// Test cases are kept in sync with SanitizerTest.php
|
||
var text = 'foo тест_#%!\'()[]:<>',
|
||
legacyEncoded = 'foo_.D1.82.D0.B5.D1.81.D1.82_.23.25.21.27.28.29.5B.5D:.3C.3E',
|
||
html5Encoded = 'foo_тест_#%!\'()[]:<>',
|
||
// Settings: this is wgFragmentMode
|
||
legacy = [ 'legacy' ],
|
||
legacyNew = [ 'legacy', 'html5' ],
|
||
newLegacy = [ 'html5', 'legacy' ],
|
||
allNew = [ 'html5' ];
|
||
|
||
[
|
||
// Pure legacy: how MW worked before 2017
|
||
[ legacy, text, legacyEncoded ],
|
||
// Transition to a new world: legacy links with HTML5 fallback
|
||
[ legacyNew, text, legacyEncoded ],
|
||
// New world: HTML5 links, legacy fallbacks
|
||
[ newLegacy, text, html5Encoded ],
|
||
// Distant future: no legacy fallbacks
|
||
[ allNew, text, html5Encoded ]
|
||
].forEach( function ( testCase ) {
|
||
mw.util.setOptionsForTest( { FragmentMode: testCase[ 0 ] } );
|
||
|
||
assert.strictEqual( util.escapeIdForLink( testCase[ 1 ] ), testCase[ 2 ] );
|
||
} );
|
||
} );
|
||
|
||
QUnit.test( 'wikiUrlencode', function ( assert ) {
|
||
assert.strictEqual( util.wikiUrlencode( 'Test:A & B/Here' ), 'Test:A_%26_B/Here' );
|
||
// See also wfUrlencodeTest.php#provideURLS
|
||
// eslint-disable-next-line no-jquery/no-each-util
|
||
$.each( {
|
||
'+': '%2B',
|
||
'&': '%26',
|
||
'=': '%3D',
|
||
':': ':',
|
||
';@$-_.!*': ';@$-_.!*',
|
||
'/': '/',
|
||
'~': '~',
|
||
'[]': '%5B%5D',
|
||
'<>': '%3C%3E',
|
||
'\'': '%27'
|
||
}, function ( input, output ) {
|
||
assert.strictEqual( util.wikiUrlencode( input ), output );
|
||
} );
|
||
} );
|
||
|
||
QUnit.test( 'getUrl', function ( assert ) {
|
||
var href;
|
||
mw.config.set( {
|
||
wgScript: '/w/index.php',
|
||
wgArticlePath: '/wiki/$1',
|
||
wgPageName: 'Foobar'
|
||
} );
|
||
|
||
href = util.getUrl( 'Sandbox' );
|
||
assert.strictEqual( href, '/wiki/Sandbox', 'simple title' );
|
||
|
||
href = util.getUrl( 'Foo:Sandbox? 5+5=10! (test)/sub ' );
|
||
assert.strictEqual( href, '/wiki/Foo:Sandbox%3F_5%2B5%3D10!_(test)/sub_', 'complex title' );
|
||
|
||
// T149767
|
||
href = util.getUrl( 'My$$test$$$$$title' );
|
||
assert.strictEqual( href, '/wiki/My$$test$$$$$title', 'title with multiple consecutive dollar signs' );
|
||
|
||
href = util.getUrl();
|
||
assert.strictEqual( href, '/wiki/Foobar', 'default title' );
|
||
|
||
href = util.getUrl( null, { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Foobar&action=edit', 'default title with query string' );
|
||
|
||
href = util.getUrl( 'Sandbox', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Sandbox&action=edit', 'simple title with query string' );
|
||
|
||
// Test fragments
|
||
href = util.getUrl( 'Foo:Sandbox#Fragment', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Foo:Sandbox&action=edit#Fragment', 'namespaced title with query string and fragment' );
|
||
|
||
href = util.getUrl( 'Sandbox#', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Sandbox&action=edit', 'title with query string and empty fragment' );
|
||
|
||
href = util.getUrl( 'Sandbox', {} );
|
||
assert.strictEqual( href, '/wiki/Sandbox', 'title with empty query string' );
|
||
|
||
href = util.getUrl( '#Fragment' );
|
||
assert.strictEqual( href, '/wiki/#Fragment', 'empty title with fragment' );
|
||
|
||
href = util.getUrl( '#Fragment', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?action=edit#Fragment', 'empty title with query string and fragment' );
|
||
|
||
mw.util.setOptionsForTest( { FragmentMode: [ 'legacy' ] } );
|
||
href = util.getUrl( 'Foo:Sandbox \xC4#Fragment \xC4', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Foo:Sandbox_%C3%84&action=edit#Fragment_.C3.84', 'title with query string, fragment, and special characters' );
|
||
|
||
mw.util.setOptionsForTest( { FragmentMode: [ 'html5' ] } );
|
||
href = util.getUrl( 'Foo:Sandbox \xC4#Fragment \xC4', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Foo:Sandbox_%C3%84&action=edit#Fragment_Ä', 'title with query string, fragment, and special characters' );
|
||
|
||
href = util.getUrl( 'Foo:%23#Fragment', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?title=Foo:%2523&action=edit#Fragment', 'title containing %23 (#), fragment, and a query string' );
|
||
|
||
mw.util.setOptionsForTest( { FragmentMode: [ 'legacy' ] } );
|
||
href = util.getUrl( '#+&=:;@$-_.!*/[]<>\'§', { action: 'edit' } );
|
||
assert.strictEqual( 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' );
|
||
|
||
mw.util.setOptionsForTest( { FragmentMode: [ 'html5' ] } );
|
||
href = util.getUrl( '#+&=:;@$-_.!*/[]<>\'§', { action: 'edit' } );
|
||
assert.strictEqual( href, '/w/index.php?action=edit#+&=:;@$-_.!*/[]<>\'§', 'fragment with various characters' );
|
||
} );
|
||
|
||
QUnit.test( 'wikiScript', function ( assert ) {
|
||
mw.util.setOptionsForTest( {
|
||
LoadScript: '/w/l.php'
|
||
} );
|
||
mw.config.set( {
|
||
// customized wgScript for T41103
|
||
wgScript: '/w/i.php',
|
||
wgScriptPath: '/w'
|
||
} );
|
||
|
||
assert.strictEqual( util.wikiScript(), mw.config.get( 'wgScript' ),
|
||
'wikiScript() returns wgScript'
|
||
);
|
||
assert.strictEqual( util.wikiScript( 'index' ), mw.config.get( 'wgScript' ),
|
||
'wikiScript( index ) returns wgScript'
|
||
);
|
||
assert.strictEqual( util.wikiScript( 'load' ), '/w/l.php',
|
||
'wikiScript( load ) returns /w/l.php'
|
||
);
|
||
assert.strictEqual( util.wikiScript( 'api' ), '/w/api.php', 'API path' );
|
||
} );
|
||
|
||
QUnit.test( 'addCSS', function ( assert ) {
|
||
var $el, style;
|
||
$el = $( '<div>' ).attr( 'id', 'mw-addcsstest' ).appendTo( '#qunit-fixture' );
|
||
|
||
style = util.addCSS( '#mw-addcsstest { visibility: hidden; }' );
|
||
assert.strictEqual( typeof style, 'object', 'addCSS returned an object' );
|
||
assert.strictEqual( style.disabled, false, 'property "disabled" is available and set to false' );
|
||
|
||
assert.strictEqual( $el.css( 'visibility' ), 'hidden', 'Added style properties are in effect' );
|
||
|
||
// Clean up
|
||
$( style.ownerNode ).remove();
|
||
} );
|
||
|
||
QUnit.test( 'getParamValue', function ( assert ) {
|
||
var url;
|
||
|
||
url = 'http://example.org/?foo=wrong&foo=right#&foo=bad';
|
||
assert.strictEqual( util.getParamValue( 'foo', url ), 'right', 'Use latest one, ignore hash' );
|
||
assert.strictEqual( util.getParamValue( 'bar', url ), null, 'Return null when not found' );
|
||
|
||
url = 'http://example.org/#&foo=bad';
|
||
assert.strictEqual( util.getParamValue( 'foo', url ), null, 'Ignore hash if param is not in querystring but in hash (T29427)' );
|
||
|
||
url = 'example.org?' + $.param( { TEST: 'a b+c' } );
|
||
assert.strictEqual( util.getParamValue( 'TEST', url ), 'a b+c', 'T32441: getParamValue must understand "+" encoding of space' );
|
||
|
||
url = 'example.org?' + $.param( { TEST: 'a b+c d' } ); // check for sloppy code from r95332 :)
|
||
assert.strictEqual( util.getParamValue( 'TEST', url ), 'a b+c d', 'T32441: getParamValue must understand "+" encoding of space (multiple spaces)' );
|
||
} );
|
||
|
||
QUnit.test( '$content', function ( assert ) {
|
||
assert.ok( util.$content instanceof $, 'mw.util.$content instance of jQuery' );
|
||
assert.strictEqual( util.$content.length, 1, 'mw.util.$content must have length of 1' );
|
||
} );
|
||
|
||
/**
|
||
* Portlet names are prefixed with 'p-test' to avoid conflict with core
|
||
* when running the test suite under a wiki page.
|
||
* Previously, test elements where invisible to the selector since only
|
||
* one element can have a given id.
|
||
*/
|
||
QUnit.test( 'addPortletLink', function ( assert ) {
|
||
var tbRL, cuQuux, $cuQuux, tbMW, $tbMW, tbRLDM, caFoo,
|
||
addedAfter, tbRLDMnonexistentid, tbRLDMemptyjquery;
|
||
|
||
$( '#qunit-fixture' ).append(
|
||
'<div class="portlet" id="p-test-tb">' +
|
||
'<h3>Toolbox</h3>' +
|
||
'<ul class="body"></ul>' +
|
||
'</div>' +
|
||
'<div class="portlet" id="p-test-custom">' +
|
||
'<h3>Views</h3>' +
|
||
'<ul class="body">' +
|
||
'<li id="c-foo"><a href="#">Foo</a></li>' +
|
||
'<li id="c-barmenu">' +
|
||
'<ul>' +
|
||
'<li id="c-bar-baz"><a href="#">Baz</a></a>' +
|
||
'</ul>' +
|
||
'</li>' +
|
||
'</ul>' +
|
||
'</div>' +
|
||
'<div id="p-test-views" class="vectorTabs">' +
|
||
'<h3>Views</h3>' +
|
||
'<ul></ul>' +
|
||
'</div>'
|
||
);
|
||
|
||
tbRL = util.addPortletLink( 'p-test-tb', 'https://example.org/next',
|
||
'Next', 't-rl', 'More info about Example Next ', 'l'
|
||
);
|
||
assert.strictEqual( tbRL.nodeType, 1, 'returns a DOM Node' );
|
||
assert.strictEqual( tbRL.nodeName, 'LI', 'returns a list item element' );
|
||
|
||
tbMW = util.addPortletLink( 'p-test-tb', '//example.org/',
|
||
'Example.org', 't-xmp', 'Go to Example', 'x', tbRL );
|
||
$tbMW = $( tbMW );
|
||
assert.propEqual(
|
||
$tbMW.getAttrs(),
|
||
{
|
||
id: 't-xmp'
|
||
},
|
||
'List item attributes'
|
||
);
|
||
assert.propEqual(
|
||
$tbMW.find( 'a' ).getAttrs(),
|
||
{
|
||
href: '//example.org/',
|
||
title: 'Go to Example [test-x]',
|
||
accesskey: 'x'
|
||
},
|
||
'Anchor link attributes'
|
||
);
|
||
assert.strictEqual(
|
||
$tbMW.closest( '.portlet' ).attr( 'id' ),
|
||
'p-test-tb',
|
||
'Parent portlet ID'
|
||
);
|
||
assert.strictEqual(
|
||
$tbMW.next()[ 0 ],
|
||
tbRL,
|
||
'Next node (set as Node object)'
|
||
);
|
||
assert.strictEqual(
|
||
$tbMW.find( 'span' ).length,
|
||
0,
|
||
'No <span> wrap for porlets without vectorTabs class'
|
||
);
|
||
|
||
cuQuux = util.addPortletLink( 'p-test-custom', '#', 'Quux', null, 'Example [shift-x]', 'q' );
|
||
$cuQuux = $( cuQuux );
|
||
assert.strictEqual(
|
||
$cuQuux.find( 'a' ).attr( 'title' ),
|
||
'Example [test-q]',
|
||
'Title has new accesskey and label'
|
||
);
|
||
assert.strictEqual(
|
||
$( '#p-test-custom #c-barmenu ul li' ).length,
|
||
1,
|
||
'No items added to unrelated <ul> elsewhere in the portlet (T37082)'
|
||
);
|
||
|
||
tbRLDM = util.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
|
||
'Default modules', 't-rldm', 'List of all default modules ', 'd', '#t-rl' );
|
||
assert.strictEqual( $( tbRLDM ).next()[ 0 ], tbRL, 'Next node (set as CSS selector)' );
|
||
|
||
caFoo = util.addPortletLink( 'p-test-views', '#', 'Foo' );
|
||
assert.strictEqual( $( caFoo ).find( 'span' ).length, 1, 'Added <span> element for porlet with vectorTabs class' );
|
||
|
||
addedAfter = util.addPortletLink( 'p-test-tb', '#', 'After foo', 'post-foo', 'After foo', null, $( tbRL ) );
|
||
assert.strictEqual( $( addedAfter ).next()[ 0 ], tbRL, 'Next node (set as jQuery object)' );
|
||
|
||
tbRLDMnonexistentid = util.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
|
||
'Default modules', 't-rldm-nonexistent', 'List of all default modules ', 'd', '#t-rl-nonexistent' );
|
||
assert.strictEqual(
|
||
tbRLDMnonexistentid,
|
||
$( '#p-test-tb li' ).last()[ 0 ],
|
||
'Next node as non-matching CSS selector falls back to appending'
|
||
);
|
||
|
||
tbRLDMemptyjquery = util.addPortletLink( 'p-test-tb', '//mediawiki.org/wiki/RL/DM',
|
||
'Default modules', 't-rldm-empty-jquery', 'List of all default modules ', 'd', $( '#t-rl-nonexistent' ) );
|
||
assert.strictEqual(
|
||
tbRLDMemptyjquery,
|
||
$( '#p-test-tb li' ).last()[ 0 ],
|
||
'Next node as empty jQuery object falls back to appending'
|
||
);
|
||
} );
|
||
|
||
QUnit.test( 'validateEmail', function ( assert ) {
|
||
assert.strictEqual( util.validateEmail( '' ), null, 'Should return null for empty string ' );
|
||
assert.strictEqual( util.validateEmail( 'user@localhost' ), true, 'Return true for a valid e-mail address' );
|
||
|
||
// testEmailWithCommasAreInvalids
|
||
assert.strictEqual( util.validateEmail( 'user,foo@example.org' ), false, 'Emails with commas are invalid' );
|
||
assert.strictEqual( util.validateEmail( 'userfoo@ex,ample.org' ), false, 'Emails with commas are invalid' );
|
||
|
||
// testEmailWithHyphens
|
||
assert.strictEqual( util.validateEmail( 'user-foo@example.org' ), true, 'Emails may contain a hyphen' );
|
||
assert.strictEqual( util.validateEmail( 'userfoo@ex-ample.org' ), true, 'Emails may contain a hyphen' );
|
||
} );
|
||
|
||
QUnit.test( 'isIPv6Address', function ( assert ) {
|
||
IPV6_CASES.forEach( function ( ipCase ) {
|
||
assert.strictEqual( util.isIPv6Address( ipCase[ 1 ] ), ipCase[ 0 ], ipCase[ 2 ] );
|
||
} );
|
||
} );
|
||
|
||
QUnit.test( 'isIPv4Address', function ( assert ) {
|
||
IPV4_CASES.forEach( function ( ipCase ) {
|
||
assert.strictEqual( util.isIPv4Address( ipCase[ 1 ] ), ipCase[ 0 ], ipCase[ 2 ] );
|
||
} );
|
||
} );
|
||
|
||
QUnit.test( 'isIPAddress', function ( assert ) {
|
||
IPV4_CASES.forEach( function ( ipCase ) {
|
||
assert.strictEqual( util.isIPv4Address( ipCase[ 1 ] ), ipCase[ 0 ], ipCase[ 2 ] );
|
||
} );
|
||
|
||
IPV6_CASES.forEach( function ( ipCase ) {
|
||
assert.strictEqual( util.isIPv6Address( ipCase[ 1 ] ), ipCase[ 0 ], ipCase[ 2 ] );
|
||
} );
|
||
} );
|
||
}() );
|