diff --git a/.changeset/quiet-eggs-leave.md b/.changeset/quiet-eggs-leave.md new file mode 100644 index 000000000..555ab2fb7 --- /dev/null +++ b/.changeset/quiet-eggs-leave.md @@ -0,0 +1,5 @@ +--- +"eslint-plugin-svelte": minor +--- + +feat: add `svelte/no-loss-of-prop-reactivity` rule diff --git a/README.md b/README.md index d7ba245c5..a1cf5ec20 100644 --- a/README.md +++ b/README.md @@ -341,6 +341,7 @@ These rules relate to better ways of doing things to help you avoid problems: | [svelte/button-has-type](https://sveltejs.github.io/eslint-plugin-svelte/rules/button-has-type/) | disallow usage of button without an explicit type attribute | | | [svelte/no-at-debug-tags](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-at-debug-tags/) | disallow the use of `{@debug}` | :star: | | [svelte/no-immutable-reactive-statements](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-immutable-reactive-statements/) | disallow reactive statements that don't reference reactive values. | | +| [svelte/no-loss-of-prop-reactivity](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-loss-of-prop-reactivity/) | disallow the use of props that potentially lose reactivity | | | [svelte/no-reactive-functions](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-reactive-functions/) | it's not necessary to define functions in reactive statements | :bulb: | | [svelte/no-reactive-literals](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-reactive-literals/) | don't assign literal values in reactive statements | :bulb: | | [svelte/no-unused-svelte-ignore](https://sveltejs.github.io/eslint-plugin-svelte/rules/no-unused-svelte-ignore/) | disallow unused svelte-ignore comments | :star: | diff --git a/docs/rules.md b/docs/rules.md index 276f34ff8..805c9796e 100644 --- a/docs/rules.md +++ b/docs/rules.md @@ -54,6 +54,7 @@ These rules relate to better ways of doing things to help you avoid problems: | [svelte/button-has-type](./rules/button-has-type.md) | disallow usage of button without an explicit type attribute | | | [svelte/no-at-debug-tags](./rules/no-at-debug-tags.md) | disallow the use of `{@debug}` | :star: | | [svelte/no-immutable-reactive-statements](./rules/no-immutable-reactive-statements.md) | disallow reactive statements that don't reference reactive values. | | +| [svelte/no-loss-of-prop-reactivity](./rules/no-loss-of-prop-reactivity.md) | disallow the use of props that potentially lose reactivity | | | [svelte/no-reactive-functions](./rules/no-reactive-functions.md) | it's not necessary to define functions in reactive statements | :bulb: | | [svelte/no-reactive-literals](./rules/no-reactive-literals.md) | don't assign literal values in reactive statements | :bulb: | | [svelte/no-unused-svelte-ignore](./rules/no-unused-svelte-ignore.md) | disallow unused svelte-ignore comments | :star: | diff --git a/docs/rules/no-loss-of-prop-reactivity.md b/docs/rules/no-loss-of-prop-reactivity.md new file mode 100644 index 000000000..cc89d410c --- /dev/null +++ b/docs/rules/no-loss-of-prop-reactivity.md @@ -0,0 +1,42 @@ +--- +pageClass: "rule-details" +sidebarDepth: 0 +title: "svelte/no-loss-of-prop-reactivity" +description: "disallow the use of props that potentially lose reactivity" +--- + +# svelte/no-loss-of-prop-reactivity + +> disallow the use of props that potentially lose reactivity + +- :exclamation: **_This rule has not been released yet._** + +## :book: Rule Details + +This rule reports the use of props that potentially lose reactivity. + + + + + +```svelte + +``` + + + +## :wrench: Options + +Nothing. + +## :mag: Implementation + +- [Rule source](https://github.com/sveltejs/eslint-plugin-svelte/blob/main/src/rules/no-loss-of-prop-reactivity.ts) +- [Test source](https://github.com/sveltejs/eslint-plugin-svelte/blob/main/tests/src/rules/no-loss-of-prop-reactivity.ts) diff --git a/src/rules/no-loss-of-prop-reactivity.ts b/src/rules/no-loss-of-prop-reactivity.ts new file mode 100644 index 000000000..e0fc2c8a1 --- /dev/null +++ b/src/rules/no-loss-of-prop-reactivity.ts @@ -0,0 +1,152 @@ +import type { AST } from "svelte-eslint-parser" +import { createRule } from "../utils" +import type { TSESTree } from "@typescript-eslint/types" +import { findAttribute, findVariable } from "../utils/ast-utils" +import { getStaticAttributeValue } from "../utils/ast-utils" + +export default createRule("no-loss-of-prop-reactivity", { + meta: { + docs: { + description: "disallow the use of props that potentially lose reactivity", + category: "Best Practices", + recommended: false, + }, + schema: [], + messages: { + potentiallyLoseReactivity: + "Referencing props that potentially lose reactivity should be avoided.", + }, + type: "suggestion", + }, + create(context) { + if (!context.parserServices.isSvelte) { + return {} + } + let contextModule: AST.SvelteScriptElement | null = null + let scriptNode: AST.SvelteScriptElement | null = null + const reactiveStatements: AST.SvelteReactiveStatement[] = [] + const functions: TSESTree.FunctionLike[] = [] + const props = new Set() + return { + SvelteScriptElement(node: AST.SvelteScriptElement) { + const contextAttr = findAttribute(node, "context") + if (contextAttr && getStaticAttributeValue(contextAttr) === "module") { + contextModule = node + return + } + + scriptNode = node + }, + SvelteReactiveStatement(node: AST.SvelteReactiveStatement) { + reactiveStatements.push(node) + }, + ":function"(node: TSESTree.FunctionLike) { + functions.push(node) + }, + ExportNamedDeclaration(node: TSESTree.ExportNamedDeclaration) { + if ( + contextModule && + contextModule.range[0] < node.range[0] && + node.range[1] < contextModule.range[1] + ) { + return + } + if ( + node.declaration?.type === "VariableDeclaration" && + (node.declaration.kind === "let" || node.declaration.kind === "var") + ) { + for (const decl of node.declaration.declarations) { + if (decl.id.type === "Identifier") { + props.add(decl.id) + } + } + } + for (const spec of node.specifiers) { + if (spec.exportKind === "type") { + continue + } + if (isMutableProp(spec.local)) { + props.add(spec.exported) + } + } + }, + "Program:exit"() { + for (const prop of props) { + const variable = findVariable(context, prop) + if ( + !variable || + // ignore multiple definitions + variable.defs.length > 1 + ) { + return + } + for (const reference of variable.references) { + if (reference.isWrite()) continue + const id = reference.identifier as TSESTree.Identifier + if ( + variable.defs.some( + (def) => + def.node.range[0] <= id.range[0] && + id.range[1] <= def.node.range[1], + ) + ) { + // The reference is in the variable definition. + continue + } + if (isInReactivityScope(id)) continue + + context.report({ + node: id, + messageId: "potentiallyLoseReactivity", + }) + } + } + }, + } + + /** Checks whether given prop id is mutable variable or not */ + function isMutableProp(id: TSESTree.Identifier) { + const variable = findVariable(context, id) + if (!variable || variable.defs.length === 0) { + return false + } + return variable.defs.every((def) => { + if (def.type !== "Variable") { + return false + } + return def.parent.kind === "let" || def.parent.kind === "var" + }) + } + + /** Checks whether given id is in potentially reactive scope or not */ + function isInReactivityScope(id: TSESTree.Identifier) { + if ( + !scriptNode || + id.range[1] <= scriptNode.range[0] || + scriptNode.range[1] <= id.range[0] + ) { + // The reference is in the template. + return true + } + if ( + reactiveStatements.some( + (node) => + node.range[0] <= id.range[0] && id.range[1] <= node.range[1], + ) + ) { + // The reference is in the reactive statement. + return true + } + if ( + functions.some( + (node) => + node.range[0] <= id.range[0] && id.range[1] <= node.range[1], + ) + ) { + // The reference is in the function. + return true + } + return false + } + }, +}) diff --git a/src/utils/rules.ts b/src/utils/rules.ts index 7dac11750..11cc8eb08 100644 --- a/src/utils/rules.ts +++ b/src/utils/rules.ts @@ -29,6 +29,7 @@ import noExportLoadInSvelteModuleInKitPages from "../rules/no-export-load-in-sve import noExtraReactiveCurlies from "../rules/no-extra-reactive-curlies" import noImmutableReactiveStatements from "../rules/no-immutable-reactive-statements" import noInnerDeclarations from "../rules/no-inner-declarations" +import noLossOfPropReactivity from "../rules/no-loss-of-prop-reactivity" import noNotFunctionHandler from "../rules/no-not-function-handler" import noObjectInTextMustaches from "../rules/no-object-in-text-mustaches" import noReactiveFunctions from "../rules/no-reactive-functions" @@ -88,6 +89,7 @@ export const rules = [ noExtraReactiveCurlies, noImmutableReactiveStatements, noInnerDeclarations, + noLossOfPropReactivity, noNotFunctionHandler, noObjectInTextMustaches, noReactiveFunctions, diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml new file mode 100644 index 000000000..76cfa5393 --- /dev/null +++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-errors.yaml @@ -0,0 +1,4 @@ +- message: Referencing props that potentially lose reactivity should be avoided. + line: 4 + column: 17 + suggestions: null diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte new file mode 100644 index 000000000..d28686862 --- /dev/null +++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/invalid/prop01-input.svelte @@ -0,0 +1,5 @@ + diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte new file mode 100644 index 000000000..e4db2dec8 --- /dev/null +++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/keypad01-input.svelte @@ -0,0 +1,32 @@ + + +
+ + + + + +
+ + diff --git a/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte new file mode 100644 index 000000000..bab40db47 --- /dev/null +++ b/tests/fixtures/rules/no-loss-of-prop-reactivity/valid/prop01-input.svelte @@ -0,0 +1,5 @@ + diff --git a/tests/src/rules/no-loss-of-prop-reactivity.ts b/tests/src/rules/no-loss-of-prop-reactivity.ts new file mode 100644 index 000000000..20fd103a6 --- /dev/null +++ b/tests/src/rules/no-loss-of-prop-reactivity.ts @@ -0,0 +1,16 @@ +import { RuleTester } from "eslint" +import rule from "../../../src/rules/no-loss-of-prop-reactivity" +import { loadTestCases } from "../../utils/utils" + +const tester = new RuleTester({ + parserOptions: { + ecmaVersion: 2020, + sourceType: "module", + }, +}) + +tester.run( + "no-loss-of-prop-reactivity", + rule as any, + loadTestCases("no-loss-of-prop-reactivity"), +)