diff --git a/.nycrc b/.nycrc
index b9d7b70b..27772eac 100644
--- a/.nycrc
+++ b/.nycrc
@@ -4,7 +4,6 @@
],
"exclude": [
"lib/rules/no-hide-core-modules.js",
- "lib/rules/no-unsupported-features.js",
"lib/converted-esm/*.js"
],
"reporter": [
diff --git a/README.md b/README.md
index 021ce524..60b4b531 100644
--- a/README.md
+++ b/README.md
@@ -133,7 +133,6 @@ For [Shareable Configs](https://eslint.org/docs/latest/developer-guide/shareable
| [no-unpublished-bin](docs/rules/no-unpublished-bin.md) | disallow `bin` files that npm ignores | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
| [no-unpublished-import](docs/rules/no-unpublished-import.md) | disallow `import` declarations which import private modules | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
| [no-unpublished-require](docs/rules/no-unpublished-require.md) | disallow `require()` expressions which import private modules | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
-| [no-unsupported-features](docs/rules/no-unsupported-features.md) | disallow unsupported ECMAScript features on the specified version | | | ❌ |
| [no-unsupported-features/es-builtins](docs/rules/no-unsupported-features/es-builtins.md) | disallow unsupported ECMAScript built-ins on the specified version | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
| [no-unsupported-features/es-syntax](docs/rules/no-unsupported-features/es-syntax.md) | disallow unsupported ECMAScript syntax on the specified version | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
| [no-unsupported-features/node-builtins](docs/rules/no-unsupported-features/node-builtins.md) | disallow unsupported Node.js built-in APIs on the specified version | ☑️ 🟢 ✅ ☑️ 🟢 ✅ | | |
diff --git a/docs/rules/no-unsupported-features.md b/docs/rules/no-unsupported-features.md
deleted file mode 100644
index 69a62a27..00000000
--- a/docs/rules/no-unsupported-features.md
+++ /dev/null
@@ -1,308 +0,0 @@
-# Disallow unsupported ECMAScript features on the specified version (`n/no-unsupported-features`)
-
-❌ This rule is deprecated. It was replaced by [`n/no-unsupported-features/es-syntax`](no-unsupported-features/es-syntax.md),[`n/no-unsupported-features/es-builtins`](no-unsupported-features/es-builtins.md).
-
-
-
-Node.js doesn't support all ECMAScript standard features.
-This rule reports when you used unsupported ECMAScript 2015-2018 features on the specified Node.js version.
-
-> ※ About ECMAScript 2018, this rule reports only features which have arrived at stage 4 until 2018-02-01.
-> It needs a major version bump in order to cover newer features.
-
-## 📖 Rule Details
-
-⚠️ This rule expects to be used with the following configuration:
-
-```json
-{
- "env": {"es6": true},
- "parserOptions": {"ecmaVersion": 2018}
-}
-```
-
-⚠️ This rule reads the [engines] field of `package.json` to detect Node.js version.
-
-I recommend a use of the [engines] field since it's the official way to indicate what Node.js versions your module is supporting.
-For example of `package.json`:
-
-```json
-{
- "name": "your-module",
- "version": "1.0.0",
- "engines": {
- "node": ">=6.0.0"
- }
-}
-```
-
-If the [engines] field is omitted, this rule chooses `4` since it's the minimum version the community is maintaining.
-
-Examples of 👎 **incorrect** code for this rule:
-
-```js
-/*eslint n/no-unsupported-features: ["error", {version: 4}]*/
-/*eslint-env es6*/
-
-function foo(a = 1) { /*error Default Parameters are not supported yet on Node v4.*/
- //...
-}
-
-function foo(...a) { /*error Rest Parameters are not supported yet on Node v4.*/
- //...
-}
-
-var a = [...b]; /*error Spread Operators are not supported yet on Node v4.*/
-var a = /foo/y; /*error RegExp 'y' Flags are not supported yet on Node v4.*/
-var a = /foo/u; /*error RegExp 'u' Flags are not supported yet on Node v4.*/
-var {a, b} = c; /*error Destructuring are not supported yet on Node v4.*/
-var {a, b} = c; /*error Destructuring are not supported yet on Node v4.*/
-
-let a = 1; /*error 'let' Declarations in non-strict mode are not supported yet on Node v4.*/
-const a = 1; /*error 'const' Declarations in non-strict mode are not supported yet on Node v4.*/
-class A {} /*error Classes in non-strict mode are not supported yet on Node v4.*/
-
-if (a) {
- function foo() { /*error Block-Scoped Functions in non-strict mode are not supported yet on Node v4.*/
- //...
- }
-}
-
-var p = new Proxy(o, { /*error Proxy is not supported yet on Node v4.*/
- //...
-});
-```
-
-Examples of 👍 **correct** code for this rule:
-
-```js
-/*eslint n/no-unsupported-features: ["error", {version: 4}]*/
-/*eslint-env es6*/
-
-for (var a of list) {
- //...
-}
-
-var a = `hello, ${world}!`;
-
-function foo() {
- "use strict";
-
- let a = 1;
- const b = 2;
-
- class A {
- //...
- }
-
- if (c) {
- function bar() {
- //...
- }
- }
-}
-
-var p = new Promise((resolve, reject) => {
- //...
-});
-```
-
-### Options
-
-```json
-{
- "n/no-unsupported-features": ["error", {
- "version": 4,
- "ignores": []
- }]
-}
-```
-
-#### version
-
-As mentioned above, this rule reads the [engines] field of `package.json` to detect Node.js version.
-Also, you can overwrite the version by `version` option.
-The `version` option accepts the following version number:
-
-- `0.10`
-- `0.12`
-- `4`
-- `5`
-- `6`
-- `6.5` ... `Symbol.hasInstance` and `Symbol.species`.
-- `7` ... Exponential operators, `Object.values`, `Object.entries`, and `Object.getOwnPropertyDescriptors`.
-- `7.6` ... Async functions.
-- `8` ... Trailing commas in functions.
-- `8.3` ... Rest/Spread proeprties.
-- `9.0` ... Illegal escape sequences in taggled templates, RegExp 's' flags, RegExp lookbehind assertions, `SharedArrayBuffer`, and `Atomics`.
-- `10.0` ... RegExp named capture groups, RegExp Unicode property escapes, Async generators, and `for-await-of` loops.
-
-#### ignores
-
-If you are using transpilers, maybe you want to ignore the warnings about some features.
-You can use this `ignores` option to ignore the given features.
-The `"ignores"` option accepts an array of the following strings.
-
-- `"syntax"` (group)
- - `"defaultParameters"`
- - `"restParameters"`
- - `"spreadOperators"`
- - `"objectLiteralExtensions"`
- - `"forOf"`
- - `"binaryNumberLiterals"`
- - `"octalNumberLiterals"`
- - `"templateStrings"`
- - `"regexpY"`
- - `"regexpU"`
- - `"destructuring"`
- - `"unicodeCodePointEscapes"`
- - `"new.target"`
- - `"const"`
- - `"let"`
- - `"blockScopedFunctions"`
- - `"arrowFunctions"`
- - `"generatorFunctions"`
- - `"classes"`
- - `"modules"`
- - `"exponentialOperators"`
- - `"asyncAwait"`
- - `"trailingCommasInFunctions"`
- - `"templateLiteralRevision"`
- - `"regexpS"`
- - `"regexpNamedCaptureGroups"`
- - `"regexpLookbehind"`
- - `"regexpUnicodeProperties"`
- - `"restProperties"`
- - `"spreadProperties"`
- - `"asyncGenerators"`
- - `"forAwaitOf"`
-- `"runtime"` (group)
- - `"globalObjects"` (group)
- - `"typedArrays"` (group)
- - `"Int8Array"`
- - `"Uint8Array"`
- - `"Uint8ClampedArray"`
- - `"Int16Array"`
- - `"Uint16Array"`
- - `"Int32Array"`
- - `"Uint32Array"`
- - `"Float32Array"`
- - `"Float64Array"`
- - `"DataView"`
- - `"Map"`
- - `"Set"`
- - `"WeakMap"`
- - `"WeakSet"`
- - `"Proxy"`
- - `"Reflect"`
- - `"Promise"`
- - `"Symbol"`
- - `"SharedArrayBuffer"`
- - `"Atomics"`
- - `"staticMethods"` (group)
- - `"Object.*"` (group)
- - `"Object.assign"`
- - `"Object.is"`
- - `"Object.getOwnPropertySymbols"`
- - `"Object.setPrototypeOf"`
- - `"Object.values"`
- - `"Object.entries"`
- - `"Object.getOwnPropertyDescriptors"`
- - `"String.*"` (group)
- - `"String.raw"`
- - `"String.fromCodePoint"`
- - `"Array.*"` (group)
- - `"Array.from"`
- - `"Array.of"`
- - `"Number.*"` (group)
- - `"Number.isFinite"`
- - `"Number.isInteger"`
- - `"Number.isSafeInteger"`
- - `"Number.isNaN"`
- - `"Number.EPSILON"`
- - `"Number.MIN_SAFE_INTEGER"`
- - `"Number.MAX_SAFE_INTEGER"`
- - `"Math.*"` (group)
- - `"Math.clz32"`
- - `"Math.imul"`
- - `"Math.sign"`
- - `"Math.log10"`
- - `"Math.log2"`
- - `"Math.log1p"`
- - `"Math.expm1"`
- - `"Math.cosh"`
- - `"Math.sinh"`
- - `"Math.tanh"`
- - `"Math.acosh"`
- - `"Math.asinh"`
- - `"Math.atanh"`
- - `"Math.trunc"`
- - `"Math.fround"`
- - `"Math.cbrt"`
- - `"Math.hypot"`
- - `"Symbol.*"` (group)
- - `"Symbol.hasInstance"`
- - `"Symbol.isConcatSpreadablec"`
- - `"Symbol.iterator"`
- - `"Symbol.species"`
- - `"Symbol.replace"`
- - `"Symbol.search"`
- - `"Symbol.split"`
- - `"Symbol.match"`
- - `"Symbol.toPrimitive"`
- - `"Symbol.toStringTag"`
- - `"Symbol.unscopables"`
- - `"Atomics.*"` (group)
- - `"Atomics.add"`
- - `"Atomics.and"`
- - `"Atomics.compareExchange"`
- - `"Atomics.exchange"`
- - `"Atomics.wait"`
- - `"Atomics.wake"`
- - `"Atomics.isLockFree"`
- - `"Atomics.load"`
- - `"Atomics.or"`
- - `"Atomics.store"`
- - `"Atomics.sub"`
- - `"Atomics.xor"`
- - `"extends"` (group)
- - `"extendsArray"`
- - `"extendsRegExp"`
- - `"extendsFunction"`
- - `"extendsPromise"`
- - `"extendsBoolean"`
- - `"extendsNumber"`
- - `"extendsString"`
- - `"extendsMap"`
- - `"extendsSet"`
- - `"extendsNull"`
-
-If a group value is given, all sub items of the value are ignored.
-e.g. if `"String.*"` is given then `"String.raw"` and `"String.fromCodePoint"` are ignored.
-
-Examples of 👍 **correct** code for the `"ignores"` option:
-
-```js
-/*eslint n/no-unsupported-features: ["error", {version: 4, ignores: ["defaultParameters"]}]*/
-/*eslint-env es6*/
-
-function foo(a = 1) {
- //...
-}
-```
-
-## ⚠️ Known Limitations
-
-This rule cannot report non-static things.
-E.g., a use of instance methods.
-
-## 📚 Further Reading
-
--
-
-[engines]: https://docs.npmjs.com/files/package.json#engines
-
-## 🔎 Implementation
-
-- [Rule source](../../lib/rules/no-unsupported-features.js)
-- [Test source](../../tests/lib/rules/no-unsupported-features.js)
diff --git a/lib/index.js b/lib/index.js
index def1bbf6..23cd1937 100644
--- a/lib/index.js
+++ b/lib/index.js
@@ -46,7 +46,6 @@ const rules = {
// Deprecated rules.
"no-hide-core-modules": require("./rules/no-hide-core-modules"),
- "no-unsupported-features": require("./rules/no-unsupported-features"),
}
const mod = {
diff --git a/lib/rules/no-unsupported-features.js b/lib/rules/no-unsupported-features.js
deleted file mode 100644
index 810945ef..00000000
--- a/lib/rules/no-unsupported-features.js
+++ /dev/null
@@ -1,1552 +0,0 @@
-/**
- * @author Toru Nagashima
- * See LICENSE file in root directory for full license.
- */
-"use strict"
-
-const semver = require("semver")
-const {
- getInnermostScope,
- getPropertyName,
-} = require("@eslint-community/eslint-utils")
-const getPackageJson = require("../util/get-package-json")
-
-const VERSION_MAP = new Map([
- [0.1, "0.10.0"],
- [0.12, "0.12.0"],
- [4, "4.0.0"],
- [5, "5.0.0"],
- [6, "6.0.0"],
- [6.5, "6.5.0"],
- [7, "7.0.0"],
- [7.6, "7.6.0"],
- [8, "8.0.0"],
- [8.3, "8.3.0"],
- [9, "9.0.0"],
- [10, "10.0.0"],
-])
-const VERSION_SCHEMA = {
- anyOf: [
- { enum: Array.from(VERSION_MAP.keys()) },
- {
- type: "string",
- pattern: "^(?:0|[1-9]\\d*)\\.(?:0|[1-9]\\d*)\\.(?:0|[1-9]\\d*)$",
- },
- ],
-}
-const DEFAULT_VERSION = "4.0.0"
-const FUNC_TYPE = /^(?:Arrow)?Function(?:Declaration|Expression)$/u
-const CLASS_TYPE = /^Class(?:Declaration|Expression)$/u
-const DESTRUCTURING_PARENT_TYPE =
- /^(?:Function(?:Declaration|Expression)|ArrowFunctionExpression|AssignmentExpression|VariableDeclarator)$/u
-const TOPLEVEL_SCOPE_TYPE = /^(?:global|function|module)$/u
-const BINARY_NUMBER = /^0[bB]/u
-const OCTAL_NUMBER = /^0[oO]/u
-const UNICODE_ESC = /(\\+)u\{[0-9a-fA-F]+?\}/gu
-const GET_OR_SET = /^(?:g|s)et$/u
-const NEW_BUILTIN_TYPES = [
- "Int8Array",
- "Uint8Array",
- "Uint8ClampedArray",
- "Int16Array",
- "Uint16Array",
- "Int32Array",
- "Uint32Array",
- "Float32Array",
- "Float64Array",
- "DataView",
- "Map",
- "Set",
- "WeakMap",
- "WeakSet",
- "Proxy",
- "Reflect",
- "Promise",
- "Symbol",
- "SharedArrayBuffer",
- "Atomics",
-]
-const SUBCLASSING_TEST_TARGETS = [
- "Array",
- "RegExp",
- "Function",
- "Promise",
- "Boolean",
- "Number",
- "String",
- "Map",
- "Set",
-]
-const PROPERTY_TEST_TARGETS = {
- Object: [
- "assign",
- "is",
- "getOwnPropertySymbols",
- "setPrototypeOf",
- "values",
- "entries",
- "getOwnPropertyDescriptors",
- ],
- String: ["raw", "fromCodePoint"],
- Array: ["from", "of"],
- Number: [
- "isFinite",
- "isInteger",
- "isSafeInteger",
- "isNaN",
- "EPSILON",
- "MIN_SAFE_INTEGER",
- "MAX_SAFE_INTEGER",
- ],
- Math: [
- "clz32",
- "imul",
- "sign",
- "log10",
- "log2",
- "log1p",
- "expm1",
- "cosh",
- "sinh",
- "tanh",
- "acosh",
- "asinh",
- "atanh",
- "trunc",
- "fround",
- "cbrt",
- "hypot",
- ],
- Symbol: [
- "hasInstance",
- "isConcatSpreadablec",
- "iterator",
- "species",
- "replace",
- "search",
- "split",
- "match",
- "toPrimitive",
- "toStringTag",
- "unscopables",
- ],
- Atomics: [
- "add",
- "and",
- "compareExchange",
- "exchange",
- "wait",
- "wake",
- "isLockFree",
- "load",
- "or",
- "store",
- "sub",
- "xor",
- ],
-}
-const REGEXP_NAMED_GROUP = /(\\*)\(\?<[_$\w]/u
-const REGEXP_LOOKBEHIND = /(\\*)\(\?<[=!]/u
-const REGEXP_UNICODE_PROPERTY = /(\\*)\\[pP]\{.+?\}/u
-const FEATURES = {
- defaultParameters: {
- alias: ["syntax"],
- name: "Default parameters",
- node: "6.0.0",
- },
- restParameters: {
- alias: ["syntax"],
- name: "Rest parameters",
- node: "6.0.0",
- },
- spreadOperators: {
- alias: ["syntax"],
- name: "Spread operators",
- node: "5.0.0",
- },
- objectLiteralExtensions: {
- alias: ["syntax"],
- name: "Object literal extensions",
- node: "4.0.0",
- },
- objectPropertyShorthandOfGetSet: {
- alias: ["syntax", "objectLiteralExtensions"],
- name: "Property shorthand of 'get' and 'set'",
- node: "6.0.0",
- },
- forOf: {
- alias: ["syntax"],
- name: "'for..of' loops",
- node: "0.12.0",
- },
- binaryNumberLiterals: {
- alias: ["syntax"],
- name: "Binary number literals",
- node: "4.0.0",
- },
- octalNumberLiterals: {
- alias: ["syntax"],
- name: "Octal number literals",
- node: "4.0.0",
- },
- templateStrings: {
- alias: ["syntax"],
- name: "Template strings",
- node: "4.0.0",
- },
- regexpY: {
- alias: ["syntax"],
- name: "RegExp 'y' flags",
- node: "6.0.0",
- },
- regexpU: {
- alias: ["syntax"],
- name: "RegExp 'u' flags",
- node: "6.0.0",
- },
- destructuring: {
- alias: ["syntax"],
- name: "Destructuring",
- node: "6.0.0",
- },
- unicodeCodePointEscapes: {
- alias: ["syntax"],
- name: "Unicode code point escapes",
- node: "4.0.0",
- },
- "new.target": {
- alias: ["syntax"],
- name: "'new.target'",
- node: "5.0.0",
- },
- const: {
- alias: ["syntax"],
- name: "'const' declarations",
- node: {
- sloppy: "6.0.0",
- strict: "4.0.0",
- },
- },
- let: {
- alias: ["syntax"],
- name: "'let' declarations",
- node: {
- sloppy: "6.0.0",
- strict: "4.0.0",
- },
- },
- blockScopedFunctions: {
- alias: ["syntax"],
- name: "Block-scoped functions",
- node: {
- sloppy: "6.0.0",
- strict: "4.0.0",
- },
- },
- arrowFunctions: {
- alias: ["syntax"],
- name: "Arrow functions",
- node: "4.0.0",
- },
- generatorFunctions: {
- alias: ["syntax"],
- name: "Generator functions",
- node: "4.0.0",
- },
- classes: {
- alias: ["syntax"],
- name: "Classes",
- node: {
- sloppy: "6.0.0",
- strict: "4.0.0",
- },
- },
- modules: {
- alias: ["syntax"],
- name: "Import and export declarations",
- node: null,
- },
- exponentialOperators: {
- alias: ["syntax"],
- name: "Exponential operators (**)",
- node: "7.0.0",
- },
- asyncAwait: {
- alias: ["syntax"],
- name: "Async functions",
- node: "7.6.0",
- },
- trailingCommasInFunctions: {
- alias: ["syntax"],
- name: "Trailing commas in functions",
- node: "8.0.0",
- },
- //------------------------------------------
- templateLiteralRevision: {
- alias: ["syntax"],
- name: "Illegal escape sequences in taggled templates",
- node: "9.0.0",
- },
- regexpS: {
- alias: ["syntax"],
- name: "RegExp 's' flags",
- node: "9.0.0",
- },
- regexpNamedCaptureGroups: {
- alias: ["syntax"],
- name: "RegExp named capture groups",
- node: "10.0.0",
- },
- regexpLookbehind: {
- alias: ["syntax"],
- name: "RegExp lookbehind assertions",
- node: "9.0.0",
- },
- regexpUnicodeProperties: {
- alias: ["syntax"],
- name: "RegExp Unicode property escapes",
- node: "10.0.0",
- },
- restProperties: {
- alias: ["syntax"],
- name: "Rest properties",
- node: "8.3.0",
- },
- spreadProperties: {
- alias: ["syntax"],
- name: "Spread properties",
- node: "8.3.0",
- },
- asyncGenerators: {
- alias: ["syntax"],
- name: "Async generators",
- node: "10.0.0",
- },
- forAwaitOf: {
- alias: ["syntax"],
- name: "for-await-of loops",
- node: "10.0.0",
- },
-
- Int8Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Int8Array'",
- singular: true,
- node: "0.12.0",
- },
- Uint8Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Uint8Array'",
- singular: true,
- node: "0.12.0",
- },
- Uint8ClampedArray: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Uint8ClampedArray'",
- singular: true,
- node: "0.12.0",
- },
- Int16Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Int16Array'",
- singular: true,
- node: "0.12.0",
- },
- Uint16Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Uint16Array'",
- singular: true,
- node: "0.12.0",
- },
- Int32Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Int32Array'",
- singular: true,
- node: "0.12.0",
- },
- Uint32Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Uint32Array'",
- singular: true,
- node: "0.12.0",
- },
- Float32Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Float32Array'",
- singular: true,
- node: "0.12.0",
- },
- Float64Array: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'Float64Array'",
- singular: true,
- node: "0.12.0",
- },
- DataView: {
- alias: ["runtime", "globalObjects", "typedArrays"],
- name: "'DataView'",
- singular: true,
- node: "0.12.0",
- },
- Map: {
- alias: ["runtime", "globalObjects"],
- name: "'Map'",
- singular: true,
- node: "0.12.0",
- },
- Set: {
- alias: ["runtime", "globalObjects"],
- name: "'Set'",
- singular: true,
- node: "0.12.0",
- },
- WeakMap: {
- alias: ["runtime", "globalObjects"],
- name: "'WeakMap'",
- singular: true,
- node: "0.12.0",
- },
- WeakSet: {
- alias: ["runtime", "globalObjects"],
- name: "'WeakSet'",
- singular: true,
- node: "0.12.0",
- },
- Proxy: {
- alias: ["runtime", "globalObjects"],
- name: "'Proxy'",
- singular: true,
- node: "6.0.0",
- },
- Reflect: {
- alias: ["runtime", "globalObjects"],
- name: "'Reflect'",
- singular: true,
- node: "6.0.0",
- },
- Promise: {
- alias: ["runtime", "globalObjects"],
- name: "'Promise'",
- singular: true,
- node: "0.12.0",
- },
- Symbol: {
- alias: ["runtime", "globalObjects"],
- name: "'Symbol'",
- singular: true,
- node: "0.12.0",
- },
- SharedArrayBuffer: {
- alias: ["runtime", "globalObjects"],
- name: "'SharedArrayBuffer'",
- singular: true,
- node: "9.0.0",
- },
- Atomics: {
- alias: ["runtime", "globalObjects"],
- name: "'Atomics'",
- singular: true,
- node: "9.0.0",
- },
-
- "Object.assign": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.assign'",
- singular: true,
- node: "4.0.0",
- },
- "Object.is": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.is'",
- singular: true,
- node: "0.12.0",
- },
- "Object.getOwnPropertySymbols": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.getOwnPropertySymbols'",
- singular: true,
- node: "0.12.0",
- },
- "Object.setPrototypeOf": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.setPrototypeOf'",
- singular: true,
- node: "0.12.0",
- },
- "Object.values": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.values'",
- singular: true,
- node: "7.0.0",
- },
- "Object.entries": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.entries'",
- singular: true,
- node: "7.0.0",
- },
- "Object.getOwnPropertyDescriptors": {
- alias: ["runtime", "staticMethods", "Object.*"],
- name: "'Object.getOwnPropertyDescriptors'",
- singular: true,
- node: "7.0.0",
- },
-
- "String.raw": {
- alias: ["runtime", "staticMethods", "String.*"],
- name: "'String.raw'",
- singular: true,
- node: "4.0.0",
- },
- "String.fromCodePoint": {
- alias: ["runtime", "staticMethods", "String.*"],
- name: "'String.fromCodePoint'",
- singular: true,
- node: "4.0.0",
- },
-
- "Array.from": {
- alias: ["runtime", "staticMethods", "Array.*"],
- name: "'Array.from'",
- singular: true,
- node: "4.0.0",
- },
- "Array.of": {
- alias: ["runtime", "staticMethods", "Array.*"],
- name: "'Array.of'",
- singular: true,
- node: "4.0.0",
- },
-
- "Number.isFinite": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.isFinite'",
- singular: true,
- node: "0.10.0",
- },
- "Number.isInteger": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.isInteger'",
- singular: true,
- node: "0.12.0",
- },
- "Number.isSafeInteger": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.isSafeInteger'",
- singular: true,
- node: "0.12.0",
- },
- "Number.isNaN": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.isNaN'",
- singular: true,
- node: "0.10.0",
- },
- "Number.EPSILON": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.EPSILON'",
- singular: true,
- node: "0.12.0",
- },
- "Number.MIN_SAFE_INTEGER": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.MIN_SAFE_INTEGER'",
- singular: true,
- node: "0.12.0",
- },
- "Number.MAX_SAFE_INTEGER": {
- alias: ["runtime", "staticMethods", "Number.*"],
- name: "'Number.MAX_SAFE_INTEGER'",
- singular: true,
- node: "0.12.0",
- },
-
- "Math.clz32": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.clz32'",
- singular: true,
- node: "0.12.0",
- },
- "Math.imul": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.imul'",
- singular: true,
- node: "0.12.0",
- },
- "Math.sign": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.sign'",
- singular: true,
- node: "0.12.0",
- },
- "Math.log10": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.log10'",
- singular: true,
- node: "0.12.0",
- },
- "Math.log2": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.log2'",
- singular: true,
- node: "0.12.0",
- },
- "Math.log1p": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.log1p'",
- singular: true,
- node: "0.12.0",
- },
- "Math.expm1": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.expm1'",
- singular: true,
- node: "0.12.0",
- },
- "Math.cosh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.cosh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.sinh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.sinh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.tanh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.tanh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.acosh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.acosh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.asinh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.asinh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.atanh": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.atanh'",
- singular: true,
- node: "0.12.0",
- },
- "Math.trunc": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.trunc'",
- singular: true,
- node: "0.12.0",
- },
- "Math.fround": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.fround'",
- singular: true,
- node: "0.12.0",
- },
- "Math.cbrt": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.cbrt'",
- singular: true,
- node: "0.12.0",
- },
- "Math.hypot": {
- alias: ["runtime", "staticMethods", "Math.*"],
- name: "'Math.hypot'",
- singular: true,
- node: "0.12.0",
- },
-
- "Symbol.hasInstance": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.hasInstance'",
- singular: true,
- node: "6.5.0",
- },
- "Symbol.isConcatSpreadablec": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.isConcatSpreadablec'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.iterator": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.iterator'",
- singular: true,
- node: "0.12.0",
- },
- "Symbol.species": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.species'",
- singular: true,
- node: "6.5.0",
- },
- "Symbol.replace": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.replace'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.search": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.search'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.split": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.split'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.match": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.match'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.toPrimitive": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.toPrimitive'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.toStringTag": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.toStringTag'",
- singular: true,
- node: "6.0.0",
- },
- "Symbol.unscopables": {
- alias: ["runtime", "staticMethods", "Symbol.*"],
- name: "'Symbol.unscopables'",
- singular: true,
- node: "4.0.0",
- },
-
- "Atomics.add": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.add'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.and": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.and'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.compareExchange": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.compareExchange'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.exchange": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.exchange'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.wait": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.wait'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.wake": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.wake'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.isLockFree": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.isLockFree'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.load": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.load'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.or": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.or'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.store": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.store'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.sub": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.sub'",
- singular: true,
- node: "9.0.0",
- },
- "Atomics.xor": {
- alias: ["runtime", "staticMethods", "Atomics.*"],
- name: "'Atomics.xor'",
- singular: true,
- node: "9.0.0",
- },
-
- extendsArray: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Array'",
- singular: true,
- node: "6.0.0",
- },
- extendsRegExp: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'RegExp'",
- singular: true,
- node: "5.0.0",
- },
- extendsFunction: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Function'",
- singular: true,
- node: "6.0.0",
- },
- extendsPromise: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Promise'",
- singular: true,
- node: "5.0.0",
- },
- extendsBoolean: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Boolean'",
- singular: true,
- node: "4.0.0",
- },
- extendsNumber: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Number'",
- singular: true,
- node: "4.0.0",
- },
- extendsString: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'String'",
- singular: true,
- node: "4.0.0",
- },
- extendsMap: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Map'",
- singular: true,
- node: "4.0.0",
- },
- extendsSet: {
- alias: ["runtime", "extends"],
- name: "Subclassing of 'Set'",
- singular: true,
- node: "4.0.0",
- },
- extendsNull: {
- alias: ["runtime", "extends"],
- name: "'extends null'",
- singular: true,
- node: null,
- },
-}
-const OPTIONS = Object.keys(FEATURES)
-
-/**
- * Gets default version configuration of this rule.
- *
- * This finds and reads 'package.json' file, then parses 'engines.node' field.
- * If it's nothing, this returns null.
- *
- * @param {string} filename - The file name of the current linting file.
- * @returns {string} The default version configuration.
- */
-function getDefaultVersion(filename) {
- const info = getPackageJson(filename)
- const nodeVersion = info && info.engines && info.engines.node
-
- return semver.validRange(nodeVersion) || DEFAULT_VERSION
-}
-
-/**
- * Gets values of the `ignores` option.
- *
- * @returns {string[]} Values of the `ignores` option.
- */
-function getIgnoresEnum() {
- return Object.keys(
- OPTIONS.reduce((retv, key) => {
- for (const alias of FEATURES[key].alias) {
- retv[alias] = true
- }
- retv[key] = true
- return retv
- }, Object.create(null))
- )
-}
-
-/**
- * Checks whether a given key should be ignored or not.
- *
- * @param {string} key - A key to check.
- * @param {string[]} ignores - An array of keys and aliases to be ignored.
- * @returns {boolean} `true` if the key should be ignored.
- */
-function isIgnored(key, ignores) {
- return (
- ignores.indexOf(key) !== -1 ||
- FEATURES[key].alias.some(alias => ignores.indexOf(alias) !== -1)
- )
-}
-
-/**
- * Parses the options.
- *
- * @param {number|string|object|undefined} options - An option object to parse.
- * @param {number} defaultVersion - The default version to use if the version option was omitted.
- * @returns {object} Parsed value.
- */
-function parseOptions(options, defaultVersion) {
- let version = null
- let range = null
- let ignores = []
-
- if (typeof options === "number") {
- version = VERSION_MAP.get(options)
- } else if (typeof options === "string") {
- version = options
- } else if (typeof options === "object") {
- version =
- typeof options.version === "number"
- ? VERSION_MAP.get(options.version)
- : options.version
-
- ignores = options.ignores || []
- }
-
- range = semver.validRange(version ? `>=${version}` : defaultVersion)
- if (!version) {
- version = defaultVersion
- }
-
- return Object.freeze({
- version,
- features: Object.freeze(
- OPTIONS.reduce((retv, key) => {
- const feature = FEATURES[key]
-
- if (isIgnored(key, ignores)) {
- retv[key] = Object.freeze({
- name: feature.name,
- singular: Boolean(feature.singular),
- supported: true,
- supportedInStrict: true,
- })
- } else if (typeof feature.node === "string") {
- retv[key] = Object.freeze({
- name: feature.name,
- singular: Boolean(feature.singular),
- supported: !semver.intersects(
- range,
- `<${feature.node}`
- ),
- supportedInStrict: !semver.intersects(
- range,
- `<${feature.node}`
- ),
- })
- } else {
- retv[key] = Object.freeze({
- name: feature.name,
- singular: Boolean(feature.singular),
- supported:
- feature.node != null &&
- feature.node.sloppy != null &&
- !semver.intersects(
- range,
- `<${feature.node.sloppy}`
- ),
- supportedInStrict:
- feature.node != null &&
- feature.node.strict != null &&
- !semver.intersects(
- range,
- `<${feature.node.strict}`
- ),
- })
- }
-
- return retv
- }, Object.create(null))
- ),
- })
-}
-
-/**
- * Find the scope that a given node belongs to.
- * @param {Scope} initialScope The initial scope to find.
- * @param {Node} node The AST node.
- * @returns {Scope} The scope that the node belongs to.
- */
-function normalizeScope(initialScope, node) {
- let scope = getInnermostScope(initialScope, node)
-
- while (scope && scope.block === node) {
- scope = scope.upper
- }
-
- return scope
-}
-
-/**
- * Checks whether the given string has `\u{90ABCDEF}`-like escapes.
- *
- * @param {string} raw - The string to check.
- * @returns {boolean} `true` if the string has Unicode code point escapes.
- */
-function hasUnicodeCodePointEscape(raw) {
- let match = null
-
- UNICODE_ESC.lastIndex = 0
- while ((match = UNICODE_ESC.exec(raw)) != null) {
- if (match[1].length % 2 === 1) {
- return true
- }
- }
-
- return false
-}
-
-/**
- * Check a given string has a given pattern.
- * @param {string} s A string to check.
- * @param {RegExp} pattern A RegExp object to check.
- * @returns {boolean} `true` if the string has the pattern.
- */
-function hasPattern(s, pattern) {
- const m = pattern.exec(s)
- return m != null && (m[1] || "").length % 2 === 0
-}
-
-module.exports = {
- meta: {
- docs: {
- description:
- "disallow unsupported ECMAScript features on the specified version",
- recommended: false,
-
- url: "/~https://github.com/eslint-community/eslint-plugin-n/blob/HEAD/docs/rules/no-unsupported-features.md",
- },
- type: "problem",
- deprecated: true,
- replacedBy: [
- "n/no-unsupported-features/es-syntax",
- "n/no-unsupported-features/es-builtins",
- ],
- fixable: null,
- schema: [
- {
- anyOf: [
- VERSION_SCHEMA.anyOf[0],
- VERSION_SCHEMA.anyOf[1],
- {
- type: "object",
- properties: {
- version: VERSION_SCHEMA,
- ignores: {
- type: "array",
- items: { enum: getIgnoresEnum() },
- uniqueItems: true,
- },
- },
- additionalProperties: false,
- },
- ],
- },
- ],
- messages: {
- unsupported:
- "{{feature}} {{be}} not supported yet on Node {{version}}.",
- },
- },
- create(context) {
- const sourceCode = context.sourceCode ?? context.getSourceCode() // TODO: just use context.sourceCode when dropping eslint < v9
- const supportInfo = parseOptions(
- context.options[0],
- getDefaultVersion(context.filename ?? context.getFilename())
- )
-
- /**
- * Gets the references of the specified global variables.
- *
- * @param {string[]} names - Variable names to get.
- * @returns {void}
- */
- function* getReferences(names) {
- const globalScope =
- sourceCode.getScope?.(sourceCode.ast) ?? context.getScope() //TODO: remove context.getScope() when dropping support for ESLint < v9
-
- for (const name of names) {
- const variable = globalScope.set.get(name)
-
- if (variable && variable.defs.length === 0) {
- yield* variable.references
- }
- }
- }
-
- /**
- * Checks whether the given function has trailing commas or not.
- *
- * @param {ASTNode} node - The function node to check.
- * @returns {boolean} `true` if the function has trailing commas.
- */
- function hasTrailingCommaForFunction(node) {
- const length = node.params.length
-
- return (
- length >= 1 &&
- sourceCode.getTokenAfter(node.params[length - 1]).value === ","
- )
- }
-
- /**
- * Checks whether the given call expression has trailing commas or not.
- *
- * @param {ASTNode} node - The call expression node to check.
- * @returns {boolean} `true` if the call expression has trailing commas.
- */
- function hasTrailingCommaForCall(node) {
- return (
- node.arguments.length >= 1 &&
- sourceCode.getLastToken(node, 1).value === ","
- )
- }
-
- /**
- * Checks whether the given class extends from null or not.
- *
- * @param {ASTNode} node - The class node to check.
- * @returns {boolean} `true` if the class extends from null.
- */
- function extendsNull(node) {
- return (
- node.superClass != null &&
- node.superClass.type === "Literal" &&
- node.superClass.value === null
- )
- }
-
- /**
- * Reports a given node if the specified feature is not supported.
- *
- * @param {ASTNode} node - A node to be reported.
- * @param {string} key - A feature name to report.
- * @returns {void}
- */
- function report(node, key) {
- const globalScope =
- sourceCode.getScope?.(node) ?? context.getScope() //TODO: remove context.getScope() when dropping support for ESLint < v9
- const version = supportInfo.version
- const feature = supportInfo.features[key]
- if (feature.supported) {
- return
- }
-
- if (!feature.supportedInStrict) {
- context.report({
- node,
- messageId: "unsupported",
- data: {
- feature: feature.name,
- be: feature.singular ? "is" : "are",
- version,
- },
- })
- } else if (!normalizeScope(globalScope, node).isStrict) {
- context.report({
- node,
- messageId: "unsupported",
- data: {
- feature: `${feature.name} in non-strict mode`,
- be: feature.singular ? "is" : "are",
- version,
- },
- })
- }
- }
-
- /**
- * Validate RegExp syntax.
- * @param {string} pattern A RegExp pattern to check.
- * @param {string} flags A RegExp flags to check.
- * @param {ASTNode} node A node to report.
- * @returns {void}
- */
- function validateRegExp(pattern, flags, node) {
- if (typeof pattern === "string") {
- if (hasPattern(pattern, REGEXP_NAMED_GROUP)) {
- report(node, "regexpNamedCaptureGroups")
- }
- if (hasPattern(pattern, REGEXP_LOOKBEHIND)) {
- report(node, "regexpLookbehind")
- }
- if (hasPattern(pattern, REGEXP_UNICODE_PROPERTY)) {
- report(node, "regexpUnicodeProperties")
- }
- }
- if (typeof flags === "string") {
- if (flags.indexOf("y") !== -1) {
- report(node, "regexpY")
- }
- if (flags.indexOf("u") !== -1) {
- report(node, "regexpU")
- }
- if (flags.indexOf("s") !== -1) {
- report(node, "regexpS")
- }
- }
- }
-
- /**
- * Validate RegExp syntax in a RegExp literal.
- * @param {ASTNode} node A Literal node to check.
- * @returns {void}
- */
- function validateRegExpLiteral(node) {
- validateRegExp(node.regex.pattern, node.regex.flags, node)
- }
-
- /**
- * Validate RegExp syntax in the first argument of `new RegExp()`.
- * @param {ASTNode} node A NewExpression node to check.
- * @returns {void}
- */
- function validateRegExpString(node) {
- const patternNode = node.arguments[0]
- const flagsNode = node.arguments[1]
- const pattern =
- patternNode &&
- patternNode.type === "Literal" &&
- typeof patternNode.value === "string"
- ? patternNode.value
- : null
- const flags =
- flagsNode &&
- flagsNode.type === "Literal" &&
- typeof flagsNode.value === "string"
- ? flagsNode.value
- : null
- validateRegExp(pattern, flags, node)
- }
-
- return {
- "Program:exit"() {
- // Check new global variables.
- for (const name of NEW_BUILTIN_TYPES) {
- for (const reference of getReferences([name])) {
- // Ignore if it's using new static methods.
- const node = reference.identifier
- const parentNode = node.parent
- const properties = PROPERTY_TEST_TARGETS[name]
- if (
- properties &&
- parentNode.type === "MemberExpression"
- ) {
- const propertyName = getPropertyName(parentNode)
- if (properties.indexOf(propertyName) !== -1) {
- continue
- }
- }
-
- report(reference.identifier, name)
- }
- }
-
- // Check static methods.
- for (const reference of getReferences(
- Object.keys(PROPERTY_TEST_TARGETS)
- )) {
- const node = reference.identifier
- const parentNode = node.parent
- if (
- parentNode.type !== "MemberExpression" ||
- parentNode.object !== node
- ) {
- continue
- }
-
- const objectName = node.name
- const properties = PROPERTY_TEST_TARGETS[objectName]
- const propertyName = getPropertyName(parentNode)
- if (
- propertyName &&
- properties.indexOf(propertyName) !== -1
- ) {
- report(parentNode, `${objectName}.${propertyName}`)
- }
- }
-
- // Check subclassing
- for (const reference of getReferences(
- SUBCLASSING_TEST_TARGETS
- )) {
- const node = reference.identifier
- const parentNode = node.parent
- if (
- CLASS_TYPE.test(parentNode.type) &&
- parentNode.superClass === node
- ) {
- report(node, `extends${node.name}`)
- }
- }
- },
-
- ArrowFunctionExpression(node) {
- report(node, "arrowFunctions")
- if (node.async) {
- report(node, "asyncAwait")
- }
- if (hasTrailingCommaForFunction(node)) {
- report(node, "trailingCommasInFunctions")
- }
- },
-
- AssignmentPattern(node) {
- if (FUNC_TYPE.test(node.parent.type)) {
- report(node, "defaultParameters")
- }
- },
-
- FunctionDeclaration(node) {
- const scope = (
- sourceCode.getScope?.(node) ?? context.getScope()
- ).upper //TODO: remove context.getScope() when dropping support for ESLint < v9
- if (!TOPLEVEL_SCOPE_TYPE.test(scope.type)) {
- report(node, "blockScopedFunctions")
- }
- if (node.generator) {
- report(node, "generatorFunctions")
- }
- if (node.async) {
- report(node, "asyncAwait")
- }
- if (hasTrailingCommaForFunction(node)) {
- report(node, "trailingCommasInFunctions")
- }
- if (node.async && node.generator) {
- report(node, "asyncGenerators")
- }
- },
-
- FunctionExpression(node) {
- if (node.generator) {
- report(node, "generatorFunctions")
- }
- if (node.async) {
- report(node, "asyncAwait")
- }
- if (hasTrailingCommaForFunction(node)) {
- report(node, "trailingCommasInFunctions")
- }
- if (node.async && node.generator) {
- report(node, "asyncGenerators")
- }
- },
-
- MetaProperty(node) {
- const meta = node.meta.name || node.meta
- const property = node.property.name || node.property
- if (meta === "new" && property === "target") {
- report(node, "new.target")
- }
- },
-
- ClassDeclaration(node) {
- report(node, "classes")
-
- if (extendsNull(node)) {
- report(node, "extendsNull")
- }
- },
-
- ClassExpression(node) {
- report(node, "classes")
-
- if (extendsNull(node)) {
- report(node, "extendsNull")
- }
- },
-
- ForOfStatement(node) {
- report(node, "forOf")
- if (node.await) {
- report(node, "forAwaitOf")
- }
- },
-
- VariableDeclaration(node) {
- if (node.kind === "const") {
- report(node, "const")
- } else if (node.kind === "let") {
- report(node, "let")
- }
- },
-
- ArrayPattern(node) {
- if (DESTRUCTURING_PARENT_TYPE.test(node.parent.type)) {
- report(node, "destructuring")
- }
- },
-
- AssignmentExpression(node) {
- if (node.operator === "**=") {
- report(node, "exponentialOperators")
- }
- },
-
- AwaitExpression(node) {
- report(node, "asyncAwait")
- },
-
- BinaryExpression(node) {
- if (node.operator === "**") {
- report(node, "exponentialOperators")
- }
- },
-
- CallExpression(node) {
- if (hasTrailingCommaForCall(node)) {
- report(node, "trailingCommasInFunctions")
- }
- },
-
- Identifier(node) {
- const raw = sourceCode.getText(node)
- if (hasUnicodeCodePointEscape(raw)) {
- report(node, "unicodeCodePointEscapes")
- }
- },
-
- Literal(node) {
- if (typeof node.value === "number") {
- if (BINARY_NUMBER.test(node.raw)) {
- report(node, "binaryNumberLiterals")
- } else if (OCTAL_NUMBER.test(node.raw)) {
- report(node, "octalNumberLiterals")
- }
- } else if (typeof node.value === "string") {
- if (hasUnicodeCodePointEscape(node.raw)) {
- report(node, "unicodeCodePointEscapes")
- }
- } else if (node.regex) {
- validateRegExpLiteral(node)
- }
- },
-
- NewExpression(node) {
- if (
- node.callee.type === "Identifier" &&
- node.callee.name === "RegExp"
- ) {
- validateRegExpString(node)
- }
- if (hasTrailingCommaForCall(node)) {
- report(node, "trailingCommasInFunctions")
- }
- },
-
- ObjectPattern(node) {
- if (DESTRUCTURING_PARENT_TYPE.test(node.parent.type)) {
- report(node, "destructuring")
- }
- },
-
- Property(node) {
- if (
- node.parent.type === "ObjectExpression" &&
- (node.computed || node.shorthand || node.method)
- ) {
- if (node.shorthand && GET_OR_SET.test(node.key.name)) {
- report(node, "objectPropertyShorthandOfGetSet")
- } else {
- report(node, "objectLiteralExtensions")
- }
- }
- },
-
- RestElement(node) {
- if (FUNC_TYPE.test(node.parent.type)) {
- report(node, "restParameters")
- } else if (node.parent.type === "ObjectPattern") {
- report(node, "restProperties")
- }
- },
-
- SpreadElement(node) {
- if (node.parent.type === "ObjectExpression") {
- report(node, "spreadProperties")
- } else {
- report(node, "spreadOperators")
- }
- },
-
- TemplateElement(node) {
- if (node.value.cooked == null) {
- report(node, "templateLiteralRevision")
- }
- },
-
- TemplateLiteral(node) {
- report(node, "templateStrings")
- },
-
- ExperimentalRestProperty(node) {
- report(node, "restProperties")
- },
-
- ExperimentalSpreadProperty(node) {
- report(node, "spreadProperties")
- },
-
- RestProperty(node) {
- report(node, "restProperties")
- },
-
- SpreadProperty(node) {
- report(node, "spreadProperties")
- },
-
- ExportAllDeclaration(node) {
- report(node, "modules")
- },
-
- ExportDefaultDeclaration(node) {
- report(node, "modules")
- },
-
- ExportNamedDeclaration(node) {
- report(node, "modules")
- },
-
- ImportDeclaration(node) {
- report(node, "modules")
- },
- }
- },
-}
diff --git a/tests/lib/rules/no-unsupported-features.js b/tests/lib/rules/no-unsupported-features.js
deleted file mode 100644
index 8351e7f9..00000000
--- a/tests/lib/rules/no-unsupported-features.js
+++ /dev/null
@@ -1,1395 +0,0 @@
-/**
- * @author Toru Nagashima
- * See LICENSE file in root directory for full license.
- */
-"use strict"
-
-const path = require("path")
-const RuleTester = require("#eslint-rule-tester").RuleTester
-const rule = require("../../../lib/rules/no-unsupported-features")
-
-const VERSION_MAP = new Map([
- [0.1, "0.10.0"],
- [0.12, "0.12.0"],
- [4, "4.0.0"],
- [5, "5.0.0"],
- [6, "6.0.0"],
- [6.5, "6.5.0"],
- [7, "7.0.0"],
- [7.6, "7.6.0"],
- [8, "8.0.0"],
- [8.3, "8.3.0"],
- [9, "9.0.0"],
- [10, "10.0.0"],
-])
-
-/**
- * Creates test pattern.
- *
- * @param {{valid: object[], invalid: object[]}} retv - Actual test patterns.
- * @param {object} pattern - Original test pattern.
- * @returns {{valid: object[], invalid: object[]}} retv.
- */
-function convertPattern(retv, pattern) {
- let i = 0
-
- // If this test is on script mode, it should do this test on module mode as well.
- if (
- !pattern.modules &&
- pattern.code.indexOf("'use strict'") !== 0 &&
- pattern.name.indexOf("non-strict") === -1
- ) {
- convertPattern(
- retv,
- Object.create(pattern, { modules: { value: true } })
- )
- }
-
- // Creates error messages.
- const errors = []
- for (i = 0; i < pattern.errors; ++i) {
- errors.push(
- `${pattern.name} ${
- pattern.singular ? "is" : "are"
- } not supported yet on Node `
- )
- }
-
- // Creates each pattern of Node versions.
- for (const version of VERSION_MAP.keys()) {
- const versionText = VERSION_MAP.get(version)
-
- // Skips if ignored
- if (pattern.ignores && pattern.ignores.indexOf(version) !== -1) {
- continue
- }
-
- if (version >= pattern.supported) {
- // If this is supported, add to a valid pattern.
- retv.valid.push({
- code: `/*${pattern.name}: ${versionText}*/ ${pattern.code}`,
- options: [version],
- languageOptions: {
- ecmaVersion: 2018,
- sourceType: pattern.modules ? "module" : "script",
- globals: { SharedArrayBuffer: false, Atomics: false },
- },
- })
- } else {
- // If this is not supported, add to a valid pattern with a "ignores" option.
- ;[].push.apply(
- retv.valid,
- pattern.keys.map(key => ({
- code: `/*${pattern.name}: ${versionText}, ignores: ["${key}"]*/ ${pattern.code}`,
- options: [{ version, ignores: [key] }],
- languageOptions: {
- ecmaVersion: 2018,
- sourceType: pattern.modules ? "module" : "script",
- globals: { SharedArrayBuffer: false, Atomics: false },
- },
- }))
- )
-
- // If this is not supported, add to a invalid pattern.
- retv.invalid.push({
- code: `/*${pattern.name}: ${versionText}*/ ${pattern.code}`,
- options: [version],
- languageOptions: {
- ecmaVersion: 2018,
- sourceType: pattern.modules ? "module" : "script",
- globals: { SharedArrayBuffer: false, Atomics: false },
- },
- errors: errors.map(message => `${message + versionText}.`),
- })
- }
- }
-
- return retv
-}
-
-/**
- * Makes a file path to a fixture.
- * @param {string} name - A name.
- * @returns {string} A file path to a fixture.
- */
-function fixture(name) {
- return path.resolve(
- __dirname,
- "../../fixtures/no-unsupported-features",
- name
- )
-}
-
-const ruleTester = new RuleTester()
-ruleTester.run(
- "no-unsupported-features",
- rule,
- [
- {
- keys: ["defaultParameters", "syntax"],
- name: "Default parameters",
- code: "function foo(a = 1) {} ;(function(a = 1) {})()",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["restParameters", "syntax"],
- name: "Rest parameters",
- code: "function foo(a, ...b) {} ;(function(a, ...b) {})()",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["spreadOperators", "syntax"],
- name: "Spread operators",
- code: "foo(...a); foo([...a, ...b])",
- errors: 3,
- supported: 5,
- },
- {
- keys: ["objectLiteralExtensions", "syntax"],
- name: "Object literal extensions",
- code: "var obj = {[a]: 0, b, c() {}, get [d]() {}, set [d](v) {}}",
- errors: 5,
- supported: 4,
- },
- {
- keys: [
- "objectPropertyShorthandOfGetSet",
- "objectLiteralExtensions",
- "syntax",
- ],
- name: "Property shorthand of 'get' and 'set'",
- code: "var obj = {get, set}",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["forOf", "syntax"],
- name: "'for..of' loops",
- code: "for (var a of []) {}",
- errors: 1,
- supported: 0.12,
- },
- {
- keys: ["binaryNumberLiterals", "syntax"],
- name: "Binary number literals",
- code: "var a = 0b10 === 2 && 0B10 === 2",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["octalNumberLiterals", "syntax"],
- name: "Octal number literals",
- code: "var a = 0o10 === 8 && 0O10 === 8",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["templateStrings", "syntax"],
- name: "Template strings",
- code: "`hello, ${world}!`; foo`tagged`;",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["regexpY", "syntax"],
- name: "RegExp 'y' flags",
- code: "new RegExp('', 'y'); (/a/y)",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["regexpU", "syntax"],
- name: "RegExp 'u' flags",
- code: "new RegExp('', 'u'); (/a/u)",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["destructuring", "syntax"],
- name: "Destructuring",
- code: "var [[a], [b = 1]] = [], {c: {c}, d: {d = 1}} = {};",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["destructuring", "syntax"],
- name: "Destructuring",
- code: "[[a], [b = 1]] = []; ({c: {c}, d: {d = 1}} = {})",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["destructuring", "syntax"],
- name: "Destructuring",
- code: "function foo([[a], [b = 1]], {c: {c}, d: {d = 1}}) {}",
- errors: 2,
- supported: 6,
- },
- {
- keys: ["unicodeCodePointEscapes", "syntax"],
- name: "Unicode code point escapes",
- code: "var \\u{102C0} = { \\u{102C0} : '\\u{1d306}' };",
- errors: 3,
- supported: 4,
- },
- {
- keys: ["new.target", "syntax"],
- name: "'new.target'",
- code: "function Foo() { new.target; } ;(function() { new.target; })()",
- errors: 2,
- supported: 5,
- },
- {
- keys: ["const", "syntax"],
- name: "'const' declarations",
- code: "'use strict'; const a = 0;",
- errors: 1,
- supported: 4,
- },
- {
- keys: ["const", "syntax"],
- name: "'const' declarations",
- code: "const a = 0;",
- errors: 1,
- supported: 4,
- modules: true,
- },
- {
- keys: ["const", "syntax"],
- name: "'const' declarations in non-strict mode",
- code: "const a = 0;",
- errors: 1,
- supported: 6,
- ignores: [0.1, 0.12],
- },
- {
- keys: ["let", "syntax"],
- name: "'let' declarations",
- code: "'use strict'; let a = 0;",
- errors: 1,
- supported: 4,
- },
- {
- keys: ["let", "syntax"],
- name: "'let' declarations",
- code: "let a = 0;",
- errors: 1,
- supported: 4,
- modules: true,
- },
- {
- keys: ["let", "syntax"],
- name: "'let' declarations in non-strict mode",
- code: "let a = 0;",
- errors: 1,
- supported: 6,
- ignores: [0.1, 0.12],
- },
- {
- keys: ["blockScopedFunctions", "syntax"],
- name: "Block-scoped functions",
- code: "'use strict'; { function foo() {} } if (a) { function foo() {} }",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["blockScopedFunctions", "syntax"],
- name: "Block-scoped functions",
- code: "{ function foo() {} } if (a) { function foo() {} }",
- errors: 2,
- supported: 4,
- modules: true,
- },
- {
- keys: ["blockScopedFunctions", "syntax"],
- name: "Block-scoped functions in non-strict mode",
- code: "{ function foo() {} } if (a) { function foo() {} }",
- errors: 2,
- supported: 6,
- ignores: [0.1, 0.12],
- },
- {
- keys: ["arrowFunctions", "syntax"],
- name: "Arrow functions",
- code: "var a = () => 1",
- errors: 1,
- supported: 4,
- },
- {
- keys: ["classes", "syntax"],
- name: "Classes",
- code: "'use strict'; class A {} new (class{})()",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["classes", "syntax"],
- name: "Classes",
- code: "class A {} new (class{})()",
- errors: 2,
- supported: 4,
- modules: true,
- },
- {
- keys: ["classes", "syntax"],
- name: "Classes in non-strict mode",
- code: "class A {} new (class{})()",
- errors: 2,
- supported: 6,
- ignores: [0.1, 0.12],
- },
- {
- keys: ["generatorFunctions", "syntax"],
- name: "Generator functions",
- code: "function* foo() {} ;(function*() {})();",
- errors: 2,
- supported: 4,
- },
- {
- keys: ["modules", "syntax"],
- name: "Import and export declarations",
- code: "import foo from 'foo'; export default 0; export {foo}; export * from 'foo';",
- errors: 4,
- supported: NaN,
- modules: true,
- },
- {
- keys: ["exponentialOperators", "syntax"],
- name: "Exponential operators (**)",
- code: "var a = 2 ** 10; a **= 10;",
- errors: 2,
- supported: 7,
- },
- {
- keys: ["asyncAwait", "syntax"],
- name: "Async functions",
- code: [
- "async function foo() { await obj; };",
- "(async function() { await obj; });",
- "(async () => { await obj; });",
- "({async foo() { await obj; }});",
- "class A { async foo() { await obj; } };",
- ].join("\n"),
- errors: 10,
- supported: 7.6,
- ignores: [0.1, 0.12, 4, 5],
- },
- {
- keys: ["trailingCommasInFunctions", "syntax"],
- name: "Trailing commas in functions",
- code: [
- "function foo(a,) {}",
- "(function(a,) {});",
- "((a,) => {});",
- "({ foo(a,) {} });",
- "class A { foo(a,) {} };",
- "foo(a,);",
- "new Foo(a,);",
- ].join("\n"),
- errors: 7,
- supported: 8,
- ignores: [0.1, 0.12, 4, 5],
- },
- {
- keys: ["templateLiteralRevision", "syntax"],
- name: "Illegal escape sequences in taggled templates",
- code: [
- "tag`\\01\\1\\xg\\xAg\\u0\\u0g\\u00g\\u000g\\u{g\\u{0\\u{110000}${0}\\0`",
- ].join("\n"),
- errors: 1,
- supported: 9,
- ignores: [0.1, 0.12, 4, 5],
- },
- {
- keys: ["regexpS", "syntax"],
- name: "RegExp 's' flags",
- code: "new RegExp('', 's'); (/a/s)",
- errors: 2,
- supported: 9,
- },
- {
- keys: ["regexpNamedCaptureGroups", "syntax"],
- name: "RegExp named capture groups",
- code: [
- "new RegExp('(?b)');",
- //TODO: Espree has not supported this syntax yet.
- // ";(/(?b)/)",
- ].join("\n"),
- errors: 1,
- supported: 10,
- },
- {
- keys: ["regexpLookbehind", "syntax"],
- name: "RegExp lookbehind assertions",
- code: [
- "new RegExp('(?<=a)b')",
- "new RegExp('(? 1",
- },
- {
- filename: fixture("gte-4.4.0-lt-5.0.0/a.js"),
- code: "var a = () => 1",
- },
- {
- filename: fixture("hat-4.1.2/a.js"),
- code: "var a = () => 1",
- },
- {
- code: "'\\\\u{0123}'",
- },
- {
- filename: fixture("gte-4.0.0/a.js"),
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- options: [{ ignores: ["asyncAwait"] }],
- },
- {
- filename: fixture("gte-7.6.0/a.js"),
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- },
- {
- filename: fixture("gte-7.10.0/a.js"),
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- },
- {
- filename: fixture("invalid/a.js"),
- code: "var a = () => 1",
- },
- {
- filename: fixture("nothing/a.js"),
- code: "var a = () => 1",
- },
- {
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- options: ["7.10.0"],
- },
- {
- filename: fixture("without-node/a.js"),
- code: "var a = () => 1",
- },
- ],
- invalid: [
- {
- filename: fixture("gte-0.12.8/a.js"),
- code: "var a = () => 1",
- errors: [
- "Arrow functions are not supported yet on Node >=0.12.8.",
- ],
- },
- {
- filename: fixture("invalid/a.js"),
- code: "var a = (b,) => 1",
- languageOptions: { ecmaVersion: 2017 },
- errors: [
- "Trailing commas in functions are not supported yet on Node 4.0.0.",
- ],
- },
- {
- filename: fixture("lt-6.0.0/a.js"),
- code: "var a = () => 1",
- languageOptions: { ecmaVersion: 2017 },
- errors: [
- "Arrow functions are not supported yet on Node <6.0.0.",
- ],
- },
- {
- filename: fixture("nothing/a.js"),
- code: "var a = (b,) => 1",
- languageOptions: { ecmaVersion: 2017 },
- errors: [
- "Trailing commas in functions are not supported yet on Node 4.0.0.",
- ],
- },
- {
- filename: fixture("gte-7.5.0/a.js"),
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- errors: [
- "Async functions are not supported yet on Node >=7.5.0.",
- ],
- },
- {
- code: "var a = async () => 1",
- languageOptions: { ecmaVersion: 2017 },
- options: ["7.1.0"],
- errors: [
- "Async functions are not supported yet on Node 7.1.0.",
- ],
- },
- ],
- })
-)