"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 _helpers = require("../helpers"); var _constants = require("../constants"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } const apis = _astMetadataInferer.default; // @TODO Import this type from ast-metadata-inferer after migrating this project to TypeScript const mdnRecords = new Map(apis.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, _helpers.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 `${_constants.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 = apis // 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/mdn-provider.ts"],"names":["apis","apiMetadata","mdnRecords","Map","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 apiMetadata from \"ast-metadata-inferer\";\nimport semver from \"semver\";\nimport { ApiMetadata } from \"ast-metadata-inferer/lib/types\";\nimport { reverseTargetMappings } from \"../helpers\";\nimport { STANDARD_TARGET_NAME_MAPPING } from \"../constants\";\nimport { AstMetadataApiWithTargetsResolver, Target } from \"../types\";\n\nconst apis = apiMetadata as ApiMetadata[];\n\n// @TODO Import this type from ast-metadata-inferer after migrating this project to TypeScript\nconst mdnRecords: Map<string, ApiMetadata> = new Map(\n  apis.map((e) => [e.protoChainId, e])\n);\n\ninterface 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\n/**\n * Map ids of mdn targets to their \"common/friendly\" name\n */\nconst targetIdMappings: Readonly<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: AstMetadataApiWithTargetsResolver,\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: AstMetadataApiWithTargetsResolver,\n  targets: Target[]\n): string[] {\n  return targets\n    .filter((target) => !isSupportedByMDN(node, target))\n    .map(formatTargetNames);\n}\n\nfunction getMetadataName(metadata: ApiMetadata) {\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<AstMetadataApiWithTargetsResolver> = apis\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"]}