This is the first day of my participation in the More text Challenge. For details, see more text Challenge
Afraid of what truth is infinite, into an inch of an inch of joy. Hello, everyone. I am @Luozhu, a lifelong practitioner who loves programming and life.
rollup -c
A brief process
Plug-in system related modules
- Graph: globally unique Graph that contains entries and dependencies, operations, caching, etc. Is the heart of rollup
- PluginDriver: a plug-in driver that invokes the plug-in and provides context for the plug-in environment
Analysis of plug-in mechanism
An overview of the
A Rollup plug-in is an object consisting of one or more properties, building hook functions, and output hook functions. The plug-in also needs to conform to some official conventions. A plug-in should be published as a package that exports a function that can be called with plugin-specific options and that returns an object.
Plugins allow you to customize Rollup’s behavior, such as converting code before packaging or looking for third-party packages in your node_modules folder.
Official plugins are maintained in the Rollup/Plugins repository, while community-selected plugins are maintained in Rollup/Awesome. If you want to suggest a plugin, please submit a PR.
A simple example
The following plug-ins can intercept any virtual-Module import without accessing the file system. This is necessary if you want to use Rollup in your browser, for example. It can even be used to replace entry points, as shown in the example.
// index.js
export default function myExample() {
return {
name: 'my-example'.// The name is displayed in warnings and errors
resolveId(source) {
if (source === 'virtual-module') {
return source; // This indicates that Rollup should not ask other plug-ins or check the file system to find this ID.
}
return null; // Other ids should be processed the usual way
},
load(id) {
if (id === 'virtual-module') {
return 'export default "This is virtual!" '; // The source code for "virtual-module"
}
return null; // Other ids should be processed the usual way}}; }// rollup.config.js
import myExample from './index';
export default {
input: 'virtual-module'.// is parsed by our plugin
plugins: [myExample()],
output: [{file: 'bundle.js'.format: 'es',}]};Copy the code
convention
- Plug-ins should have a clear name and must be carried
rollup-plugin-
Prefix. - in
package.json
Contained in therollup-plugin
The keyword. - Plugins should be tested and we recommend libraries like Mocha or Ava that support promises out of the box.
- Use asynchronous methods whenever possible.
- Write plug-in documentation in English
- If appropriate, make sure your plugin outputs the correct sourcemap
- If your plugin uses ‘virtual modules’ (such as helper functions), add it to the module name
\ 0
Prefix. This prevents other plug-ins from executing it.
Hook function
The core of the rollup plugin is the hook function. There are two types of hook functions:
Building hook functions
In order to interact with the build process, your plug-in object needs to contain some build hook functions. Build hooks are functions that are called at each stage of a build. Build hook functions can affect how a build is executed, provide information about a build, or modify a build after it is completed. There are different build hook functions in Rollup:
async
: this hook can also return a promise that resolves to a value of the same type; Otherwise, the hook will be marked assync
.first
If multiple plugins implement the hook, the hooks will be run in turn until the hook returns a nonull
Or notundefined
The value of the.sequential
If multiple plugins implement this hook, all plugins will run in the order specified. If a hook is asynchronous, subsequent hooks of this type will wait until the current hook is resolved.parallel
If multiple plugins implement this hook, all plugins will run in the order specified. If a hook is asynchronous, subsequent hooks of this type will run in parallel without waiting for the current hook.
Build hook functions are executed during the build phase. They are triggered by rollup.rollup(inputOptions). They focus on locating, supplying, and converting input files before they are processed by Rollup. The first hook in the build phase is options, and the last hook is always buildEnd, unless there is a build error, in which case the closeBundle will be called after that.
In addition, in watch mode, the watchChange hook can be fired at any time to notify that a new run will be fired after the current run has produced its output. In addition, the closeWatcher hook function is fired when watcher is closed.
The output generates the hook function
The output build hook function provides information about the generated package and is executed immediately after the build is complete. They work in the same way and are of the same type as the build hook functions, but are called by bundle.generate(output) or bundle.write(outputOptions), respectively. Plug-ins that generate hooks using only the output can also be passed in through the output option because they only run on certain outputs.
The first hook function in the output build phase is outputOptions, if the output goes through bundle.generate(…) The first hook function is generateBundle if the output is delivered via bundle.write(…) The last hook function is writeBundle, and the last hook function is renderError if an error occurs during the output build phase.
In addition, the closeBundle can be called as the last hook, but it is the user’s responsibility to trigger it manually by calling bundle.close(). The CLI will always ensure that this happens.
The hook function loads the implementation
There are nine hook loading functions in the PluginDriver. This is mainly because there are synchronous and asynchronous versions of hooks for each class.
1. HookFirst:
Load the hook function of type FIRST in scenarios such as resolveId, resolveAssetUrl, and so on
function hookFirst<H extends keyof PluginHooks.R = ReturnType<PluginHooks[H] > > (hookName: H, args: Args<PluginHooks[H]>, replaceContext? : ReplaceContext |null, skip? : number |null
) :EnsurePromise<R> {
// Initialize the promise
let promise: Promise<any> = Promise.resolve();
// This.plugins are initialized when the Graph is instantiated
for (let i = 0; i < this.plugins.length; i++) {
if (skip === i) continue;
// Overwrite the previous promise. In other words, execute the hook function sequentially
promise = promise.then((result: any) = > {
// If null or undefined is returned, stop running and return the result
if(result ! =null) return result;
// Execute the hook function
return this.runHook(hookName, args as any[], i, false, replaceContext);
});
}
// Return the hooked promise
return promise;
}
Copy the code
2. HookFirstSync:
A synchronous version of hookFirst. The scenarios include resolveFileUrl, resolveImportMeta, and so on
function hookFirstSync<H extends keyof PluginHooks.R = ReturnType<PluginHooks[H] > > (hookName: H, args: Args
, replaceContext? : ReplaceContext
[h]>) :R {
for (let i = 0; i < this.plugins.length; i++) {
// Synchronous version of runHook
const result = this.runHookSync(hookName, args, i, replaceContext);
// If null or undefined is returned, stop running and return the result
if(result ! =null) return result as any;
}
// Otherwise null is returned
return null as any;
}
Copy the code
3.hookSeq
Load sequential type hook functions, which are different from hookFirst in that they cannot be interrupted. Use scenarios include onWrite, generateBundle, and so on
async function hookSeq<H extends keyof PluginHooks> (hookName: H, args: Args<PluginHooks[H]>, replaceContext? : ReplaceContext,// hookFirst uses the skip parameter to decide whether to skip a hook function
) :Promise<void> {
let promise: Promise<void> = Promise.resolve();
for (let i = 0; i < this.plugins.length; i++)
promise = promise.then(() = >
this.runHook<void>(hookName, args as any[], i, false, replaceContext),
);
return promise;
}
Copy the code
4.hookSeqSync
HookSeq synchronization version, instead of constructing a Promise, uses runHookSync directly to execute the hook function. The application scenarios include closeWatcher, watchChange, etc.
hookSeqSync<H extends SyncPluginHooks & SequentialPluginHooks>(
hookName: H,
args: Parameters<PluginHooks[H]>, replaceContext? : ReplaceContext ):void {
for (const plugin of this.plugins) {
this.runHookSync(hookName, args, plugin, replaceContext); }}Copy the code
5.hookReduceArg0
The reduce operation is performed on the first item of the ARG. Scenarios: Options, renderChunk, and so on
function hookReduceArg0<H extends keyof PluginHooks.V.R = ReturnType<PluginHooks[H] > > (
hookName: H,
[arg0, ...args]: any[], // Take the first argument of the passed array and place the rest in an arrayreduce: Reduce<V, R>, replaceContext? : ReplaceContext// Replace the context in which the plugin is currently invoked
) {
let promise = Promise.resolve(arg0); // Source.code is returned by default
for (let i = 0; i < this.plugins.length; i++) {
// The first promise will only receive the arg0 passed above
// Each subsequent promise accepts the value of source.code processed by the previous plugin
promise = promise.then(arg0= > {
const hookPromise = this.runHook(hookName, [arg0, ...args], i, false, replaceContext);
// Return arg0 if no promise is returned
if(! hookPromise)return arg0;
// Result is the return value of plug-in execution
return hookPromise.then((result: any) = >
reduce.call(this.pluginContexts[i], arg0, result, this.plugins[i])
);
});
}
return promise;
}
Copy the code
6.hookReduceArg0Sync
HookReduceArg0 Synchronous version, such as Transform and generateBundle
7.hookParallel
Execute hooks in parallel without waiting for the current hook to complete. Usage Scenario: buildEnd, buildStart, moduleParsed.
hookParallel<H extends AsyncPluginHooks & ParallelPluginHooks>(
hookName: H,
args: Parameters<PluginHooks[H]>, replaceContext? : ReplaceContext ):Promise<void> {
const promises: Promise<void= > [] [];for (const plugin of this.plugins) {
const hookPromise = this.runHook(hookName, args, plugin, false, replaceContext);
if(! hookPromise)continue;
promises.push(hookPromise);
}
return Promise.all(promises).then(() = > {});
}
Copy the code
runHook
The hook function is loaded with runHook or runHookSync. The hook function is loaded with runHook or runHookSync.
function runHook<T> (
hookName: string,
args: any[],
pluginIndex: number,
permitValues: boolean, hookContext? : ReplaceContext |null.) :Promise<T> {
this.previousHooks.add(hookName);
// Find the current plugin
const plugin = this.plugins[pluginIndex];
// Find the current executed hooks function defined in the plugin
const hook = (plugin as any)[hookName];
if(! hook)return undefined as any;
PluginContexts is an array that holds the context of each plug-in
let context = this.pluginContexts[pluginIndex];
// Plugin context to distinguish between different hook functions
if (hookContext) {
context = hookContext(context, plugin);
}
return Promise.resolve()
.then(() = > {
// The license value allows the return value, instead of a function hook, to be loaded using hookReduceValue or hookReduceValueSync.
if (typeofhook ! = ='function') {
if (permitValues) return hook;
return error({
code: 'INVALID_PLUGIN_HOOK'.message: `Error running plugin hook ${hookName} for ${plugin.name}, expected a function hook.`}); }// Return the result of plug-in execution by passing in the plug-in context and parameters
return hook.apply(context, args);
})
.catch(err= > throwPluginError(err, plugin.name, { hook: hookName }));
}
Copy the code
The core depends on
- Yargs-parser: Powerful option parsing plugin used by Yargs
- Source-map-support: This module supports stack Sourcemap support through the V8 stack tracing API
conclusion
Rollup’s plug-ins are similar to other large frameworks in that they provide a uniform interface and implement the idea of convention over configuration. The nine hook loading functions make plug-in development very flexible for Rollup, but they also come with a learning cost.
Compared to WebPack, Rollup’s plugin system is self-styled and makes no distinction between plugins and Loaders.
At the heart of the Rollup plug-in mechanism are various hook functions during the build phase and output generation phase. Internal implementation of asynchronous hook scheduling based on Promise.
The source code for Rollup is all mashed up in one library, making it a big read, and the management of modules and utility functions looks random. And we weren’t able to directly port any of its tools to our project. In contrast, WebPack’s plug-in system encapsulated as a plug-in tapable was a great way to learn and use it.
This article was first published on the “official website of Luozhu”, as well as on the public account “Luozhu Morning Teahouse” and “Gold Column”.