Skip to content

Commit

Permalink
feat(linter) eslint-plugin-unicorn no await in promise methods (#2963)
Browse files Browse the repository at this point in the history
  • Loading branch information
camc314 authored Apr 14, 2024
1 parent 98a3acd commit 01e64bf
Show file tree
Hide file tree
Showing 3 changed files with 234 additions and 1 deletion.
4 changes: 3 additions & 1 deletion crates/oxc_linter/src/rules.rs
Original file line number Diff line number Diff line change
Expand Up @@ -227,6 +227,7 @@ mod unicorn {
pub mod no_array_for_each;
pub mod no_array_reduce;
pub mod no_await_expression_member;
pub mod no_await_in_promise_methods;
pub mod no_console_spaces;
pub mod no_document_cookie;
pub mod no_empty_file;
Expand Down Expand Up @@ -527,6 +528,7 @@ oxc_macros::declare_all_lint_rules! {
unicorn::no_array_for_each,
unicorn::no_array_reduce,
unicorn::no_await_expression_member,
unicorn::no_await_in_promise_methods,
unicorn::no_console_spaces,
unicorn::no_document_cookie,
unicorn::no_empty_file,
Expand All @@ -548,7 +550,6 @@ oxc_macros::declare_all_lint_rules! {
unicorn::no_typeof_undefined,
unicorn::no_unnecessary_await,
unicorn::no_unreadable_array_destructuring,
unicorn::prefer_node_protocol,
unicorn::no_unreadable_iife,
unicorn::no_useless_fallback_in_spread,
unicorn::no_useless_length_check,
Expand All @@ -565,6 +566,7 @@ oxc_macros::declare_all_lint_rules! {
unicorn::prefer_blob_reading_methods,
unicorn::prefer_code_point,
unicorn::prefer_date_now,
unicorn::prefer_node_protocol,
unicorn::prefer_dom_node_append,
unicorn::prefer_dom_node_dataset,
unicorn::prefer_dom_node_remove,
Expand Down
137 changes: 137 additions & 0 deletions crates/oxc_linter/src/rules/unicorn/no_await_in_promise_methods.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,137 @@
use oxc_ast::{
ast::{Argument, ArrayExpressionElement, Expression},
AstKind,
};
use oxc_diagnostics::{
miette::{self, Diagnostic},
thiserror::{self, Error},
};
use oxc_macros::declare_oxc_lint;
use oxc_span::Span;

use crate::{ast_util::is_method_call, context::LintContext, rule::Rule, AstNode};

#[derive(Debug, Error, Diagnostic)]
#[error("eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.{1}()` should not be awaited.")]
#[diagnostic(severity(warning), help("Remove the `await`"))]
struct NoAwaitInPromiseMethodsDiagnostic(#[label] pub Span, String);

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

declare_oxc_lint!(
/// ### What it does
///
/// Disallow using `await` in `Promise` method parameters
///
/// ### Why is this bad?
///
/// Using `await` on promises passed as arguments to `Promise.all()`, `Promise.allSettled()`, `Promise.any()`, or `Promise.race()` is likely a mistake.
///
/// ### Example
/// Bad
///
/// ```js
/// Promise.all([await promise, anotherPromise]);
/// Promise.allSettled([await promise, anotherPromise]);
/// Promise.any([await promise, anotherPromise]);
/// Promise.race([await promise, anotherPromise]);
/// ```
///
/// Good
///
/// ```js
/// Promise.all([promise, anotherPromise]);
/// Promise.allSettled([promise, anotherPromise]);
/// Promise.any([promise, anotherPromise]);
/// Promise.race([promise, anotherPromise]);
/// ```
NoAwaitInPromiseMethods,
correctness
);

impl Rule for NoAwaitInPromiseMethods {
fn run<'a>(&self, node: &AstNode<'a>, ctx: &LintContext<'a>) {
let AstKind::CallExpression(call_expr) = node.kind() else {
return;
};

if !is_method_call(
call_expr,
Some(&["Promise"]),
Some(&["all", "allSettled", "any", "race"]),
Some(1),
Some(1),
) {
return;
}

let Some(Argument::Expression(first_argument)) = call_expr.arguments.first() else {
return;
};
let first_argument = first_argument.without_parenthesized();
let Expression::ArrayExpression(first_argument_array_expr) = first_argument else {
return;
};

for element in &first_argument_array_expr.elements {
if let ArrayExpressionElement::Expression(element_expr) = element {
if let Expression::AwaitExpression(await_expr) =
element_expr.without_parenthesized()
{
let property_name = call_expr
.callee
.get_member_expr()
.expect("callee is a member expression")
.static_property_name()
.expect("callee is a static property");

ctx.diagnostic(NoAwaitInPromiseMethodsDiagnostic(
Span::new(await_expr.span.start, await_expr.span.start + 5),
property_name.to_string(),
));
}
}
}
}
}

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

let pass = vec![
"Promise.all([promise1, promise2, promise3, promise4])",
"Promise.allSettled([promise1, promise2, promise3, promise4])",
"Promise.any([promise1, promise2, promise3, promise4])",
"Promise.race([promise1, promise2, promise3, promise4])",
"Promise.all(...[await promise])",
"Promise.all([await promise], extraArguments)",
"Promise.all()",
"Promise.all(notArrayExpression)",
"Promise.all([,])",
"Promise[all]([await promise])",
"Promise.notListedMethod([await promise])",
"NotPromise.all([await promise])",
"Promise.all([(await promise, 0)])",
"new Promise.all([await promise])",
"globalThis.Promise.all([await promise])",
];

let fail = vec![
"Promise.all([await promise])",
"Promise.allSettled([await promise])",
"Promise.any([await promise])",
"Promise.race([await promise])",
"Promise.all([, await promise])",
"Promise.all([await promise,])",
"Promise.all([await promise],)",
"Promise.all([await (0, promise)],)",
"Promise.all([await (( promise ))])",
"Promise.all([await await promise])",
"Promise.all([...foo, await promise1, await promise2])",
"Promise.all([await /* comment*/ promise])",
];

Tester::new(NoAwaitInPromiseMethods::NAME, pass, fail).test_and_snapshot();
}
94 changes: 94 additions & 0 deletions crates/oxc_linter/src/snapshots/no_await_in_promise_methods.snap
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
---
source: crates/oxc_linter/src/tester.rs
expression: no_await_in_promise_methods
---
eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.allSettled()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:21]
1Promise.allSettled([await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.any()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.any([await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.race()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:15]
1Promise.race([await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:16]
1Promise.all([, await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await promise,])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await promise],)
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await (0, promise)],)
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await (( promise ))])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await await promise])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:22]
1Promise.all([...foo, await promise1, await promise2])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:38]
1Promise.all([...foo, await promise1, await promise2])
· ─────
╰────
help: Remove the `await`

eslint-plugin-unicorn(no-await-in-promise-methods): Promise in `Promise.all()` should not be awaited.
╭─[no_await_in_promise_methods.tsx:1:14]
1Promise.all([await /* comment*/ promise])
· ─────
╰────
help: Remove the `await`

0 comments on commit 01e64bf

Please sign in to comment.