Handle exprAllowed before ObjectLike is parsed (#12267)
* fix: disallow expression after `}` is consumed in parseObjectLike * refactor: avoid accessing this.prodParam in context update
This commit is contained in:
@@ -241,15 +241,15 @@ export default class ExpressionParser extends LValParser {
|
||||
const startLoc = this.state.startLoc;
|
||||
if (this.isContextual("yield")) {
|
||||
if (this.prodParam.hasYield) {
|
||||
// If we have [Yield] production, `yield` will start a YieldExpression thus
|
||||
// regex is allowed following. Otherwise `yield` is an identifier and regex
|
||||
// is disallowed in tt.name.updateContext
|
||||
this.state.exprAllowed = true;
|
||||
let left = this.parseYield();
|
||||
if (afterLeftParse) {
|
||||
left = afterLeftParse.call(this, left, startPos, startLoc);
|
||||
}
|
||||
return left;
|
||||
} else {
|
||||
// The tokenizer will assume an expression is allowed after
|
||||
// `yield`, but this isn't that kind of yield
|
||||
this.state.exprAllowed = false;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1603,14 +1603,13 @@ export default class ExpressionParser extends LValParser {
|
||||
node.properties = [];
|
||||
this.next();
|
||||
|
||||
while (!this.eat(close)) {
|
||||
while (!this.match(close)) {
|
||||
if (first) {
|
||||
first = false;
|
||||
} else {
|
||||
this.expect(tt.comma);
|
||||
if (this.match(close)) {
|
||||
this.addExtra(node, "trailingComma", this.state.lastTokStart);
|
||||
this.next();
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -1637,6 +1636,13 @@ export default class ExpressionParser extends LValParser {
|
||||
node.properties.push(prop);
|
||||
}
|
||||
|
||||
// The tokenizer uses `braceIsBlock` to detect whether `{` starts a block statement.
|
||||
// If `{` is a block statement, `exprAllowed` will be `true`.
|
||||
// However the tokenizer can not handle edge cases like `0 ? a : { a : 1 } / 2`, here
|
||||
// we update `exprAllowed` when an object-like is parsed.
|
||||
this.state.exprAllowed = false;
|
||||
this.next();
|
||||
|
||||
this.state.inFSharpPipelineDirectBody = oldInFSharpPipelineDirectBody;
|
||||
let type = "ObjectExpression";
|
||||
if (isPattern) {
|
||||
|
||||
@@ -40,6 +40,14 @@ export const types: {
|
||||
};
|
||||
|
||||
// Token-specific context update code
|
||||
// Note that we should avoid accessing `this.prodParam` in context update,
|
||||
// because it is executed immediately when last token is consumed, which may be
|
||||
// before `this.prodParam` is updated. e.g.
|
||||
// ```
|
||||
// function *g() { () => yield / 2 }
|
||||
// ```
|
||||
// When `=>` is eaten, the context update of `yield` is executed, however,
|
||||
// `this.prodParam` still has `[Yield]` production because it is not yet updated
|
||||
|
||||
tt.parenR.updateContext = tt.braceR.updateContext = function () {
|
||||
if (this.state.context.length === 1) {
|
||||
@@ -59,11 +67,10 @@ tt.name.updateContext = function (prevType) {
|
||||
let allowed = false;
|
||||
if (prevType !== tt.dot) {
|
||||
if (
|
||||
(this.state.value === "of" &&
|
||||
!this.state.exprAllowed &&
|
||||
prevType !== tt._function &&
|
||||
prevType !== tt._class) ||
|
||||
(this.state.value === "yield" && this.prodParam.hasYield)
|
||||
this.state.value === "of" &&
|
||||
!this.state.exprAllowed &&
|
||||
prevType !== tt._function &&
|
||||
prevType !== tt._class
|
||||
) {
|
||||
allowed = true;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user