"use strict"; require("core-js/modules/es.array.flat"); require("core-js/modules/es.array.unscopables.flat"); Object.defineProperty(exports, "__esModule", { value: true }); exports.isSupportedByMDN = isSupportedByMDN; exports.getUnsupportedTargets = getUnsupportedTargets; exports.default = void 0; var _astMetadataInferer = _interopRequireDefault(require("ast-metadata-inferer")); var _semver = _interopRequireDefault(require("semver")); var _Versioning = require("../Versioning"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } const mdnRecords = new Map(_astMetadataInferer.default.map(e => [e.protoChainId, e])); /** * Map ids of mdn targets to their "common/friendly" name */ const targetIdMappings = { chrome: "chrome", firefox: "firefox", opera: "opera", safari: "safari", safari_ios: "ios_saf", ie: "ie", edge_mobile: "ie_mob", edge: "edge", opera_android: "and_opera", chrome_android: "and_chrome", firefox_android: "and_firefox", webview_android: "and_webview", samsunginternet_android: "and_samsung", nodejs: "node" }; const reversedTargetMappings = (0, _Versioning.reverseTargetMappings)(targetIdMappings); /** * Take a target's id and return it's full name by using `targetNameMappings` * ex. {target: and_ff, version: 40} => 'Android FireFox 40' */ function formatTargetNames(target) { return `${_Versioning.STANDARD_TARGET_NAME_MAPPING[target.target]} ${target.version}`; } /** * Convert '9' => '9.0.0' */ function customCoerce(version) { return version.length === 1 ? [version, 0, 0].join(".") : version; } /* * Return if MDN supports the API or not */ function isSupportedByMDN(node, { version, target: mdnTarget }) { const target = reversedTargetMappings[mdnTarget]; // If no record could be found, return true. Rules might not // be found because they could belong to another provider if (!mdnRecords.has(node.protoChainId)) return true; const record = mdnRecords.get(node.protoChainId); if (!record || !record.compat.support) return true; const compatRecord = record.compat.support[target]; if (!compatRecord) return true; if (!Array.isArray(compatRecord) && !("version_added" in compatRecord)) return true; const { version_added: versionAdded } = Array.isArray(compatRecord) ? compatRecord.find(e => "version_added" in e) : compatRecord; // If a version is true then it is supported but version is unsure if (typeof versionAdded === "boolean") return versionAdded; if (versionAdded === null) return true; // Special case for Safari TP: TP is always gte than any other releases if (target === "safari") { if (version === "TP") return true; if (versionAdded === "TP") return false; } // A browser supports an API if its version is greater than or equal // to the first version of the browser that API was added in const semverCurrent = _semver.default.coerce(customCoerce(version)); const semverAdded = _semver.default.coerce(customCoerce(versionAdded)); // semver.coerce() might be null for non-semvers (other than Safari TP) // Just warn and treat features as supported here for now to avoid lint from // crashing if (!semverCurrent) { // eslint-disable-next-line no-console console.warn(`eslint-plugin-compat: A non-semver target "${target} ${version}" matched for the feature ${node.protoChainId}, skipping. You're welcome to submit this log to https://github.com/amilajack/eslint-plugin-compat/issues for analysis.`); return true; } if (!versionAdded) { // eslint-disable-next-line no-console console.warn(`eslint-plugin-compat: The feature ${node.protoChainId} is supported since a non-semver target "${target} ${versionAdded}", skipping. You're welcome to submit this log to https://github.com/amilajack/eslint-plugin-compat/issues for analysis.`); return true; } return _semver.default.gte(semverCurrent, semverAdded); } /** * Return an array of all unsupported targets */ function getUnsupportedTargets(node, targets) { return targets.filter(target => !isSupportedByMDN(node, target)).map(formatTargetNames); } function getMetadataName(metadata) { switch (metadata.protoChain.length) { case 1: { return metadata.protoChain[0]; } default: return `${metadata.protoChain.join(".")}()`; } } const MdnProvider = _astMetadataInferer.default // Create entries for each ast node type .map(metadata => metadata.astNodeTypes.map(astNodeType => ({ ...metadata, name: getMetadataName(metadata), id: metadata.protoChainId, protoChainId: metadata.protoChainId, astNodeType, object: metadata.protoChain[0], // @TODO Handle cases where 'prototype' is in protoChain property: metadata.protoChain[1] }))) // Flatten the array of arrays .flat() // Add rule and target support logic for each entry .map(rule => ({ ...rule, getUnsupportedTargets })); var _default = MdnProvider; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/providers/MdnProvider.js"],"names":["mdnRecords","Map","AstMetadata","map","e","protoChainId","targetIdMappings","chrome","firefox","opera","safari","safari_ios","ie","edge_mobile","edge","opera_android","chrome_android","firefox_android","webview_android","samsunginternet_android","nodejs","reversedTargetMappings","formatTargetNames","target","STANDARD_TARGET_NAME_MAPPING","version","customCoerce","length","join","isSupportedByMDN","node","mdnTarget","has","record","get","compat","support","compatRecord","Array","isArray","version_added","versionAdded","find","semverCurrent","semver","coerce","semverAdded","console","warn","gte","getUnsupportedTargets","targets","filter","getMetadataName","metadata","protoChain","MdnProvider","astNodeTypes","astNodeType","name","id","object","property","flat","rule"],"mappings":"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","sourcesContent":["import AstMetadata from \"ast-metadata-inferer\";\nimport semver from \"semver\";\nimport {\n  STANDARD_TARGET_NAME_MAPPING,\n  reverseTargetMappings\n} from \"../Versioning\";\nimport type { Node, Targets, Target } from \"../LintTypes\";\n\n// @TODO Import this type from ast-metadata-inferer after migrating this project to TypeScript\ntype AstMetadataRecordType = {\n  apiType: \"js-api\" | \"css-api\",\n  type: \"js-api\" | \"css-api\",\n  protoChain: Array<string>,\n  protoChainId: string,\n  astNodeTypes: Array<string>,\n  isStatic: boolean,\n  compat: {\n    support: {\n      [browserName: string]: {\n        // If a version is true then it is supported but version is unsure\n        version_added: string | boolean\n      }\n    },\n    [x: string]: any\n  }\n};\n\nconst mdnRecords: Map<string, AstMetadataRecordType> = new Map(\n  AstMetadata.map(e => [e.protoChainId, e])\n);\n\n/**\n * Map ids of mdn targets to their \"common/friendly\" name\n */\nconst targetIdMappings = {\n  chrome: \"chrome\",\n  firefox: \"firefox\",\n  opera: \"opera\",\n  safari: \"safari\",\n  safari_ios: \"ios_saf\",\n  ie: \"ie\",\n  edge_mobile: \"ie_mob\",\n  edge: \"edge\",\n  opera_android: \"and_opera\",\n  chrome_android: \"and_chrome\",\n  firefox_android: \"and_firefox\",\n  webview_android: \"and_webview\",\n  samsunginternet_android: \"and_samsung\",\n  nodejs: \"node\"\n};\n\nconst reversedTargetMappings = reverseTargetMappings(targetIdMappings);\n\n/**\n * Take a target's id and return it's full name by using `targetNameMappings`\n * ex. {target: and_ff, version: 40} => 'Android FireFox 40'\n */\nfunction formatTargetNames(target: Target): string {\n  return `${STANDARD_TARGET_NAME_MAPPING[target.target]} ${target.version}`;\n}\n\n/**\n * Convert '9' => '9.0.0'\n */\nfunction customCoerce(version: string): string {\n  return version.length === 1 ? [version, 0, 0].join(\".\") : version;\n}\n\n/*\n * Return if MDN supports the API or not\n */\nexport function isSupportedByMDN(\n  node: Node,\n  { version, target: mdnTarget }: Target\n): boolean {\n  const target = reversedTargetMappings[mdnTarget];\n  // If no record could be found, return true. Rules might not\n  // be found because they could belong to another provider\n  if (!mdnRecords.has(node.protoChainId)) return true;\n  const record = mdnRecords.get(node.protoChainId);\n  if (!record || !record.compat.support) return true;\n  const compatRecord = record.compat.support[target];\n  if (!compatRecord) return true;\n  if (!Array.isArray(compatRecord) && !(\"version_added\" in compatRecord))\n    return true;\n  const { version_added: versionAdded } = Array.isArray(compatRecord)\n    ? compatRecord.find(e => \"version_added\" in e)\n    : compatRecord;\n\n  // If a version is true then it is supported but version is unsure\n  if (typeof versionAdded === \"boolean\") return versionAdded;\n  if (versionAdded === null) return true;\n\n  // Special case for Safari TP: TP is always gte than any other releases\n  if (target === \"safari\") {\n    if (version === \"TP\") return true;\n    if (versionAdded === \"TP\") return false;\n  }\n  // A browser supports an API if its version is greater than or equal\n  // to the first version of the browser that API was added in\n  const semverCurrent = semver.coerce(customCoerce(version));\n  const semverAdded = semver.coerce(customCoerce(versionAdded));\n\n  // semver.coerce() might be null for non-semvers (other than Safari TP)\n  // Just warn and treat features as supported here for now to avoid lint from\n  // crashing\n  if (!semverCurrent) {\n    // eslint-disable-next-line no-console\n    console.warn(\n      `eslint-plugin-compat: A non-semver target \"${target} ${version}\" matched for the feature ${node.protoChainId}, skipping. You're welcome to submit this log to https://github.com/amilajack/eslint-plugin-compat/issues for analysis.`\n    );\n    return true;\n  }\n  if (!versionAdded) {\n    // eslint-disable-next-line no-console\n    console.warn(\n      `eslint-plugin-compat: The feature ${node.protoChainId} is supported since a non-semver target \"${target} ${versionAdded}\", skipping. You're welcome to submit this log to https://github.com/amilajack/eslint-plugin-compat/issues for analysis.`\n    );\n    return true;\n  }\n  return semver.gte(semverCurrent, semverAdded);\n}\n\n/**\n * Return an array of all unsupported targets\n */\nexport function getUnsupportedTargets(\n  node: Node,\n  targets: Targets\n): Array<string> {\n  return targets\n    .filter(target => !isSupportedByMDN(node, target))\n    .map(formatTargetNames);\n}\n\nfunction getMetadataName(metadata: Node) {\n  switch (metadata.protoChain.length) {\n    case 1: {\n      return metadata.protoChain[0];\n    }\n    default:\n      return `${metadata.protoChain.join(\".\")}()`;\n  }\n}\n\nconst MdnProvider: Array<Node> = AstMetadata\n  // Create entries for each ast node type\n  .map(metadata =>\n    metadata.astNodeTypes.map(astNodeType => ({\n      ...metadata,\n      name: getMetadataName(metadata),\n      id: metadata.protoChainId,\n      protoChainId: metadata.protoChainId,\n      astNodeType,\n      object: metadata.protoChain[0],\n      // @TODO Handle cases where 'prototype' is in protoChain\n      property: metadata.protoChain[1]\n    }))\n  )\n  // Flatten the array of arrays\n  .flat()\n  // Add rule and target support logic for each entry\n  .map(rule => ({\n    ...rule,\n    getUnsupportedTargets\n  }));\n\nexport default MdnProvider;\n"]}