( function () { /* eslint-disable camelcase */ var formatText, formatParse, formatnumTests, specialCharactersPageName, expectedListUsers, expectedListUsersSitename, expectedLinkPagenamee, expectedEntrypoints, mwLanguageCache = {}, hasOwn = Object.hasOwnProperty; // When the expected result is the same in both modes function assertBothModes( assert, parserArguments, expectedResult, assertMessage ) { assert.strictEqual( formatText.apply( null, parserArguments ), expectedResult, assertMessage + ' when format is \'text\'' ); assert.strictEqual( formatParse.apply( null, parserArguments ), expectedResult, assertMessage + ' when format is \'parse\'' ); } QUnit.module( 'mediawiki.jqueryMsg', QUnit.newMwEnvironment( { setup: function () { this.originalMwLanguage = mw.language; this.parserDefaults = mw.jqueryMsg.getParserDefaults(); mw.jqueryMsg.setParserDefaults( { magic: { PAGENAME: '2 + 2', PAGENAMEE: mw.util.wikiUrlencode( '2 + 2' ), SITENAME: 'Wiki' } } ); specialCharactersPageName = '"Who" wants to be a millionaire & live on \'Exotic Island\'?'; expectedListUsers = '注册用户'; expectedListUsersSitename = '注册用户' + 'Wiki'; expectedLinkPagenamee = 'Test'; expectedEntrypoints = 'index.php'; formatText = mw.jqueryMsg.getMessageFunction( { format: 'text' } ); formatParse = mw.jqueryMsg.getMessageFunction( { format: 'parse' } ); }, teardown: function () { mw.language = this.originalMwLanguage; mw.jqueryMsg.setParserDefaults( this.parserDefaults ); }, config: { wgArticlePath: '/wiki/$1', wgNamespaceIds: { template: 10, template_talk: 11, // Localised szablon: 10, dyskusja_szablonu: 11 }, wgFormattedNamespaces: { // Localised 10: 'Szablon', 11: 'Dyskusja szablonu' } }, // Messages that are reused in multiple tests messages: { // The values for gender are not significant, // what matters is which of the values is choosen by the parser 'gender-msg': '$1: {{GENDER:$2|blue|pink|green}}', 'gender-msg-currentuser': '{{GENDER:|blue|pink|green}}', 'plural-msg': 'Found $1 {{PLURAL:$1|item|items}}', // See https://phabricator.wikimedia.org/T71993 'plural-msg-explicit-forms-nested': 'Found {{PLURAL:$1|$1 results|0=no results in {{SITENAME}}|1=$1 result}}', // Assume the grammar form grammar_case_foo is not valid in any language 'grammar-msg': 'Przeszukaj {{GRAMMAR:grammar_case_foo|{{SITENAME}}}}', 'formatnum-msg': '{{formatnum:$1}}', 'portal-url': 'Project:Community portal', 'see-portal-url': '{{Int:portal-url}} is an important community page.', 'jquerymsg-test-statistics-users': '注册[[Special:ListUsers|用户]]', 'jquerymsg-test-statistics-users-sitename': '注册[[Special:ListUsers|用户{{SITENAME}}]]', 'jquerymsg-test-link-pagenamee': '[https://example.org/wiki/Foo?bar=baz#val/{{PAGENAMEE}} Test]', 'jquerymsg-test-version-entrypoints-index-php': '[https://www.mediawiki.org/wiki/Manual:index.php index.php]', 'external-link-replace': 'Foo [$1 bar]', 'external-link-plural': 'Foo {{PLURAL:$1|is [$2 one]|are [$2 some]|2=[$2 two]|3=three|4=a=b}} things.', 'plural-only-explicit-forms': 'It is a {{PLURAL:$1|1=single|2=double}} room.', 'plural-empty-explicit-form': 'There is me{{PLURAL:$1|0=| and other people}}.' } } ) ); /** * Be careful to no run this in parallel as it uses a global identifier (mw.language) * to transport the module back to the test. It musn't be overwritten concurrentely. * * This function caches the mw.language data to avoid having to request the same module * multiple times. There is more than one test case for any given language. */ function getMwLanguage( langCode ) { if ( !hasOwn.call( mwLanguageCache, langCode ) ) { mwLanguageCache[ langCode ] = $.ajax( { url: mw.util.wikiScript( 'load' ), data: { skin: mw.config.get( 'skin' ), lang: langCode, debug: mw.config.get( 'debug' ), modules: 'mediawiki.language', only: 'scripts' }, dataType: 'script', cache: true } ).then( function () { return mw.language; } ); } return mwLanguageCache[ langCode ]; } /** * @param {Function[]} tasks List of functions that perform tasks * that may be asynchronous. Invoke the callback parameter when done. */ function process( tasks ) { function abort() { tasks.splice( 0, tasks.length ); // eslint-disable-next-line no-use-before-define next(); } function next() { var task; if ( !tasks ) { // This happens if after the process is completed, one of our callbacks is // invoked. This can happen if a test timed out but the process was still // running. In that case, ignore it. Don't invoke complete() a second time. return; } task = tasks.shift(); if ( task ) { task( next, abort ); } else { // Remove tasks list to indicate the process is final. tasks = null; } } next(); } QUnit.test( 'Replace', function ( assert ) { mw.messages.set( 'simple', 'Foo $1 baz $2' ); assert.strictEqual( formatParse( 'simple' ), 'Foo $1 baz $2', 'Replacements with no substitutes' ); assert.strictEqual( formatParse( 'simple', 'bar' ), 'Foo bar baz $2', 'Replacements with less substitutes' ); assert.strictEqual( formatParse( 'simple', 'bar', 'quux' ), 'Foo bar baz quux', 'Replacements with all substitutes' ); mw.messages.set( 'plain-input', 'x$1y<z' ); assert.strictEqual( formatParse( 'plain-input', 'bar' ), '<foo foo="foo">xbary&lt;</foo>z', 'Input is not considered html' ); mw.messages.set( 'plain-replace', 'Foo $1' ); assert.strictEqual( formatParse( 'plain-replace', '>' ), 'Foo <bar bar="bar">&gt;</bar>', 'Replacement is not considered html' ); mw.messages.set( 'object-replace', 'Foo $1' ); assert.strictEqual( formatParse( 'object-replace', $( '
>
' ) ), 'Foo
>
', 'jQuery objects are preserved as raw html' ); assert.strictEqual( formatParse( 'object-replace', $( '
>
' ).get( 0 ) ), 'Foo
>
', 'HTMLElement objects are preserved as raw html' ); assert.strictEqual( formatParse( 'object-replace', $( '
>
' ).toArray() ), 'Foo
>
', 'HTMLElement[] arrays are preserved as raw html' ); assert.strictEqual( formatParse( 'external-link-replace', 'http://example.org/?x=y&z' ), 'Foo bar', 'Href is not double-escaped in wikilink function' ); assert.strictEqual( formatParse( 'external-link-plural', 1, 'http://example.org' ), 'Foo is one things.', 'Link is expanded inside plural and is not escaped html' ); assert.strictEqual( formatParse( 'external-link-plural', 2, 'http://example.org' ), 'Foo two things.', 'Link is expanded inside an explicit plural form and is not escaped html' ); assert.strictEqual( formatParse( 'external-link-plural', 3 ), 'Foo three things.', 'A simple explicit plural form co-existing with complex explicit plural forms' ); assert.strictEqual( formatParse( 'external-link-plural', 4, 'http://example.org' ), 'Foo a=b things.', 'Only first equal sign is used as delimiter for explicit plural form. Repeated equal signs does not create issue' ); assert.strictEqual( formatParse( 'external-link-plural', 6, 'http://example.org' ), 'Foo are some things.', 'Plural fallback to the "other" plural form' ); assert.strictEqual( formatParse( 'plural-only-explicit-forms', 2 ), 'It is a double room.', 'Plural with explicit forms alone.' ); } ); QUnit.test( 'Plural', function ( assert ) { assert.strictEqual( formatParse( 'plural-msg', 0 ), 'Found 0 items', 'Plural test for english with zero as count' ); assert.strictEqual( formatParse( 'plural-msg', 1 ), 'Found 1 item', 'Singular test for english' ); assert.strictEqual( formatParse( 'plural-msg', 2 ), 'Found 2 items', 'Plural test for english' ); assert.strictEqual( formatParse( 'plural-msg-explicit-forms-nested', 6 ), 'Found 6 results', 'Plural message with explicit plural forms' ); assert.strictEqual( formatParse( 'plural-msg-explicit-forms-nested', 0 ), 'Found no results in Wiki', 'Plural message with explicit plural forms, with nested {{SITENAME}}' ); assert.strictEqual( formatParse( 'plural-msg-explicit-forms-nested', 1 ), 'Found 1 result', 'Plural message with explicit plural forms with placeholder nested' ); assert.strictEqual( formatParse( 'plural-empty-explicit-form', 0 ), 'There is me.' ); assert.strictEqual( formatParse( 'plural-empty-explicit-form', 1 ), 'There is me and other people.' ); assert.strictEqual( formatParse( 'plural-empty-explicit-form', 2 ), 'There is me and other people.' ); } ); QUnit.test( 'Gender', function ( assert ) { var originalGender = mw.user.options.get( 'gender' ); // TODO: These tests should be for mw.msg once mw.msg integrated with mw.jqueryMsg // TODO: English may not be the best language for these tests. Use a language like Arabic or Russian mw.user.options.set( 'gender', 'male' ); assert.strictEqual( formatParse( 'gender-msg', 'Bob', 'male' ), 'Bob: blue', 'Masculine from string "male"' ); assert.strictEqual( formatParse( 'gender-msg', 'Bob', mw.user ), 'Bob: blue', 'Masculine from mw.user object' ); assert.strictEqual( formatParse( 'gender-msg-currentuser' ), 'blue', 'Masculine for current user' ); mw.user.options.set( 'gender', 'female' ); assert.strictEqual( formatParse( 'gender-msg', 'Alice', 'female' ), 'Alice: pink', 'Feminine from string "female"' ); assert.strictEqual( formatParse( 'gender-msg', 'Alice', mw.user ), 'Alice: pink', 'Feminine from mw.user object' ); assert.strictEqual( formatParse( 'gender-msg-currentuser' ), 'pink', 'Feminine for current user' ); mw.user.options.set( 'gender', 'unknown' ); assert.strictEqual( formatParse( 'gender-msg', 'Foo', mw.user ), 'Foo: green', 'Neutral from mw.user object' ); assert.strictEqual( formatParse( 'gender-msg', 'User' ), 'User: green', 'Neutral when no parameter given' ); assert.strictEqual( formatParse( 'gender-msg', 'User', 'unknown' ), 'User: green', 'Neutral from string "unknown"' ); assert.strictEqual( formatParse( 'gender-msg-currentuser' ), 'green', 'Neutral for current user' ); mw.messages.set( 'gender-msg-one-form', '{{GENDER:$1|User}}: $2 {{PLURAL:$2|edit|edits}}' ); assert.strictEqual( formatParse( 'gender-msg-one-form', 'male', 10 ), 'User: 10 edits', 'Gender neutral and plural form' ); assert.strictEqual( formatParse( 'gender-msg-one-form', 'female', 1 ), 'User: 1 edit', 'Gender neutral and singular form' ); mw.messages.set( 'gender-msg-lowercase', '{{gender:$1|he|she}} is awesome' ); assert.strictEqual( formatParse( 'gender-msg-lowercase', 'male' ), 'he is awesome', 'Gender masculine' ); assert.strictEqual( formatParse( 'gender-msg-lowercase', 'female' ), 'she is awesome', 'Gender feminine' ); mw.messages.set( 'gender-msg-wrong', '{{gender}} test' ); assert.strictEqual( formatParse( 'gender-msg-wrong', 'female' ), ' test', 'Invalid syntax should result in {{gender}} simply being stripped away' ); mw.user.options.set( 'gender', originalGender ); } ); QUnit.test( 'Case changing', function ( assert ) { mw.messages.set( 'to-lowercase', '{{lc:thIS hAS MEsSed uP CapItaliZatiON}}' ); assert.strictEqual( formatParse( 'to-lowercase' ), 'this has messed up capitalization', 'To lowercase' ); mw.messages.set( 'to-caps', '{{uc:thIS hAS MEsSed uP CapItaliZatiON}}' ); assert.strictEqual( formatParse( 'to-caps' ), 'THIS HAS MESSED UP CAPITALIZATION', 'To caps' ); mw.messages.set( 'uc-to-lcfirst', '{{lcfirst:THis hAS MEsSed uP CapItaliZatiON}}' ); mw.messages.set( 'lc-to-lcfirst', '{{lcfirst:thIS hAS MEsSed uP CapItaliZatiON}}' ); assert.strictEqual( formatParse( 'uc-to-lcfirst' ), 'tHis hAS MEsSed uP CapItaliZatiON', 'Lcfirst caps' ); assert.strictEqual( formatParse( 'lc-to-lcfirst' ), 'thIS hAS MEsSed uP CapItaliZatiON', 'Lcfirst lowercase' ); mw.messages.set( 'uc-to-ucfirst', '{{ucfirst:THis hAS MEsSed uP CapItaliZatiON}}' ); mw.messages.set( 'lc-to-ucfirst', '{{ucfirst:thIS hAS MEsSed uP CapItaliZatiON}}' ); assert.strictEqual( formatParse( 'uc-to-ucfirst' ), 'THis hAS MEsSed uP CapItaliZatiON', 'Ucfirst caps' ); assert.strictEqual( formatParse( 'lc-to-ucfirst' ), 'ThIS hAS MEsSed uP CapItaliZatiON', 'Ucfirst lowercase' ); mw.messages.set( 'mixed-to-sentence', '{{ucfirst:{{lc:thIS hAS MEsSed uP CapItaliZatiON}}}}' ); assert.strictEqual( formatParse( 'mixed-to-sentence' ), 'This has messed up capitalization', 'To sentence case' ); mw.messages.set( 'all-caps-except-first', '{{lcfirst:{{uc:thIS hAS MEsSed uP CapItaliZatiON}}}}' ); assert.strictEqual( formatParse( 'all-caps-except-first' ), 'tHIS HAS MESSED UP CAPITALIZATION', 'To opposite sentence case' ); } ); QUnit.test( 'Grammar', function ( assert ) { assert.strictEqual( formatParse( 'grammar-msg' ), 'Przeszukaj Wiki', 'Grammar Test with sitename' ); mw.messages.set( 'grammar-msg-wrong-syntax', 'Przeszukaj {{GRAMMAR:grammar_case_xyz}}' ); assert.strictEqual( formatParse( 'grammar-msg-wrong-syntax' ), 'Przeszukaj ', 'Grammar Test with wrong grammar template syntax' ); } ); QUnit.test( 'Match PHP parser', function ( assert ) { var tasks; mw.messages.set( mw.libs.phpParserData.messages ); tasks = mw.libs.phpParserData.tests.map( function ( test ) { var done = assert.async(); return function ( next, abort ) { getMwLanguage( test.lang ) .then( function ( langClass ) { var parser; mw.config.set( 'wgUserLanguage', test.lang ); parser = new mw.jqueryMsg.Parser( { language: langClass } ); assert.strictEqual( parser.parse( test.key, test.args ).html(), test.result, test.name ); }, function () { assert.ok( false, 'Language "' + test.lang + '" failed to load.' ); } ) .then( done, done ) .then( next, abort ); }; } ); process( tasks ); } ); QUnit.test( 'Links', function ( assert ) { var testCases, expectedDisambiguationsText, expectedMultipleBars, expectedSpecialCharacters; // The below three are all identical to or based on real messages. For disambiguations-text, // the bold was removed because it is not yet implemented. assert.htmlEqual( formatParse( 'jquerymsg-test-statistics-users' ), expectedListUsers, 'Piped wikilink' ); expectedDisambiguationsText = 'The following pages contain at least one link to a disambiguation page.\nThey may have to link to a more appropriate page instead.\nA page is treated as a disambiguation page if it uses a template that is linked from ' + 'MediaWiki:Disambiguationspage.'; mw.messages.set( 'disambiguations-text', 'The following pages contain at least one link to a disambiguation page.\nThey may have to link to a more appropriate page instead.\nA page is treated as a disambiguation page if it uses a template that is linked from [[MediaWiki:Disambiguationspage]].' ); assert.htmlEqual( formatParse( 'disambiguations-text' ), expectedDisambiguationsText, 'Wikilink without pipe' ); assert.htmlEqual( formatParse( 'jquerymsg-test-version-entrypoints-index-php' ), expectedEntrypoints, 'External link' ); // Pipe trick is not supported currently, but should not parse as text either. mw.messages.set( 'pipe-trick', '[[Tampa, Florida|]]' ); mw.messages.set( 'reverse-pipe-trick', '[[|Tampa, Florida]]' ); mw.messages.set( 'empty-link', '[[]]' ); this.suppressWarnings(); assert.strictEqual( formatParse( 'pipe-trick' ), '[[Tampa, Florida|]]', 'Pipe trick should not be parsed.' ); assert.strictEqual( formatParse( 'reverse-pipe-trick' ), '[[|Tampa, Florida]]', 'Reverse pipe trick should not be parsed.' ); assert.strictEqual( formatParse( 'empty-link' ), '[[]]', 'Empty link should not be parsed.' ); this.restoreWarnings(); expectedMultipleBars = 'Main|Page'; mw.messages.set( 'multiple-bars', '[[Main Page|Main|Page]]' ); assert.htmlEqual( formatParse( 'multiple-bars' ), expectedMultipleBars, 'Bar in anchor' ); expectedSpecialCharacters = '"Who" wants to be a millionaire & live on 'Exotic Island'?'; mw.messages.set( 'special-characters', '[[' + specialCharactersPageName + ']]' ); assert.htmlEqual( formatParse( 'special-characters' ), expectedSpecialCharacters, 'Special characters' ); mw.messages.set( 'leading-colon', '[[:File:Foo.jpg]]' ); assert.htmlEqual( formatParse( 'leading-colon' ), 'File:Foo.jpg', 'Leading colon in links is stripped' ); assert.htmlEqual( formatParse( 'jquerymsg-test-statistics-users-sitename' ), expectedListUsersSitename, 'Piped wikilink with parser function in the text' ); assert.htmlEqual( formatParse( 'jquerymsg-test-link-pagenamee' ), expectedLinkPagenamee, 'External link with parser function in the URL' ); testCases = [ [ 'extlink-html-full', 'asd [http://example.org Example] asd', 'asd Example asd' ], [ 'extlink-html-partial', 'asd [http://example.org foo Example bar] asd', 'asd foo Example bar asd' ], [ 'wikilink-html-full', 'asd [[Example|Example]] asd', 'asd Example asd' ], [ 'wikilink-html-partial', 'asd [[Example|foo Example bar]] asd', 'asd foo Example bar asd' ] ]; testCases.forEach( function ( testCase ) { var key = testCase[ 0 ], input = testCase[ 1 ], output = testCase[ 2 ]; mw.messages.set( key, input ); assert.htmlEqual( formatParse( key ), output, 'HTML in links: ' + key ); } ); } ); QUnit.test( 'Replacements in links', function ( assert ) { var testCases = [ [ 'extlink-param-href-full', 'asd [$1 Example] asd', 'asd Example asd' ], [ 'extlink-param-href-partial', 'asd [$1/example Example] asd', 'asd Example asd' ], [ 'extlink-param-text-full', 'asd [http://example.org $2] asd', 'asd Text asd' ], [ 'extlink-param-text-partial', 'asd [http://example.org Example $2] asd', 'asd Example Text asd' ], [ 'extlink-param-both-full', 'asd [$1 $2] asd', 'asd Text asd' ], [ 'extlink-param-both-partial', 'asd [$1/example Example $2] asd', 'asd Example Text asd' ], [ 'wikilink-param-href-full', 'asd [[$1|Example]] asd', 'asd Example asd' ], [ 'wikilink-param-href-partial', 'asd [[$1/Test|Example]] asd', 'asd Example asd' ], [ 'wikilink-param-text-full', 'asd [[Example|$2]] asd', 'asd Text asd' ], [ 'wikilink-param-text-partial', 'asd [[Example|Example $2]] asd', 'asd Example Text asd' ], [ 'wikilink-param-both-full', 'asd [[$1|$2]] asd', 'asd Text asd' ], [ 'wikilink-param-both-partial', 'asd [[$1/Test|Example $2]] asd', 'asd Example Text asd' ], [ 'wikilink-param-unpiped-full', 'asd [[$1]] asd', 'asd Example asd' ], [ 'wikilink-param-unpiped-partial', 'asd [[$1/Test]] asd', 'asd Example/Test asd' ] ]; testCases.forEach( function ( testCase ) { var key = testCase[ 0 ], input = testCase[ 1 ], output = testCase[ 2 ], paramHref = key.slice( 0, 8 ) === 'wikilink' ? 'Example' : 'http://example.com', paramText = 'Text'; mw.messages.set( key, input ); assert.htmlEqual( formatParse( key, paramHref, paramText ), output, 'Replacements in links: ' + key ); } ); } ); // Tests that {{-transformation vs. general parsing are done as requested QUnit.test( 'Curly brace transformation', function ( assert ) { var oldUserLang = mw.config.get( 'wgUserLanguage' ); assertBothModes( assert, [ 'gender-msg', 'Bob', 'male' ], 'Bob: blue', 'gender is resolved' ); assertBothModes( assert, [ 'plural-msg', 5 ], 'Found 5 items', 'plural is resolved' ); assertBothModes( assert, [ 'grammar-msg' ], 'Przeszukaj Wiki', 'grammar is resolved' ); mw.config.set( 'wgUserLanguage', 'en' ); assertBothModes( assert, [ 'formatnum-msg', '987654321.654321' ], '987,654,321.654', 'formatnum is resolved' ); // Test non-{{ wikitext, where behavior differs // Wikilink assert.strictEqual( formatText( 'jquerymsg-test-statistics-users' ), mw.messages.get( 'jquerymsg-test-statistics-users' ), 'Internal link message unchanged when format is \'text\'' ); assert.htmlEqual( formatParse( 'jquerymsg-test-statistics-users' ), expectedListUsers, 'Internal link message parsed when format is \'parse\'' ); // External link assert.strictEqual( formatText( 'jquerymsg-test-version-entrypoints-index-php' ), mw.messages.get( 'jquerymsg-test-version-entrypoints-index-php' ), 'External link message unchanged when format is \'text\'' ); assert.htmlEqual( formatParse( 'jquerymsg-test-version-entrypoints-index-php' ), expectedEntrypoints, 'External link message processed when format is \'parse\'' ); // External link with parameter assert.strictEqual( formatText( 'external-link-replace', 'http://example.com' ), 'Foo [http://example.com bar]', 'External link message only substitutes parameter when format is \'text\'' ); assert.htmlEqual( formatParse( 'external-link-replace', 'http://example.com' ), 'Foo bar', 'External link message processed when format is \'parse\'' ); assert.htmlEqual( formatParse( 'external-link-replace', $( '' ) ), 'Foo bar', 'External link message processed as jQuery object when format is \'parse\'' ); assert.htmlEqual( formatParse( 'external-link-replace', function () {} ), 'Foo bar', 'External link message processed as function when format is \'parse\'' ); mw.config.set( 'wgUserLanguage', oldUserLang ); } ); QUnit.test( 'Int', function ( assert ) { var newarticletextSource = 'You have followed a link to a page that does not exist yet. To create the page, start typing in the box below (see the [[{{Int:Foobar}}|foobar]] for more info). If you are here by mistake, click your browser\'s back button.', expectedNewarticletext, helpPageTitle = 'Help:Foobar'; mw.messages.set( 'foobar', helpPageTitle ); expectedNewarticletext = 'You have followed a link to a page that does not exist yet. To create the page, start typing in the box below (see the ' + 'foobar for more info). If you are here by mistake, click your browser\'s back button.'; mw.messages.set( 'newarticletext', newarticletextSource ); assert.htmlEqual( formatParse( 'newarticletext' ), expectedNewarticletext, 'Link with nested message' ); assert.strictEqual( formatParse( 'see-portal-url' ), 'Project:Community portal is an important community page.', 'Nested message' ); mw.messages.set( 'newarticletext-lowercase', newarticletextSource.replace( 'Int:Helppage', 'int:helppage' ) ); assert.htmlEqual( formatParse( 'newarticletext-lowercase' ), expectedNewarticletext, 'Link with nested message, lowercase include' ); mw.messages.set( 'uses-missing-int', '{{int:doesnt-exist}}' ); assert.strictEqual( formatParse( 'uses-missing-int' ), '⧼doesnt-exist⧽', 'int: where nested message does not exist' ); } ); QUnit.test( 'Ns', function ( assert ) { mw.messages.set( 'ns-template-talk', '{{ns:Template talk}}' ); assert.strictEqual( formatParse( 'ns-template-talk' ), 'Dyskusja szablonu', 'ns: returns localised namespace when used with a canonical namespace name' ); mw.messages.set( 'ns-10', '{{ns:10}}' ); assert.strictEqual( formatParse( 'ns-10' ), 'Szablon', 'ns: returns localised namespace when used with a namespace number' ); mw.messages.set( 'ns-unknown', '{{ns:doesnt-exist}}' ); assert.strictEqual( formatParse( 'ns-unknown' ), '', 'ns: returns empty string for unknown namespace name' ); mw.messages.set( 'ns-in-a-link', '[[{{ns:template}}:Foo]]' ); assert.strictEqual( formatParse( 'ns-in-a-link' ), 'Szablon:Foo', 'ns: works when used inside a wikilink' ); } ); // Tests that getMessageFunction is used for non-plain messages with curly braces or // square brackets, but not otherwise. QUnit.test( 'mw.Message.prototype.parser monkey-patch', function ( assert ) { var oldGMF, outerCalled, innerCalled; mw.messages.set( { 'curly-brace': '{{int:message}}', 'single-square-bracket': '[https://www.mediawiki.org/ MediaWiki]', 'double-square-bracket': '[[Some page]]', regular: 'Other message' } ); oldGMF = mw.jqueryMsg.getMessageFunction; mw.jqueryMsg.getMessageFunction = function () { outerCalled = true; return function () { innerCalled = true; }; }; function verifyGetMessageFunction( key, format, shouldCall ) { var message; outerCalled = false; innerCalled = false; message = mw.message( key ); message[ format ](); assert.strictEqual( outerCalled, shouldCall, 'Outer function called for ' + key ); assert.strictEqual( innerCalled, shouldCall, 'Inner function called for ' + key ); delete mw.messages[ format ]; } verifyGetMessageFunction( 'curly-brace', 'parse', true ); verifyGetMessageFunction( 'curly-brace', 'plain', false ); verifyGetMessageFunction( 'single-square-bracket', 'parse', true ); verifyGetMessageFunction( 'single-square-bracket', 'plain', false ); verifyGetMessageFunction( 'double-square-bracket', 'parse', true ); verifyGetMessageFunction( 'double-square-bracket', 'plain', false ); verifyGetMessageFunction( 'regular', 'parse', false ); verifyGetMessageFunction( 'regular', 'plain', false ); verifyGetMessageFunction( 'jquerymsg-test-pagetriage-del-talk-page-notify-summary', 'plain', false ); verifyGetMessageFunction( 'jquerymsg-test-categorytree-collapse-bullet', 'plain', false ); verifyGetMessageFunction( 'jquerymsg-test-wikieditor-toolbar-help-content-signature-result', 'plain', false ); mw.jqueryMsg.getMessageFunction = oldGMF; } ); formatnumTests = [ { lang: 'en', number: 987654321.654321, result: '987,654,321.654', description: 'formatnum test for English, decimal separator' }, { lang: 'ar', number: 987654321.654321, result: '٩٨٧٬٦٥٤٬٣٢١٫٦٥٤', description: 'formatnum test for Arabic, with decimal separator' }, { lang: 'ar', number: '٩٨٧٦٥٤٣٢١٫٦٥٤٣٢١', result: '987654321', integer: true, description: 'formatnum test for Arabic, with decimal separator, reverse' }, { lang: 'ar', number: -12.89, result: '-١٢٫٨٩', description: 'formatnum test for Arabic, negative number' }, { lang: 'ar', number: '-١٢٫٨٩', result: '-12', integer: true, description: 'formatnum test for Arabic, negative number, reverse' }, { lang: 'nl', number: 987654321.654321, result: '987.654.321,654', description: 'formatnum test for Nederlands, decimal separator' }, { lang: 'nl', number: -12.89, result: '-12,89', description: 'formatnum test for Nederlands, negative number' }, { lang: 'nl', number: '.89', result: '0,89', description: 'formatnum test for Nederlands' }, { lang: 'nl', number: 'invalidnumber', result: 'invalidnumber', description: 'formatnum test for Nederlands, invalid number' }, { lang: 'ml', number: '1000000000', result: '1,00,00,00,000', description: 'formatnum test for Malayalam' }, { lang: 'ml', number: '-1000000000', result: '-1,00,00,00,000', description: 'formatnum test for Malayalam, negative number' }, /* * This will fail because of wrong pattern for ml in MW(different from CLDR) { lang: 'ml', number: '1000000000.000', result: '1,00,00,00,000.000', description: 'formatnum test for Malayalam with decimal place' }, */ { lang: 'hi', number: '123456789.123456789', result: '१२,३४,५६,७८९', description: 'formatnum test for Hindi' }, { lang: 'hi', number: '१२,३४,५६,७८९', result: '१२,३४,५६,७८९', description: 'formatnum test for Hindi, Devanagari digits passed' }, { lang: 'hi', number: '१,२३,४५६', result: '123456', integer: true, description: 'formatnum test for Hindi, Devanagari digits passed to get integer value' } ]; QUnit.test( 'formatnum', function ( assert ) { var queue; mw.messages.set( 'formatnum-msg', '{{formatnum:$1}}' ); mw.messages.set( 'formatnum-msg-int', '{{formatnum:$1|R}}' ); queue = formatnumTests.map( function ( test ) { var done = assert.async(); return function ( next, abort ) { getMwLanguage( test.lang ) .then( function ( langClass ) { var parser; // The unit tests perform hot-reloading of mw.language (in hacky way). // For the languages/*.js script files to work, they need to statically // access mw.language.getData() for the "current" language. mw.language = langClass; mw.config.set( 'wgUserLanguage', test.lang ); parser = new mw.jqueryMsg.Parser( { language: langClass } ); assert.strictEqual( parser.parse( test.integer ? 'formatnum-msg-int' : 'formatnum-msg', [ test.number ] ).html(), test.result, test.description ); }, function () { assert.ok( false, 'Language "' + test.lang + '" failed to load' ); } ) .then( done, done ) .then( next, abort ); }; } ); process( queue ); } ); // HTML in wikitext QUnit.test( 'HTML', function ( assert ) { mw.messages.set( 'jquerymsg-italics-msg', 'Very important' ); assertBothModes( assert, [ 'jquerymsg-italics-msg' ], mw.messages.get( 'jquerymsg-italics-msg' ), 'Simple italics unchanged' ); mw.messages.set( 'jquerymsg-bold-msg', 'Strong speaker' ); assertBothModes( assert, [ 'jquerymsg-bold-msg' ], mw.messages.get( 'jquerymsg-bold-msg' ), 'Simple bold unchanged' ); mw.messages.set( 'jquerymsg-bold-italics-msg', 'It is key' ); assertBothModes( assert, [ 'jquerymsg-bold-italics-msg' ], mw.messages.get( 'jquerymsg-bold-italics-msg' ), 'Bold and italics nesting order preserved' ); mw.messages.set( 'jquerymsg-italics-bold-msg', 'It is vital' ); assertBothModes( assert, [ 'jquerymsg-italics-bold-msg' ], mw.messages.get( 'jquerymsg-italics-bold-msg' ), 'Italics and bold nesting order preserved' ); mw.messages.set( 'jquerymsg-italics-with-link', 'An italicized [[link|wiki-link]]' ); assert.htmlEqual( formatParse( 'jquerymsg-italics-with-link' ), 'An italicized wiki-link', 'Italics with link inside in parse mode' ); assert.strictEqual( formatText( 'jquerymsg-italics-with-link' ), mw.messages.get( 'jquerymsg-italics-with-link' ), 'Italics with link unchanged in text mode' ); mw.messages.set( 'jquerymsg-italics-id-class', 'Foo' ); assert.htmlEqual( formatParse( 'jquerymsg-italics-id-class' ), mw.messages.get( 'jquerymsg-italics-id-class' ), 'ID and class are allowed' ); mw.messages.set( 'jquerymsg-italics-onclick', 'Foo' ); assert.htmlEqual( formatParse( 'jquerymsg-italics-onclick' ), '<i onclick="alert(\'foo\')">Foo</i>', 'element with onclick is escaped because it is not allowed' ); mw.messages.set( 'jquerymsg-script-msg', '' ); assert.htmlEqual( formatParse( 'jquerymsg-script-msg' ), '<script >alert( "Who put this tag here?" );</script>', 'Tag outside whitelist escaped in parse mode' ); assert.strictEqual( formatText( 'jquerymsg-script-msg' ), mw.messages.get( 'jquerymsg-script-msg' ), 'Tag outside whitelist unchanged in text mode' ); mw.messages.set( 'jquerymsg-script-link-msg', '' ); assert.htmlEqual( formatParse( 'jquerymsg-script-link-msg' ), '<script>bar</script>', 'Script tag text is escaped because that element is not allowed, but link inside is still HTML' ); mw.messages.set( 'jquerymsg-mismatched-html', 'test' ); assert.htmlEqual( formatParse( 'jquerymsg-mismatched-html' ), '<i class="important">test</b>', 'Mismatched HTML start and end tag treated as text' ); mw.messages.set( 'jquerymsg-script-and-external-link', ' [http://example.com Foo bar]' ); assert.htmlEqual( formatParse( 'jquerymsg-script-and-external-link' ), '<script>alert( "jquerymsg-script-and-external-link test" );</script> Foo bar', 'HTML tags in external links not interfering with escaping of other tags' ); mw.messages.set( 'jquerymsg-link-script', '[http://example.com ]' ); assert.htmlEqual( formatParse( 'jquerymsg-link-script' ), '<script>alert( "jquerymsg-link-script test" );</script>', 'Non-whitelisted HTML tag in external link anchor treated as text' ); // Intentionally not using htmlEqual for the quote tests mw.messages.set( 'jquerymsg-double-quotes-preserved', 'Double' ); assert.strictEqual( formatParse( 'jquerymsg-double-quotes-preserved' ), mw.messages.get( 'jquerymsg-double-quotes-preserved' ), 'Attributes with double quotes are preserved as such' ); mw.messages.set( 'jquerymsg-single-quotes-normalized-to-double', 'Single' ); assert.strictEqual( formatParse( 'jquerymsg-single-quotes-normalized-to-double' ), 'Single', 'Attributes with single quotes are normalized to double' ); mw.messages.set( 'jquerymsg-escaped-double-quotes-attribute', 'Styled' ); assert.htmlEqual( formatParse( 'jquerymsg-escaped-double-quotes-attribute' ), mw.messages.get( 'jquerymsg-escaped-double-quotes-attribute' ), 'Escaped attributes are parsed correctly' ); mw.messages.set( 'jquerymsg-escaped-single-quotes-attribute', 'Styled' ); assert.htmlEqual( formatParse( 'jquerymsg-escaped-single-quotes-attribute' ), mw.messages.get( 'jquerymsg-escaped-single-quotes-attribute' ), 'Escaped attributes are parsed correctly' ); mw.messages.set( 'jquerymsg-wikitext-contents-parsed', '[http://example.com Example]' ); assert.htmlEqual( formatParse( 'jquerymsg-wikitext-contents-parsed' ), 'Example', 'Contents of valid tag are treated as wikitext, so external link is parsed' ); mw.messages.set( 'jquerymsg-wikitext-contents-script', '' ); assert.htmlEqual( formatParse( 'jquerymsg-wikitext-contents-script' ), '<script>Script inside</script>', 'Contents of valid tag are treated as wikitext, so invalid HTML element is treated as text' ); mw.messages.set( 'jquerymsg-unclosed-tag', 'Foobar' ); assert.htmlEqual( formatParse( 'jquerymsg-unclosed-tag' ), 'Foo<tag>bar', 'Nonsupported unclosed tags are escaped' ); mw.messages.set( 'jquerymsg-self-closing-tag', 'Foobar' ); assert.htmlEqual( formatParse( 'jquerymsg-self-closing-tag' ), 'Foo<tag/>bar', 'Self-closing tags don\'t cause a parse error' ); mw.messages.set( 'jquerymsg-asciialphabetliteral-regression', '>>="dir">asd' ); assert.htmlEqual( formatParse( 'jquerymsg-asciialphabetliteral-regression' ), '>>="dir">asd', 'Regression test for bad "asciiAlphabetLiteral" definition' ); mw.messages.set( 'jquerymsg-entities1', 'A&B' ); mw.messages.set( 'jquerymsg-entities2', 'A>B' ); mw.messages.set( 'jquerymsg-entities3', 'A→B' ); assert.htmlEqual( formatParse( 'jquerymsg-entities1' ), 'A&B', 'Lone "&" is escaped in text' ); assert.htmlEqual( formatParse( 'jquerymsg-entities2' ), 'A&gt;B', '">" entity is double-escaped in text' // (WHY?) ); assert.htmlEqual( formatParse( 'jquerymsg-entities3' ), 'A&rarr;B', '"→" entity is double-escaped in text' ); mw.messages.set( 'jquerymsg-entities-attr1', '' ); mw.messages.set( 'jquerymsg-entities-attr2', '' ); mw.messages.set( 'jquerymsg-entities-attr3', '' ); assert.htmlEqual( formatParse( 'jquerymsg-entities-attr1' ), '', 'Lone "&" is escaped in attribute' ); assert.htmlEqual( formatParse( 'jquerymsg-entities-attr2' ), '', '">" entity is not double-escaped in attribute' // (WHY?) ); assert.htmlEqual( formatParse( 'jquerymsg-entities-attr3' ), '', '"→" entity is double-escaped in attribute' ); } ); QUnit.test( 'Nowiki', function ( assert ) { mw.messages.set( 'jquerymsg-nowiki-link', 'Foo [[bar]] baz.' ); assert.strictEqual( formatParse( 'jquerymsg-nowiki-link' ), 'Foo [[bar]] baz.', 'Link inside nowiki is not parsed' ); mw.messages.set( 'jquerymsg-nowiki-htmltag', 'Foo bar baz.' ); assert.strictEqual( formatParse( 'jquerymsg-nowiki-htmltag' ), 'Foo <b>bar</b> baz.', 'HTML inside nowiki is not parsed and escaped' ); mw.messages.set( 'jquerymsg-nowiki-template', 'Foo {{bar}} baz.' ); assert.strictEqual( formatParse( 'jquerymsg-nowiki-template' ), 'Foo {{bar}} baz.', 'Template inside nowiki is not parsed and does not cause a parse error' ); } ); QUnit.test( 'Behavior in case of invalid wikitext', function ( assert ) { var logSpy; mw.messages.set( 'invalid-wikitext', '{{FAIL}}' ); this.suppressWarnings(); logSpy = this.sandbox.spy( mw.log, 'warn' ); assert.strictEqual( formatParse( 'invalid-wikitext' ), '<b>{{FAIL}}</b>', 'Invalid wikitext: \'parse\' format' ); assert.strictEqual( formatText( 'invalid-wikitext' ), '{{FAIL}}', 'Invalid wikitext: \'text\' format' ); assert.strictEqual( logSpy.callCount, 2, 'mw.log.warn calls' ); } ); QUnit.test( 'Non-string parameters to various functions', function ( assert ) { var i, cases; // For jquery-param-grammar mw.language.setData( 'en', 'grammarTransformations', { test: [ [ 'x', 'y' ] ] } ); cases = [ { key: 'jquery-param-wikilink', msg: '[[$1]] [[$1|a]]', expected: 'x a' }, { key: 'jquery-param-plural', msg: '{{PLURAL:$1|a|b}}', expected: 'b' }, { key: 'jquery-param-gender', msg: '{{GENDER:$1|a|b}}', expected: 'a' }, { key: 'jquery-param-grammar', msg: '{{GRAMMAR:test|$1}}', expected: '{{GRAMMAR:test|$1}}' }, { key: 'jquery-param-int', msg: '{{int:$1}}', expected: '{{int:$1}}' }, { key: 'jquery-param-ns', msg: '{{ns:$1}}', expected: '' }, { key: 'jquery-param-formatnum', msg: '{{formatnum:$1}}', expected: '[object Object]' }, { key: 'jquery-param-case', msg: '{{lc:$1}} {{uc:$1}} {{lcfirst:$1}} {{ucfirst:$1}}', expected: 'x X x X' } ]; for ( i = 0; i < cases.length; i++ ) { mw.messages.set( cases[ i ].key, cases[ i ].msg ); assert.strictEqual( mw.message( cases[ i ].key, $( '' ).text( 'x' ) ).parse(), cases[ i ].expected, cases[ i ].key ); } } ); QUnit.test( 'Integration', function ( assert ) { var expected, msg; expected = 'Bold!'; mw.messages.set( 'integration-test', '[[Bold]]!' ); assert.strictEqual( mw.message( 'integration-test' ).parse(), expected, 'mw.message().parse() works correctly' ); assert.strictEqual( $( '' ).msg( 'integration-test' ).html(), expected, 'jQuery plugin $.fn.msg() works correctly' ); mw.messages.set( 'integration-test-extlink', '[$1 Link]' ); msg = mw.message( 'integration-test-extlink', $( '' ).attr( 'href', 'http://example.com/' ) ); msg.parse(); // Not a no-op assert.strictEqual( msg.parse(), 'Link', 'Calling .parse() multiple times does not duplicate link contents' ); } ); QUnit.test( 'setParserDefaults', function ( assert ) { mw.jqueryMsg.setParserDefaults( { magic: { FOO: 'foo', BAR: 'bar' } } ); assert.deepEqual( mw.jqueryMsg.getParserDefaults().magic, { FOO: 'foo', BAR: 'bar' }, 'setParserDefaults is shallow by default' ); mw.jqueryMsg.setParserDefaults( { magic: { BAZ: 'baz' } }, true ); assert.deepEqual( mw.jqueryMsg.getParserDefaults().magic, { FOO: 'foo', BAR: 'bar', BAZ: 'baz' }, 'setParserDefaults is deep if requested' ); } ); }() );