-
-
Notifications
You must be signed in to change notification settings - Fork 368
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fixes: #36
- Loading branch information
Showing
6 changed files
with
446 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
# Enforce the use of built-in methods instead of unnecessary polyfills | ||
💼 This rule is enabled in the ✅ `recommended` [config](https://github.com/sindresorhus/eslint-plugin-unicorn#preset-configs). | ||
<!-- end auto-generated rule header --> | ||
|
||
<!-- Do not manually modify RULE_NOTICE part. Run: `npm run generate-rule-notices` --> | ||
<!-- RULE_NOTICE --> | ||
|
||
✅ _This rule is part of the [recommended](https://github.com/sindresorhus/eslint-plugin-unicorn#recommended-config) config._ | ||
|
||
<!-- /RULE_NOTICE --> | ||
|
||
This rules helps to use existing methods instead of using extra polyfills. | ||
|
||
## Fail | ||
|
||
package.json | ||
|
||
```json | ||
{ | ||
"engines": { | ||
"node": ">=8" | ||
} | ||
} | ||
``` | ||
|
||
```js | ||
const assign = require('object-assign'); | ||
``` | ||
|
||
## Pass | ||
|
||
package.json | ||
|
||
```json | ||
{ | ||
"engines": { | ||
"node": "4" | ||
} | ||
} | ||
``` | ||
|
||
```js | ||
const assign = require('object-assign'); // Passes as Object.assign is not supported | ||
``` | ||
|
||
## Options | ||
|
||
Type: `object` | ||
|
||
### targets | ||
|
||
Type: `string | string[] | object` | ||
|
||
The `targets` option allows to specify the target versions. This option could be a Browserlist query or a targets object, see [core-js-compat `targets` option](https://github.com/zloirock/core-js/tree/HEAD/packages/core-js-compat#targets-option) for more informations. | ||
|
||
If the option is not specified, target versions are defined using the [`browserlist`](https://browsersl.ist) field, or as a last resort, the `engines` field in `package.json`. | ||
|
||
```js | ||
"unicorn/no-unnecessary-polyfills": [ | ||
"error", | ||
{ | ||
"targets": "node >=12" | ||
} | ||
] | ||
``` | ||
|
||
```js | ||
"unicorn/no-unnecessary-polyfills": [ | ||
"error", | ||
{ | ||
"targets": [ | ||
"node 14.1.0", | ||
"chrome 95" | ||
] | ||
} | ||
] | ||
``` | ||
|
||
```js | ||
"unicorn/no-unnecessary-polyfills": [ | ||
"error", | ||
{ | ||
"targets": { | ||
"node": "current", | ||
"firefox": "15" | ||
} | ||
} | ||
] | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,178 @@ | ||
'use strict'; | ||
const path = require('node:path'); | ||
const readPkgUp = require('read-pkg-up'); | ||
const coreJsCompat = require('core-js-compat'); | ||
const {camelCase} = require('lodash'); | ||
const isStaticRequire = require('./ast/is-static-require.js'); | ||
|
||
const {data: compatData, entries: coreJsEntries} = coreJsCompat; | ||
|
||
const MESSAGE_ID_POLYFILL = 'unnecessaryPolyfill'; | ||
const MESSAGE_ID_CORE_JS = 'unnecessaryCoreJsModule'; | ||
const messages = { | ||
[MESSAGE_ID_POLYFILL]: 'Use built-in instead.', | ||
[MESSAGE_ID_CORE_JS]: | ||
'All polyfilled features imported from `{{coreJsModule}}` are available as built-ins. Use the built-ins instead.', | ||
}; | ||
|
||
const additionalPolyfillPatterns = { | ||
'es.promise.finally': '|(p-finally)', | ||
'es.object.set-prototype-of': '|(setprototypeof)', | ||
'es.string.code-point-at': '|(code-point-at)', | ||
}; | ||
|
||
const prefixes = '(mdn-polyfills/|polyfill-)'; | ||
const suffixes = '(-polyfill)'; | ||
const delimiter = '(\\.|-|\\.prototype\\.|/)?'; | ||
|
||
const polyfills = Object.keys(compatData).map(feature => { | ||
let [ecmaVersion, constructorName, methodName = ''] = feature.split('.'); | ||
|
||
if (ecmaVersion === 'es') { | ||
ecmaVersion = '(es\\d*)'; | ||
} | ||
|
||
constructorName = `(${constructorName}|${camelCase(constructorName)})`; | ||
methodName &&= `(${methodName}|${camelCase(methodName)})`; | ||
|
||
const methodOrConstructor = methodName || constructorName; | ||
|
||
const patterns = [ | ||
`^((${prefixes}?(`, | ||
methodName && `(${ecmaVersion}${delimiter}${constructorName}${delimiter}${methodName})|`, // Ex: es6-array-copy-within | ||
methodName && `(${constructorName}${delimiter}${methodName})|`, // Ex: array-copy-within | ||
`(${ecmaVersion}${delimiter}${constructorName}))`, // Ex: es6-array | ||
`${suffixes}?)|`, | ||
`(${prefixes}${methodOrConstructor}|${methodOrConstructor}${suffixes})`, // Ex: polyfill-copy-within / polyfill-promise | ||
`${additionalPolyfillPatterns[feature] || ''})$`, | ||
]; | ||
|
||
return { | ||
feature, | ||
pattern: new RegExp(patterns.join(''), 'i'), | ||
}; | ||
}); | ||
|
||
function getTargets(options, dirname) { | ||
if (options?.targets) { | ||
return options.targets; | ||
} | ||
|
||
/** @type {readPkgUp.ReadResult | undefined} */ | ||
let packageResult; | ||
try { | ||
packageResult = readPkgUp.sync({normalize: false, cwd: dirname}); | ||
} catch { | ||
// This can happen if package.json files have comments in them etc. | ||
packageResult = undefined; | ||
} | ||
|
||
if (!packageResult) { | ||
return; | ||
} | ||
|
||
const {browserlist, engines} = packageResult.packageJson; | ||
return browserlist || engines; | ||
} | ||
|
||
function create(context) { | ||
const targets = getTargets(context.options[0], path.dirname(context.filename)); | ||
if (!targets) { | ||
return {}; | ||
} | ||
|
||
let unavailableFeatures; | ||
try { | ||
unavailableFeatures = coreJsCompat({targets}).list; | ||
} catch { | ||
// This can happen if the targets are invalid or use unsupported syntax like `{node:'*'}`. | ||
return {}; | ||
} | ||
|
||
const checkFeatures = features => !features.every(feature => unavailableFeatures.includes(feature)); | ||
|
||
return { | ||
Literal(node) { | ||
if ( | ||
!( | ||
(['ImportDeclaration', 'ImportExpression'].includes(node.parent.type) && node.parent.source === node) | ||
|| (isStaticRequire(node.parent) && node.parent.arguments[0] === node) | ||
) | ||
) { | ||
return; | ||
} | ||
|
||
const importedModule = node.value; | ||
if (typeof importedModule !== 'string' || ['.', '/'].includes(importedModule[0])) { | ||
return; | ||
} | ||
|
||
const coreJsModuleFeatures = coreJsEntries[importedModule.replace('core-js-pure', 'core-js')]; | ||
|
||
if (coreJsModuleFeatures) { | ||
if (coreJsModuleFeatures.length > 1) { | ||
if (checkFeatures(coreJsModuleFeatures)) { | ||
return { | ||
node, | ||
messageId: MESSAGE_ID_CORE_JS, | ||
data: { | ||
coreJsModule: importedModule, | ||
}, | ||
}; | ||
} | ||
} else if (!unavailableFeatures.includes(coreJsModuleFeatures[0])) { | ||
return {node, messageId: MESSAGE_ID_POLYFILL}; | ||
} | ||
|
||
return; | ||
} | ||
|
||
const polyfill = polyfills.find(({pattern}) => pattern.test(importedModule)); | ||
if (polyfill) { | ||
const [, namespace, method = ''] = polyfill.feature.split('.'); | ||
const [, features] = Object.entries(coreJsEntries).find( | ||
entry => entry[0] === `core-js/full/${namespace}${method && '/'}${method}`, | ||
); | ||
if (checkFeatures(features)) { | ||
return {node, messageId: MESSAGE_ID_POLYFILL}; | ||
} | ||
} | ||
}, | ||
}; | ||
} | ||
|
||
const schema = [ | ||
{ | ||
type: 'object', | ||
additionalProperties: false, | ||
required: ['targets'], | ||
properties: { | ||
targets: { | ||
oneOf: [ | ||
{ | ||
type: 'string', | ||
}, | ||
{ | ||
type: 'array', | ||
}, | ||
{ | ||
type: 'object', | ||
}, | ||
], | ||
}, | ||
}, | ||
}, | ||
]; | ||
|
||
/** @type {import('eslint').Rule.RuleModule} */ | ||
module.exports = { | ||
create, | ||
meta: { | ||
type: 'suggestion', | ||
docs: { | ||
description: 'Enforce the use of built-in methods instead of unnecessary polyfills.', | ||
}, | ||
schema, | ||
messages, | ||
}, | ||
}; |
Oops, something went wrong.