| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 | /*	MIT License http://www.opensource.org/licenses/mit-license.php	Author Tobias Koppers @sokra*/"use strict";const EnvironmentNotSupportAsyncWarning = require("../EnvironmentNotSupportAsyncWarning");const { JAVASCRIPT_MODULE_TYPE_ESM } = require("../ModuleTypeConstants");const DynamicExports = require("./DynamicExports");const HarmonyCompatibilityDependency = require("./HarmonyCompatibilityDependency");const HarmonyExports = require("./HarmonyExports");/** @typedef {import("../Module").BuildMeta} BuildMeta *//** @typedef {import("../javascript/JavascriptParser")} JavascriptParser *//** @typedef {import("./HarmonyModulesPlugin").HarmonyModulesPluginOptions} HarmonyModulesPluginOptions */module.exports = class HarmonyDetectionParserPlugin {	/**	 * @param {HarmonyModulesPluginOptions} options options	 */	constructor(options) {		const { topLevelAwait = false } = options || {};		this.topLevelAwait = topLevelAwait;	}	/**	 * @param {JavascriptParser} parser the parser	 * @returns {void}	 */	apply(parser) {		parser.hooks.program.tap("HarmonyDetectionParserPlugin", ast => {			const isStrictHarmony =				parser.state.module.type === JAVASCRIPT_MODULE_TYPE_ESM;			const isHarmony =				isStrictHarmony ||				ast.body.some(					statement =>						statement.type === "ImportDeclaration" ||						statement.type === "ExportDefaultDeclaration" ||						statement.type === "ExportNamedDeclaration" ||						statement.type === "ExportAllDeclaration"				);			if (isHarmony) {				const module = parser.state.module;				const compatDep = new HarmonyCompatibilityDependency();				compatDep.loc = {					start: {						line: -1,						column: 0					},					end: {						line: -1,						column: 0					},					index: -3				};				module.addPresentationalDependency(compatDep);				DynamicExports.bailout(parser.state);				HarmonyExports.enable(parser.state, isStrictHarmony);				parser.scope.isStrict = true;			}		});		parser.hooks.topLevelAwait.tap("HarmonyDetectionParserPlugin", () => {			const module = parser.state.module;			if (!this.topLevelAwait) {				throw new Error(					"The top-level-await experiment is not enabled (set experiments.topLevelAwait: true to enable it)"				);			}			if (!HarmonyExports.isEnabled(parser.state)) {				throw new Error(					"Top-level-await is only supported in EcmaScript Modules"				);			}			/** @type {BuildMeta} */			(module.buildMeta).async = true;			EnvironmentNotSupportAsyncWarning.check(				module,				parser.state.compilation.runtimeTemplate,				"topLevelAwait"			);		});		/**		 * @returns {boolean | undefined} true if in harmony		 */		const skipInHarmony = () => {			if (HarmonyExports.isEnabled(parser.state)) {				return true;			}		};		/**		 * @returns {null | undefined} null if in harmony		 */		const nullInHarmony = () => {			if (HarmonyExports.isEnabled(parser.state)) {				return null;			}		};		const nonHarmonyIdentifiers = ["define", "exports"];		for (const identifier of nonHarmonyIdentifiers) {			parser.hooks.evaluateTypeof				.for(identifier)				.tap("HarmonyDetectionParserPlugin", nullInHarmony);			parser.hooks.typeof				.for(identifier)				.tap("HarmonyDetectionParserPlugin", skipInHarmony);			parser.hooks.evaluate				.for(identifier)				.tap("HarmonyDetectionParserPlugin", nullInHarmony);			parser.hooks.expression				.for(identifier)				.tap("HarmonyDetectionParserPlugin", skipInHarmony);			parser.hooks.call				.for(identifier)				.tap("HarmonyDetectionParserPlugin", skipInHarmony);		}	}};
 |