Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(linter): eslint-plugin-vitest/no-import-node-test #4440

Merged
merged 11 commits into from
Jul 29, 2024
1 change: 1 addition & 0 deletions crates/oxc_linter/src/options.rs
Original file line number Diff line number Diff line change
Expand Up @@ -362,6 +362,7 @@ impl LintOptions {
}
false
}
"vitest" => self.vitest_plugin,
"jsx_a11y" => self.jsx_a11y_plugin,
"nextjs" => self.nextjs_plugin,
"react_perf" => self.react_perf_plugin,
Expand Down
5 changes: 5 additions & 0 deletions crates/oxc_linter/src/rules.rs
Original file line number Diff line number Diff line change
Expand Up @@ -440,6 +440,10 @@ mod promise {
pub mod param_names;
}

mod vitest {
pub mod no_import_node_test;
}

oxc_macros::declare_all_lint_rules! {
eslint::array_callback_return,
eslint::constructor_super,
Expand Down Expand Up @@ -835,4 +839,5 @@ oxc_macros::declare_all_lint_rules! {
promise::avoid_new,
promise::no_new_statics,
promise::param_names,
vitest::no_import_node_test,
}
81 changes: 81 additions & 0 deletions crates/oxc_linter/src/rules/vitest/no_import_node_test.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
use oxc_ast::AstKind;
use oxc_diagnostics::OxcDiagnostic;
use oxc_macros::declare_oxc_lint;
use oxc_semantic::AstNode;
use oxc_span::Span;

use crate::{context::LintContext, rule::Rule};

fn no_import_node_test(span0: Span) -> OxcDiagnostic {
OxcDiagnostic::warn("disallow importing `node:test`")
.with_help("Import from `vitest` instead of `node:test`")
.with_label(span0)
}

#[derive(Debug, Default, Clone)]
pub struct NoImportNodeTest;

declare_oxc_lint!(
/// ### What it does
///
/// This rule warns when `node:test` is imported (usually accidentally). With `--fix`, it will replace the import with `vitest`.
///
/// ### Examples
///
/// ```javascript
/// // invalid
/// import { test } from 'node:test'
/// import { expect } from 'vitest'
///
/// test('foo', () => {
/// expect(1).toBe(1)
/// })
/// ```
///
/// ```javascript
/// // valid
/// import { test, expect } from 'vitest'
///
/// test('foo', () => {
/// expect(1).toBe(1)
/// })
/// ```
NoImportNodeTest,
style,
);

impl Rule for NoImportNodeTest {
fn run<'a>(&self, node: &AstNode<'a>, ctx: &LintContext<'a>) {
if let AstKind::ImportDeclaration(import_decl) = node.kind() {
if import_decl.source.value.eq("node:test") {
let span = import_decl.source.span;

ctx.diagnostic_with_fix(no_import_node_test(span), |fixer| {
fixer.replace(span, "\"vitest\"")
});
}
};
}
}

#[test]
fn test() {
use crate::tester::Tester;

let pass = vec![(r#"import { test } from "vitest""#, None)];

let fail = vec![
(r#"import { test } from "node:test""#, None),
("import * as foo from 'node:test'", None),
];

let fix = vec![
(r#"import { test } from "node:test""#, r#"import { test } from "vitest""#, None),
(r#"import * as foo from "node:test""#, r#"import * as foo from "vitest""#, None),
];

Tester::new(NoImportNodeTest::NAME, pass, fail)
.with_vitest_plugin(true)
.expect_fix(fix)
.test_and_snapshot();
}
17 changes: 17 additions & 0 deletions crates/oxc_linter/src/snapshots/no_import_node_test.snap
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
---
source: crates/oxc_linter/src/tester.rs
assertion_line: 216
---
⚠ eslint-plugin-vitest(no-import-node-test): disallow importing `node:test`
╭─[no_import_node_test.tsx:1:22]
1 │ import { test } from "node:test"
· ───────────
╰────
help: Import from `vitest` instead of `node:test`

⚠ eslint-plugin-vitest(no-import-node-test): disallow importing `node:test`
╭─[no_import_node_test.tsx:1:22]
1 │ import * as foo from 'node:test'
· ───────────
╰────
help: Import from `vitest` instead of `node:test`