160 lines
3.4 KiB
JavaScript
160 lines
3.4 KiB
JavaScript
// Manually “tree shaken” from:
|
||
// <https://github.com/nodejs/node/blob/45f5c9b/lib/internal/modules/esm/get_format.js>
|
||
// Last checked on: Nov 2, 2023.
|
||
|
||
import {fileURLToPath} from 'node:url'
|
||
import {getPackageType} from './resolve-get-package-type.js'
|
||
import {codes} from './errors.js'
|
||
|
||
const {ERR_UNKNOWN_FILE_EXTENSION} = codes
|
||
|
||
const hasOwnProperty = {}.hasOwnProperty
|
||
|
||
/** @type {Record<string, string>} */
|
||
const extensionFormatMap = {
|
||
// @ts-expect-error: hush.
|
||
__proto__: null,
|
||
'.cjs': 'commonjs',
|
||
'.js': 'module',
|
||
'.json': 'json',
|
||
'.mjs': 'module'
|
||
}
|
||
|
||
/**
|
||
* @param {string | null} mime
|
||
* @returns {string | null}
|
||
*/
|
||
function mimeToFormat(mime) {
|
||
if (
|
||
mime &&
|
||
/\s*(text|application)\/javascript\s*(;\s*charset=utf-?8\s*)?/i.test(mime)
|
||
)
|
||
return 'module'
|
||
if (mime === 'application/json') return 'json'
|
||
return null
|
||
}
|
||
|
||
/**
|
||
* @callback ProtocolHandler
|
||
* @param {URL} parsed
|
||
* @param {{parentURL: string, source?: Buffer}} context
|
||
* @param {boolean} ignoreErrors
|
||
* @returns {string | null | void}
|
||
*/
|
||
|
||
/**
|
||
* @type {Record<string, ProtocolHandler>}
|
||
*/
|
||
const protocolHandlers = {
|
||
// @ts-expect-error: hush.
|
||
__proto__: null,
|
||
'data:': getDataProtocolModuleFormat,
|
||
'file:': getFileProtocolModuleFormat,
|
||
'http:': getHttpProtocolModuleFormat,
|
||
'https:': getHttpProtocolModuleFormat,
|
||
'node:'() {
|
||
return 'builtin'
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @param {URL} parsed
|
||
*/
|
||
function getDataProtocolModuleFormat(parsed) {
|
||
const {1: mime} = /^([^/]+\/[^;,]+)[^,]*?(;base64)?,/.exec(
|
||
parsed.pathname
|
||
) || [null, null, null]
|
||
return mimeToFormat(mime)
|
||
}
|
||
|
||
/**
|
||
* Returns the file extension from a URL.
|
||
*
|
||
* Should give similar result to
|
||
* `require('node:path').extname(require('node:url').fileURLToPath(url))`
|
||
* when used with a `file:` URL.
|
||
*
|
||
* @param {URL} url
|
||
* @returns {string}
|
||
*/
|
||
function extname(url) {
|
||
const pathname = url.pathname
|
||
let index = pathname.length
|
||
|
||
while (index--) {
|
||
const code = pathname.codePointAt(index)
|
||
|
||
if (code === 47 /* `/` */) {
|
||
return ''
|
||
}
|
||
|
||
if (code === 46 /* `.` */) {
|
||
return pathname.codePointAt(index - 1) === 47 /* `/` */
|
||
? ''
|
||
: pathname.slice(index)
|
||
}
|
||
}
|
||
|
||
return ''
|
||
}
|
||
|
||
/**
|
||
* @type {ProtocolHandler}
|
||
*/
|
||
function getFileProtocolModuleFormat(url, _context, ignoreErrors) {
|
||
const ext = extname(url)
|
||
|
||
if (ext === '.js') {
|
||
const packageType = getPackageType(url)
|
||
|
||
if (packageType !== 'none') {
|
||
return packageType
|
||
}
|
||
|
||
return 'commonjs'
|
||
}
|
||
|
||
if (ext === '') {
|
||
const packageType = getPackageType(url)
|
||
|
||
// Legacy behavior
|
||
if (packageType === 'none' || packageType === 'commonjs') {
|
||
return 'commonjs'
|
||
}
|
||
|
||
// Note: we don’t implement WASM, so we don’t need
|
||
// `getFormatOfExtensionlessFile` from `formats`.
|
||
return 'module'
|
||
}
|
||
|
||
const format = extensionFormatMap[ext]
|
||
if (format) return format
|
||
|
||
// Explicit undefined return indicates load hook should rerun format check
|
||
if (ignoreErrors) {
|
||
return undefined
|
||
}
|
||
|
||
const filepath = fileURLToPath(url)
|
||
throw new ERR_UNKNOWN_FILE_EXTENSION(ext, filepath)
|
||
}
|
||
|
||
function getHttpProtocolModuleFormat() {
|
||
// To do: HTTPS imports.
|
||
}
|
||
|
||
/**
|
||
* @param {URL} url
|
||
* @param {{parentURL: string}} context
|
||
* @returns {string | null}
|
||
*/
|
||
export function defaultGetFormatWithoutErrors(url, context) {
|
||
const protocol = url.protocol
|
||
|
||
if (!hasOwnProperty.call(protocolHandlers, protocol)) {
|
||
return null
|
||
}
|
||
|
||
return protocolHandlers[protocol](url, context, true) || null
|
||
}
|