+
Skip to content
This repository was archived by the owner on Aug 31, 2023. It is now read-only.

feat(rome_js_analyze): add noExtraNonNullAssertion rule #3797

Merged
merged 6 commits into from
Nov 23, 2022
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
1 change: 1 addition & 0 deletions crates/rome_diagnostics_categories/src/categories.rs
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ define_dategories! {
"lint/nursery/noDupeKeys":"https://docs.rome.tools/lint/rules/noDupeKeys",
"lint/nursery/noEmptyInterface": "https://docs.rome.tools/lint/rules/noEmptyInterface",
"lint/nursery/noExplicitAny": "https://docs.rome.tools/lint/rules/noExplicitAny",
"lint/nursery/noExtraNonNullAssertion":"https://docs.rome.tools/lint/rules/noExtraNonNullAssertion",
"lint/nursery/noHeaderScope": "https://docs.rome.tools/lint/rules/noHeaderScope",
"lint/nursery/noInvalidConstructorSuper": "https://docs.rome.tools/lint/rules/noInvalidConstructorSuper",
"lint/nursery/noPrecisionLoss": "https://docs.rome.tools/lint/rules/noPrecisionLoss",
Expand Down
3 changes: 2 additions & 1 deletion crates/rome_js_analyze/src/analyzers/nursery.rs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
@@ -0,0 +1,142 @@
use rome_analyze::context::RuleContext;
use rome_analyze::{declare_rule, ActionCategory, Ast, Rule, RuleDiagnostic};
use rome_console::markup;
use rome_diagnostics::Applicability;
use rome_js_syntax::{
JsAnyAssignment, JsAnyExpression, TsNonNullAssertionAssignment, TsNonNullAssertionExpression,
};
use rome_rowan::{declare_node_union, AstNode, BatchMutationExt};

use crate::JsRuleAction;

declare_rule! {
/// Prevents the wrong usage of the non-null assertion operator (`!`) in TypeScript files.
///
/// > The `!` non-null assertion operator in TypeScript is used to assert that a value's type does not include `null` or `undefined`. Using the operator any more than once on a single value does nothing.
///
/// Source: https://typescript-eslint.io/rules/no-extra-non-null-assertion
///
/// ## Examples
///
/// ### Invalid
/// ```ts,expect_diagnostic
/// const bar = foo!!.bar;
/// ```
///
/// ```ts,expect_diagnostic
/// function fn(bar?: { n: number }) {
/// return bar!?.n;
/// }
/// ```
///
/// ```ts,expect_diagnostic
/// function fn(bar?: { n: number }) {
/// return ((bar!))?.();
/// }
/// ```
///
/// ### Valid
/// ```ts
/// const bar = foo!.bar;
///
/// obj?.string!.trim();
///
/// function fn(key: string | null) {
/// const obj = {};
/// return obj?.[key!];
/// }
/// ```
///
pub(crate) NoExtraNonNullAssertion {
version: "11.0.0",
name: "noExtraNonNullAssertion",
recommended: false,
}
}

declare_node_union! {
pub(crate) TsAnyNonNullAssertion = TsNonNullAssertionAssignment | TsNonNullAssertionExpression
}

impl Rule for NoExtraNonNullAssertion {
type Query = Ast<TsAnyNonNullAssertion>;
type State = ();
type Signals = Option<Self::State>;
type Options = ();

fn run(ctx: &RuleContext<Self>) -> Self::Signals {
let node = ctx.query();

match node {
TsAnyNonNullAssertion::TsNonNullAssertionAssignment(_) => {
let parent = node.parent::<JsAnyAssignment>()?;

// Cases considered as invalid:
// - TsNonNullAssertionAssignment > TsNonNullAssertionAssignment
if matches!(parent, JsAnyAssignment::TsNonNullAssertionAssignment(_)) {
return Some(());
}
}
TsAnyNonNullAssertion::TsNonNullAssertionExpression(_) => {
let parent = node.parent::<JsAnyExpression>()?;

// Cases considered as invalid:
// - TsNonNullAssertionAssignment > TsNonNullAssertionExpression
// - TsNonNullAssertionExpression > TsNonNullAssertionExpression
// - JsCallExpression[optional] > TsNonNullAssertionExpression
// - JsStaticMemberExpression[optional] > TsNonNullAssertionExpression
let has_extra_non_assertion = match parent.omit_parentheses() {
JsAnyExpression::JsAssignmentExpression(expr) => expr
.left()
.ok()?
.as_js_any_assignment()?
.as_ts_non_null_assertion_assignment()
.is_some(),
JsAnyExpression::TsNonNullAssertionExpression(_) => true,
JsAnyExpression::JsStaticMemberExpression(expr) => expr.is_optional(),
JsAnyExpression::JsCallExpression(expr) => expr.is_optional(),
_ => false,
};

if has_extra_non_assertion {
return Some(());
}
}
}

None
}

fn diagnostic(ctx: &RuleContext<Self>, _: &Self::State) -> Option<RuleDiagnostic> {
let diagnostic = RuleDiagnostic::new(
rule_category!(),
ctx.query().range(),
"Forbidden extra non-null assertion.",
);

Some(diagnostic)
}

fn action(ctx: &RuleContext<Self>, _: &Self::State) -> Option<JsRuleAction> {
let mut mutation = ctx.root().begin();
let node = ctx.query();

let excl_token = match node {
TsAnyNonNullAssertion::TsNonNullAssertionAssignment(assignment) => {
assignment.excl_token().ok()?
}
TsAnyNonNullAssertion::TsNonNullAssertionExpression(expression) => {
expression.excl_token().ok()?
}
};

mutation.remove_token(excl_token);

Some(JsRuleAction {
category: ActionCategory::QuickFix,
applicability: Applicability::Always,
message: markup! { "Remove extra non-null assertion." }.to_owned(),
mutation,
})
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
// @ts-expect-error
const bar = case1!!.bar;

function case2(bar: number | undefined) {
const bar1: number = bar!!!;
}

function case3(bar?: { n: number }) {
return bar!?.n;
}

function case4(bar?: { n: number }) {
return bar!?.();
}

const bar2 = (case5!)!.bar;

function case6(bar?: { n: number }) {
return (bar!)?.n;
}

function case7(bar?: { n: number }) {
return (bar)!?.n;
}

function case8(bar?: { n: number }) {
return ((bar!))?.();
}

class Case9 {
method() {
this.property!!;
}
}

case10!!.prop = null;

case11!?.[computedField];

case12!?.[a.b!!];

case13!!! = null

case14!! = null

if (case15!!) {}

if (!case16!!) {}
Loading
点击 这是indexloc提供的php浏览器服务,不要输入任何密码和下载