123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409 |
- /* eslint no-console:0 */
- 'use strict'
- // Imports
- import pathUtil from 'path'
- import semver from 'semver'
- import { errtion, stringify, simplifyRange } from './util.js'
- // As Errlop uses Editions, we should use a specific Errlop edition
- // As otherwise, the circular reference may fail on some machines
- // https://github.com/bevry/errlop/issues/2
- import Errlop from 'errlop'
- /**
- * The current node version that we are operating within.
- * It is compared in {@link requireEdition} against {@link Edition.engines}.
- */
- const NODE_VERSION: string = process.versions.node
- /**
- * Set the environment variable `EDITIONS_VERBOSE` to output debugging information to stderr on how editions selected the edition it did.
- * Values of `yes` and `true` are supported.
- * @example env EDITIONS_VERBOSE=true node mypackage/index.js
- */
- type EDITIONS_VERBOSE = string | boolean | undefined
- /**
- * Whether or not {@link EDITIONS_VERBOSE} is enabled.
- * @type {bollean}
- * @private
- */
- const VERBOSE: boolean =
- (process.env.EDITIONS_VERBOSE as EDITIONS_VERBOSE) === true ||
- (process.env.EDITIONS_VERBOSE as EDITIONS_VERBOSE) === 'yes' ||
- (process.env.EDITIONS_VERBOSE as EDITIONS_VERBOSE) === 'true' ||
- false
- /**
- * Set the environment variable `EDITIONS_TAG_BLACKLIST` to the tags you wish to blacklist, and editions will skip editions that contain them.
- * For backwards compatibility `EDITIONS_SYNTAX_BLACKLIST` is also supported.
- * It is compared in {@link requireEdition} against {@link Edition.tags}.
- * The value of this is stored locally in the {@link BLACKLIST} cache.
- * @example env EDITIONS_TAG_BLACKLIST=esnext,typescript,coffeescript node mypackage/index.js
- */
- type EDITIONS_TAG_BLACKLIST = string
- /**
- * A list of the blacklisted tags.
- * Data imported from {@link EDITIONS_TAG_BLACKLIST}.
- */
- const BLACKLIST: string[] =
- (process.env.EDITIONS_TAG_BLACKLIST as EDITIONS_TAG_BLACKLIST)?.split(
- /[, ]+/g
- ) ||
- (process.env.EDITIONS_SYNTAX_BLACKLIST as EDITIONS_TAG_BLACKLIST)?.split(
- /[, ]+/g
- )
- /**
- * A mapping of blacklisted tags to their reasons.
- * Keys are the tags.
- * Values are the error instances that contain the reasoning for why/how that tag is/became blacklisted.
- * Data imported from {@link EDITIONS_TAG_BLACKLIST}.
- */
- const blacklist: { [tag: string]: Error } = {}
- /** Edition entries must conform to the following specification. */
- export interface Edition {
- /**
- * Use this property to decribe the edition in human readable terms. Such as what it does and who it is for. It is used to reference the edition in user facing reporting, such as error messages.
- * @example "esnext source code with require for modules"
- */
- description: String
- /**
- * The location to where this directory is located. It should be a relative path from the `package.json` file.
- * @example "source"
- */
- directory: string
- /**
- * The default entry location for this edition, relative to the edition's directory.
- * @example "index.js"
- */
- entry?: string
- /**
- * Any keywords you wish to associate to the edition. Useful for various ecosystem tooling, such as automatic ESNext lint configuration if the `esnext` tag is present in the source edition tags. Consumers also make use of this via {@link EDITIONS_TAG_BLACKLIST} for preventing loading editions that contain a blacklisted tag. Previously this field was named `syntaxes`.
- * @example ["javascript", "esnext", "require"]
- */
- tags?: string[]
- /** @alias tags */
- syntaxes?: Edition['tags']
- /**
- * This field is used to specific which Node.js and Browser environments this edition supports. If `false` this edition does not support either. If `node` is a string, it should be a semver range of node.js versions that the edition targets. If `browsers` is a string, it should be a [browserlist](https://github.com/browserslist/browserslist) value of the specific browser values the edition targets. If `node` or `browsers` is true, it indicates that this edition is compatible with those environments.
- * @example
- * {
- * "description": "esnext source code with require for modules",
- * "directory": "source",
- * "entry": "index.js",
- * "tags": [
- * "javascript",
- * "esnext",
- * "require"
- * ],
- * "engines": {
- * "node": ">=6",
- * "browsers": "defaults"
- * }
- * }
- */
- engines?: false | { [engine: string]: string | boolean }
- }
- /** These are the various options that you can use to customise the behaviour of certain methods. */
- export interface EditionOptions {
- /** The require method of the calling module, used to ensure require paths remain correct. */
- require: Function
- /** If provided, this is used for debugging. */
- packagePath?: string
- /** If provided, any error loading an edition will be logged. By default, errors are only logged if all editions failed. If not provided, process.env.EDITIONS_VERBOSE is used. */
- verbose?: boolean
- /** If `true`, then only exact version matches will be loaded. If `false`, then likely matches using {@link simplifyRange} will be evaluated, with a fallback to the last. If missing, then `true` is attempted first and if no result, then `false` is attempted. */
- strict?: boolean
- /** If provided, this will be the cwd for entries. */
- cwd?: string
- /** If provided, should be a relative path to the entry point of the edition. */
- entry?: string
- /** If provided, should be the name of the package that we are loading the editions for. */
- package?: string
- /** If not provided, will use process.stderr instead. It is the stream that verbose errors are logged to. */
- stderr?: NodeJS.WritableStream
- }
- // Create the mapping of blacklisted tags and their reasonings
- if (BLACKLIST) {
- for (let i = 0; i < BLACKLIST.length; ++i) {
- const tag = BLACKLIST[i].trim().toLowerCase()
- blacklist[tag] = errtion({
- message: `The EDITIONS_TAG_BLACKLIST (aka EDITIONS_SYNTAX_BLACKLIST) environment variable blacklisted the tag [${tag}]`,
- code: 'blacklisted-tag'
- })
- }
- }
- // Blacklist the tag 'esnext' if our node version is below 0.12
- if (semver.satisfies(NODE_VERSION, '<0.12')) {
- blacklist.esnext = new Error(
- 'The esnext tag is skipped on early node versions as attempting to use esnext features will output debugging information on these node versions'
- )
- }
- /**
- * Attempt to load a specific {@link Edition}.
- * @returns The result of the loaded edition.
- * @throws An error if the edition failed to load.
- */
- export function loadEdition(edition: Edition, opts: EditionOptions): any {
- const entry = pathUtil.resolve(
- opts.cwd || '',
- edition.directory,
- opts.entry || edition.entry || ''
- )
- if (opts.require == null) {
- throw errtion({
- message: `Skipped edition [${edition.description}] as opts.require was not provided, this is probably due to a testing misconfiguration.`,
- code: 'unsupported-edition-require'
- })
- }
- try {
- return opts.require(entry)
- } catch (loadError) {
- // Note the error with more details
- throw errtion(
- {
- message: `Skipped edition [${edition.description}] at entry [${entry}] because it failed to load`,
- code: 'unsupported-edition-tried'
- },
- loadError
- )
- }
- }
- /**
- * Attempt to require an {@link Edition}, based on its compatibility with the current environment, such as {@link NODE_VERSION} and {@link EDITIONS_TAG_BLACKLIST} compatibility.
- * If compatibility is established with the environment, it will load the edition using {@link loadEdition}.
- * @returns The result of the loaded edition
- * @throws An error if the edition failed to load
- */
- export function requireEdition(edition: Edition, opts: EditionOptions): any {
- // Verify the edition is valid
- if (
- !edition.description ||
- !edition.directory ||
- !edition.entry ||
- edition.engines == null
- ) {
- throw errtion({
- message: `Each edition must have its [description, directory, entry, engines] fields defined, yet all it had was [${Object.keys(
- edition
- ).join(', ')}]`,
- code: 'unsupported-edition-malformed',
- level: 'fatal'
- })
- }
- // Handle strict omission
- if (opts.strict == null) {
- try {
- return requireEdition(edition, { ...opts, strict: true })
- } catch (err) {
- return requireEdition(edition, { ...opts, strict: false })
- }
- }
- // Verify tag support
- // Convert tags into a sorted lowercase string
- const tags = (edition.tags || edition.syntaxes || [])
- .map(i => i.toLowerCase())
- .sort()
- for (let index = 0; index < tags.length; index++) {
- const tag = tags[index]
- const blacklisted = blacklist[tag]
- if (blacklisted) {
- throw errtion(
- {
- message: `Skipping edition [${edition.description}] because it contained a blacklisted tag [${tag}]`,
- code: 'unsupported-edition-backlisted-tag'
- },
- blacklisted
- )
- }
- }
- // Verify engine support
- if (edition.engines === false) {
- throw errtion({
- message: `Skipping edition [${edition.description}] because its engines field was false`,
- code: 'unsupported-edition-engine'
- })
- }
- if (!edition.engines.node) {
- throw errtion({
- message: `Skipping edition [${edition.description}] because its .engines.node field was falsey`,
- code: 'unsupported-edition-engines-node'
- })
- }
- if (opts.strict) {
- if (edition.engines.node === true) {
- throw errtion({
- message: `Skipping edition [${edition.description}] because its .engines.node field was true yet we are in strict mode`,
- code: 'unsupported-edition-engines-node-version-true'
- })
- } else if (semver.satisfies(NODE_VERSION, edition.engines.node) === false) {
- throw errtion({
- message: `Skipping edition [${
- edition.description
- }] because our current node version [${NODE_VERSION}] is not supported by its specific range [${stringify(
- edition.engines.node
- )}]`,
- code: 'unsupported-edition-engines-node-version-specific'
- })
- }
- } else if (edition.engines.node !== true) {
- const simplifiedRange = simplifyRange(edition.engines.node)
- if (semver.satisfies(NODE_VERSION, simplifiedRange) === false) {
- throw errtion({
- message: `Skipping edition [${
- edition.description
- }] because our current node version [${NODE_VERSION}] is not supported by its simplified range [${stringify(
- simplifiedRange
- )}]`,
- code: 'unsupported-edition-engines-node-version-simplified'
- })
- }
- }
- // Load the edition
- return loadEdition(edition, opts)
- }
- /**
- * Cycles through a list of editions, returning the require result of the first suitable {@link Edition} that it was able to load.
- * Editions should be ordered from most preferable first, to least desirable last.
- * Providing the editions configuration is valid, individual edition handling is forwarded to {@link requireEdition}.
- * @returns The result of the loaded edition.
- * @throws An error if a suitable edition was unable to be resolved.
- */
- export function requireEditions(
- editions: Edition[],
- opts: EditionOptions
- ): any {
- // Check
- if (!editions || editions.length === 0) {
- if (opts.packagePath) {
- throw errtion({
- message: `There were no editions specified for package [${opts.packagePath}]`,
- code: 'unsupported-editions-missing'
- })
- } else {
- throw errtion({
- message: 'There were no editions specified',
- code: 'unsupported-editions-missing'
- })
- }
- }
- // Handle strict omission
- if (opts.strict == null) {
- try {
- return requireEditions(editions, { ...opts, strict: true })
- } catch (err) {
- return requireEditions(editions, { ...opts, strict: false })
- }
- }
- // Whether or not we should be verbose
- const verbose = opts.verbose == null ? VERBOSE : opts.verbose
- // Capture the load result, the last error, and the fallback option
- let result,
- loaded = false,
- editionsError = null,
- fallbackEdition = null
- // Cycle through the editions determing the above
- for (let i = 0; i < editions.length; ++i) {
- const edition = editions[i]
- try {
- result = requireEdition(edition, opts)
- loaded = true
- break
- } catch (editionError) {
- if (editionError.level === 'fatal') {
- editionsError = editionError
- break
- } else if (editionsError) {
- editionsError = errtion(editionsError, editionError)
- } else {
- editionsError = editionError
- }
- // make the fallback edition one that we don't bother loading due to its engines
- // also: don't assume that .code is accessible, as it may not be, even if it should be, due to the way different environments behave
- if (
- String(editionError.code || '').indexOf(
- 'unsupported-edition-engines-node-version'
- ) === 0
- ) {
- fallbackEdition = edition
- }
- }
- }
- // if no edition was suitable for our environment, then try the fallback if it exists
- // that is to say, ignore its engines.node
- if (opts.strict === false && loaded === false && fallbackEdition) {
- try {
- result = loadEdition(fallbackEdition, opts)
- loaded = true
- } catch (editionError) {
- editionsError = new Errlop(editionError, editionsError)
- }
- }
- // if we were able to load something, then provide it
- if (loaded) {
- // make note of any errors if desired
- if (editionsError && verbose) {
- const stderr = opts.stderr || process.stderr
- stderr.write(editionsError.stack + '\n')
- }
- return result
- }
- // otherwise, provide the error
- else if (editionsError) {
- if (opts.packagePath) {
- throw errtion(
- {
- message: `There were no suitable editions for package [${opts.packagePath}]`,
- code: 'unsupported-editions-tried'
- },
- editionsError
- )
- } else {
- throw errtion(
- {
- message: 'There were no suitable editions',
- code: 'unsupported-editions-tried'
- },
- editionsError
- )
- }
- }
- }
- /**
- * Cycle through the editions for a package and require the correct one.
- * Providing the package configuration is valid, editions handling is forwarded to {@link requireEditions}.
- * @returns The result of the loaded edition.
- * @throws An error if a suitable edition was unable to be resolved.
- */
- export function requirePackage(
- cwd: EditionOptions['cwd'],
- require: EditionOptions['require'],
- entry: EditionOptions['entry']
- ): any {
- // Load the package.json file to fetch `name` for debugging and `editions` for loading
- const packagePath = pathUtil.resolve(cwd || '', 'package.json')
- const { editions } = require(packagePath)
- const opts = { packagePath, cwd, require, entry }
- return requireEditions(editions, opts)
- }
|