This reverts commit 0a2b996278.
Reason for revert: Broke postgres tests.
Change-Id: I27d8e0c807ad5f0748b9611a4f3df84cc213fbe1
367 lines
9 KiB
PHP
367 lines
9 KiB
PHP
<?php
|
|
|
|
class JavaScriptMinifierTest extends PHPUnit\Framework\TestCase {
|
|
|
|
use MediaWikiCoversValidator;
|
|
|
|
protected function tearDown() {
|
|
parent::tearDown();
|
|
// Reset
|
|
$this->setMaxLineLength( 1000 );
|
|
}
|
|
|
|
private function setMaxLineLength( $val ) {
|
|
$classReflect = new ReflectionClass( JavaScriptMinifier::class );
|
|
$propertyReflect = $classReflect->getProperty( 'maxLineLength' );
|
|
$propertyReflect->setAccessible( true );
|
|
$propertyReflect->setValue( JavaScriptMinifier::class, $val );
|
|
}
|
|
|
|
public static function provideCases() {
|
|
return [
|
|
|
|
// Basic whitespace and comments that should be stripped entirely
|
|
[ "\r\t\f \v\n\r", "" ],
|
|
[ "/* Foo *\n*bar\n*/", "" ],
|
|
|
|
/**
|
|
* Slashes used inside block comments (T28931).
|
|
* At some point there was a bug that caused this comment to be ended at '* /',
|
|
* causing /M... to be left as the beginning of a regex.
|
|
*/
|
|
[
|
|
"/**\n * Foo\n * {\n * 'bar' : {\n * "
|
|
. "//Multiple rules with configurable operators\n * 'baz' : false\n * }\n */",
|
|
"" ],
|
|
|
|
/**
|
|
* ' Foo \' bar \
|
|
* baz \' quox ' .
|
|
*/
|
|
[
|
|
"' Foo \\' bar \\\n baz \\' quox ' .length",
|
|
"' Foo \\' bar \\\n baz \\' quox '.length"
|
|
],
|
|
[
|
|
"\" Foo \\\" bar \\\n baz \\\" quox \" .length",
|
|
"\" Foo \\\" bar \\\n baz \\\" quox \".length"
|
|
],
|
|
[ "// Foo b/ar baz", "" ],
|
|
[
|
|
"/ Foo \\/ bar [ / \\] / ] baz / .length",
|
|
"/ Foo \\/ bar [ / \\] / ] baz /.length"
|
|
],
|
|
|
|
// HTML comments
|
|
[ "<!-- Foo bar", "" ],
|
|
[ "<!-- Foo --> bar", "" ],
|
|
[ "--> Foo", "" ],
|
|
[ "x --> y", "x-->y" ],
|
|
|
|
// Semicolon insertion
|
|
[ "(function(){return\nx;})", "(function(){return\nx;})" ],
|
|
[ "throw\nx;", "throw\nx;" ],
|
|
[ "while(p){continue\nx;}", "while(p){continue\nx;}" ],
|
|
[ "while(p){break\nx;}", "while(p){break\nx;}" ],
|
|
[ "var\nx;", "var x;" ],
|
|
[ "x\ny;", "x\ny;" ],
|
|
[ "x\n++y;", "x\n++y;" ],
|
|
[ "x\n!y;", "x\n!y;" ],
|
|
[ "x\n{y}", "x\n{y}" ],
|
|
[ "x\n+y;", "x+y;" ],
|
|
[ "x\n(y);", "x(y);" ],
|
|
[ "5.\nx;", "5.\nx;" ],
|
|
[ "0xFF.\nx;", "0xFF.x;" ],
|
|
[ "5.3.\nx;", "5.3.x;" ],
|
|
|
|
// Cover failure case for incomplete hex literal
|
|
[ "0x;", false, false ],
|
|
|
|
// Cover failure case for number with no digits after E
|
|
[ "1.4E", false, false ],
|
|
|
|
// Cover failure case for number with several E
|
|
[ "1.4EE2", false, false ],
|
|
[ "1.4EE", false, false ],
|
|
|
|
// Cover failure case for number with several E (nonconsecutive)
|
|
// FIXME: This is invalid, but currently tolerated
|
|
[ "1.4E2E3", "1.4E2 E3", false ],
|
|
|
|
// Semicolon insertion between an expression having an inline
|
|
// comment after it, and a statement on the next line (T29046).
|
|
[
|
|
"var a = this //foo bar \n for ( b = 0; c < d; b++ ) {}",
|
|
"var a=this\nfor(b=0;c<d;b++){}"
|
|
],
|
|
|
|
// Cover failure case of incomplete regexp at end of file (T75556)
|
|
// FIXME: This is invalid, but currently tolerated
|
|
[ "*/", "*/", false ],
|
|
|
|
// Cover failure case of incomplete char class in regexp (T75556)
|
|
// FIXME: This is invalid, but currently tolerated
|
|
[ "/a[b/.test", "/a[b/.test", false ],
|
|
|
|
// Cover failure case of incomplete string at end of file (T75556)
|
|
// FIXME: This is invalid, but currently tolerated
|
|
[ "'a", "'a", false ],
|
|
|
|
// Token separation
|
|
[ "x in y", "x in y" ],
|
|
[ "/x/g in y", "/x/g in y" ],
|
|
[ "x in 30", "x in 30" ],
|
|
[ "x + ++ y", "x+ ++y" ],
|
|
[ "x ++ + y", "x++ +y" ],
|
|
[ "x / /y/.exec(z)", "x/ /y/.exec(z)" ],
|
|
|
|
// State machine
|
|
[ "/ x/g", "/ x/g" ],
|
|
[ "(function(){return/ x/g})", "(function(){return/ x/g})" ],
|
|
[ "+/ x/g", "+/ x/g" ],
|
|
[ "++/ x/g", "++/ x/g" ],
|
|
[ "x/ x/g", "x/x/g" ],
|
|
[ "(/ x/g)", "(/ x/g)" ],
|
|
[ "if(/ x/g);", "if(/ x/g);" ],
|
|
[ "(x/ x/g)", "(x/x/g)" ],
|
|
[ "([/ x/g])", "([/ x/g])" ],
|
|
[ "+x/ x/g", "+x/x/g" ],
|
|
[ "{}/ x/g", "{}/ x/g" ],
|
|
[ "+{}/ x/g", "+{}/x/g" ],
|
|
[ "(x)/ x/g", "(x)/x/g" ],
|
|
[ "if(x)/ x/g", "if(x)/ x/g" ],
|
|
[ "for(x;x;{}/ x/g);", "for(x;x;{}/x/g);" ],
|
|
[ "x;x;{}/ x/g", "x;x;{}/ x/g" ],
|
|
[ "x:{}/ x/g", "x:{}/ x/g" ],
|
|
[ "switch(x){case y?z:{}/ x/g:{}/ x/g;}", "switch(x){case y?z:{}/x/g:{}/ x/g;}" ],
|
|
[ "function x(){}/ x/g", "function x(){}/ x/g" ],
|
|
[ "+function x(){}/ x/g", "+function x(){}/x/g" ],
|
|
|
|
// Multiline quoted string
|
|
[ "var foo=\"\\\nblah\\\n\";", "var foo=\"\\\nblah\\\n\";" ],
|
|
|
|
// Multiline quoted string followed by string with spaces
|
|
[
|
|
"var foo=\"\\\nblah\\\n\";\nvar baz = \" foo \";\n",
|
|
"var foo=\"\\\nblah\\\n\";var baz=\" foo \";"
|
|
],
|
|
|
|
// URL in quoted string ( // is not a comment)
|
|
[
|
|
"aNode.setAttribute('href','http://foo.bar.org/baz');",
|
|
"aNode.setAttribute('href','http://foo.bar.org/baz');"
|
|
],
|
|
|
|
// URL in quoted string after multiline quoted string
|
|
[
|
|
"var foo=\"\\\nblah\\\n\";\naNode.setAttribute('href','http://foo.bar.org/baz');",
|
|
"var foo=\"\\\nblah\\\n\";aNode.setAttribute('href','http://foo.bar.org/baz');"
|
|
],
|
|
|
|
// Division vs. regex nastiness
|
|
[
|
|
"alert( (10+10) / '/'.charCodeAt( 0 ) + '//' );",
|
|
"alert((10+10)/'/'.charCodeAt(0)+'//');"
|
|
],
|
|
[ "if(1)/a /g.exec('Pa ss');", "if(1)/a /g.exec('Pa ss');" ],
|
|
|
|
// Unicode letter characters should pass through ok in identifiers (T33187)
|
|
[ "var KaŝSkatolVal = {}", 'var KaŝSkatolVal={}' ],
|
|
|
|
// Per spec unicode char escape values should work in identifiers,
|
|
// as long as it's a valid char. In future it might get normalized.
|
|
[ "var Ka\\u015dSkatolVal = {}", 'var Ka\\u015dSkatolVal={}' ],
|
|
|
|
// Some structures that might look invalid at first sight
|
|
[ "var a = 5.;", "var a=5.;" ],
|
|
[ "5.0.toString();", "5.0.toString();" ],
|
|
[ "5..toString();", "5..toString();" ],
|
|
// Cover failure case for too many decimal points
|
|
[ "5...toString();", false ],
|
|
[ "5.\n.toString();", '5..toString();' ],
|
|
|
|
// Boolean minification (!0 / !1)
|
|
[ "var a = { b: true };", "var a={b:!0};" ],
|
|
[ "var a = { true: 12 };", "var a={true:12};" ],
|
|
[ "a.true = 12;", "a.true=12;" ],
|
|
[ "a.foo = true;", "a.foo=!0;" ],
|
|
[ "a.foo = false;", "a.foo=!1;" ],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideCases
|
|
* @covers JavaScriptMinifier::minify
|
|
* @covers JavaScriptMinifier::parseError
|
|
*/
|
|
public function testMinifyOutput( $code, $expectedOutput, $expectedValid = true ) {
|
|
$minified = JavaScriptMinifier::minify( $code );
|
|
|
|
// JSMin+'s parser will throw an exception if output is not valid JS.
|
|
// suppression of warnings needed for stupid crap
|
|
if ( $expectedValid ) {
|
|
Wikimedia\suppressWarnings();
|
|
$parser = new JSParser();
|
|
Wikimedia\restoreWarnings();
|
|
$parser->parse( $minified, 'minify-test.js', 1 );
|
|
}
|
|
|
|
$this->assertEquals(
|
|
$expectedOutput,
|
|
$minified,
|
|
"Minified output should be in the form expected."
|
|
);
|
|
}
|
|
|
|
public static function provideLineBreaker() {
|
|
return [
|
|
[
|
|
// Regression tests for T34548.
|
|
// Must not break between 'E' and '+'.
|
|
'var name = 1.23456789E55;',
|
|
[
|
|
'var',
|
|
'name',
|
|
'=',
|
|
'1.23456789E55',
|
|
';',
|
|
],
|
|
],
|
|
[
|
|
'var name = 1.23456789E+5;',
|
|
[
|
|
'var',
|
|
'name',
|
|
'=',
|
|
'1.23456789E+5',
|
|
';',
|
|
],
|
|
],
|
|
[
|
|
'var name = 1.23456789E-5;',
|
|
[
|
|
'var',
|
|
'name',
|
|
'=',
|
|
'1.23456789E-5',
|
|
';',
|
|
],
|
|
],
|
|
[
|
|
// Must not break before '++'
|
|
'if(x++);',
|
|
[
|
|
'if',
|
|
'(',
|
|
'x++',
|
|
')',
|
|
';',
|
|
],
|
|
],
|
|
[
|
|
// Regression test for T201606.
|
|
// Must not break between 'return' and Expression.
|
|
// Was caused by bad state after '{}' in property value.
|
|
<<<JAVASCRIPT
|
|
call( function () {
|
|
try {
|
|
} catch (e) {
|
|
obj = {
|
|
key: 1 ? 0 : {}
|
|
};
|
|
}
|
|
return name === 'input';
|
|
} );
|
|
JAVASCRIPT
|
|
,
|
|
[
|
|
'call',
|
|
'(',
|
|
'function',
|
|
'(',
|
|
')',
|
|
'{',
|
|
'try',
|
|
'{',
|
|
'}',
|
|
'catch',
|
|
'(',
|
|
'e',
|
|
')',
|
|
'{',
|
|
'obj',
|
|
'=',
|
|
'{',
|
|
'key',
|
|
':',
|
|
'1',
|
|
'?',
|
|
'0',
|
|
':',
|
|
'{',
|
|
'}',
|
|
'}',
|
|
';',
|
|
'}',
|
|
// The return Statement:
|
|
// return [no LineTerminator here] Expression
|
|
'return name',
|
|
'===',
|
|
"'input'",
|
|
';',
|
|
'}',
|
|
')',
|
|
';',
|
|
]
|
|
],
|
|
[
|
|
// Regression test for T201606.
|
|
// Must not break between 'return' and Expression.
|
|
// Was caused by bad state after a ternary in the expression value
|
|
// for a key in an object literal.
|
|
<<<JAVASCRIPT
|
|
call( {
|
|
key: 1 ? 0 : function () {
|
|
return this;
|
|
}
|
|
} );
|
|
JAVASCRIPT
|
|
,
|
|
[
|
|
'call',
|
|
'(',
|
|
'{',
|
|
'key',
|
|
':',
|
|
'1',
|
|
'?',
|
|
'0',
|
|
':',
|
|
'function',
|
|
'(',
|
|
')',
|
|
'{',
|
|
'return this',
|
|
';',
|
|
'}',
|
|
'}',
|
|
')',
|
|
';',
|
|
]
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideLineBreaker
|
|
* @covers JavaScriptMinifier::minify
|
|
*/
|
|
public function testLineBreaker( $code, array $expectedLines ) {
|
|
$this->setMaxLineLength( 1 );
|
|
$actual = JavaScriptMinifier::minify( $code );
|
|
$this->assertEquals(
|
|
array_merge( [ '' ], $expectedLines ),
|
|
explode( "\n", $actual )
|
|
);
|
|
}
|
|
}
|