Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
21 changes: 18 additions & 3 deletions src/prism.c
Original file line number Diff line number Diff line change
Expand Up @@ -19047,8 +19047,11 @@ parse_parentheses(pm_parser_t *parser, pm_binding_power_t binding_power, uint8_t
lex_state_set(parser, PM_LEX_STATE_ENDARG);
}

parser_lex(parser);
/* Pop before consuming the closing `)` so the following token (e.g. a
* `do`) is lexed in the enclosing context rather than as a block
* belonging to the parenthesized expression. */
pm_accepts_block_stack_pop(parser);
parser_lex(parser);
pop_block_exits(parser, previous_block_exits);

if (PM_NODE_TYPE_P(statement, PM_MULTI_TARGET_NODE) || PM_NODE_TYPE_P(statement, PM_SPLAT_NODE)) {
Expand Down Expand Up @@ -19319,14 +19322,20 @@ parse_expression_prefix(pm_parser_t *parser, pm_binding_power_t binding_power, u

accept1(parser, PM_TOKEN_NEWLINE);

/* Pop before consuming the closing `]` so the following token (e.g.
* a `do`) is lexed in the enclosing context rather than as a block
* belonging to the array's interior. Otherwise a `do` block would
* wrongly bind to a command with an array argument, as in
* `foo(m [] do end)`. */
pm_accepts_block_stack_pop(parser);

if (!accept1(parser, PM_TOKEN_BRACKET_RIGHT)) {
PM_PARSER_ERR_TOKEN_FORMAT(parser, &parser->current, PM_ERR_ARRAY_TERM, pm_token_str(parser->current.type));
parser->previous.start = parser->previous.end;
parser->previous.type = 0;
}

pm_array_node_close_set(parser, array, &parser->previous);
pm_accepts_block_stack_pop(parser);

return UP(array);
}
Expand Down Expand Up @@ -20645,6 +20654,11 @@ parse_expression_prefix(pm_parser_t *parser, pm_binding_power_t binding_power, u
}

parser_warn_indentation_mismatch(parser, opening_newline_index, &operator, false, false);

/* Pop before consuming the closing `}` so the following token
* (e.g. a `do`) is lexed in the enclosing context rather than
* as a block belonging to the lambda's interior. */
pm_accepts_block_stack_pop(parser);
expect1_opening(parser, PM_TOKEN_BRACE_RIGHT, PM_ERR_LAMBDA_TERM_BRACE, &opening);
} else {
expect1(parser, PM_TOKEN_KEYWORD_DO, PM_ERR_LAMBDA_OPEN);
Expand All @@ -20661,6 +20675,8 @@ parse_expression_prefix(pm_parser_t *parser, pm_binding_power_t binding_power, u
parser_warn_indentation_mismatch(parser, opening_newline_index, &operator, false, false);
}

/* As with the brace case above, pop before consuming `end`. */
pm_accepts_block_stack_pop(parser);
expect1_opening(parser, PM_TOKEN_KEYWORD_END, PM_ERR_LAMBDA_TERM_END, &operator);
}

Expand All @@ -20669,7 +20685,6 @@ parse_expression_prefix(pm_parser_t *parser, pm_binding_power_t binding_power, u
pm_node_t *parameters = parse_blocklike_parameters(parser, UP(block_parameters), &operator, &parser->previous);

pm_parser_scope_pop(parser);
pm_accepts_block_stack_pop(parser);

return UP(pm_lambda_node_create(parser, &locals, &operator, &opening, &parser->previous, parameters, body));
}
Expand Down
21 changes: 21 additions & 0 deletions test/prism/errors/do_block_in_command_args.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
foo(m [] do end)
^~ unexpected 'do'; expected a `)` to close the arguments
^~ unexpected 'do', expecting end-of-input
^~ unexpected 'do', ignoring it
^~~ unexpected 'end', ignoring it
^ unexpected ')', ignoring it

foo(m -> {} do end)
^~ unexpected 'do'; expected a `)` to close the arguments
^~ unexpected 'do', expecting end-of-input
^~ unexpected 'do', ignoring it
^~~ unexpected 'end', ignoring it
^ unexpected ')', ignoring it

foo(m (1) do end)
^~ unexpected 'do'; expected a `)` to close the arguments
^~ unexpected 'do', expecting end-of-input
^~ unexpected 'do', ignoring it
^~~ unexpected 'end', ignoring it
^ unexpected ')', ignoring it

Loading