Skip to content

Commit 7f9ac14

Browse files
committed
feat(power-assert-formatter): add support for yield statements
From what I can tell, there is no way to get esprima to parse a yield statement that is not wrapped in a generator function. To work around this: try to parse normally first, then try parsing by wrapping with a generator statement. If it works the second time, then we assume it is due to a yield statement.
1 parent 5fc8773 commit 7f9ac14

File tree

3 files changed

+89
-10
lines changed

3 files changed

+89
-10
lines changed

lib/traverse.js

+30-3
Original file line numberDiff line numberDiff line change
@@ -24,14 +24,14 @@ ContextTraversal.prototype.traverse = function () {
2424
};
2525

2626
function onEachEsNode(arg, jsCode, callback) {
27-
var jsAST = esprima.parse(jsCode, {sourceType: 'module', tolerant: true, loc: true, tokens: true, raw: true});
28-
var tokens = jsAST.tokens;
27+
var parseResult = parse(jsCode);
28+
var tokens = parseResult.tokens;
2929
var espathToValue = reduce(arg.events, function (accum, ev) {
3030
accum[ev.espath] = ev.value;
3131
return accum;
3232
}, {});
3333
var nodeStack = [];
34-
estraverse.traverse(extractExpressionFrom(jsAST), {
34+
estraverse.traverse(parseResult.expression, {
3535
enter: function (currentNode, parentNode) {
3636
var esNode = new EsNode(this.path(), currentNode, parentNode, espathToValue, jsCode, tokens);
3737
if (1 < nodeStack.length) {
@@ -46,6 +46,33 @@ function onEachEsNode(arg, jsCode, callback) {
4646
});
4747
}
4848

49+
function parserOptions() {
50+
return {sourceType: 'module', tolerant: true, loc: true, tokens: true, raw: true};
51+
}
52+
53+
function wrappedInGenerator(jsCode) {
54+
return 'function *wrapper() {\n' + jsCode + '\n}';
55+
}
56+
57+
function parse(jsCode) {
58+
var ast;
59+
var top;
60+
try {
61+
// Try parsing the raw line of code, without wrapping.
62+
top = ast = esprima.parse(jsCode, parserOptions());
63+
} catch (e) {
64+
// Esprima will throw an error if YieldExpressions are not inside a generator function.
65+
// Since we only have a single line of code to work with; wrap it with a dummy generator function;
66+
// then unwrap the original code from the resulting AST.
67+
ast = esprima.parse(wrappedInGenerator(jsCode), parserOptions());
68+
top = ast.body[0].body;
69+
}
70+
return {
71+
tokens: ast.tokens,
72+
expression: extractExpressionFrom(top)
73+
};
74+
}
75+
4976
function extractExpressionFrom (tree) {
5077
var expressionStatement = tree.body[0];
5178
var expression = expressionStatement.expression;

package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -27,11 +27,12 @@
2727
"babel-core": "^5.8.22",
2828
"babel-plugin-espower": "^1.0.0",
2929
"blanket": "^1.1.7",
30+
"bluebird": "^3.0.5",
3031
"browserify": "^11.0.1",
3132
"del": "^1.2.1",
3233
"empower": "^1.0.0",
3334
"escodegen": "^1.6.1",
34-
"espower": "^1.0.6",
35+
"espower": "^1.1.0",
3536
"gulp": "^3.9.0",
3637
"gulp-derequire": "^2.1.0",
3738
"gulp-dereserve": "^0.2.1",
@@ -44,6 +45,7 @@
4445
"licensify": "^2.0.1",
4546
"mocha": "^2.2.5",
4647
"mocha-lcov-reporter": "^0.0.2",
48+
"regenerator": "^0.8.41",
4749
"through2": "^2.0.0",
4850
"vinyl-source-stream": "^1.1.0"
4951
},

test/es6_test.js

+56-6
Original file line numberDiff line numberDiff line change
@@ -16,12 +16,28 @@ function weave (line) {
1616
}).code;
1717
}
1818

19-
function assertPowerAssertContextFormatting (body, expectedLines) {
20-
try {
21-
body();
22-
baseAssert.fail('AssertionError should be thrown');
23-
} catch (e) {
24-
baseAssert.equal(e.message, expectedLines.join('\n'));
19+
function assertPowerAssertContextFormatting (body, expectedLines, done) {
20+
if (done) {
21+
baseAssert.equal(body.length, 1, 'body should accept a "done" callback');
22+
body(function (e) {
23+
try {
24+
if (!e) {
25+
baseAssert.fail('AssertionError should be thrown');
26+
}
27+
baseAssert.equal(e.message, expectedLines.join('\n'));
28+
} catch (err) {
29+
return done(err);
30+
}
31+
done();
32+
});
33+
} else {
34+
baseAssert.equal(body.length, 0, 'assertPowerAssertContextFormatting must be passed a "done" callback if "body" needs to run async');
35+
try {
36+
body();
37+
baseAssert.fail('AssertionError should be thrown');
38+
} catch (e) {
39+
baseAssert.equal(e.message, expectedLines.join('\n'));
40+
}
2541
}
2642
}
2743

@@ -91,4 +107,38 @@ suite('ES6 features', function () {
91107
]);
92108
});
93109

110+
test('Yield Statements', function (done) {
111+
assertPowerAssertContextFormatting(function (done) {
112+
var Promise = require('bluebird');
113+
var regeneratorRuntime = require('regenerator/runtime-module');
114+
var big = 'big';
115+
eval(weave([
116+
'function bigOrSmall(size) {',
117+
' return Promise.resolve(size > 100 ? "big" : "small");',
118+
'}',
119+
'',
120+
'function *myGenerator (input) {',
121+
' assert((yield bigOrSmall(input)) === big);',
122+
'}',
123+
'',
124+
'var gen = myGenerator(3);',
125+
'gen.next().value.then((val) => gen.next(val)).catch(done);'
126+
].join('\n')));
127+
}, [
128+
' # test/some_test.js:6',
129+
' ',
130+
' assert((yield bigOrSmall(input)) === big)',
131+
' | | | | ',
132+
' | | | "big"',
133+
' "small" 3 false ',
134+
' ',
135+
' --- [string] big',
136+
' +++ [string] yield bigOrSmall(input)',
137+
' @@ -1,3 +1,5 @@',
138+
' -big',
139+
' +small',
140+
' ',
141+
' '
142+
], done);
143+
});
94144
});

0 commit comments

Comments
 (0)