Add semantic token highlighting for 'that' variable and comment file references in VS Code extension Add Phone_Text_Input and Currency_Input components with formatting utilities Implement client widgets, form standardization, and soft delete functionality Add modal scroll lock and update documentation Implement comprehensive modal system with form integration and validation Fix modal component instantiation using jQuery plugin API Implement modal system with responsive sizing, queuing, and validation support Implement form submission with validation, error handling, and loading states Implement country/state selectors with dynamic data loading and Bootstrap styling Revert Rsx::Route() highlighting in Blade/PHP files Target specific PHP scopes for Rsx::Route() highlighting in Blade Expand injection selector for Rsx::Route() highlighting Add custom syntax highlighting for Rsx::Route() and Rsx.Route() calls Update jqhtml packages to v2.2.165 Add bundle path validation for common mistakes (development mode only) Create Ajax_Select_Input widget and Rsx_Reference_Data controller Create Country_Select_Input widget with default country support Initialize Tom Select on Select_Input widgets Add Tom Select bundle for enhanced select dropdowns Implement ISO 3166 geographic data system for country/region selection Implement widget-based form system with disabled state support 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
797 lines
22 KiB
JavaScript
Executable File
797 lines
22 KiB
JavaScript
Executable File
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Sergey Melyukov @smelukov
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
const path = require("path");
|
|
const { RawSource } = require("webpack-sources");
|
|
const ConcatenationScope = require("../ConcatenationScope");
|
|
const Generator = require("../Generator");
|
|
const {
|
|
ASSET_AND_CSS_URL_TYPES,
|
|
ASSET_AND_JS_AND_CSS_URL_TYPES,
|
|
ASSET_AND_JS_TYPES,
|
|
ASSET_TYPES,
|
|
CSS_URL_TYPES,
|
|
JS_AND_CSS_URL_TYPES,
|
|
JS_TYPES,
|
|
NO_TYPES
|
|
} = require("../ModuleSourceTypesConstants");
|
|
const { ASSET_MODULE_TYPE } = require("../ModuleTypeConstants");
|
|
const RuntimeGlobals = require("../RuntimeGlobals");
|
|
const CssUrlDependency = require("../dependencies/CssUrlDependency");
|
|
const createHash = require("../util/createHash");
|
|
const { makePathsRelative } = require("../util/identifier");
|
|
const memoize = require("../util/memoize");
|
|
const nonNumericOnlyHash = require("../util/nonNumericOnlyHash");
|
|
|
|
const getMimeTypes = memoize(() => require("mime-types"));
|
|
|
|
/** @typedef {import("webpack-sources").Source} Source */
|
|
/** @typedef {import("../../declarations/WebpackOptions").AssetGeneratorDataUrlOptions} AssetGeneratorDataUrlOptions */
|
|
/** @typedef {import("../../declarations/WebpackOptions").AssetGeneratorOptions} AssetGeneratorOptions */
|
|
/** @typedef {import("../../declarations/WebpackOptions").AssetModuleFilename} AssetModuleFilename */
|
|
/** @typedef {import("../../declarations/WebpackOptions").AssetModuleOutputPath} AssetModuleOutputPath */
|
|
/** @typedef {import("../../declarations/WebpackOptions").AssetResourceGeneratorOptions} AssetResourceGeneratorOptions */
|
|
/** @typedef {import("../../declarations/WebpackOptions").RawPublicPath} RawPublicPath */
|
|
/** @typedef {import("../ChunkGraph")} ChunkGraph */
|
|
/** @typedef {import("../Compilation").AssetInfo} AssetInfo */
|
|
/** @typedef {import("../Generator").GenerateContext} GenerateContext */
|
|
/** @typedef {import("../Generator").UpdateHashContext} UpdateHashContext */
|
|
/** @typedef {import("../Module").NameForCondition} NameForCondition */
|
|
/** @typedef {import("../Module").BuildInfo} BuildInfo */
|
|
/** @typedef {import("../Module").ConcatenationBailoutReasonContext} ConcatenationBailoutReasonContext */
|
|
/** @typedef {import("../Module").SourceTypes} SourceTypes */
|
|
/** @typedef {import("../ModuleGraph")} ModuleGraph */
|
|
/** @typedef {import("../NormalModule")} NormalModule */
|
|
/** @typedef {import("../RuntimeTemplate")} RuntimeTemplate */
|
|
/** @typedef {import("../util/Hash")} Hash */
|
|
/** @typedef {import("../util/runtime").RuntimeSpec} RuntimeSpec */
|
|
|
|
/**
|
|
* @template T
|
|
* @template U
|
|
* @param {null | string | T[] | Set<T> | undefined} a a
|
|
* @param {null | string | U[] | Set<U> | undefined} b b
|
|
* @returns {T[] & U[]} array
|
|
*/
|
|
const mergeMaybeArrays = (a, b) => {
|
|
const set = new Set();
|
|
if (Array.isArray(a)) for (const item of a) set.add(item);
|
|
else set.add(a);
|
|
if (Array.isArray(b)) for (const item of b) set.add(item);
|
|
else set.add(b);
|
|
return [...set];
|
|
};
|
|
|
|
/**
|
|
* @param {AssetInfo} a a
|
|
* @param {AssetInfo} b b
|
|
* @returns {AssetInfo} object
|
|
*/
|
|
const mergeAssetInfo = (a, b) => {
|
|
/** @type {AssetInfo} */
|
|
const result = { ...a, ...b };
|
|
for (const key of Object.keys(a)) {
|
|
if (key in b) {
|
|
if (a[key] === b[key]) continue;
|
|
switch (key) {
|
|
case "fullhash":
|
|
case "chunkhash":
|
|
case "modulehash":
|
|
case "contenthash":
|
|
result[key] = mergeMaybeArrays(a[key], b[key]);
|
|
break;
|
|
case "immutable":
|
|
case "development":
|
|
case "hotModuleReplacement":
|
|
case "javascriptModule":
|
|
result[key] = a[key] || b[key];
|
|
break;
|
|
case "related":
|
|
result[key] = mergeRelatedInfo(
|
|
/** @type {NonNullable<AssetInfo["related"]>} */
|
|
(a[key]),
|
|
/** @type {NonNullable<AssetInfo["related"]>} */
|
|
(b[key])
|
|
);
|
|
break;
|
|
default:
|
|
throw new Error(`Can't handle conflicting asset info for ${key}`);
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
|
|
/**
|
|
* @param {NonNullable<AssetInfo["related"]>} a a
|
|
* @param {NonNullable<AssetInfo["related"]>} b b
|
|
* @returns {NonNullable<AssetInfo["related"]>} object
|
|
*/
|
|
const mergeRelatedInfo = (a, b) => {
|
|
const result = { ...a, ...b };
|
|
for (const key of Object.keys(a)) {
|
|
if (key in b) {
|
|
if (a[key] === b[key]) continue;
|
|
result[key] = mergeMaybeArrays(a[key], b[key]);
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
|
|
/**
|
|
* @param {"base64" | false} encoding encoding
|
|
* @param {Source} source source
|
|
* @returns {string} encoded data
|
|
*/
|
|
const encodeDataUri = (encoding, source) => {
|
|
/** @type {string | undefined} */
|
|
let encodedContent;
|
|
|
|
switch (encoding) {
|
|
case "base64": {
|
|
encodedContent = source.buffer().toString("base64");
|
|
break;
|
|
}
|
|
case false: {
|
|
const content = source.source();
|
|
|
|
if (typeof content !== "string") {
|
|
encodedContent = content.toString("utf8");
|
|
}
|
|
|
|
encodedContent = encodeURIComponent(
|
|
/** @type {string} */
|
|
(encodedContent)
|
|
).replace(
|
|
/[!'()*]/g,
|
|
(character) =>
|
|
`%${/** @type {number} */ (character.codePointAt(0)).toString(16)}`
|
|
);
|
|
break;
|
|
}
|
|
default:
|
|
throw new Error(`Unsupported encoding '${encoding}'`);
|
|
}
|
|
|
|
return encodedContent;
|
|
};
|
|
|
|
/**
|
|
* @param {"base64" | false} encoding encoding
|
|
* @param {string} content content
|
|
* @returns {Buffer} decoded content
|
|
*/
|
|
const decodeDataUriContent = (encoding, content) => {
|
|
const isBase64 = encoding === "base64";
|
|
|
|
if (isBase64) {
|
|
return Buffer.from(content, "base64");
|
|
}
|
|
|
|
// If we can't decode return the original body
|
|
try {
|
|
return Buffer.from(decodeURIComponent(content), "ascii");
|
|
} catch (_) {
|
|
return Buffer.from(content, "ascii");
|
|
}
|
|
};
|
|
|
|
const DEFAULT_ENCODING = "base64";
|
|
|
|
class AssetGenerator extends Generator {
|
|
/**
|
|
* @param {ModuleGraph} moduleGraph the module graph
|
|
* @param {AssetGeneratorOptions["dataUrl"]=} dataUrlOptions the options for the data url
|
|
* @param {AssetModuleFilename=} filename override for output.assetModuleFilename
|
|
* @param {RawPublicPath=} publicPath override for output.assetModulePublicPath
|
|
* @param {AssetModuleOutputPath=} outputPath the output path for the emitted file which is not included in the runtime import
|
|
* @param {boolean=} emit generate output asset
|
|
*/
|
|
constructor(
|
|
moduleGraph,
|
|
dataUrlOptions,
|
|
filename,
|
|
publicPath,
|
|
outputPath,
|
|
emit
|
|
) {
|
|
super();
|
|
this.dataUrlOptions = dataUrlOptions;
|
|
this.filename = filename;
|
|
this.publicPath = publicPath;
|
|
this.outputPath = outputPath;
|
|
this.emit = emit;
|
|
this._moduleGraph = moduleGraph;
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module
|
|
* @param {RuntimeTemplate} runtimeTemplate runtime template
|
|
* @returns {string} source file name
|
|
*/
|
|
static getSourceFileName(module, runtimeTemplate) {
|
|
return makePathsRelative(
|
|
runtimeTemplate.compilation.compiler.context,
|
|
/** @type {string} */
|
|
(module.getResource()),
|
|
runtimeTemplate.compilation.compiler.root
|
|
).replace(/^\.\//, "");
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module
|
|
* @param {RuntimeTemplate} runtimeTemplate runtime template
|
|
* @returns {[string, string]} return full hash and non-numeric full hash
|
|
*/
|
|
static getFullContentHash(module, runtimeTemplate) {
|
|
const hash = createHash(runtimeTemplate.outputOptions.hashFunction);
|
|
|
|
if (runtimeTemplate.outputOptions.hashSalt) {
|
|
hash.update(runtimeTemplate.outputOptions.hashSalt);
|
|
}
|
|
|
|
const source = module.originalSource();
|
|
|
|
if (source) {
|
|
hash.update(source.buffer());
|
|
}
|
|
|
|
if (module.error) {
|
|
hash.update(module.error.toString());
|
|
}
|
|
|
|
const fullContentHash = hash.digest(
|
|
runtimeTemplate.outputOptions.hashDigest
|
|
);
|
|
|
|
const contentHash = nonNumericOnlyHash(
|
|
fullContentHash,
|
|
runtimeTemplate.outputOptions.hashDigestLength
|
|
);
|
|
|
|
return [fullContentHash, contentHash];
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module for which the code should be generated
|
|
* @param {Pick<AssetResourceGeneratorOptions, "filename" | "outputPath">} generatorOptions generator options
|
|
* @param {{ runtime: RuntimeSpec, runtimeTemplate: RuntimeTemplate, chunkGraph: ChunkGraph }} generateContext context for generate
|
|
* @param {string} contentHash the content hash
|
|
* @returns {{ filename: string, originalFilename: string, assetInfo: AssetInfo }} info
|
|
*/
|
|
static getFilenameWithInfo(
|
|
module,
|
|
generatorOptions,
|
|
{ runtime, runtimeTemplate, chunkGraph },
|
|
contentHash
|
|
) {
|
|
const assetModuleFilename =
|
|
generatorOptions.filename ||
|
|
runtimeTemplate.outputOptions.assetModuleFilename;
|
|
|
|
const sourceFilename = AssetGenerator.getSourceFileName(
|
|
module,
|
|
runtimeTemplate
|
|
);
|
|
let { path: filename, info: assetInfo } =
|
|
runtimeTemplate.compilation.getAssetPathWithInfo(assetModuleFilename, {
|
|
module,
|
|
runtime,
|
|
filename: sourceFilename,
|
|
chunkGraph,
|
|
contentHash
|
|
});
|
|
|
|
const originalFilename = filename;
|
|
|
|
if (generatorOptions.outputPath) {
|
|
const { path: outputPath, info } =
|
|
runtimeTemplate.compilation.getAssetPathWithInfo(
|
|
generatorOptions.outputPath,
|
|
{
|
|
module,
|
|
runtime,
|
|
filename: sourceFilename,
|
|
chunkGraph,
|
|
contentHash
|
|
}
|
|
);
|
|
filename = path.posix.join(outputPath, filename);
|
|
assetInfo = mergeAssetInfo(assetInfo, info);
|
|
}
|
|
|
|
return { originalFilename, filename, assetInfo };
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module for which the code should be generated
|
|
* @param {Pick<AssetResourceGeneratorOptions, "publicPath">} generatorOptions generator options
|
|
* @param {GenerateContext} generateContext context for generate
|
|
* @param {string} filename the filename
|
|
* @param {AssetInfo} assetInfo the asset info
|
|
* @param {string} contentHash the content hash
|
|
* @returns {{ assetPath: string, assetInfo: AssetInfo }} asset path and info
|
|
*/
|
|
static getAssetPathWithInfo(
|
|
module,
|
|
generatorOptions,
|
|
{ runtime, runtimeTemplate, type, chunkGraph, runtimeRequirements },
|
|
filename,
|
|
assetInfo,
|
|
contentHash
|
|
) {
|
|
const sourceFilename = AssetGenerator.getSourceFileName(
|
|
module,
|
|
runtimeTemplate
|
|
);
|
|
|
|
let assetPath;
|
|
|
|
if (generatorOptions.publicPath !== undefined && type === "javascript") {
|
|
const { path, info } = runtimeTemplate.compilation.getAssetPathWithInfo(
|
|
generatorOptions.publicPath,
|
|
{
|
|
module,
|
|
runtime,
|
|
filename: sourceFilename,
|
|
chunkGraph,
|
|
contentHash
|
|
}
|
|
);
|
|
assetInfo = mergeAssetInfo(assetInfo, info);
|
|
assetPath = JSON.stringify(path + filename);
|
|
} else if (
|
|
generatorOptions.publicPath !== undefined &&
|
|
type === "css-url"
|
|
) {
|
|
const { path, info } = runtimeTemplate.compilation.getAssetPathWithInfo(
|
|
generatorOptions.publicPath,
|
|
{
|
|
module,
|
|
runtime,
|
|
filename: sourceFilename,
|
|
chunkGraph,
|
|
contentHash
|
|
}
|
|
);
|
|
assetInfo = mergeAssetInfo(assetInfo, info);
|
|
assetPath = path + filename;
|
|
} else if (type === "javascript") {
|
|
// add __webpack_require__.p
|
|
runtimeRequirements.add(RuntimeGlobals.publicPath);
|
|
assetPath = runtimeTemplate.concatenation(
|
|
{ expr: RuntimeGlobals.publicPath },
|
|
filename
|
|
);
|
|
} else if (type === "css-url") {
|
|
const compilation = runtimeTemplate.compilation;
|
|
const path =
|
|
compilation.outputOptions.publicPath === "auto"
|
|
? CssUrlDependency.PUBLIC_PATH_AUTO
|
|
: compilation.getAssetPath(compilation.outputOptions.publicPath, {
|
|
hash: compilation.hash
|
|
});
|
|
|
|
assetPath = path + filename;
|
|
}
|
|
|
|
return {
|
|
// eslint-disable-next-line object-shorthand
|
|
assetPath: /** @type {string} */ (assetPath),
|
|
assetInfo: { sourceFilename, ...assetInfo }
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module for which the bailout reason should be determined
|
|
* @param {ConcatenationBailoutReasonContext} context context
|
|
* @returns {string | undefined} reason why this module can't be concatenated, undefined when it can be concatenated
|
|
*/
|
|
getConcatenationBailoutReason(module, context) {
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module
|
|
* @returns {string} mime type
|
|
*/
|
|
getMimeType(module) {
|
|
if (typeof this.dataUrlOptions === "function") {
|
|
throw new Error(
|
|
"This method must not be called when dataUrlOptions is a function"
|
|
);
|
|
}
|
|
|
|
/** @type {string | boolean | undefined} */
|
|
let mimeType =
|
|
/** @type {AssetGeneratorDataUrlOptions} */
|
|
(this.dataUrlOptions).mimetype;
|
|
if (mimeType === undefined) {
|
|
const ext = path.extname(
|
|
/** @type {NameForCondition} */
|
|
(module.nameForCondition())
|
|
);
|
|
if (
|
|
module.resourceResolveData &&
|
|
module.resourceResolveData.mimetype !== undefined
|
|
) {
|
|
mimeType =
|
|
module.resourceResolveData.mimetype +
|
|
module.resourceResolveData.parameters;
|
|
} else if (ext) {
|
|
mimeType = getMimeTypes().lookup(ext);
|
|
|
|
if (typeof mimeType !== "string") {
|
|
throw new Error(
|
|
"DataUrl can't be generated automatically, " +
|
|
`because there is no mimetype for "${ext}" in mimetype database. ` +
|
|
'Either pass a mimetype via "generator.mimetype" or ' +
|
|
'use type: "asset/resource" to create a resource file instead of a DataUrl'
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (typeof mimeType !== "string") {
|
|
throw new Error(
|
|
"DataUrl can't be generated automatically. " +
|
|
'Either pass a mimetype via "generator.mimetype" or ' +
|
|
'use type: "asset/resource" to create a resource file instead of a DataUrl'
|
|
);
|
|
}
|
|
|
|
return /** @type {string} */ (mimeType);
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module for which the code should be generated
|
|
* @returns {string} DataURI
|
|
*/
|
|
generateDataUri(module) {
|
|
const source = /** @type {Source} */ (module.originalSource());
|
|
|
|
let encodedSource;
|
|
|
|
if (typeof this.dataUrlOptions === "function") {
|
|
encodedSource = this.dataUrlOptions.call(null, source.source(), {
|
|
filename: /** @type {string} */ (module.getResource()),
|
|
module
|
|
});
|
|
} else {
|
|
let encoding =
|
|
/** @type {AssetGeneratorDataUrlOptions} */
|
|
(this.dataUrlOptions).encoding;
|
|
if (
|
|
encoding === undefined &&
|
|
module.resourceResolveData &&
|
|
module.resourceResolveData.encoding !== undefined
|
|
) {
|
|
encoding = module.resourceResolveData.encoding;
|
|
}
|
|
if (encoding === undefined) {
|
|
encoding = DEFAULT_ENCODING;
|
|
}
|
|
const mimeType = this.getMimeType(module);
|
|
|
|
let encodedContent;
|
|
|
|
if (
|
|
module.resourceResolveData &&
|
|
module.resourceResolveData.encoding === encoding &&
|
|
decodeDataUriContent(
|
|
module.resourceResolveData.encoding,
|
|
/** @type {string} */ (module.resourceResolveData.encodedContent)
|
|
).equals(source.buffer())
|
|
) {
|
|
encodedContent = module.resourceResolveData.encodedContent;
|
|
} else {
|
|
encodedContent = encodeDataUri(
|
|
/** @type {"base64" | false} */ (encoding),
|
|
source
|
|
);
|
|
}
|
|
|
|
encodedSource = `data:${mimeType}${
|
|
encoding ? `;${encoding}` : ""
|
|
},${encodedContent}`;
|
|
}
|
|
|
|
return encodedSource;
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module module for which the code should be generated
|
|
* @param {GenerateContext} generateContext context for generate
|
|
* @returns {Source | null} generated code
|
|
*/
|
|
generate(module, generateContext) {
|
|
const {
|
|
type,
|
|
getData,
|
|
runtimeTemplate,
|
|
runtimeRequirements,
|
|
concatenationScope
|
|
} = generateContext;
|
|
|
|
let content;
|
|
|
|
const needContent = type === "javascript" || type === "css-url";
|
|
|
|
const data = getData ? getData() : undefined;
|
|
|
|
if (
|
|
/** @type {BuildInfo} */
|
|
(module.buildInfo).dataUrl &&
|
|
needContent
|
|
) {
|
|
const encodedSource = this.generateDataUri(module);
|
|
content =
|
|
type === "javascript" ? JSON.stringify(encodedSource) : encodedSource;
|
|
|
|
if (data) {
|
|
data.set("url", { [type]: content, ...data.get("url") });
|
|
}
|
|
} else {
|
|
const [fullContentHash, contentHash] = AssetGenerator.getFullContentHash(
|
|
module,
|
|
runtimeTemplate
|
|
);
|
|
|
|
if (data) {
|
|
data.set("fullContentHash", fullContentHash);
|
|
data.set("contentHash", contentHash);
|
|
}
|
|
|
|
/** @type {BuildInfo} */
|
|
(module.buildInfo).fullContentHash = fullContentHash;
|
|
|
|
const { originalFilename, filename, assetInfo } =
|
|
AssetGenerator.getFilenameWithInfo(
|
|
module,
|
|
{ filename: this.filename, outputPath: this.outputPath },
|
|
generateContext,
|
|
contentHash
|
|
);
|
|
|
|
if (data) {
|
|
data.set("filename", filename);
|
|
}
|
|
|
|
let { assetPath, assetInfo: newAssetInfo } =
|
|
AssetGenerator.getAssetPathWithInfo(
|
|
module,
|
|
{ publicPath: this.publicPath },
|
|
generateContext,
|
|
originalFilename,
|
|
assetInfo,
|
|
contentHash
|
|
);
|
|
|
|
if (data && (type === "javascript" || type === "css-url")) {
|
|
data.set("url", { [type]: assetPath, ...data.get("url") });
|
|
}
|
|
|
|
if (data) {
|
|
const oldAssetInfo = data.get("assetInfo");
|
|
|
|
if (oldAssetInfo) {
|
|
newAssetInfo = mergeAssetInfo(oldAssetInfo, newAssetInfo);
|
|
}
|
|
}
|
|
|
|
if (data) {
|
|
data.set("assetInfo", newAssetInfo);
|
|
}
|
|
|
|
// Due to code generation caching module.buildInfo.XXX can't used to store such information
|
|
// It need to be stored in the code generation results instead, where it's cached too
|
|
// TODO webpack 6 For back-compat reasons we also store in on module.buildInfo
|
|
/** @type {BuildInfo} */
|
|
(module.buildInfo).filename = filename;
|
|
|
|
/** @type {BuildInfo} */
|
|
(module.buildInfo).assetInfo = newAssetInfo;
|
|
|
|
content = assetPath;
|
|
}
|
|
|
|
if (type === "javascript") {
|
|
if (concatenationScope) {
|
|
concatenationScope.registerNamespaceExport(
|
|
ConcatenationScope.NAMESPACE_OBJECT_EXPORT
|
|
);
|
|
|
|
return new RawSource(
|
|
`${runtimeTemplate.renderConst()} ${
|
|
ConcatenationScope.NAMESPACE_OBJECT_EXPORT
|
|
} = ${content};`
|
|
);
|
|
}
|
|
|
|
runtimeRequirements.add(RuntimeGlobals.module);
|
|
|
|
return new RawSource(`${RuntimeGlobals.module}.exports = ${content};`);
|
|
} else if (type === "css-url") {
|
|
return null;
|
|
}
|
|
|
|
return /** @type {Source} */ (module.originalSource());
|
|
}
|
|
|
|
/**
|
|
* @param {Error} error the error
|
|
* @param {NormalModule} module module for which the code should be generated
|
|
* @param {GenerateContext} generateContext context for generate
|
|
* @returns {Source | null} generated code
|
|
*/
|
|
generateError(error, module, generateContext) {
|
|
switch (generateContext.type) {
|
|
case "asset": {
|
|
return new RawSource(error.message);
|
|
}
|
|
case "javascript": {
|
|
return new RawSource(
|
|
`throw new Error(${JSON.stringify(error.message)});`
|
|
);
|
|
}
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module fresh module
|
|
* @returns {SourceTypes} available types (do not mutate)
|
|
*/
|
|
getTypes(module) {
|
|
/** @type {Set<string>} */
|
|
const sourceTypes = new Set();
|
|
const connections = this._moduleGraph.getIncomingConnections(module);
|
|
|
|
for (const connection of connections) {
|
|
if (!connection.originModule) {
|
|
continue;
|
|
}
|
|
|
|
sourceTypes.add(connection.originModule.type.split("/")[0]);
|
|
}
|
|
|
|
if ((module.buildInfo && module.buildInfo.dataUrl) || this.emit === false) {
|
|
if (sourceTypes.size > 0) {
|
|
if (sourceTypes.has("javascript") && sourceTypes.has("css")) {
|
|
return JS_AND_CSS_URL_TYPES;
|
|
} else if (sourceTypes.has("css")) {
|
|
return CSS_URL_TYPES;
|
|
}
|
|
return JS_TYPES;
|
|
}
|
|
|
|
return NO_TYPES;
|
|
}
|
|
|
|
if (sourceTypes.size > 0) {
|
|
if (sourceTypes.has("javascript") && sourceTypes.has("css")) {
|
|
return ASSET_AND_JS_AND_CSS_URL_TYPES;
|
|
} else if (sourceTypes.has("css")) {
|
|
return ASSET_AND_CSS_URL_TYPES;
|
|
}
|
|
return ASSET_AND_JS_TYPES;
|
|
}
|
|
|
|
return ASSET_TYPES;
|
|
}
|
|
|
|
/**
|
|
* @param {NormalModule} module the module
|
|
* @param {string=} type source type
|
|
* @returns {number} estimate size of the module
|
|
*/
|
|
getSize(module, type) {
|
|
switch (type) {
|
|
case ASSET_MODULE_TYPE: {
|
|
const originalSource = module.originalSource();
|
|
|
|
if (!originalSource) {
|
|
return 0;
|
|
}
|
|
|
|
return originalSource.size();
|
|
}
|
|
default:
|
|
if (module.buildInfo && module.buildInfo.dataUrl) {
|
|
const originalSource = module.originalSource();
|
|
|
|
if (!originalSource) {
|
|
return 0;
|
|
}
|
|
|
|
// roughly for data url
|
|
// Example: m.exports="data:image/png;base64,ag82/f+2=="
|
|
// 4/3 = base64 encoding
|
|
// 34 = ~ data url header + footer + rounding
|
|
return originalSource.size() * 1.34 + 36;
|
|
}
|
|
// it's only estimated so this number is probably fine
|
|
// Example: m.exports=r.p+"0123456789012345678901.ext"
|
|
return 42;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Hash} hash hash that will be modified
|
|
* @param {UpdateHashContext} updateHashContext context for updating hash
|
|
*/
|
|
updateHash(hash, updateHashContext) {
|
|
const { module } = updateHashContext;
|
|
|
|
if (
|
|
/** @type {BuildInfo} */
|
|
(module.buildInfo).dataUrl
|
|
) {
|
|
hash.update("data-url");
|
|
// this.dataUrlOptions as function should be pure and only depend on input source and filename
|
|
// therefore it doesn't need to be hashed
|
|
if (typeof this.dataUrlOptions === "function") {
|
|
const ident = /** @type {{ ident?: string }} */ (this.dataUrlOptions)
|
|
.ident;
|
|
if (ident) hash.update(ident);
|
|
} else {
|
|
const dataUrlOptions =
|
|
/** @type {AssetGeneratorDataUrlOptions} */
|
|
(this.dataUrlOptions);
|
|
if (
|
|
dataUrlOptions.encoding &&
|
|
dataUrlOptions.encoding !== DEFAULT_ENCODING
|
|
) {
|
|
hash.update(dataUrlOptions.encoding);
|
|
}
|
|
if (dataUrlOptions.mimetype) hash.update(dataUrlOptions.mimetype);
|
|
// computed mimetype depends only on module filename which is already part of the hash
|
|
}
|
|
} else {
|
|
hash.update("resource");
|
|
|
|
const { module, chunkGraph, runtime } = updateHashContext;
|
|
const runtimeTemplate =
|
|
/** @type {NonNullable<UpdateHashContext["runtimeTemplate"]>} */
|
|
(updateHashContext.runtimeTemplate);
|
|
|
|
const pathData = {
|
|
module,
|
|
runtime,
|
|
filename: AssetGenerator.getSourceFileName(module, runtimeTemplate),
|
|
chunkGraph,
|
|
contentHash: runtimeTemplate.contentHashReplacement
|
|
};
|
|
|
|
if (typeof this.publicPath === "function") {
|
|
hash.update("path");
|
|
const assetInfo = {};
|
|
hash.update(this.publicPath(pathData, assetInfo));
|
|
hash.update(JSON.stringify(assetInfo));
|
|
} else if (this.publicPath) {
|
|
hash.update("path");
|
|
hash.update(this.publicPath);
|
|
} else {
|
|
hash.update("no-path");
|
|
}
|
|
|
|
const assetModuleFilename =
|
|
this.filename || runtimeTemplate.outputOptions.assetModuleFilename;
|
|
const { path: filename, info } =
|
|
runtimeTemplate.compilation.getAssetPathWithInfo(
|
|
assetModuleFilename,
|
|
pathData
|
|
);
|
|
hash.update(filename);
|
|
hash.update(JSON.stringify(info));
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = AssetGenerator;
|