/**
|
|
* @author Toru Nagashima
|
|
* See LICENSE file in root directory for full license.
|
|
*/
|
|
"use strict"
|
|
|
|
const { Range, lt, major } = require("semver") //eslint-disable-line no-unused-vars
|
|
const { ReferenceTracker } = require("eslint-utils")
|
|
const getConfiguredNodeVersion = require("./get-configured-node-version")
|
|
const getSemverRange = require("./get-semver-range")
|
|
|
|
/**
|
|
* @typedef {Object} SupportInfo
|
|
* @property {string | null} supported The stably supported version. If `null` is present, it hasn't been supported yet.
|
|
* @property {string[]} [backported] The backported versions.
|
|
* @property {string} [experimental] The added version as experimental.
|
|
*/
|
|
|
|
/**
|
|
* Parses the options.
|
|
* @param {RuleContext} context The rule context.
|
|
* @returns {{version:Range,ignores:Set<string>}} Parsed value.
|
|
*/
|
|
function parseOptions(context) {
|
|
const raw = context.options[0] || {}
|
|
const filePath = context.getFilename()
|
|
const version = getConfiguredNodeVersion(raw.version, filePath)
|
|
const ignores = new Set(raw.ignores || [])
|
|
|
|
return Object.freeze({ version, ignores })
|
|
}
|
|
|
|
/**
|
|
* Check if it has been supported.
|
|
* @param {SupportInfo} info The support info.
|
|
* @param {Range} configured The configured version range.
|
|
*/
|
|
function isSupported({ backported, supported }, configured) {
|
|
if (
|
|
backported &&
|
|
backported.length >= 2 &&
|
|
!backported.every((v, i) => i === 0 || lt(backported[i - 1], v))
|
|
) {
|
|
throw new Error("Invalid BackportConfiguration")
|
|
}
|
|
|
|
if (supported == null) {
|
|
return false
|
|
}
|
|
if (backported == null || backported.length === 0) {
|
|
return !configured.intersects(getSemverRange(`<${supported}`))
|
|
}
|
|
|
|
return !configured.intersects(
|
|
getSemverRange(
|
|
[...backported, supported]
|
|
.map((v, i) => (i === 0 ? `<${v}` : `>=${major(v)}.0.0 <${v}`))
|
|
.join(" || ")
|
|
)
|
|
)
|
|
}
|
|
|
|
/**
|
|
* Get the formatted text of a given supported version.
|
|
* @param {SupportInfo} info The support info.
|
|
*/
|
|
function supportedVersionToString({ backported, supported }) {
|
|
if (supported == null) {
|
|
return "(none yet)"
|
|
}
|
|
if (backported == null || backported.length === 0) {
|
|
return supported
|
|
}
|
|
return `${supported} (backported: ^${backported.join(", ^")})`
|
|
}
|
|
|
|
/**
|
|
* Verify the code to report unsupported APIs.
|
|
* @param {RuleContext} context The rule context.
|
|
* @param {{modules:object,globals:object}} trackMap The map for APIs to report.
|
|
* @returns {void}
|
|
*/
|
|
module.exports = function checkUnsupportedBuiltins(context, trackMap) {
|
|
const options = parseOptions(context)
|
|
const tracker = new ReferenceTracker(context.getScope(), { mode: "legacy" })
|
|
const references = [
|
|
...tracker.iterateCjsReferences(trackMap.modules || {}),
|
|
...tracker.iterateEsmReferences(trackMap.modules || {}),
|
|
...tracker.iterateGlobalReferences(trackMap.globals || {}),
|
|
]
|
|
|
|
for (const { node, path, info } of references) {
|
|
const name = path.join(".")
|
|
const supported = isSupported(info, options.version)
|
|
|
|
if (!supported && !options.ignores.has(name)) {
|
|
context.report({
|
|
node,
|
|
messageId: "unsupported",
|
|
data: {
|
|
name,
|
|
supported: supportedVersionToString(info),
|
|
version: options.version.raw,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
}
|