Standardize settings file naming and relocate documentation files Fix code quality violations from rsx:check Reorganize user_management directory into logical subdirectories Move Quill Bundle to core and align with Tom Select pattern Simplify Site Settings page to focus on core site information Complete Phase 5: Multi-tenant authentication with login flow and site selection Add route query parameter rule and synchronize filename validation logic Fix critical bug in UpdateNpmCommand causing missing JavaScript stubs Implement filename convention rule and resolve VS Code auto-rename conflict Implement js-sanitizer RPC server to eliminate 900+ Node.js process spawns Implement RPC server architecture for JavaScript parsing WIP: Add RPC server infrastructure for JS parsing (partial implementation) Update jqhtml terminology from destroy to stop, fix datagrid DOM preservation Add JQHTML-CLASS-01 rule and fix redundant class names Improve code quality rules and resolve violations Remove legacy fatal error format in favor of unified 'fatal' error type Filter internal keys from window.rsxapp output Update button styling and comprehensive form/modal documentation Add conditional fly-in animation for modals Fix non-deterministic bundle compilation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
337 lines
9.6 KiB
JavaScript
Executable File
337 lines
9.6 KiB
JavaScript
Executable File
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
const RuntimeGlobals = require("../RuntimeGlobals");
|
|
const JavascriptModulesPlugin = require("../javascript/JavascriptModulesPlugin");
|
|
|
|
/** @typedef {import("webpack-sources").Source} Source */
|
|
/** @typedef {import("../../declarations/WebpackOptions").LibraryOptions} LibraryOptions */
|
|
/** @typedef {import("../../declarations/WebpackOptions").LibraryType} LibraryType */
|
|
/** @typedef {import("../Chunk")} Chunk */
|
|
/** @typedef {import("../ChunkGraph")} ChunkGraph */
|
|
/** @typedef {import("../Compilation")} Compilation */
|
|
/** @typedef {import("../Compilation").ChunkHashContext} ChunkHashContext */
|
|
/** @typedef {import("../Compiler")} Compiler */
|
|
/** @typedef {import("../Module")} Module */
|
|
/** @typedef {import("../Module").RuntimeRequirements} RuntimeRequirements */
|
|
/** @typedef {import("../javascript/JavascriptModulesPlugin").RenderContext} RenderContext */
|
|
/** @typedef {import("../javascript/JavascriptModulesPlugin").StartupRenderContext} StartupRenderContext */
|
|
/** @typedef {import("../javascript/JavascriptModulesPlugin").ModuleRenderContext} ModuleRenderContext */
|
|
/** @typedef {import("../util/Hash")} Hash */
|
|
|
|
const COMMON_LIBRARY_NAME_MESSAGE =
|
|
"Common configuration options that specific library names are 'output.library[.name]', 'entry.xyz.library[.name]', 'ModuleFederationPlugin.name' and 'ModuleFederationPlugin.library[.name]'.";
|
|
|
|
/**
|
|
* @template T
|
|
* @typedef {object} LibraryContext
|
|
* @property {Compilation} compilation
|
|
* @property {ChunkGraph} chunkGraph
|
|
* @property {T} options
|
|
*/
|
|
|
|
/**
|
|
* @typedef {object} AbstractLibraryPluginOptions
|
|
* @property {string} pluginName name of the plugin
|
|
* @property {LibraryType} type used library type
|
|
*/
|
|
|
|
/**
|
|
* @template T
|
|
*/
|
|
class AbstractLibraryPlugin {
|
|
/**
|
|
* @param {AbstractLibraryPluginOptions} options options
|
|
*/
|
|
constructor({ pluginName, type }) {
|
|
this._pluginName = pluginName;
|
|
this._type = type;
|
|
this._parseCache = new WeakMap();
|
|
}
|
|
|
|
/**
|
|
* Apply the plugin
|
|
* @param {Compiler} compiler the compiler instance
|
|
* @returns {void}
|
|
*/
|
|
apply(compiler) {
|
|
const { _pluginName } = this;
|
|
compiler.hooks.thisCompilation.tap(_pluginName, (compilation) => {
|
|
compilation.hooks.finishModules.tap(
|
|
{ name: _pluginName, stage: 10 },
|
|
() => {
|
|
for (const [
|
|
name,
|
|
{
|
|
dependencies: deps,
|
|
options: { library }
|
|
}
|
|
] of compilation.entries) {
|
|
const options = this._parseOptionsCached(
|
|
library !== undefined
|
|
? library
|
|
: compilation.outputOptions.library
|
|
);
|
|
if (options !== false) {
|
|
const dep = deps[deps.length - 1];
|
|
if (dep) {
|
|
const module = compilation.moduleGraph.getModule(dep);
|
|
if (module) {
|
|
this.finishEntryModule(module, name, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
);
|
|
|
|
/**
|
|
* @param {Chunk} chunk chunk
|
|
* @returns {T | false} options for the chunk
|
|
*/
|
|
const getOptionsForChunk = (chunk) => {
|
|
if (compilation.chunkGraph.getNumberOfEntryModules(chunk) === 0) {
|
|
return false;
|
|
}
|
|
const options = chunk.getEntryOptions();
|
|
const library = options && options.library;
|
|
return this._parseOptionsCached(
|
|
library !== undefined ? library : compilation.outputOptions.library
|
|
);
|
|
};
|
|
|
|
if (
|
|
this.render !== AbstractLibraryPlugin.prototype.render ||
|
|
this.runtimeRequirements !==
|
|
AbstractLibraryPlugin.prototype.runtimeRequirements
|
|
) {
|
|
compilation.hooks.additionalChunkRuntimeRequirements.tap(
|
|
_pluginName,
|
|
(chunk, set, { chunkGraph }) => {
|
|
const options = getOptionsForChunk(chunk);
|
|
if (options !== false) {
|
|
this.runtimeRequirements(chunk, set, {
|
|
options,
|
|
compilation,
|
|
chunkGraph
|
|
});
|
|
}
|
|
}
|
|
);
|
|
}
|
|
|
|
const hooks = JavascriptModulesPlugin.getCompilationHooks(compilation);
|
|
|
|
if (this.render !== AbstractLibraryPlugin.prototype.render) {
|
|
hooks.render.tap(_pluginName, (source, renderContext) => {
|
|
const options = getOptionsForChunk(renderContext.chunk);
|
|
if (options === false) return source;
|
|
return this.render(source, renderContext, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
});
|
|
}
|
|
|
|
if (
|
|
this.embedInRuntimeBailout !==
|
|
AbstractLibraryPlugin.prototype.embedInRuntimeBailout
|
|
) {
|
|
hooks.embedInRuntimeBailout.tap(
|
|
_pluginName,
|
|
(module, renderContext) => {
|
|
const options = getOptionsForChunk(renderContext.chunk);
|
|
if (options === false) return;
|
|
return this.embedInRuntimeBailout(module, renderContext, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
}
|
|
);
|
|
}
|
|
|
|
if (
|
|
this.strictRuntimeBailout !==
|
|
AbstractLibraryPlugin.prototype.strictRuntimeBailout
|
|
) {
|
|
hooks.strictRuntimeBailout.tap(_pluginName, (renderContext) => {
|
|
const options = getOptionsForChunk(renderContext.chunk);
|
|
if (options === false) return;
|
|
return this.strictRuntimeBailout(renderContext, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
});
|
|
}
|
|
|
|
if (
|
|
this.renderModuleContent !==
|
|
AbstractLibraryPlugin.prototype.renderModuleContent
|
|
) {
|
|
hooks.renderModuleContent.tap(
|
|
_pluginName,
|
|
(source, module, renderContext) =>
|
|
this.renderModuleContent(source, module, renderContext, {
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
})
|
|
);
|
|
}
|
|
|
|
if (
|
|
this.renderStartup !== AbstractLibraryPlugin.prototype.renderStartup
|
|
) {
|
|
hooks.renderStartup.tap(
|
|
_pluginName,
|
|
(source, module, renderContext) => {
|
|
const options = getOptionsForChunk(renderContext.chunk);
|
|
if (options === false) return source;
|
|
return this.renderStartup(source, module, renderContext, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
}
|
|
);
|
|
}
|
|
|
|
hooks.chunkHash.tap(_pluginName, (chunk, hash, context) => {
|
|
const options = getOptionsForChunk(chunk);
|
|
if (options === false) return;
|
|
this.chunkHash(chunk, hash, context, {
|
|
options,
|
|
compilation,
|
|
chunkGraph: compilation.chunkGraph
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {LibraryOptions=} library normalized library option
|
|
* @returns {T | false} preprocess as needed by overriding
|
|
*/
|
|
_parseOptionsCached(library) {
|
|
if (!library) return false;
|
|
if (library.type !== this._type) return false;
|
|
const cacheEntry = this._parseCache.get(library);
|
|
if (cacheEntry !== undefined) return cacheEntry;
|
|
const result = this.parseOptions(library);
|
|
this._parseCache.set(library, result);
|
|
return result;
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
/**
|
|
* @abstract
|
|
* @param {LibraryOptions} library normalized library option
|
|
* @returns {T | false} preprocess as needed by overriding
|
|
*/
|
|
parseOptions(library) {
|
|
const AbstractMethodError = require("../AbstractMethodError");
|
|
|
|
throw new AbstractMethodError();
|
|
}
|
|
|
|
/**
|
|
* @param {Module} module the exporting entry module
|
|
* @param {string} entryName the name of the entrypoint
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {void}
|
|
*/
|
|
finishEntryModule(module, entryName, libraryContext) {}
|
|
|
|
/**
|
|
* @param {Module} module the exporting entry module
|
|
* @param {RenderContext} renderContext render context
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {string | undefined} bailout reason
|
|
*/
|
|
embedInRuntimeBailout(module, renderContext, libraryContext) {
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* @param {RenderContext} renderContext render context
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {string | undefined} bailout reason
|
|
*/
|
|
strictRuntimeBailout(renderContext, libraryContext) {
|
|
return undefined;
|
|
}
|
|
|
|
/**
|
|
* @param {Chunk} chunk the chunk
|
|
* @param {RuntimeRequirements} set runtime requirements
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {void}
|
|
*/
|
|
runtimeRequirements(chunk, set, libraryContext) {
|
|
if (this.render !== AbstractLibraryPlugin.prototype.render) {
|
|
set.add(RuntimeGlobals.returnExportsFromRuntime);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Source} source source
|
|
* @param {RenderContext} renderContext render context
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {Source} source with library export
|
|
*/
|
|
render(source, renderContext, libraryContext) {
|
|
return source;
|
|
}
|
|
|
|
/**
|
|
* @param {Source} source source
|
|
* @param {Module} module module
|
|
* @param {StartupRenderContext} renderContext render context
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {Source} source with library export
|
|
*/
|
|
renderStartup(source, module, renderContext, libraryContext) {
|
|
return source;
|
|
}
|
|
|
|
/**
|
|
* @param {Source} source source
|
|
* @param {Module} module module
|
|
* @param {ModuleRenderContext} renderContext render context
|
|
* @param {Omit<LibraryContext<T>, 'options'>} libraryContext context
|
|
* @returns {Source} source with library export
|
|
*/
|
|
renderModuleContent(source, module, renderContext, libraryContext) {
|
|
return source;
|
|
}
|
|
|
|
/**
|
|
* @param {Chunk} chunk the chunk
|
|
* @param {Hash} hash hash
|
|
* @param {ChunkHashContext} chunkHashContext chunk hash context
|
|
* @param {LibraryContext<T>} libraryContext context
|
|
* @returns {void}
|
|
*/
|
|
chunkHash(chunk, hash, chunkHashContext, libraryContext) {
|
|
const options = this._parseOptionsCached(
|
|
libraryContext.compilation.outputOptions.library
|
|
);
|
|
hash.update(this._pluginName);
|
|
hash.update(JSON.stringify(options));
|
|
}
|
|
}
|
|
|
|
AbstractLibraryPlugin.COMMON_LIBRARY_NAME_MESSAGE = COMMON_LIBRARY_NAME_MESSAGE;
|
|
|
|
module.exports = AbstractLibraryPlugin;
|