You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
207 lines
27 KiB
207 lines
27 KiB
"use strict";
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
value: true
|
|
});
|
|
exports.default = void 0;
|
|
|
|
var _fs = _interopRequireDefault(require("fs"));
|
|
|
|
var _findUp = _interopRequireDefault(require("find-up"));
|
|
|
|
var _lodash = _interopRequireDefault(require("lodash.memoize"));
|
|
|
|
var _helpers = require("../helpers");
|
|
|
|
var _providers = require("../providers");
|
|
|
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
|
|
/*
|
|
* Step 2) Logic that handles AST traversal
|
|
* Does not handle looking up the API
|
|
* Handles checking what kinds of eslint nodes should be linted
|
|
* Tells eslint to lint certain nodes (lintCallExpression, lintMemberExpression, lintNewExpression)
|
|
* Gets protochain for the ESLint nodes the plugin is interested in
|
|
*/
|
|
function getName(node) {
|
|
switch (node.type) {
|
|
case "NewExpression":
|
|
{
|
|
return node.callee.name;
|
|
}
|
|
|
|
case "MemberExpression":
|
|
{
|
|
return node.object.name;
|
|
}
|
|
|
|
case "ExpressionStatement":
|
|
{
|
|
return node.expression.name;
|
|
}
|
|
|
|
case "CallExpression":
|
|
{
|
|
return node.callee.name;
|
|
}
|
|
|
|
default:
|
|
throw new Error("not found");
|
|
}
|
|
}
|
|
|
|
function generateErrorName(rule) {
|
|
if (rule.name) return rule.name;
|
|
if (rule.property) return `${rule.object}.${rule.property}()`;
|
|
return rule.object;
|
|
}
|
|
|
|
const getPolyfillSet = (0, _lodash.default)(polyfillArrayJSON => new Set(JSON.parse(polyfillArrayJSON)));
|
|
|
|
function isPolyfilled(context, rule) {
|
|
var _context$settings;
|
|
|
|
if (!((_context$settings = context.settings) === null || _context$settings === void 0 ? void 0 : _context$settings.polyfills)) return false;
|
|
const polyfills = getPolyfillSet(JSON.stringify(context.settings.polyfills));
|
|
return (// v2 allowed users to select polyfills based off their caniuseId. This is
|
|
polyfills.has(rule.id) || // no longer supported. Keeping this here to avoid breaking changes.
|
|
polyfills.has(rule.protoChainId) || // Check if polyfill is provided (ex. `Promise.all`)
|
|
polyfills.has(rule.protoChain[0]) // Check if entire API is polyfilled (ex. `Promise`)
|
|
|
|
);
|
|
}
|
|
|
|
const items = [// Babel configs
|
|
"babel.config.json", "babel.config.js", ".babelrc", ".babelrc.json", ".babelrc.js", // TS configs
|
|
"tsconfig.json"];
|
|
/**
|
|
* Determine if a user has a TS or babel config. This is used to infer if a user is transpiling their code.
|
|
* If transpiling code, do not lint ES APIs. We assume that all transpiled code is polyfilled.
|
|
* @TODO Use @babel/core to find config. See https://github.com/babel/babel/discussions/11602
|
|
* @param dir @
|
|
*/
|
|
|
|
function isUsingTranspiler(context) {
|
|
var _context$parserOption;
|
|
|
|
// If tsconfig config exists in parser options, assume transpilation
|
|
if (((_context$parserOption = context.parserOptions) === null || _context$parserOption === void 0 ? void 0 : _context$parserOption.tsconfigRootDir) === true) return true;
|
|
const dir = context.getFilename();
|
|
|
|
const configPath = _findUp.default.sync(items, {
|
|
cwd: dir
|
|
});
|
|
|
|
if (configPath) return true;
|
|
|
|
const pkgPath = _findUp.default.sync("package.json", {
|
|
cwd: dir
|
|
}); // Check if babel property exists
|
|
|
|
|
|
if (pkgPath) {
|
|
const pkg = JSON.parse(_fs.default.readFileSync(pkgPath).toString());
|
|
return !!pkg.babel;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
var _default = {
|
|
meta: {
|
|
docs: {
|
|
description: "Ensure cross-browser API compatibility",
|
|
category: "Compatibility",
|
|
url: "https://github.com/amilajack/eslint-plugin-compat/blob/master/docs/rules/compat.md",
|
|
recommended: true
|
|
},
|
|
type: "problem",
|
|
schema: [{
|
|
type: "string"
|
|
}]
|
|
},
|
|
|
|
create(context) {
|
|
var _context$settings2, _context$settings3, _context$settings4, _context$settings5, _context$settings5$po;
|
|
|
|
// Determine lowest targets from browserslist config, which reads user's
|
|
// package.json config section. Use config from eslintrc for testing purposes
|
|
const browserslistConfig = ((_context$settings2 = context.settings) === null || _context$settings2 === void 0 ? void 0 : _context$settings2.browsers) || ((_context$settings3 = context.settings) === null || _context$settings3 === void 0 ? void 0 : _context$settings3.targets) || context.options[0];
|
|
const lintAllEsApis = ((_context$settings4 = context.settings) === null || _context$settings4 === void 0 ? void 0 : _context$settings4.lintAllEsApis) === true || // Attempt to infer polyfilling of ES APIs from ts or babel config
|
|
!((_context$settings5 = context.settings) === null || _context$settings5 === void 0 ? void 0 : (_context$settings5$po = _context$settings5.polyfills) === null || _context$settings5$po === void 0 ? void 0 : _context$settings5$po.includes("es:all")) && !isUsingTranspiler(context);
|
|
const browserslistTargets = (0, _helpers.parseBrowsersListVersion)((0, _helpers.determineTargetsFromConfig)(context.getFilename(), browserslistConfig));
|
|
|
|
/**
|
|
* A small optimization that only lints APIs that are not supported by targeted browsers.
|
|
* For example, if the user is targeting chrome 50, which supports the fetch API, it is
|
|
* wasteful to lint calls to fetch.
|
|
*/
|
|
const getRulesForTargets = (0, _lodash.default)(targetsJSON => {
|
|
const result = {
|
|
CallExpression: [],
|
|
NewExpression: [],
|
|
MemberExpression: [],
|
|
ExpressionStatement: []
|
|
};
|
|
const targets = JSON.parse(targetsJSON);
|
|
|
|
_providers.nodes.filter(node => {
|
|
return lintAllEsApis ? true : node.kind !== "es";
|
|
}).forEach(node => {
|
|
if (!node.getUnsupportedTargets(node, targets).length) return;
|
|
result[node.astNodeType].push(node);
|
|
});
|
|
|
|
return result;
|
|
}); // Stringify to support memoization; browserslistConfig is always an array of new objects.
|
|
|
|
const targetedRules = getRulesForTargets(JSON.stringify(browserslistTargets));
|
|
const errors = [];
|
|
|
|
const handleFailingRule = (node, eslintNode) => {
|
|
if (isPolyfilled(context, node)) return;
|
|
errors.push({
|
|
node: eslintNode,
|
|
message: [generateErrorName(node), "is not supported in", node.getUnsupportedTargets(node, browserslistTargets).join(", ")].join(" ")
|
|
});
|
|
};
|
|
|
|
const identifiers = new Set();
|
|
return {
|
|
CallExpression: _helpers.lintCallExpression.bind(null, context, handleFailingRule, targetedRules.CallExpression),
|
|
NewExpression: _helpers.lintNewExpression.bind(null, context, handleFailingRule, targetedRules.NewExpression),
|
|
ExpressionStatement: _helpers.lintExpressionStatement.bind(null, context, handleFailingRule, [...targetedRules.MemberExpression, ...targetedRules.CallExpression]),
|
|
MemberExpression: _helpers.lintMemberExpression.bind(null, context, handleFailingRule, [...targetedRules.MemberExpression, ...targetedRules.CallExpression, ...targetedRules.NewExpression]),
|
|
|
|
// Keep track of all the defined variables. Do not report errors for nodes that are not defined
|
|
Identifier(node) {
|
|
if (node.parent) {
|
|
const {
|
|
type
|
|
} = node.parent;
|
|
|
|
if (type === "Property" || // ex. const { Set } = require('immutable');
|
|
type === "FunctionDeclaration" || // ex. function Set() {}
|
|
type === "VariableDeclarator" || // ex. const Set = () => {}
|
|
type === "ClassDeclaration" || // ex. class Set {}
|
|
type === "ImportDefaultSpecifier" || // ex. import Set from 'set';
|
|
type === "ImportSpecifier" || // ex. import {Set} from 'set';
|
|
type === "ImportDeclaration" // ex. import {Set} from 'set';
|
|
) {
|
|
identifiers.add(node.name);
|
|
}
|
|
}
|
|
},
|
|
|
|
"Program:exit": () => {
|
|
// Get a map of all the variables defined in the root scope (not the global scope)
|
|
// const variablesMap = context.getScope().childScopes.map(e => e.set)[0];
|
|
errors.filter(error => !identifiers.has(getName(error.node))).forEach(node => context.report(node));
|
|
}
|
|
};
|
|
}
|
|
|
|
};
|
|
exports.default = _default;
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/rules/compat.ts"],"names":["getName","node","type","callee","name","object","expression","Error","generateErrorName","rule","property","getPolyfillSet","polyfillArrayJSON","Set","JSON","parse","isPolyfilled","context","settings","polyfills","stringify","has","id","protoChainId","protoChain","items","isUsingTranspiler","parserOptions","tsconfigRootDir","dir","getFilename","configPath","findUp","sync","cwd","pkgPath","pkg","fs","readFileSync","toString","babel","meta","docs","description","category","url","recommended","schema","create","browserslistConfig","browsers","targets","options","lintAllEsApis","includes","browserslistTargets","getRulesForTargets","targetsJSON","result","CallExpression","NewExpression","MemberExpression","ExpressionStatement","nodes","filter","kind","forEach","getUnsupportedTargets","length","astNodeType","push","targetedRules","errors","handleFailingRule","eslintNode","message","join","identifiers","lintCallExpression","bind","lintNewExpression","lintExpressionStatement","lintMemberExpression","Identifier","parent","add","error","report"],"mappings":"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","sourcesContent":["/*\n * Step 2) Logic that handles AST traversal\n * Does not handle looking up the API\n * Handles checking what kinds of eslint nodes should be linted\n *   Tells eslint to lint certain nodes  (lintCallExpression, lintMemberExpression, lintNewExpression)\n *   Gets protochain for the ESLint nodes the plugin is interested in\n */\nimport fs from \"fs\";\nimport findUp from \"find-up\";\nimport memoize from \"lodash.memoize\";\nimport {\n  lintCallExpression,\n  lintMemberExpression,\n  lintNewExpression,\n  lintExpressionStatement,\n  parseBrowsersListVersion,\n  determineTargetsFromConfig,\n} from \"../helpers\"; // will be deprecated and introduced to this file\nimport {\n  ESLintNode,\n  AstMetadataApiWithTargetsResolver,\n  BrowserListConfig,\n  HandleFailingRule,\n  Context,\n} from \"../types\";\nimport { nodes } from \"../providers\";\n\ntype ESLint = {\n  [astNodeTypeName: string]: (node: ESLintNode) => void;\n};\n\nfunction getName(node: ESLintNode): string {\n  switch (node.type) {\n    case \"NewExpression\": {\n      return node.callee.name;\n    }\n    case \"MemberExpression\": {\n      return node.object.name;\n    }\n    case \"ExpressionStatement\": {\n      return node.expression.name;\n    }\n    case \"CallExpression\": {\n      return node.callee.name;\n    }\n    default:\n      throw new Error(\"not found\");\n  }\n}\n\nfunction generateErrorName(rule: AstMetadataApiWithTargetsResolver): string {\n  if (rule.name) return rule.name;\n  if (rule.property) return `${rule.object}.${rule.property}()`;\n  return rule.object;\n}\n\nconst getPolyfillSet = memoize(\n  (polyfillArrayJSON: string): Set<String> =>\n    new Set(JSON.parse(polyfillArrayJSON))\n);\n\nfunction isPolyfilled(\n  context: Context,\n  rule: AstMetadataApiWithTargetsResolver\n): boolean {\n  if (!context.settings?.polyfills) return false;\n  const polyfills = getPolyfillSet(JSON.stringify(context.settings.polyfills));\n  return (\n    // v2 allowed users to select polyfills based off their caniuseId. This is\n    polyfills.has(rule.id) || // no longer supported. Keeping this here to avoid breaking changes.\n    polyfills.has(rule.protoChainId) || // Check if polyfill is provided (ex. `Promise.all`)\n    polyfills.has(rule.protoChain[0]) // Check if entire API is polyfilled (ex. `Promise`)\n  );\n}\n\nconst items = [\n  // Babel configs\n  \"babel.config.json\",\n  \"babel.config.js\",\n  \".babelrc\",\n  \".babelrc.json\",\n  \".babelrc.js\",\n  // TS configs\n  \"tsconfig.json\",\n];\n\n/**\n * Determine if a user has a TS or babel config. This is used to infer if a user is transpiling their code.\n * If transpiling code, do not lint ES APIs. We assume that all transpiled code is polyfilled.\n * @TODO Use @babel/core to find config. See https://github.com/babel/babel/discussions/11602\n * @param dir @\n */\nfunction isUsingTranspiler(context: Context): boolean {\n  // If tsconfig config exists in parser options, assume transpilation\n  if (context.parserOptions?.tsconfigRootDir === true) return true;\n  const dir = context.getFilename();\n  const configPath = findUp.sync(items, {\n    cwd: dir,\n  });\n  if (configPath) return true;\n  const pkgPath = findUp.sync(\"package.json\", {\n    cwd: dir,\n  });\n  // Check if babel property exists\n  if (pkgPath) {\n    const pkg = JSON.parse(fs.readFileSync(pkgPath).toString());\n    return !!pkg.babel;\n  }\n  return false;\n}\n\nexport default {\n  meta: {\n    docs: {\n      description: \"Ensure cross-browser API compatibility\",\n      category: \"Compatibility\",\n      url:\n        \"https://github.com/amilajack/eslint-plugin-compat/blob/master/docs/rules/compat.md\",\n      recommended: true,\n    },\n    type: \"problem\",\n    schema: [{ type: \"string\" }],\n  },\n  create(context: Context): ESLint {\n    // Determine lowest targets from browserslist config, which reads user's\n    // package.json config section. Use config from eslintrc for testing purposes\n    const browserslistConfig: BrowserListConfig =\n      context.settings?.browsers ||\n      context.settings?.targets ||\n      context.options[0];\n\n    const lintAllEsApis: boolean =\n      context.settings?.lintAllEsApis === true ||\n      // Attempt to infer polyfilling of ES APIs from ts or babel config\n      (!context.settings?.polyfills?.includes(\"es:all\") &&\n        !isUsingTranspiler(context));\n    const browserslistTargets = parseBrowsersListVersion(\n      determineTargetsFromConfig(context.getFilename(), browserslistConfig)\n    );\n\n    type RulesFilteredByTargets = {\n      CallExpression: AstMetadataApiWithTargetsResolver[];\n      NewExpression: AstMetadataApiWithTargetsResolver[];\n      MemberExpression: AstMetadataApiWithTargetsResolver[];\n      ExpressionStatement: AstMetadataApiWithTargetsResolver[];\n    };\n\n    /**\n     * A small optimization that only lints APIs that are not supported by targeted browsers.\n     * For example, if the user is targeting chrome 50, which supports the fetch API, it is\n     * wasteful to lint calls to fetch.\n     */\n    const getRulesForTargets = memoize(\n      (targetsJSON: string): RulesFilteredByTargets => {\n        const result = {\n          CallExpression: [],\n          NewExpression: [],\n          MemberExpression: [],\n          ExpressionStatement: [],\n        };\n        const targets = JSON.parse(targetsJSON);\n\n        nodes\n          .filter((node) => {\n            return lintAllEsApis ? true : node.kind !== \"es\";\n          })\n          .forEach((node) => {\n            if (!node.getUnsupportedTargets(node, targets).length) return;\n            result[node.astNodeType as keyof RulesFilteredByTargets].push(node);\n          });\n\n        return result;\n      }\n    );\n\n    // Stringify to support memoization; browserslistConfig is always an array of new objects.\n    const targetedRules = getRulesForTargets(\n      JSON.stringify(browserslistTargets)\n    );\n\n    type Error = {\n      message: string;\n      node: ESLintNode;\n    };\n\n    const errors: Error[] = [];\n\n    const handleFailingRule: HandleFailingRule = (\n      node: AstMetadataApiWithTargetsResolver,\n      eslintNode: ESLintNode\n    ) => {\n      if (isPolyfilled(context, node)) return;\n      errors.push({\n        node: eslintNode,\n        message: [\n          generateErrorName(node),\n          \"is not supported in\",\n          node.getUnsupportedTargets(node, browserslistTargets).join(\", \"),\n        ].join(\" \"),\n      });\n    };\n\n    const identifiers = new Set();\n\n    return {\n      CallExpression: lintCallExpression.bind(\n        null,\n        context,\n        handleFailingRule,\n        targetedRules.CallExpression\n      ),\n      NewExpression: lintNewExpression.bind(\n        null,\n        context,\n        handleFailingRule,\n        targetedRules.NewExpression\n      ),\n      ExpressionStatement: lintExpressionStatement.bind(\n        null,\n        context,\n        handleFailingRule,\n        [...targetedRules.MemberExpression, ...targetedRules.CallExpression]\n      ),\n      MemberExpression: lintMemberExpression.bind(\n        null,\n        context,\n        handleFailingRule,\n        [\n          ...targetedRules.MemberExpression,\n          ...targetedRules.CallExpression,\n          ...targetedRules.NewExpression,\n        ]\n      ),\n      // Keep track of all the defined variables. Do not report errors for nodes that are not defined\n      Identifier(node: ESLintNode) {\n        if (node.parent) {\n          const { type } = node.parent;\n          if (\n            type === \"Property\" || // ex. const { Set } = require('immutable');\n            type === \"FunctionDeclaration\" || // ex. function Set() {}\n            type === \"VariableDeclarator\" || // ex. const Set = () => {}\n            type === \"ClassDeclaration\" || // ex. class Set {}\n            type === \"ImportDefaultSpecifier\" || // ex. import Set from 'set';\n            type === \"ImportSpecifier\" || // ex. import {Set} from 'set';\n            type === \"ImportDeclaration\" // ex. import {Set} from 'set';\n          ) {\n            identifiers.add(node.name);\n          }\n        }\n      },\n      \"Program:exit\": () => {\n        // Get a map of all the variables defined in the root scope (not the global scope)\n        // const variablesMap = context.getScope().childScopes.map(e => e.set)[0];\n        errors\n          .filter((error) => !identifiers.has(getName(error.node)))\n          .forEach((node) => context.report(node));\n      },\n    };\n  },\n};\n"]}
|