Multiple separate builds should form a single application. These separate builds should not have dependencies between each other, so they can be developed and deployed individually.
This is often known as Micro-Frontends, but is not limited to that.
We distinguish between local and remote modules. Local modules are normal modules which are part of the current build. Remote modules are modules that are not part of the current build and loaded from a so-called container at the runtime.
Loading remote modules is considered asynchronous operation. When using a remote module these asynchronous operations will be placed in the next chunk loading operation(s) that is between the remote module and the entrypoint. It's not possible to use a remote module without a chunk loading operation.
A chunk loading operation is usually an import()
call, but older constructs like require.ensure
or require([...])
are supported as well.
A container is created through a container entry, which exposes asynchronous access to the specific modules. The exposed access is separated into two steps:
Step 1 will be done during the chunk loading. Step 2 will be done during the module evaluation interleaved with other (local and remote) modules. This way, evaluation order is unaffected by converting a module from local to remote or the other way around.
It is possible to nest a container. Containers can use modules from other containers. Circular dependencies between containers are also possible.
A container is able to flag selected local modules as "overridable". A consumer of the container is able to provide "overrides", which are modules that replace one of the overridable modules of the container. All modules of the container will use the replacement module instead of the local module when the consumer provides one. When the consumer doesn't provide a replacement module, all modules of the container will use the local one.
The container will manage overridable modules in a way that they do not need to be downloaded when they have been overridden by the consumer. This usually happens by placing them into separate chunks.
On the other hand, the provider of the replacement modules, will only provide asynchronous loading functions. It allows the container to load replacement modules only when they are needed. The provider will manage replacement modules in a way that they do not need to be downloaded at all when they are not requested by the container. This usually happens by placing them into separate chunks.
A "name" is used to identify overridable modules from the container.
Overrides are provided in a similar way as the container exposes modules, separated into two steps:
Overrides must be provided before the modules of the container are loaded. Overridables that are used in initial chunk, can only be overridden by a synchronous module override that doesn't use Promises. Once evaluated, overridables are no longer overridable.
Each build acts as a container and also consumes other builds as containers. This way each build is able to access any other exposed module by loading it from its container.
Shared modules are modules that are both overridable and provided as overrides to nested container. They usually point to the same module in each build, e.g. the same library.
The packageName
option allows setting a package name to look for a requiredVersion
. It is automatically inferred for the module requests by default, set requiredVersion
to false
when automatic infer should be disabled.
OverridablesPlugin
(low level)This plugin makes specific modules "overridable". A local API (__webpack_override__
) allows to provide overrides.
webpack.config.js
const OverridablesPlugin = require("webpack/lib/container/OverridablesPlugin");
module.exports = {
plugins: [
new OverridablesPlugin([
{
// we define an overridable module with OverridablesPlugin
test1: "./src/test1.js",
},
]),
],
};
src/index.js
__webpack_override__({
// here we override test1 module
test1: () => "I will override test1 module under src",
});
ContainerPlugin
(low level)This plugin creates an additional container entry with the specified exposed modules. It also uses the OverridablesPlugin
internally and exposes the override
API to consumer of the container.
ContainerReferencePlugin
(low level)This plugin adds specific references to containers as externals and allows to import remote modules from these containers. It also calls the override
API of these containers to provide overrides to them. Local overrides (via __webpack_override__
or override
API when build is also a container) and specified overrides are provided to all referenced containers.
ModuleFederationPlugin
(high level)This plugin combines ContainerPlugin
and ContainerReferencePlugin
. Overrides and overridables are combined into a single list of specified shared modules.
config.context
.requiredVersion
by default.requiredVersion
from package.json at this position in the graph./
in shared will match all module requests with this prefix.Each page of a Single Page Application is exposed from container build in a separate build. The application shell is also a separate build referencing all pages as remote modules. This way each page can be separately deployed. The application shell is deployed when routes are updated or new routes are added. The application shell defines commonly used libraries as shared modules to avoid duplication of them in the page builds.
Many applications share a common components library which could be built as a container with each component exposed. Each application consumes components from the components library container. Changes to the components library can be separately deployed without the need to re-deploy all applications. The application automatically uses the up-to-date version of the components library.
The container interface supports get
and init
methods. init
is a async
compatible method that is called with one argument: the shared scope object. This object is used as a shared scope in the remote container and is filled with the provided modules from a host. It can be leveraged to connect remote containers to a host container dynamically at runtime.
init.js
(async () => {
// Initializes the shared scope. Fills it with known provided modules from this build and all remotes
await __webpack_init_sharing__("default");
const container = window.someContainer; // or get the container somewhere else
// Initialize the container, it may provide shared modules
await container.init(__webpack_share_scopes__.default);
const module = await container.get("./module");
})();
The container tries to provide shared modules, but if the shared module has already been used, a warning and the provided shared module will be ignored. The container might still use it as a fallback.
This way you could dynamically load an A/B test which provides a different version of a shared module.
Example:
init.js
function loadComponent(scope, module) {
return async () => {
// Initializes the shared scope. Fills it with known provided modules from this build and all remotes
await __webpack_init_sharing__("default");
const container = window[scope]; // or get the container somewhere else
// Initialize the container, it may provide shared modules
await container.init(__webpack_share_scopes__.default);
const factory = await window[scope].get(module);
const Module = factory();
return Module;
};
}
loadComponent("abtests", "test123");
Uncaught Error: Shared module is not available for eager consumption
The application is eagerly executing an application which is operating as an omnidirectional host. There are options to choose from:
You can set the dependency as eager inside the advanced API of Module Federation, which doesn’t put the modules in an async chunk, but provides them synchronously. This allows us to use these shared modules in the initial chunk. But be careful as all provided and fallback modules will always be downloaded. It’s recommended to provide it only at one point of your application, e.g. the shell.
We strongly recommend using an asynchronous boundary. It will split out the initialization code of a larger chunk to avoid any additional round trips and improve performance in general.
For example, your entry looked like this:
index.js
import React from "react";
import ReactDOM from "react-dom";
import App from "./App";
ReactDOM.render(<App />, document.getElementById("root"));
Let's create bootstrap.js
file and move contents of the entry into it, and import that bootstrap into the entry:
index.js
+ import('./bootstrap');
- import React from 'react';
- import ReactDOM from 'react-dom';
- import App from './App';
- ReactDOM.render(<App />, document.getElementById('root'));
bootstrap.js
+ import React from 'react';
+ import ReactDOM from 'react-dom';
+ import App from './App';
+ ReactDOM.render(<App />, document.getElementById('root'));
This method works but can have limitations or drawbacks.
Setting eager: true
for dependency via the ModuleFederationPlugin
webpack.config.js
// ...
new ModuleFederationPlugin({
shared: {
...deps,
react: {
eager: true,
},
},
});
Uncaught Error: Module "./Button" does not exist in container.
It likely does not say "./Button"
, but the error message will look similar. This issue is typically seen if you are upgrading from webpack beta.16 to webpack beta.17.
Within ModuleFederationPlugin. Change the exposes from:
new ModuleFederationPlugin({
exposes: {
- 'Button': './src/Button'
+ './Button':'./src/Button'
}
});
Uncaught TypeError: fn is not a function
You are likely missing the remote container, make sure its added. If you have the container loaded for the remote you are trying to consume, but still see this error, add the host container's remote container file to the HTML as well.