Skip to content

Commit

Permalink
fix(lint): prevent crash when applying noUselessFragments unsafe fi…
Browse files Browse the repository at this point in the history
…xes (#3338)
  • Loading branch information
unvalley authored Jul 3, 2024
1 parent 1c682e0 commit f663e6b
Show file tree
Hide file tree
Showing 4 changed files with 91 additions and 12 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,8 @@ our [guidelines for writing a good changelog entry](https://github.com/biomejs/b

- Don't request alt text for elements hidden from assistive technologies ([#3316](https://github.com/biomejs/biome/issues/3316)). Contributed by @robintown

- Fix [[#3149](https://github.com/biomejs/biome/issues/3149)] crashes that occurred when applying the `noUselessFragments` unsafe fixes in certain scenarios. Contributed by @unvalley

### Parser

## v1.8.3 (2024-06-27)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,13 @@ use biome_analyze::context::RuleContext;
use biome_analyze::{declare_rule, ActionCategory, FixKind, Rule, RuleDiagnostic, RuleSource};
use biome_console::markup;
use biome_js_factory::make::{
js_expression_statement, js_string_literal_expression, jsx_expression_child, jsx_string,
jsx_string_literal, jsx_tag_expression, token, JsxExpressionChildBuilder,
js_string_literal_expression, jsx_expression_child, jsx_string, jsx_string_literal,
jsx_tag_expression, token, JsxExpressionChildBuilder,
};
use biome_js_syntax::{
AnyJsExpression, AnyJsxChild, AnyJsxElementName, AnyJsxTag, JsLanguage,
JsParenthesizedExpression, JsSyntaxKind, JsxChildList, JsxElement, JsxExpressionAttributeValue,
JsxFragment, JsxTagExpression, JsxText, T,
AnyJsxChild, AnyJsxElementName, AnyJsxTag, JsLanguage, JsParenthesizedExpression, JsSyntaxKind,
JsxChildList, JsxElement, JsxExpressionAttributeValue, JsxFragment, JsxTagExpression, JsxText,
T,
};
use biome_rowan::{declare_node_union, AstNode, AstNodeList, BatchMutation, BatchMutationExt};

Expand Down Expand Up @@ -294,13 +294,9 @@ impl Rule for NoUselessFragments {
.into_syntax()
})
} else {
child.expression().map(|expression| {
if let AnyJsExpression::JsIdentifierExpression(node) = expression {
node.into_syntax()
} else {
js_expression_statement(expression).build().into_syntax()
}
})
child
.expression()
.map(|expression| expression.into_syntax())
}
}

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
function fn(member) {
fn(<>{member.expression}</>);
fn(<>{member.expression()}</>);
(<>{1}</>).toString();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
---
source: crates/biome_js_analyze/tests/spec_tests.rs
expression: issue_3149.jsx
---
# Input
```jsx
function fn(member) {
fn(<>{member.expression}</>);
fn(<>{member.expression()}</>);
(<>{1}</>).toString();
}

```

# Diagnostics
```
issue_3149.jsx:2:6 lint/complexity/noUselessFragments FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
! Avoid using unnecessary Fragment.
1 │ function fn(member) {
> 2 │ fn(<>{member.expression}</>);
│ ^^^^^^^^^^^^^^^^^^^^^^^^
3 │ fn(<>{member.expression()}</>);
4 │ (<>{1}</>).toString();
i A fragment is redundant if it contains only one child, or if it is the child of a html element, and is not a keyed fragment.
i Unsafe fix: Remove the Fragment
2 │ ··fn(<>{member.expression}</>);
│ --- ----
```

```
issue_3149.jsx:3:6 lint/complexity/noUselessFragments FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
! Avoid using unnecessary Fragment.
1 │ function fn(member) {
2 │ fn(<>{member.expression}</>);
> 3 │ fn(<>{member.expression()}</>);
│ ^^^^^^^^^^^^^^^^^^^^^^^^^^
4 │ (<>{1}</>).toString();
5 │ }
i A fragment is redundant if it contains only one child, or if it is the child of a html element, and is not a keyed fragment.
i Unsafe fix: Remove the Fragment
3 │ ··fn(<>{member.expression()}</>);
│ --- ----
```

```
issue_3149.jsx:4:4 lint/complexity/noUselessFragments FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
! Avoid using unnecessary Fragment.
2 │ fn(<>{member.expression}</>);
3 │ fn(<>{member.expression()}</>);
> 4 │ (<>{1}</>).toString();
│ ^^^^^^^^
5 │ }
6 │
i A fragment is redundant if it contains only one child, or if it is the child of a html element, and is not a keyed fragment.
i Unsafe fix: Remove the Fragment
4 │ ··(<>{1}</>).toString();
│ --- ----
```

0 comments on commit f663e6b

Please sign in to comment.