Skip to content

Commit

Permalink
feat(linter) eslint-plugin-unicorn error message (#992)
Browse files Browse the repository at this point in the history
Co-authored-by: Boshen <[email protected]>
  • Loading branch information
camc314 and Boshen authored Oct 14, 2023
1 parent 41c55bc commit dc08a21
Show file tree
Hide file tree
Showing 3 changed files with 321 additions and 0 deletions.
2 changes: 2 additions & 0 deletions crates/oxc_linter/src/rules.rs
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,7 @@ mod jest {

mod unicorn {
pub mod catch_error_name;
pub mod error_message;
pub mod filename_case;
pub mod no_console_spaces;
pub mod no_instanceof_array;
Expand Down Expand Up @@ -232,6 +233,7 @@ oxc_macros::declare_all_lint_rules! {
jest::no_identical_title,
jest::valid_title,
unicorn::catch_error_name,
unicorn::error_message,
unicorn::no_console_spaces,
unicorn::no_instanceof_array,
unicorn::no_unnecessary_await,
Expand Down
188 changes: 188 additions & 0 deletions crates/oxc_linter/src/rules/unicorn/error_message.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,188 @@
use oxc_ast::{
ast::{Argument, CallExpression, Expression, NewExpression},
AstKind,
};
use oxc_diagnostics::{
miette::{self, Diagnostic},
thiserror::Error,
};
use oxc_macros::declare_oxc_lint;
use oxc_span::{Atom, Span};

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

#[derive(Debug, Error, Diagnostic)]
pub enum ErrorMessageDiagnostic {
#[error("eslint-plugin-unicorn(error-message): Pass a message to the {0:1} constructor.")]
MissingMessage(Atom, #[label] Span),
#[error("eslint-plugin-unicorn(error-message): Error message should not be an empty string.")]
EmptyMessage(#[label] Span),
#[error("eslint-plugin-unicorn(error-message): Error message should be a string.")]
NotString(#[label] Span),
}

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

declare_oxc_lint!(
/// ### What it does
///
/// This rule enforces a `message` value to be passed in when creating an instance of a built-in `Error` object, which leads to more readable and debuggable code.
///
/// ### Example
/// ```javascript
/// // Fail
/// throw Error()
/// throw new TypeError()
///
/// // Pass
/// throw new Error('Unexpected token')
/// throw new TypeError('Number expected')
///
///
/// ```
ErrorMessage,
style
);

impl Rule for ErrorMessage {
fn run<'a>(&self, node: &AstNode<'a>, ctx: &LintContext<'a>) {
let (callee, span, args) = match &node.kind() {
AstKind::NewExpression(NewExpression {
callee: Expression::Identifier(id),
span,
arguments,
..
})
| AstKind::CallExpression(CallExpression {
callee: Expression::Identifier(id),
span,
arguments,
..
}) => (id, *span, arguments),
_ => return,
};

if !BUILT_IN_ERRORS.contains(&callee.name.as_str()) {
return;
}

let constructor_name = &callee.name;
let message_argument_idx = usize::from(constructor_name.as_str() == "AggregateError");

// If message is `SpreadElement` or there is `SpreadElement` before message
if args.iter().enumerate().any(|(i, arg)| {
i <= message_argument_idx
&& match arg {
Argument::Expression(_) => false,
Argument::SpreadElement(_) => true,
}
}) {
return;
}

let message_argument = args.get(message_argument_idx);

let arg = match message_argument {
Some(v) => v,
None => {
return ctx.diagnostic(ErrorMessageDiagnostic::MissingMessage(
constructor_name.clone(),
span,
))
}
};

let arg = match arg {
Argument::Expression(v) => v,
Argument::SpreadElement(_) => {
return;
}
};

match arg {
Expression::StringLiteral(lit) => {
if lit.value.is_empty() {
ctx.diagnostic(ErrorMessageDiagnostic::EmptyMessage(lit.span));
}
}
Expression::TemplateLiteral(template_lit) => {
if template_lit.span.source_text(ctx.source_text()).len() == 2 {
ctx.diagnostic(ErrorMessageDiagnostic::EmptyMessage(template_lit.span));
}
}
Expression::ObjectExpression(object_expr) => {
ctx.diagnostic(ErrorMessageDiagnostic::NotString(object_expr.span));
}
Expression::ArrayExpression(array_expr) => {
ctx.diagnostic(ErrorMessageDiagnostic::NotString(array_expr.span));
}
_ => {}
}
}
}

// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error#Error_types
const BUILT_IN_ERRORS: &[&str] = &[
"Error",
"EvalError",
"RangeError",
"ReferenceError",
"SyntaxError",
"TypeError",
"URIError",
"InternalError",
"AggregateError",
];

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

let pass = vec![
("throw new Error('error')", None),
("throw new TypeError('error')", None),
("throw new MyCustomError('error')", None),
("throw new MyCustomError()", None),
("throw generateError()", None),
("throw foo()", None),
("throw err", None),
("throw 1", None),
("new Error(\"message\", 0, 0)", None),
("new Error(foo)", None),
("new Error(...foo)", None),
("new AggregateError(errors, \"message\")", None),
("new NotAggregateError(errors)", None),
("new AggregateError(...foo)", None),
("new AggregateError(...foo, \"\")", None),
("new AggregateError(errors, ...foo)", None),
("new AggregateError(errors, message, \"\")", None),
("new AggregateError(\"\", message, \"\")", None),
];

let fail = vec![
("throw new Error()", None),
("throw Error()", None),
("throw new Error('')", None),
("throw new Error(``)", None),
("const foo = new TypeError()", None),
("const foo = new SyntaxError()", None),
("throw new Error([])", None),
("throw new Error([foo])", None),
("throw new Error({})", None),
("throw new Error({foo})", None),
("const error = new RangeError;", None),
("new AggregateError(errors)", None),
("AggregateError(errors)", None),
("new AggregateError(errors, \"\")", None),
("new AggregateError(errors, ``)", None),
("new AggregateError(errors, \"\", extraArgument)", None),
("new AggregateError(errors, [])", None),
("new AggregateError(errors, [foo])", None),
("new AggregateError(errors, {})", None),
("new AggregateError(errors, {foo})", None),
("const error = new AggregateError;", None),
];

Tester::new(ErrorMessage::NAME, pass, fail).test_and_snapshot();
}
131 changes: 131 additions & 0 deletions crates/oxc_linter/src/snapshots/error_message.snap
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
---
source: crates/oxc_linter/src/tester.rs
expression: error_message
---
× eslint-plugin-unicorn(error-message): Pass a message to the Error constructor.
╭─[error_message.tsx:1:1]
1throw new Error()
· ───────────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the Error constructor.
╭─[error_message.tsx:1:1]
1throw Error()
· ───────
╰────

× eslint-plugin-unicorn(error-message): Error message should not be an empty string.
╭─[error_message.tsx:1:1]
1throw new Error('')
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should not be an empty string.
╭─[error_message.tsx:1:1]
1throw new Error(``)
· ──
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the TypeError constructor.
╭─[error_message.tsx:1:1]
1const foo = new TypeError()
· ───────────────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the SyntaxError constructor.
╭─[error_message.tsx:1:1]
1const foo = new SyntaxError()
· ─────────────────
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1throw new Error([])
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1throw new Error([foo])
· ─────
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1throw new Error({})
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1throw new Error({foo})
· ─────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the RangeError constructor.
╭─[error_message.tsx:1:1]
1const error = new RangeError;
· ──────────────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the AggregateError constructor.
╭─[error_message.tsx:1:1]
1new AggregateError(errors)
· ──────────────────────────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the AggregateError constructor.
╭─[error_message.tsx:1:1]
1AggregateError(errors)
· ──────────────────────
╰────

× eslint-plugin-unicorn(error-message): Error message should not be an empty string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, "")
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should not be an empty string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, ``)
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should not be an empty string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, "", extraArgument)
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, [])
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, [foo])
· ─────
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, {})
· ──
╰────

× eslint-plugin-unicorn(error-message): Error message should be a string.
╭─[error_message.tsx:1:1]
1new AggregateError(errors, {foo})
· ─────
╰────

× eslint-plugin-unicorn(error-message): Pass a message to the AggregateError constructor.
╭─[error_message.tsx:1:1]
1const error = new AggregateError;
· ──────────────────
╰────


0 comments on commit dc08a21

Please sign in to comment.