theme | layout | highlighter | lineNumbers |
---|---|---|---|
slidev-theme-nearform |
default |
shiki |
false |
The concept of micro frontends was first mentioned circa 2016 as an extrapolation of microservices to the frontend realm. Since then, it has become a go-to strategy for splitting a monolithic frontend codebase into smaller pieces that can be owned, worked on, and deployed independently.
Module federation is one of the most popular approaches for implementing micro frontend architecture on either the client or server side.
With module federation, each micro frontend is treated as a standalone module that can be developed, deployed, and versioned independently, which allows those modules to share and consume each other's functionality, resources, and components at runtime, improving collaboration and reusability.
Started as a Webpack Plugin, module federation has now evolved into a general concept adopted by other bundlers and frameworks including Vite and Rollup, for example.
-- Runtime Web Components: each micro frontend is mounted at a custom HTML element, and the container performs instantiation.
-- Runtime Javascript integration: somewhat similar to both the previous approach and module federation, this one includes each micro frontend onto the page using a <script>
tag; the container application becomes an entry point, decides which micro frontend to be mounted, and calls the relevant function telling the micro frontend when and where to get rendered; each build file can be deployed independently.
-- iFrames: this approach is about rendering various micro frontends in separate iframes and composing those via a container application; the most obvious benefit of this approach is complete decoupling of the application components; however, this approach also has some substantial cons like composition complexity and high potential for performance issues.
-- edge-side Composition: edge-side composition assumes that micro frontends are assembled by the edge using the Edge Side Include (ESI) specification; the biggest cons are the fact that support differs depending on the CDN, and each vendor (Akamai, CloudFlare, Fastly, etc.) has its own features and limitations.
-- dedicated Frameworks for MFE Composition: one of the easiest ways to implement micro frontend architecture is to use a dedicated framework that takes care of all the ins and outs and lets you focus on the application code; some notable examples of such frameworks are listed below:
-- client-side: SingleSPA, Qiankun (based on SingleSPA), Luigi,
-- server-side: Ara, Bit, Open Components, Piral.
-- A host is an application that includes the initial chunks of our code, the ones that will be used to bootstrap our container; the concept of module federation assumes that some components that container will render are just being referenced to a remote and not the initial bundle, which allows for smaller bundle sizes and shorter initial load times.
-- A remote is a module that is being consumed by the host, and it can be either shared components or common dependencies to be used by different hosts.
-- A bidirectional host is both a host and a remote, consuming other remotes and providing some code to other hosts.
As mentioned earlier, initially, module federation was implemented as a plugin introduced in Webpack 5. To set up Module Federation in Webpack, you need to define the federated modules in your Webpack configuration files, specify the remote entry points and expose specific modules (aka “remotes”) that you want to share with other applications. The remote entry points represent the Webpack builds that expose modules for consumption.
In the consuming application's (aka “host”) Webpack configuration, you define which federated modules you want to consume. You specify the remote entry points and the modules you want to import from those remotes. When you build and run your applications, Webpack dynamically loads the federated modules at runtime. It fetches the remote entry points, resolves the requested modules, and injects them into the consuming application. This process allows you to share code between applications without physically bundling everything together.
Note that as an application has multiple dependencies, a host can also have multiple remotes.
-- Node LTS
-- npm >= 7.24.2
git clone https://github.com/nearform/the-micro-frontends-workshop
cd the-micro-frontends-workshop && npm install
-- This workshop is made up of multiple, incremental modules (aka exercises).
-- Each module builds on top of the previous one.
-- At each step, you are asked to add features and solve problems.
-- You will find a template to base your solution on in the src/step-{n}-{name}
folder.
-- You will find the solution to each step in the src/solutions/step-{n}-{name}
folder.
-- cd src/step-{n}-{name}
-- Check out README.md
-- Install dependencies.
-- Start the development server(s).
cd src/step-01-setting-up-remote
npm install
npm start
-- There are a few key steps that need to be taken in order to expose a module for remote consumption (federation).
-- In this example, we are going to demonstrate these steps in a basic React app since any Webpack based application that supports MF will have a similar flow for enabling this feature.
-- This application on its own will work just as any other React application but it will have the ability to expose a specific part of it as a remote which we will be able to consume inside of another application in later steps of the workshop.
In order to enable module federation, we need to import ModuleFederationPlugin
from Webpack on top of this file.
const { ModuleFederationPlugin } = require('webpack').container
This plugin needs to be instantiated and configured inside the plugins section of the configuration file. The most basic configuration requires name
, fileName
and exposes
key values.
// ....
plugins: [
new ModuleFederationPlugin({
name: 'remoteAppName',
filename: 'remoteEntry.js',
exposes: { './ComponentName': './src/components/ComponentName' }
}),
],
// ...
name: 'remoteAppName',
filename: 'remoteEntry.js',
exposes: { './ComponentName': './src/components/ComponentName' }
// ...
-- name
is where we define a name to distinguish modules. This value will be used by a consumer application when defining remotes inside of it.
-- filename
can be any value, and it will be an entry point for exposed/shared modules. remoteEntry.js
is most commonly/conventionally used for this purpose.
-- In the exposes
object we define components for remote consumption. The key name should always be in form of ./ComponentName
in any application that relies on Webpack's ModuleFederationPlugin and the value should be the component's relative path to the webpack.config.js.
We need index.js
to be the app's entry point but inside of it we need to import another file, bootstrap.js
(named this way by convention) that renders the entire app. This file contains what index.js
would normally contain in a React app including createRoot()
method. To allow Module Federation we need to import it dynamically using import()
inside of index.js
. This extra layer is needed for Webpack to load all of the imports necessary to render the remote app (failure to provide it will result in an error saying something like Shared module is not available for eager consumption
).
//src/bootstrap.js
// ... React Code
const root = createRoot(document.getElementById('root'));
root.render(<App />);
//src/index.js
import('bootstrap.js')
In src
folder of the provided basic React application:
-- Import the Nav
component from src/components
folder, display it inside the App.js
file under the title and pass some links as props to it; the links props should be an array of objects like this:
const links = [
{ url: '/', label: 'Home' },
{ url: 'https://react.dev/', label: 'Learn more about React.js' },
{ url: 'https://webpack.js.org/concepts/module-federation/', label: 'Learn more about Module Federation' }
]
-- Render the entire application via the createRoot()
method inside the bootstrap.js
file and import that file in the index.js
file using the import
statement.
In webpack.config.js
file:
-- Import ModuleFederationPlugin
plugin from Webpack's container
object.
-- In exported modules, instantiate new ModuleFederationPlugin
.
-- Pass a configuration object and define values for name
, filename
, and exposes
keys. Remember that filename
uses a naming convention, and exposes
refers to the element that we want to expose.
// src/App.js
import React from 'react'
import Nav from './components/Nav'
const links = [
{ url: '/', label: 'Home' },
{ url: 'https://react.dev/', label: 'Learn more about React.js' },
{ url: 'https://webpack.js.org/concepts/module-federation/', label: 'Learn more about Module Federation' }
]
const App = () => (
<div>
<h1>Basic Remote Application</h1>
<Nav links={links} />
</div>
)
export default App
// src/bootstrap.js
import App from './App';
import React from 'react';
import { createRoot } from 'react-dom/client';
const root = createRoot(document.getElementById('root'));
root.render(<App />);
// src/index.js
import('./bootstrap')
// webpack.config.js
// .....
const { ModuleFederationPlugin } = require('webpack').container
// .....
module.exports = {
// .....
plugins: [
new ModuleFederationPlugin({
name: 'remote',
filename: 'remoteEntry.js',
exposes: {
'./Nav': './src/components/Nav',
},
}),
// .....
],
}
http://localhost:3002
And
http://localhost:3002/remoteEntry.js
For the first one, you should see a page that displays the Basic Remote Application
title with your Nav
component (a list of links).
For the second one, you should see a script
that exposes our component for remote consumption.
-- In this step we are going to set up an application as the host application. We are going to use a Next.js application for this purpose which will demonstrate Module Federation's ability to work with different frameworks.
-- In order to start consuming modules, we need to configure the plugin's remotes
parameter which can take multiple remotes.
remotes: {
remoteApp1: 'someRemoteApp@http://localhost:3002/remoteEntry.js',
remoteApp2: 'anotherRemoteApp@http://localhost:3003/remoteEntry.js',
},
-- The key names are going to be used later in the host application code when importing remote modules.
-- The value's prefix (the string before @http...
) must match remote application's name
parameter defined when instantiating its Module Federation plugin (see slide 13).
-- In order to import/consume a module into a plain webpack host applications such as any React.js based app, we need to use the import keyword (dynamic import) and refer to the remote component based on the key
name from one of the previous steps where we set up remotes in the config file. We used someRemoteApp
and anotherRemoteApp
in our example.
-- Our example import statement could look something like this:
import SomeRemoteComponent from "someRemoteApp/SomeRemoteComponent"
-- In Next.js apps in CSR (client side rendering) scenario we have to rely on dynamic()
method to dynamically import the remote component. We also need to set ssr
to false
because we will not be using server side rendering in our example.
import dynamic from 'next/dynamic'
const SomeRemoteComponent = dynamic(() => import('someRemoteApp/SomeRemoteComponen'), { ssr: false })
-- To enable Module Federation in Next.js we need to import NextFederationPlugin
in the next.config.js
file since ModuleFederationPlugin
and webpack.config.js
are not used in Next.js apps.
-- Note that NextFederationPlugin
has to be installed separately with npm install @module-federation/nextjs-mf
.
-- filename
property needs to be set using the static/chunks/{fileName}.js
pattern.
// next.config.js
const { NextFederationPlugin } = require('@module-federation/nextjs-mf');
// ...
config.plugins.push(
new NextFederationPlugin({
name: 'host',
remotes: {
remote: 'remote@http://localhost:3002/remoteEntry.js',
},
filename: 'static/chunks/remoteEntry.js'
}),
);
// ...
-- Inside nextApp/pages/index.js
file import a simple LayoutBox
component from nextApp/components/nextjs-layout-box.js
file. Notice that it takes children
props.
-- Display the LayoutBox
component below the <Head>
section, place the remote component Nav
inside of it and pass links props to it similar to Step 1.
-- Configure the application inside next.config.js
file so it uses NextFederationPlugin
.
-- We added a react-app
in this step and exposed the Nav
component (similar to step 1). Configure nextApp
so it consumes that component from port 8080
.
// next.config.js
const NextFederationPlugin = require('@module-federation/nextjs-mf');
module.exports = {
webpack(config) {
config.plugins.push(
new NextFederationPlugin({
name: 'nextApp',
remotes: {
remote: 'reactApp@http://localhost:8080/remoteEntry.js',
},
filename: 'static/chunks/remoteEntry.js'
})
)
return config
},
// your original next.config.js export
reactStrictMode: true,
}
//pages/index.js
import dynamic from 'next/dynamic'
import LayoutBox from '../components/nextjs-layout-box'
const Nav = dynamic(() => import('remote/Nav'), { ssr: false })
const links = [
{ url: '/', label: 'Home' },
{ url: 'https://nextjs.org/', label: 'Learn more about Next.js' },
]
export default function Home() {
return (
<div className={styles.container}>
<Head>
{/* Head example code */}
</Head>
<LayoutBox>
<Nav links={links} />
</LayoutBox>
</div>
)
}
In your terminal from the src/step-02-setting-up-host
folder run:
npm i && npm start
This will start both remote and host applications at the same time on ports 8081 and 8080 respectively.
http://localhost:8080
You should see the LayoutBox
component acting as a wrapper and the remote Nav
component displayed inside of it (see the screenshot on the next slide).
In this step we are going to demonstrate Module Federation's bidirectional ability to share modules between multiple apps that can act both as the host and the remote at the same time by putting together what we learned in the previous two steps.
-- Inside of Next.js
application find LayoutBox
component in nextApp/components/nextjs-layout-box.js
file as well as Table
component in nextApp/components/nextjs-table.js
.
-- Expose them as remotes in next.config.js
file using the same syntax/pattern as in Step 1.
-- Similarly, inside of React.js
find the Nav
component in src/components/Nav.jsx
as well as Title
component in src/components/Title.jsx
file.
-- Expose both components as remotes inside of webpack.config.js
file using the same syntax/pattern as in Step 1.
-- Import the LayoutBox
and the Table
component into React.js
app by configuring webpack.config.js
file using the same syntax from step 2.
-- Import the Nav
and the Title
component into Next.js
app by configuring next.config.js
file using the same syntax from step 2.
-- Import and place the LayoutBox
and the Table
component in React app's src/App.jsx
file. Make sure that the entire content is wrapped in LayoutBox
. Note that Table
takes data
prop so feel free to pass some data that will be unique for this application.
-- Import and place Nav
and the Title
component in Next app's pages/index.js
file. Pass some links
and title
props to these components that will be unique for this application.
// next.config.js
//...
new NextFederationPlugin({
name: 'nextApp',
remotes: {
remote: 'reactApp@http://localhost:8080/remoteEntry.js',
},
exposes: {
'./nextjs-layout-box': './components/nextjs-layout-box.js',
'./nextjs-table': './components/nextjs-table.js'
},
// ...
})
// webpack.config.js
// ...
new ModuleFederationPlugin({
name: 'reactApp',
filename: 'remoteEntry.js',
remotes: {
remote: 'nextApp@http://localhost:8081/_next/static/chunks/remoteEntry.js',
},
exposes: {
'./Nav': './src/components/Nav',
'./Title': './src/components/Title'
},
// ...
)}
// index.js
...
...
import dynamic from 'next/dynamic'
import LayoutBox from '../components/nextjs-layout-box'
import Table from '../components/nextjs-table'
const Nav = dynamic(() => import('remote/Nav'), { ssr: false })
const Title = dynamic(() => import('remote/Title'), { ssr: false })
export default function Home() {
return (
<div className={styles.container}>
<Head>
{/* Head example code */}
</Head>
<LayoutBox>
<Title>This is Next.js App</Title>
<Nav links={links} />
<Image width="200" src={Logo} alt="logo" />
<Table data={tableData} />
</LayoutBox>
</div>
)
}
// App.jsx
import LayoutBox from 'remote/nextjs-layout-box'
import Table from 'remote/nextjs-table'
...
...
function App() {
return (
<LayoutBox>
<Title>This is React.js App</Title>
<Nav links={links} />
<img style={{maxWidth: "200px", margin: "50px auto"}} src={Logo} alt="logo" />
<Table data={tableData} />
</LayoutBox>
)
}
export default App
-- From your browser, visit:
http://localhost:8080
You should see the React.js
app wrapped in LayoutBox
consumed from Next.js
app as well as the Nav
and Title
local components and another remote Table
component.
-- From your browser, visit:
http://localhost:8081
You should see the Next.js
app wrapped in its local LayoutBox
as well the two remote Nav
and Title
components and another local Table
component.
-- shared dependencies refer to the libraries, frameworks, or modules that are required by multiple federated modules to function properly. By sharing these dependencies, modules can avoid duplication and ensure consistency and compatibility.
-- Shared dependencies typically include runtime libraries, such as React or Angular, along with any additional utility libraries or common components that are needed by the federated modules. They are typically declared and managed in a shared configuration file, allowing modules to access and utilize them seamlessly.
Please follow the guidelines from Zack Jackson (inventor & co-creator of module federation):
-- Sharing should be done with care - since shared modules cannot be tree-shaken.
-- If you need a singleton (like things that depend on React context), then it must be shared.
-- Sharing all dependencies can lead to larger bundles so its best to consider case by case.
-- shared (object | [string]): an object or an array containing a list of dependency names that can be shared across the federated modules.
-- eager (boolean): specifies whether the dependency will be eagerly loaded and provided to other federated modules as soon as the host app starts (otherwise will be loaded lazily when first requested by the federated app).
-- singleton (boolean): whether the dependency will be considered a singleton, which means that only a single instance of it is supposed to be shared across all the federated modules.
-- requiredVersion (string): specifies the required version of the dependency, which makes any incompatible version loaded separately (not shared); note that if the singleton
property is set to true
, setting requiredVersion
will raise a warning in case of a conflict.
// webpack.config.js
const path = require('path');
const { ModuleFederationPlugin } = require('webpack').container;
const packageJsonDependencies = require('./package.json').dependencies
module.exports = {
entry: './src/index.js',
output: {
filename: 'bundle.js',
path: path.resolve(__dirname, 'dist'),
},
plugins: [
new ModuleFederationPlugin({
name: 'MyApp',
filename: 'remoteEntry.js',
exposes: { './Button': './src/components/Button' },
shared: {
react: { singleton: true, requiredVersion: packageJsonDependencies.react },
'react-dom': { singleton: true, requiredVersion: packageJsonDependencies['react-dom'] },
},
}),
],
};
In this step, we are going to demonstrate sharing dependencies (React and React DOM) across federated modules.
-- Modify Webpack config of the React app to have React and React DOM as shared dependencies.
-- Modify Next.js config of the Next.js app to have React and React DOM as shared dependencies.
-- Set the versions of these shared dependencies to be read from the respective package.json
files.
-- Make sure the dependencies are being shared as singletons.
-- Try changing the version of React and React DOM in the package.json
file of the React app to a previous one, and make sure you get a warning about version mismatch in the console.
// next.config.js
const packageJsonDependencies = require('./package.json').dependencies
// ...
new NextFederationPlugin({
// ...
shared: {
react: {
requiredVersion:
packageJsonDependencies.react,
singleton: true,
},
'react-dom': {
requiredVersion:
packageJsonDependencies['react-dom'],
singleton: true,
},
},
extraOptions: {
skipSharingNextInternals: true,
}
})
// ...
// webpack.config.js
const packageJsonDependencies = require('./package.json').dependencies
// ...
new ModuleFederationPlugin({
// ...
shared: {
react: {
requiredVersion:
packageJsonDependencies.react,
singleton: true,
},
'react-dom': {
requiredVersion:
packageJsonDependencies['react-dom'],
singleton: true,
},
},
}),
// ...
)}
When it comes to TypeScript applications, the most common problem with using external libraries (which can be federated remote modules) is that not all of them provide TypeScript types with the original code. In the context of module federation, this problem is aggravated by the fact that Webpack only loads resources from the federated module at runtime, TypeScript, however, needs those during compilation. In this case you have the following options:
-- @module-federation/native-federation-typescript,
-- @module-federation/typescript,
-- packaging your types for distribution via a package registry (e.g., npm),
-- referencing types across monorepo (if possible).
Note that TypeScript plugins are the easiest way to handle federated types. They fetch the types at compile-time and store within the project in order to make them available to tsc whenever it’s needed.
-- React Micro Frontends with Module Federation
-- Micro Frontends by Cam Jackson
-- @module-federation/typescript
-- Bundler-Agnostic Plugins to Share Federated Components for Testing Purposes by NearForm
-- Bundler-Agnostic Plugins to Share Federated Types by NearForm