Skip to content

Commit

Permalink
Warn for unexpected semicolon between else and opening brace.
Browse files Browse the repository at this point in the history
  • Loading branch information
someoneigna committed May 16, 2021
1 parent 384c318 commit a4ce1c3
Show file tree
Hide file tree
Showing 4 changed files with 39 additions and 20 deletions.
13 changes: 13 additions & 0 deletions docs/errors/E220.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
# E220: Semicolon after else may cause unexpected execution of its body.

A semicolon next to the `else` keyword can cause its body to be executed even when previous `if` statement condition is true.

if (true) {
console.log("true");
} else; {
console.log("& false");
}
> true
> & false

To avoid this behavior, remove the semicolon between the else keyword and the opening brace of the else block.
13 changes: 7 additions & 6 deletions src/quick-lint-js/error.h
Original file line number Diff line number Diff line change
Expand Up @@ -1090,6 +1090,13 @@
"expected variable name"), \
unexpected_token)) \
\
QLJS_ERROR_TYPE( \
error_unexpected_semicolon_after_else, "E202", \
{ source_code_span semicolon; }, \
.warning(QLJS_TRANSLATABLE("semicolon after else may be causing " \
"unexpected behavior"), \
semicolon)) \
\
QLJS_ERROR_TYPE( \
error_unmatched_indexing_bracket, "E055", \
{ source_code_span left_square; }, \
Expand Down Expand Up @@ -1139,12 +1146,6 @@
{ source_code_span continue_statement; }, \
.error(QLJS_TRANSLATABLE("continue can only be used inside of a loop"), \
continue_statement)) \
\
QLJS_ERROR_TYPE( \
error_redundant_semicolon_after_else, "E202", \
{ source_code_span semicolon; }, \
.warning(QLJS_TRANSLATABLE("redundant semicolon after else"), \
semicolon)) \
/* END */

namespace quick_lint_js {
Expand Down
16 changes: 10 additions & 6 deletions src/quick-lint-js/parse.h
Original file line number Diff line number Diff line change
Expand Up @@ -2515,14 +2515,18 @@ class parser {

if (this->peek().type == token_type::kw_else) {
this->skip();
if (this->peek().type == token_type::semicolon) {
source_code_span semicolon = this->peek().span();
this->error_reporter_->report(error_redundant_semicolon_after_else{
.semicolon = source_code_span(
semicolon.begin(), semicolon.end()),

bool semicolon_after_else = this->peek().type == token_type::semicolon;
const char8 *token_after_else_begin = this->peek().begin;

parse_and_visit_body();
if (semicolon_after_else &&
this->peek().type == token_type::left_curly &&
!this->peek().has_leading_newline) {
this->error_reporter_->report(error_unexpected_semicolon_after_else{
.semicolon = source_code_span(token_after_else_begin, token_after_else_begin+1)
});
}
parse_and_visit_body();
}
}

Expand Down
17 changes: 9 additions & 8 deletions test/test-parse-statement.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -578,19 +578,20 @@ TEST(test_parse, else_without_if) {
}
}

TEST(test_parse, else_redundant_semicolon) {
TEST(test_parse, else_unexpected_semicolon) {
{
spy_visitor v;
padded_string code(u8"if (cond) { body; } else; { body; }"_sv);
parser p(&code, &v);
EXPECT_TRUE(p.parse_and_visit_statement(v));
EXPECT_THAT(v.visits, ElementsAre("visit_variable_use", // cond
"visit_enter_block_scope", // (if)
"visit_variable_use", // body
"visit_exit_block_scope")); // (else)
EXPECT_THAT(v.errors, ElementsAre(ERROR_TYPE_FIELD(
error_redundant_semicolon_after_else, semicolon,
offsets_matcher(&code, strlen(u8"if (cond) { body; } else"), u8";"))));
EXPECT_THAT(v.visits, ElementsAre("visit_variable_use", // cond
"visit_enter_block_scope", // (if)
"visit_variable_use", // body
"visit_exit_block_scope")); // (if)
EXPECT_THAT(v.errors,
ElementsAre(ERROR_TYPE_FIELD(
error_unexpected_semicolon_after_else, semicolon,
offsets_matcher(&code, strlen(u8"if (cond) { body; } else"), u8";"))));
}
}

Expand Down

0 comments on commit a4ce1c3

Please sign in to comment.