Skip to content

Commit 81acb09

Browse files
committed
Warn for possible unexpected behavior when else followed by a semicolon.
1 parent 1a92c01 commit 81acb09

File tree

4 files changed

+93
-0
lines changed

4 files changed

+93
-0
lines changed

docs/errors/E204.md

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
# E204: semicolon after else may be causing unexpected behavior.
2+
3+
A semicolon next to the `else` keyword may cause its body to be executed even when previous `if` statement condition is true.
4+
5+
if (true) {
6+
console.log("true");
7+
} else; {
8+
console.log("else");
9+
}
10+
11+
Example output
12+
```
13+
> true
14+
> else
15+
```
16+
17+
To avoid this behavior, remove the semicolon between the else keyword and the opening brace of the else block.
18+
19+
if (true) {
20+
console.log("true");
21+
} else {
22+
console.log("else");
23+
}
24+
25+
Or if the else body is not required remove it completely.
26+
27+
if (true) {
28+
console.log("true");
29+
}
30+
console.log("always");

src/quick-lint-js/error.h

+13
Original file line numberDiff line numberDiff line change
@@ -209,6 +209,12 @@
209209
QLJS_TRANSLATABLE("commas are not allowed after spread parameter"), \
210210
comma)) \
211211
\
212+
QLJS_ERROR_TYPE( \
213+
error_else_has_empty_body, "E205", { source_code_span where; }, \
214+
.warning(QLJS_TRANSLATABLE("else has empty body; consider removing the " \
215+
"redundant 'else'"), \
216+
where)) \
217+
\
212218
QLJS_ERROR_TYPE( \
213219
error_else_has_no_if, "E065", { source_code_span else_token; }, \
214220
.error(QLJS_TRANSLATABLE("'else' has no corresponding 'if'"), \
@@ -1048,6 +1054,13 @@
10481054
QLJS_TRANSLATABLE("for-of loop expression cannot have semicolons"), \
10491055
semicolon)) \
10501056
\
1057+
QLJS_ERROR_TYPE( \
1058+
error_unexpected_semicolon_after_else, "E204", \
1059+
{ source_code_span semicolon; }, \
1060+
.warning(QLJS_TRANSLATABLE("semicolon after else may be causing " \
1061+
"unexpected behavior"), \
1062+
semicolon)) \
1063+
\
10511064
QLJS_ERROR_TYPE( \
10521065
error_no_digits_in_binary_number, "E049", \
10531066
{ source_code_span characters; }, \

src/quick-lint-js/parse.h

+16
Original file line numberDiff line numberDiff line change
@@ -2516,7 +2516,23 @@ class parser {
25162516

25172517
if (this->peek().type == token_type::kw_else) {
25182518
this->skip();
2519+
2520+
const token token_after_else = this->peek();
25192521
parse_and_visit_body();
2522+
2523+
if (token_after_else.type == token_type::semicolon &&
2524+
this->peek().type == token_type::left_curly) {
2525+
2526+
if (this->peek().has_leading_newline) {
2527+
this->error_reporter_->report(error_else_has_empty_body{
2528+
.where = this->peek().span()
2529+
});
2530+
} else {
2531+
this->error_reporter_->report(error_unexpected_semicolon_after_else{
2532+
.semicolon = token_after_else.span()
2533+
});
2534+
}
2535+
}
25202536
}
25212537
}
25222538

test/test-parse-statement.cpp

+34
Original file line numberDiff line numberDiff line change
@@ -578,6 +578,40 @@ TEST(test_parse, else_without_if) {
578578
}
579579
}
580580

581+
TEST(test_parse, else_unexpected_semicolon) {
582+
{
583+
spy_visitor v;
584+
padded_string code(u8"if (cond) { body; } else; { body; }"_sv);
585+
parser p(&code, &v);
586+
EXPECT_TRUE(p.parse_and_visit_statement(v));
587+
EXPECT_THAT(v.visits, ElementsAre("visit_variable_use", // cond
588+
"visit_enter_block_scope", // (if)
589+
"visit_variable_use", // body
590+
"visit_exit_block_scope")); // (if)
591+
592+
EXPECT_THAT(v.errors, ElementsAre(ERROR_TYPE_FIELD(
593+
error_unexpected_semicolon_after_else, semicolon,
594+
offsets_matcher(&code, strlen(u8"if (cond) { body; } else"), u8";"))));
595+
}
596+
}
597+
598+
TEST(test_parse, else_has_empty_body) {
599+
{
600+
spy_visitor v;
601+
padded_string code(u8"if (cond) { body; } else;\n{ }"_sv);
602+
parser p(&code, &v);
603+
EXPECT_TRUE(p.parse_and_visit_statement(v));
604+
EXPECT_THAT(v.visits, ElementsAre("visit_variable_use", // cond
605+
"visit_enter_block_scope", // (if)
606+
"visit_variable_use", // body
607+
"visit_exit_block_scope")); // (if)
608+
EXPECT_THAT(v.errors,
609+
ElementsAre(ERROR_TYPE_FIELD(
610+
error_else_has_empty_body, where,
611+
offsets_matcher(&code, strlen(u8"if (cond) { body; } else {"), u8"}"))));
612+
}
613+
}
614+
581615
TEST(test_parse, block_statement) {
582616
{
583617
spy_visitor v = parse_and_visit_statement(u8"{ }"_sv);

0 commit comments

Comments
 (0)