From 81761f80a8de04b17cca884c9751889f56b05a3b Mon Sep 17 00:00:00 2001 From: Eddy Brown Date: Mon, 4 Dec 2023 17:43:46 +0000 Subject: [PATCH] feat(biome_js_analyze): `useShorthandFunctionType` (#670) Co-authored-by: Eddy Brown Co-authored-by: Emanuele Stoppa --- Cargo.lock | 1 + .../src/categories.rs | 1 + crates/biome_js_analyze/Cargo.toml | 1 + .../biome_js_analyze/src/analyzers/nursery.rs | 2 + .../nursery/use_shorthand_function_type.rs | 193 ++++++++++++++++++ .../useShorthandFunctionType/invalid.ts | 7 + .../useShorthandFunctionType/invalid.ts.snap | 68 ++++++ .../nursery/useShorthandFunctionType/valid.ts | 30 +++ .../useShorthandFunctionType/valid.ts.snap | 39 ++++ .../src/configuration/linter/rules.rs | 37 +++- .../src/configuration/parse/json/rules.rs | 8 + .../@biomejs/backend-jsonrpc/src/workspace.ts | 5 + .../@biomejs/biome/configuration_schema.json | 7 + .../components/generated/NumberOfRules.astro | 2 +- .../src/content/docs/linter/rules/index.mdx | 1 + .../rules/use-shorthand-function-type.md | 121 +++++++++++ 16 files changed, 516 insertions(+), 7 deletions(-) create mode 100644 crates/biome_js_analyze/src/analyzers/nursery/use_shorthand_function_type.rs create mode 100644 crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts create mode 100644 crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts.snap create mode 100644 crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts create mode 100644 crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts.snap create mode 100644 website/src/content/docs/linter/rules/use-shorthand-function-type.md diff --git a/Cargo.lock b/Cargo.lock index e27789ea5bc2..48155a23afb0 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -400,6 +400,7 @@ dependencies = [ "countme", "insta", "lazy_static", + "log", "natord", "roaring", "rustc-hash", diff --git a/crates/biome_diagnostics_categories/src/categories.rs b/crates/biome_diagnostics_categories/src/categories.rs index e0bcdc2abfc9..82e12dcda2f7 100644 --- a/crates/biome_diagnostics_categories/src/categories.rs +++ b/crates/biome_diagnostics_categories/src/categories.rs @@ -115,6 +115,7 @@ define_categories! { "lint/nursery/useImportRestrictions": "https://biomejs.dev/linter/rules/use-import-restrictions", "lint/nursery/useRegexLiterals": "https://biomejs.dev/linter/rules/use-regex-literals", "lint/nursery/useValidAriaRole": "https://biomejs.dev/lint/rules/use-valid-aria-role", + "lint/nursery/useShorthandFunctionType": "https://biomejs.dev/lint/rules/use-shorthand-function-type", "lint/performance/noAccumulatingSpread": "https://biomejs.dev/linter/rules/no-accumulating-spread", "lint/performance/noDelete": "https://biomejs.dev/linter/rules/no-delete", "lint/security/noDangerouslySetInnerHtml": "https://biomejs.dev/linter/rules/no-dangerously-set-inner-html", diff --git a/crates/biome_js_analyze/Cargo.toml b/crates/biome_js_analyze/Cargo.toml index d2eeeb4fd447..873994629913 100644 --- a/crates/biome_js_analyze/Cargo.toml +++ b/crates/biome_js_analyze/Cargo.toml @@ -26,6 +26,7 @@ biome_json_syntax = { workspace = true } biome_rowan = { workspace = true } bpaf.workspace = true lazy_static = { workspace = true } +log = "0.4.20" natord = "1.0.9" roaring = "0.10.1" rustc-hash = { workspace = true } diff --git a/crates/biome_js_analyze/src/analyzers/nursery.rs b/crates/biome_js_analyze/src/analyzers/nursery.rs index f9b06e33a0f1..62595b0b840e 100644 --- a/crates/biome_js_analyze/src/analyzers/nursery.rs +++ b/crates/biome_js_analyze/src/analyzers/nursery.rs @@ -11,6 +11,7 @@ pub(crate) mod use_await; pub(crate) mod use_grouped_type_import; pub(crate) mod use_import_restrictions; pub(crate) mod use_regex_literals; +pub(crate) mod use_shorthand_function_type; declare_group! { pub (crate) Nursery { @@ -25,6 +26,7 @@ declare_group! { self :: use_grouped_type_import :: UseGroupedTypeImport , self :: use_import_restrictions :: UseImportRestrictions , self :: use_regex_literals :: UseRegexLiterals , + self :: use_shorthand_function_type :: UseShorthandFunctionType , ] } } diff --git a/crates/biome_js_analyze/src/analyzers/nursery/use_shorthand_function_type.rs b/crates/biome_js_analyze/src/analyzers/nursery/use_shorthand_function_type.rs new file mode 100644 index 000000000000..5da49c3d3641 --- /dev/null +++ b/crates/biome_js_analyze/src/analyzers/nursery/use_shorthand_function_type.rs @@ -0,0 +1,193 @@ +use crate::JsRuleAction; +use biome_analyze::{ + context::RuleContext, declare_rule, ActionCategory, Ast, FixKind, Rule, RuleDiagnostic, +}; +use biome_console::markup; +use biome_diagnostics::Applicability; +use biome_js_factory::make; +use biome_js_factory::make::ts_type_alias_declaration; +use biome_js_syntax::AnyTsType::TsThisType; +use biome_js_syntax::{ + AnyJsDeclarationClause, AnyTsReturnType, AnyTsType, TsCallSignatureTypeMember, TsFunctionType, + TsInterfaceDeclaration, TsObjectType, TsTypeMemberList, T, +}; +use biome_rowan::{AstNode, AstNodeList, BatchMutationExt, TriviaPieceKind}; + +declare_rule! { + /// Enforce using function types instead of object type with call signatures. + /// + /// TypeScript allows for two common ways to declare a type for a function: + /// + /// - Function type: `() => string` + /// - Object type with a signature: `{ (): string }` + /// + /// The function type form is generally preferred when possible for being more succinct. + /// + /// This rule suggests using a function type instead of an interface or object type literal with a single call signature. + /// + /// Source: https://typescript-eslint.io/rules/prefer-function-type/ + /// + /// ## Examples + /// + /// ### Invalid + /// + /// ```ts,expect_diagnostic + /// interface Example { + /// (): string; + /// } + /// ``` + /// + /// ```ts,expect_diagnostic + /// function foo(example: { (): number }): number { + /// return example(); + /// } + /// ``` + /// + /// ## Valid + /// + /// ```ts + /// type Example = () => string; + /// ``` + /// + /// ```ts + /// function foo(example: () => number): number { + /// return bar(); + /// } + /// ``` + /// + /// ```ts + /// // returns the function itself, not the `this` argument. + /// type ReturnsSelf2 = (arg: string) => ReturnsSelf; + /// ``` + /// + /// ```ts + /// interface Foo { + /// bar: string; + /// } + /// interface Bar extends Foo { + /// (): void; + /// } + /// ``` + /// + /// ```ts + /// // multiple call signatures (overloads) is allowed: + /// interface Overloaded { + /// (data: string): number; + /// (id: number): string; + /// } + /// // this is equivalent to Overloaded interface. + /// type Intersection = ((data: string) => number) & ((id: number) => string); + ///``` + /// + pub(crate) UseShorthandFunctionType { + version: "next", + name: "useShorthandFunctionType", + recommended: false, + fix_kind: FixKind::Safe, + } +} + +impl Rule for UseShorthandFunctionType { + type Query = Ast; + type State = (); + type Signals = Option; + type Options = (); + + fn run(ctx: &RuleContext) -> Self::Signals { + let query = ctx.query(); + + if let Some(ts_type_member_list) = query.parent::() { + // If there is more than one member, it's not a single call signature. + if ts_type_member_list.len() > 1 { + return None; + } + // If the parent is an interface with an extends clause, it's not a single call signature. + if let Some(interface_decl) = ts_type_member_list.parent::() { + if interface_decl.extends_clause().is_some() { + return None; + } + + if let AnyTsReturnType::AnyTsType(TsThisType(_)) = + query.return_type_annotation()?.ty().ok()? + { + return None; + } + } + return Some(()); + } + + None + } + + fn diagnostic(ctx: &RuleContext, _: &Self::State) -> Option { + Some(RuleDiagnostic::new(rule_category!(), ctx.query().range(), markup! { + "Use a function type instead of a call signature." + }).note(markup! { "Types containing only a call signature can be shortened to a function type." })) + } + + fn action(ctx: &RuleContext, _: &Self::State) -> Option { + let node = ctx.query(); + let mut mutation = ctx.root().begin(); + + let ts_type_member_list = node.parent::()?; + + if let Some(interface_decl) = ts_type_member_list.parent::() { + let type_alias_declaration = ts_type_alias_declaration( + make::token(T![type]).with_trailing_trivia([(TriviaPieceKind::Whitespace, " ")]), + interface_decl.id().ok()?, + make::token(T![=]).with_trailing_trivia([(TriviaPieceKind::Whitespace, " ")]), + AnyTsType::from(convert_ts_call_signature_type_member_to_function_type( + node, + )?), + ) + .build(); + + mutation.replace_node( + AnyJsDeclarationClause::from(interface_decl), + AnyJsDeclarationClause::from(type_alias_declaration), + ); + + return Some(JsRuleAction { + category: ActionCategory::QuickFix, + applicability: Applicability::Always, + message: markup! { "Alias a function type instead of using an interface with a call signature." }.to_owned(), + mutation, + }); + } + + if let Some(ts_object_type) = ts_type_member_list.parent::() { + let new_function_type = convert_ts_call_signature_type_member_to_function_type(node)?; + + mutation.replace_node( + AnyTsType::from(ts_object_type), + AnyTsType::from(new_function_type), + ); + + return Some(JsRuleAction { + category: ActionCategory::QuickFix, + applicability: Applicability::Always, + message: markup! { "Use a function type instead of an object type with a call signature." }.to_owned(), + mutation, + }); + } + + None + } +} + +fn convert_ts_call_signature_type_member_to_function_type( + node: &TsCallSignatureTypeMember, +) -> Option { + let new_node = make::ts_function_type( + make::js_parameters( + make::token(T!['(']), + node.parameters().ok()?.items(), + make::token(T![')']).with_trailing_trivia([(TriviaPieceKind::Whitespace, " ")]), + ), + make::token(T![=>]).with_trailing_trivia([(TriviaPieceKind::Whitespace, " ")]), + node.return_type_annotation()?.ty().ok()?, + ) + .build(); + + Some(new_node.with_type_parameters(node.type_parameters())) +} diff --git a/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts new file mode 100644 index 000000000000..de0e7918af17 --- /dev/null +++ b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts @@ -0,0 +1,7 @@ +interface Example { + (): string; +} + +function foo(example: { (): number }): number { + return example(); +} \ No newline at end of file diff --git a/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts.snap b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts.snap new file mode 100644 index 000000000000..4bbe2e03100e --- /dev/null +++ b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/invalid.ts.snap @@ -0,0 +1,68 @@ +--- +source: crates/biome_js_analyze/tests/spec_tests.rs +expression: invalid.ts +--- +# Input +```js +interface Example { + (): string; +} + +function foo(example: { (): number }): number { + return example(); +} +``` + +# Diagnostics +``` +invalid.ts:2:2 lint/nursery/useShorthandFunctionType FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ + + ! Use a function type instead of a call signature. + + 1 │ interface Example { + > 2 │ (): string; + │ ^^^^^^^^^^^ + 3 │ } + 4 │ + + i Types containing only a call signature can be shortened to a function type. + + i Safe fix: Alias a function type instead of using an interface with a call signature. + + 1 │ - interface·Example·{ + 2 │ - ·():·string; + 3 │ - } + 1 │ + type·Example·=·()·=>·string + 4 2 │ + 5 3 │ function foo(example: { (): number }): number { + + +``` + +``` +invalid.ts:5:25 lint/nursery/useShorthandFunctionType FIXABLE ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ + + ! Use a function type instead of a call signature. + + 3 │ } + 4 │ + > 5 │ function foo(example: { (): number }): number { + │ ^^^^^^^^^^ + 6 │ return example(); + 7 │ } + + i Types containing only a call signature can be shortened to a function type. + + i Safe fix: Use a function type instead of an object type with a call signature. + + 3 3 │ } + 4 4 │ + 5 │ - function·foo(example:·{·():·number·}):·number·{ + 5 │ + function·foo(example:·()·=>·number):·number·{ + 6 6 │ return example(); + 7 7 │ } + + +``` + + diff --git a/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts new file mode 100644 index 000000000000..bae591fde495 --- /dev/null +++ b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts @@ -0,0 +1,30 @@ +type Example = () => string; + +function foo(example: () => number): number { + return bar(); +} + +// returns the function itself, not the `this` argument. +type ReturnsSelf = (arg: string) => ReturnsSelf; + +interface Foo { + bar: string; +} + +interface Bar extends Foo { + (): void; +} + +// multiple call signatures (overloads) is allowed: +interface Overloaded { + (data: string): number; + (id: number): string; +} + +// this is equivelent to Overloaded interface. +type Intersection = ((data: string) => number) & ((id: number) => string); + +interface ReturnsSelf { + // returns the function itself, not the `this` argument. + (arg: string): this; +} \ No newline at end of file diff --git a/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts.snap b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts.snap new file mode 100644 index 000000000000..56eb2704baf5 --- /dev/null +++ b/crates/biome_js_analyze/tests/specs/nursery/useShorthandFunctionType/valid.ts.snap @@ -0,0 +1,39 @@ +--- +source: crates/biome_js_analyze/tests/spec_tests.rs +expression: valid.ts +--- +# Input +```js +type Example = () => string; + +function foo(example: () => number): number { + return bar(); +} + +// returns the function itself, not the `this` argument. +type ReturnsSelf = (arg: string) => ReturnsSelf; + +interface Foo { + bar: string; +} + +interface Bar extends Foo { + (): void; +} + +// multiple call signatures (overloads) is allowed: +interface Overloaded { + (data: string): number; + (id: number): string; +} + +// this is equivelent to Overloaded interface. +type Intersection = ((data: string) => number) & ((id: number) => string); + +interface ReturnsSelf { + // returns the function itself, not the `this` argument. + (arg: string): this; +} +``` + + diff --git a/crates/biome_service/src/configuration/linter/rules.rs b/crates/biome_service/src/configuration/linter/rules.rs index 7568fb6ed104..25aa98327e6b 100644 --- a/crates/biome_service/src/configuration/linter/rules.rs +++ b/crates/biome_service/src/configuration/linter/rules.rs @@ -2847,6 +2847,15 @@ pub struct Nursery { #[bpaf(long("use-regex-literals"), argument("on|off|warn"), optional, hide)] #[serde(skip_serializing_if = "Option::is_none")] pub use_regex_literals: Option, + #[doc = "Enforce using function types instead of object type with call signatures."] + #[bpaf( + long("use-shorthand-function-type"), + argument("on|off|warn"), + optional, + hide + )] + #[serde(skip_serializing_if = "Option::is_none")] + pub use_shorthand_function_type: Option, #[doc = "Elements with ARIA roles must use a valid, non-abstract ARIA role."] #[bpaf(long("use-valid-aria-role"), argument("on|off|warn"), optional, hide)] #[serde(skip_serializing_if = "Option::is_none")] @@ -2896,6 +2905,9 @@ impl MergeWith for Nursery { if let Some(use_regex_literals) = other.use_regex_literals { self.use_regex_literals = Some(use_regex_literals); } + if let Some(use_shorthand_function_type) = other.use_shorthand_function_type { + self.use_shorthand_function_type = Some(use_shorthand_function_type); + } if let Some(use_valid_aria_role) = other.use_valid_aria_role { self.use_valid_aria_role = Some(use_valid_aria_role); } @@ -2911,7 +2923,7 @@ impl MergeWith for Nursery { } impl Nursery { const GROUP_NAME: &'static str = "nursery"; - pub(crate) const GROUP_RULES: [&'static str; 15] = [ + pub(crate) const GROUP_RULES: [&'static str; 16] = [ "noAriaHiddenOnFocusable", "noDefaultExport", "noDuplicateJsonKeys", @@ -2926,6 +2938,7 @@ impl Nursery { "useGroupedTypeImport", "useImportRestrictions", "useRegexLiterals", + "useShorthandFunctionType", "useValidAriaRole", ]; const RECOMMENDED_RULES: [&'static str; 6] = [ @@ -2942,9 +2955,9 @@ impl Nursery { RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[4]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[9]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[11]), - RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[14]), + RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[15]), ]; - const ALL_RULES_AS_FILTERS: [RuleFilter<'static>; 15] = [ + const ALL_RULES_AS_FILTERS: [RuleFilter<'static>; 16] = [ RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[0]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[1]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[2]), @@ -2960,6 +2973,7 @@ impl Nursery { RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[12]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[13]), RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[14]), + RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[15]), ]; #[doc = r" Retrieves the recommended rules"] pub(crate) fn is_recommended(&self) -> bool { @@ -3046,11 +3060,16 @@ impl Nursery { index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[13])); } } - if let Some(rule) = self.use_valid_aria_role.as_ref() { + if let Some(rule) = self.use_shorthand_function_type.as_ref() { if rule.is_enabled() { index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[14])); } } + if let Some(rule) = self.use_valid_aria_role.as_ref() { + if rule.is_enabled() { + index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[15])); + } + } index_set } pub(crate) fn get_disabled_rules(&self) -> IndexSet { @@ -3125,11 +3144,16 @@ impl Nursery { index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[13])); } } - if let Some(rule) = self.use_valid_aria_role.as_ref() { + if let Some(rule) = self.use_shorthand_function_type.as_ref() { if rule.is_disabled() { index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[14])); } } + if let Some(rule) = self.use_valid_aria_role.as_ref() { + if rule.is_disabled() { + index_set.insert(RuleFilter::Rule(Self::GROUP_NAME, Self::GROUP_RULES[15])); + } + } index_set } #[doc = r" Checks if, given a rule name, matches one of the rules contained in this category"] @@ -3143,7 +3167,7 @@ impl Nursery { pub(crate) fn recommended_rules_as_filters() -> [RuleFilter<'static>; 6] { Self::RECOMMENDED_RULES_AS_FILTERS } - pub(crate) fn all_rules_as_filters() -> [RuleFilter<'static>; 15] { + pub(crate) fn all_rules_as_filters() -> [RuleFilter<'static>; 16] { Self::ALL_RULES_AS_FILTERS } #[doc = r" Select preset rules"] @@ -3180,6 +3204,7 @@ impl Nursery { "useGroupedTypeImport" => self.use_grouped_type_import.as_ref(), "useImportRestrictions" => self.use_import_restrictions.as_ref(), "useRegexLiterals" => self.use_regex_literals.as_ref(), + "useShorthandFunctionType" => self.use_shorthand_function_type.as_ref(), "useValidAriaRole" => self.use_valid_aria_role.as_ref(), _ => None, } diff --git a/crates/biome_service/src/configuration/parse/json/rules.rs b/crates/biome_service/src/configuration/parse/json/rules.rs index 86cc668cdd1a..d2d671f50504 100644 --- a/crates/biome_service/src/configuration/parse/json/rules.rs +++ b/crates/biome_service/src/configuration/parse/json/rules.rs @@ -991,6 +991,13 @@ impl Deserializable for Nursery { diagnostics, ); } + "useShorthandFunctionType" => { + result.use_shorthand_function_type = Deserializable::deserialize( + &value, + "useShorthandFunctionType", + diagnostics, + ); + } "useValidAriaRole" => { result.use_valid_aria_role = Deserializable::deserialize( &value, @@ -1019,6 +1026,7 @@ impl Deserializable for Nursery { "useGroupedTypeImport", "useImportRestrictions", "useRegexLiterals", + "useShorthandFunctionType", "useValidAriaRole", ], )); diff --git a/packages/@biomejs/backend-jsonrpc/src/workspace.ts b/packages/@biomejs/backend-jsonrpc/src/workspace.ts index e4834ae8af8d..458b05558508 100644 --- a/packages/@biomejs/backend-jsonrpc/src/workspace.ts +++ b/packages/@biomejs/backend-jsonrpc/src/workspace.ts @@ -837,6 +837,10 @@ export interface Nursery { * Enforce the use of the regular expression literals instead of the RegExp constructor if possible. */ useRegexLiterals?: RuleConfiguration; + /** + * Enforce using function types instead of object type with call signatures. + */ + useShorthandFunctionType?: RuleConfiguration; /** * Elements with ARIA roles must use a valid, non-abstract ARIA role. */ @@ -1521,6 +1525,7 @@ export type Category = | "lint/nursery/useImportRestrictions" | "lint/nursery/useRegexLiterals" | "lint/nursery/useValidAriaRole" + | "lint/nursery/useShorthandFunctionType" | "lint/performance/noAccumulatingSpread" | "lint/performance/noDelete" | "lint/security/noDangerouslySetInnerHtml" diff --git a/packages/@biomejs/biome/configuration_schema.json b/packages/@biomejs/biome/configuration_schema.json index fcdc498a74f4..b410a537e0fe 100644 --- a/packages/@biomejs/biome/configuration_schema.json +++ b/packages/@biomejs/biome/configuration_schema.json @@ -1214,6 +1214,13 @@ { "type": "null" } ] }, + "useShorthandFunctionType": { + "description": "Enforce using function types instead of object type with call signatures.", + "anyOf": [ + { "$ref": "#/definitions/RuleConfiguration" }, + { "type": "null" } + ] + }, "useValidAriaRole": { "description": "Elements with ARIA roles must use a valid, non-abstract ARIA role.", "anyOf": [ diff --git a/website/src/components/generated/NumberOfRules.astro b/website/src/components/generated/NumberOfRules.astro index 7131a8124a29..c537c48e13f3 100644 --- a/website/src/components/generated/NumberOfRules.astro +++ b/website/src/components/generated/NumberOfRules.astro @@ -1,2 +1,2 @@ -

Biome's linter has a total of 180 rules

\ No newline at end of file +

Biome's linter has a total of 181 rules

\ No newline at end of file diff --git a/website/src/content/docs/linter/rules/index.mdx b/website/src/content/docs/linter/rules/index.mdx index 40033ff85811..a4d62b88a864 100644 --- a/website/src/content/docs/linter/rules/index.mdx +++ b/website/src/content/docs/linter/rules/index.mdx @@ -241,4 +241,5 @@ Rules that belong to this group are not subject to semantic versionimport type when an import only has specifiers with type qualifier. | ⚠️ | | [useImportRestrictions](/linter/rules/use-import-restrictions) | Disallows package private imports. | | | [useRegexLiterals](/linter/rules/use-regex-literals) | Enforce the use of the regular expression literals instead of the RegExp constructor if possible. | ⚠️ | +| [useShorthandFunctionType](/linter/rules/use-shorthand-function-type) | Enforce using function types instead of object type with call signatures. | 🔧 | | [useValidAriaRole](/linter/rules/use-valid-aria-role) | Elements with ARIA roles must use a valid, non-abstract ARIA role. | ⚠️ | diff --git a/website/src/content/docs/linter/rules/use-shorthand-function-type.md b/website/src/content/docs/linter/rules/use-shorthand-function-type.md new file mode 100644 index 000000000000..8098c61b2e54 --- /dev/null +++ b/website/src/content/docs/linter/rules/use-shorthand-function-type.md @@ -0,0 +1,121 @@ +--- +title: useShorthandFunctionType (since vnext) +--- + +**Diagnostic Category: `lint/nursery/useShorthandFunctionType`** + +:::caution +This rule is part of the [nursery](/linter/rules/#nursery) group. +::: + +Enforce using function types instead of object type with call signatures. + +TypeScript allows for two common ways to declare a type for a function: + +- Function type: `() => string` +- Object type with a signature: `{ (): string }` + +The function type form is generally preferred when possible for being more succinct. + +This rule suggests using a function type instead of an interface or object type literal with a single call signature. + +Source: https://typescript-eslint.io/rules/prefer-function-type/ + +## Examples + +### Invalid + +```ts +interface Example { + (): string; +} +``` + +

nursery/useShorthandFunctionType.js:2:3 lint/nursery/useShorthandFunctionType  FIXABLE  ━━━━━━━━━━━━
+
+   Use a function type instead of a call signature.
+  
+    1 │ interface Example {
+  > 2 │   (): string;
+     ^^^^^^^^^^^
+    3 │ }
+    4 │ 
+  
+   Types containing only a call signature can be shortened to a function type.
+  
+   Safe fix: Alias a function type instead of using an interface with a call signature.
+  
+    1  - interface·Example·{
+    2  - ··():·string;
+    3  - }
+      1+ type·Example·=·()·=>·string
+    4 2  
+  
+
+ +```ts +function foo(example: { (): number }): number { + return example(); +} +``` + +
nursery/useShorthandFunctionType.js:1:25 lint/nursery/useShorthandFunctionType  FIXABLE  ━━━━━━━━━━━
+
+   Use a function type instead of a call signature.
+  
+  > 1 │ function foo(example: { (): number }): number {
+                           ^^^^^^^^^^
+    2 │   return example();
+    3 │ }
+  
+   Types containing only a call signature can be shortened to a function type.
+  
+   Safe fix: Use a function type instead of an object type with a call signature.
+  
+    1  - function·foo(example:·{·():·number·}):·number·{
+      1+ function·foo(example:·()·=>·number):·number·{
+    2 2    return example();
+    3 3  }
+  
+
+ +## Valid + +```ts +type Example = () => string; +``` + +```ts +function foo(example: () => number): number { + return bar(); +} +``` + +```ts +// returns the function itself, not the `this` argument. +type ReturnsSelf2 = (arg: string) => ReturnsSelf; +``` + +```ts +interface Foo { + bar: string; +} +interface Bar extends Foo { + (): void; +} +``` + +```ts +// multiple call signatures (overloads) is allowed: +interface Overloaded { + (data: string): number; + (id: number): string; +} +// this is equivalent to Overloaded interface. +type Intersection = ((data: string) => number) & ((id: number) => string); +``` + +## Related links + +- [Disable a rule](/linter/#disable-a-lint-rule) +- [Rule options](/linter/#rule-options)