You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
397 lines
14 KiB
JavaScript
397 lines
14 KiB
JavaScript
5 years ago
|
'use strict';
|
||
|
/* Simple test script that doesn't need mocha or similar - it just parses stuff and checks the returned AST */
|
||
|
var acorn = require('acorn');
|
||
|
var colors = require('colors');
|
||
|
require('../')(acorn);
|
||
|
function parse(code, pluginOptions, scriptType) {
|
||
|
if (Array.isArray(code)) {
|
||
|
code = code.join('\n');
|
||
|
}
|
||
|
return acorn.parse(code, {
|
||
|
sourceType: scriptType,
|
||
|
ecmaVersion: 8,
|
||
|
locations: true,
|
||
|
ranges: true,
|
||
|
plugins: {
|
||
|
asyncawait: pluginOptions || {}
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function isIdentThenFnDecl(ast) {
|
||
|
return ast.body[0].type === 'ExpressionStatement' && ast.body[0].expression.type === 'Identifier' && ast.body[0].expression.name === 'async' && !ast.body[1].async === true && ast.body[1].type == "FunctionDeclaration";
|
||
|
}
|
||
|
|
||
|
function isAsyncFnDecl(ast) {
|
||
|
return ast.body[0].async === true && ast.body[0].type === "FunctionDeclaration";
|
||
|
}
|
||
|
|
||
|
function isAsyncFnExpr(ast) {
|
||
|
return ast.body[0].expression.async === true && ast.body[0].expression.type === "ArrowFunctionExpression";
|
||
|
}
|
||
|
|
||
|
function isExprType(type) {
|
||
|
return function (ast, sourceType) {
|
||
|
return ast.body[0].type === 'ExpressionStatement' && ast.body[0].expression.type === type;
|
||
|
};
|
||
|
}
|
||
|
|
||
|
var tests = [
|
||
|
/* Standard behaviours */
|
||
|
{
|
||
|
desc: "Simple async function",
|
||
|
code: "async function x() { return undefined; }",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].async === true;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Simple async function expression",
|
||
|
code: "(async function (){ })",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].expression.async === true;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async function expression call (1)",
|
||
|
code: "(async function (){ }())",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].expression.callee.async === true;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async function expression call (2)",
|
||
|
code: "(async function (){ })()",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].expression.callee.async === true;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Await in async is AwaitExpression",
|
||
|
code: "async function x() { await(undefined); await undefined ; }",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].expression.type === 'AwaitExpression' && ast.body[0].body.body[1].expression.type === 'AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Await in function is identifier in 'script', illegal in 'module'",
|
||
|
code: "function x() { await(undefined); }",
|
||
|
pass: function (ast,scriptType) {
|
||
|
return scriptType === 'script'?ast.body[0].body.body[0].expression.callee.name === 'await':ast.indexOf("(1:15)")>=0;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async method {code}",
|
||
|
code: "var a = {async x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.properties[0].value.async;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async arrow",
|
||
|
code: "var a = async()=>0",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.async;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Abbreviated async arrow",
|
||
|
code: "var a = async b=>-b",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.async;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Parenthesized async arrow is a call",
|
||
|
code: "var a = async(b=>0)",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.type==='CallExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Await declaration fails in async function",
|
||
|
code: "async function x() { var await; }",
|
||
|
pass: function (ex, scriptType) {
|
||
|
return ex.indexOf("(1:25)")>=0
|
||
|
}
|
||
|
},{
|
||
|
desc: "Await function declaration fails in async function",
|
||
|
code: "async function x() { function await() {} }",
|
||
|
pass: function (ex, scriptType) {
|
||
|
return ex.indexOf("(1:30)")>=0
|
||
|
}
|
||
|
},{
|
||
|
desc: "Await reference fails in async function",
|
||
|
code: "async function x() { return 1+await; }",
|
||
|
pass: function (ex) {
|
||
|
return !!ex.match(/\(1:3[05]\)/);
|
||
|
}
|
||
|
},{
|
||
|
desc: "{code} is an async FunctionExpression",
|
||
|
code: "async ()=>0",
|
||
|
pass: isAsyncFnExpr
|
||
|
},{
|
||
|
desc: "{code} is a CallExpression",
|
||
|
code: "async(()=>0)",
|
||
|
pass: isExprType('CallExpression')
|
||
|
},{
|
||
|
desc: "{code} is an async FunctionDeclaration",
|
||
|
code: "async /* a */ function x(){}",
|
||
|
pass: isAsyncFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a reference to 'async' and a sync FunctionDeclaration",
|
||
|
code: "async /*\n*/function x(){}",
|
||
|
pass: isIdentThenFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a reference to 'async' and a sync FunctionDeclaration",
|
||
|
code: "async /* a */\nfunction x(){}",
|
||
|
pass: isIdentThenFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a reference to 'async' and a sync FunctionDeclaration",
|
||
|
code: "async\nfunction x(){}",
|
||
|
pass: isIdentThenFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a reference to 'async' and a sync FunctionDeclaration",
|
||
|
code: "async //\nfunction x(){}",
|
||
|
pass: isIdentThenFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a reference to 'async' and a sync FunctionDeclaration",
|
||
|
code: "async /*\n*/\nfunction x(){}",
|
||
|
pass: isIdentThenFnDecl
|
||
|
},{
|
||
|
desc: "{code} is a SyntaxError (when inAsyncFunction and awaitAnywhere option are defaults)",
|
||
|
code: "await x",
|
||
|
pass: function (ex, sourceType) {
|
||
|
return sourceType==='module' ? !!ex.match(/\(1:0\)/) : ex === "Unexpected token (1:6)";
|
||
|
}
|
||
|
},{
|
||
|
desc: "{code} is a CallExpression in scripts, and a SyntaxError in modules",
|
||
|
code: "await(x)",
|
||
|
pass: function(ast,sourceType) {
|
||
|
return sourceType==='module'?!!ast.match(/\(1:0\)/) :isExprType('CallExpression')(ast)
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async method 'constructor' is valid",
|
||
|
code: "var a = {async constructor(){}}",
|
||
|
pass: function (ast) {
|
||
|
var props = ast.body[0].declarations[0].init.properties ;
|
||
|
return (props[0].kind === 'init' && props[0].key.name==='constructor' && props[0].value.async)
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async class constructor fails",
|
||
|
code: "class a {async constructor(){}}",
|
||
|
pass: function (ex) {
|
||
|
return !!ex.match(/class constructor\(\) cannot be be async \(1:(15|9)\)/) || ex === "Constructor can't be an async method (1:15)";
|
||
|
}
|
||
|
},{
|
||
|
desc: "Async setter fails",
|
||
|
code: "var a = {async set x(y){}}",
|
||
|
pass: function (ex) {
|
||
|
return ex === "'set <member>(value)' cannot be be async (1:15)" || ex === "Unexpected token (1:19)";
|
||
|
}
|
||
|
},{
|
||
|
desc: "Deprecated async setter fails (use 'async set x')",
|
||
|
code: "var a = {set async x(y){}}",
|
||
|
pass: function (ex) {
|
||
|
return ex === "'set <member>(value)' cannot be be async (1:13)" || ex === "Unexpected token (1:19)";
|
||
|
}
|
||
|
},{
|
||
|
desc: "{code} getters/setters are not async",
|
||
|
code: "var a = {get x(){},set y(z){}}",
|
||
|
pass: function (ast) {
|
||
|
var props = ast.body[0].declarations[0].init.properties ;
|
||
|
return (props[0].kind === 'get' && props[0].key.name==='x' && !props[0].value.async)
|
||
|
&& (props[1].kind === 'set' && props[1].key.name==='y' && !props[1].value.async);
|
||
|
}
|
||
|
},{
|
||
|
desc: "{code} are methods, not getters/setters",
|
||
|
code: "var a = {async get(){},async set(){}}",
|
||
|
pass: function (ast) {
|
||
|
var props = ast.body[0].declarations[0].init.properties ;
|
||
|
return (props[0].kind === 'init' && props[0].key.name==='get' && props[0].value.async)
|
||
|
&& (props[1].kind === 'init' && props[1].key.name==='set' && props[1].value.async);
|
||
|
}
|
||
|
},{
|
||
|
desc: "In {code}, x is an sync getter",
|
||
|
code: "class a {get x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].kind==="get" && !ast.body[0].body.body[0].value.async && !ast.body[0].body.body[0].static ;
|
||
|
}
|
||
|
},{
|
||
|
desc: "In {code}, x is an static sync getter",
|
||
|
code: "class a {static get x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].kind==="get" && !ast.body[0].body.body[0].value.async && ast.body[0].body.body[0].static ;
|
||
|
}
|
||
|
},{
|
||
|
desc: "In {code}, x is an static sync method",
|
||
|
code: "class a {static async x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].kind==="method" && ast.body[0].body.body[0].value.async && ast.body[0].body.body[0].static ;
|
||
|
}
|
||
|
},{
|
||
|
desc: "{code} are a getters/setters, not methods",
|
||
|
code: "var a = {get async(){},set async(x){}}",
|
||
|
pass: function (ast) {
|
||
|
var props = ast.body[0].declarations[0].init.properties ;
|
||
|
return (props[0].kind === 'get' && props[0].key.name==='async' && !props[0].value.async)
|
||
|
&& (props[1].kind === 'set' && props[1].key.name==='async' && !props[1].value.async);
|
||
|
}
|
||
|
},
|
||
|
/* Extended syntax behaviour for Nodent */
|
||
|
{
|
||
|
desc: "Nodent:".grey+" In {code}, get is a static method",
|
||
|
code: "class Foo { static get(v) {} }",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].type==='MethodDefinition'
|
||
|
&& ast.body[0].body.body[0].key.name === 'get'
|
||
|
&& ast.body[0].body.body[0].kind === "method"
|
||
|
&& ast.body[0].body.body[0].static;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, get is a non-static method",
|
||
|
code: "class Foo { get(v) {} }",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].type==='MethodDefinition'
|
||
|
&& ast.body[0].body.body[0].key.name === 'get'
|
||
|
&& ast.body[0].body.body[0].kind === "method"
|
||
|
&& !ast.body[0].body.body[0].static;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, get is a non-static getter",
|
||
|
code: "class Foo { get get() {} }",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].type==='MethodDefinition'
|
||
|
&& ast.body[0].body.body[0].key.name === 'get'
|
||
|
&& ast.body[0].body.body[0].kind === "get"
|
||
|
&& !ast.body[0].body.body[0].static;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, x is an async getter",
|
||
|
code: "var a = {async get x(){ await(0) }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.properties[0].value.async
|
||
|
&& ast.body[0].declarations[0].init.properties[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code} (deprecated), x is an async getter",
|
||
|
code: "var a = {get async x(){ await 0 }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.properties[0].value.async
|
||
|
&& ast.body[0].declarations[0].init.properties[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code} (deprecated), x is an async getter",
|
||
|
code: "var a = {get async x(){ await(0) }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].declarations[0].init.properties[0].value.async
|
||
|
&& ast.body[0].declarations[0].init.properties[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, x is an async getter",
|
||
|
code: "class a {async get x(){ await 0 }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].value.async
|
||
|
&& ast.body[0].body.body[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, x is an async getter",
|
||
|
code: "class a {async get x(){ await(0) }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].value.async
|
||
|
&& ast.body[0].body.body[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code} (deprecated), x is an async getter",
|
||
|
code: "class a {get async x(){ await 0 }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].value.async
|
||
|
&& ast.body[0].body.body[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code} (deprecated), x is an async getter",
|
||
|
code: "class a {get async x(){ await(0) }}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].value.async
|
||
|
&& ast.body[0].body.body[0].value.body.body[0].expression.type==='AwaitExpression';
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code}, x is an static async getter",
|
||
|
code: "class a {static async get x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].kind==="get" && ast.body[0].body.body[0].value.async && ast.body[0].body.body[0].static ;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" In {code} (deprecated), x is an static async getter",
|
||
|
code: "class a {static get async x(){}}",
|
||
|
pass: function (ast) {
|
||
|
return ast.body[0].body.body[0].kind==="get" && ast.body[0].body.body[0].value.async && ast.body[0].body.body[0].static ;
|
||
|
}
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" {code} is an AwaitExpression when inAsyncFunction option is true",
|
||
|
code: "await(x)",
|
||
|
options: {
|
||
|
inAsyncFunction: true
|
||
|
},
|
||
|
pass: isExprType('AwaitExpression')
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" {code} is an AwaitExpression when inAsyncFunction option is true",
|
||
|
code: "await x",
|
||
|
options: {
|
||
|
inAsyncFunction: true
|
||
|
},
|
||
|
pass: isExprType('AwaitExpression')
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" {code} is a CallExpression when awaitAnywhere option is true",
|
||
|
code: "await(x)",
|
||
|
options: {
|
||
|
awaitAnywhere: true
|
||
|
},
|
||
|
pass: isExprType('CallExpression')
|
||
|
},{
|
||
|
desc: "Nodent:".grey+" {code} is an AwaitExpression when awaitAnywhere option is true",
|
||
|
code: "await x",
|
||
|
options: {
|
||
|
awaitAnywhere: true
|
||
|
},
|
||
|
pass: isExprType('AwaitExpression')
|
||
|
}];
|
||
|
// TODO: Add tests for asyncExits, noAsyncGetters
|
||
|
|
||
|
var out = {
|
||
|
true: "pass".green,
|
||
|
false: "fail".red
|
||
|
};
|
||
|
var testNumber = +process.argv[2] || 0;
|
||
|
if (testNumber) {
|
||
|
tests = [tests[testNumber - 1]];
|
||
|
} else {
|
||
|
testNumber += 1;
|
||
|
}
|
||
|
var results = {
|
||
|
true: 0,
|
||
|
false: 0
|
||
|
};
|
||
|
|
||
|
tests.forEach(function (test, idx) {
|
||
|
['script','module'].forEach(function(scriptType){
|
||
|
var code = test.code.replace(/\n/g, ' <linefeed> ');
|
||
|
var desc = test.desc.replace('{code}', code.yellow);
|
||
|
var pass = function () {
|
||
|
var p = test.pass.apply(this, arguments);
|
||
|
results[p] += 1;
|
||
|
return p;
|
||
|
};
|
||
|
var prefix = idx + testNumber + " (" + scriptType + ", acorn v" + acorn.version+")\t" ;
|
||
|
try {
|
||
|
console.log(prefix, desc, out[pass(parse(test.code, test.options, scriptType),scriptType)]);
|
||
|
} catch (ex) {
|
||
|
try {
|
||
|
console.log(prefix, desc, ex.message.cyan, out[pass(ex.message,scriptType)]);
|
||
|
} catch (ex) {
|
||
|
console.log(prefix, desc, ex.message.magenta, out[false]);
|
||
|
results.false += 1;
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
}) ;
|
||
|
console.log('');
|
||
|
if (results.true)
|
||
|
console.log((results.true + " of " + tests.length*2 + " tests passed").green);
|
||
|
if (results.false) {
|
||
|
console.log((results.false + " of " + tests.length*2 + " tests failed").red);
|
||
|
var exit = new Error("Test failed") ;
|
||
|
exit.stack = "" ;
|
||
|
throw exit ;
|
||
|
}
|