Micro-frontends with Module Federation

Building Scalable and Modular Web Applications

Luis Cameroon
6 min readSep 8, 2023

Introduction

As web applications grow in complexity, maintaining a monolithic codebase becomes challenging. Micro-frontends offer a solution by breaking down an application into smaller, self-contained modules. Integrating Micro-frontends with Module Federation allows developers to build scalable, modular, and efficient web applications. In this article, we will explore the concept of Micro-frontends with Module Federation and demonstrate its implementation using React and Vite.

What are Micro-frontends?

Micro-frontends are an architectural approach that breaks down a complex web application into smaller, independently deployable and scalable modules. Each module represents a self-contained feature or functionality, developed by different teams or using different technologies. This approach enables teams to work autonomously, promotes code reusability, and simplifies maintenance.

Introducing Module Federation

Module Federation is a powerful feature provided by Webpack, Vite/Rollup, and Snowpack, allowing the dynamic loading and sharing of JavaScript modules between applications. By combining Micro-frontends with Module Federation, we can create a distributed architecture where modules can be loaded on-demand, sharing resources and dependencies as required.

Module Federation introduces several terminologies that are essential to understand how modules interact and communicate with each other. Let’s delve into the key terminologies: Host, Bidirectional Hosts and Remote.

Module Federation Terminologies
  1. Host: In Module Federation, the Host is the application that consumes and integrates remote modules exposed by other applications. It acts as the main container for the entire application and is responsible for orchestrating the integration of remote modules. The Host application can dynamically load and render remote modules, allowing seamless integration and composition of functionality from multiple sources.
  2. Bidirectional Hosts: Bidirectional Hosts refer to Host applications that can both consume and expose modules. In these scenarios, the Host application functions as both the consumer and provider of modules. This means that the Host application can consume remote modules from other applications while also exposing its own modules for consumption by other Host or Remote applications. Bidirectional Hosts enable a more cohesive and collaborative approach, allowing modules to be shared across different applications in a bidirectional manner.
  3. Remote: A Remote module is a standalone application or module that exposes certain components, functions, or assets for consumption by other applications. It is developed and deployed independently from the Host application. Remote modules can be hosted on different domains or served from separate servers. The Host application can consume and render the exposed components or utilize the functionality provided by the Remote module.

The interaction between these terminologies can be summarized as follows: The Host application loads the Remote modules at runtime, using Module Federation’s dynamic loading capabilities. The Host can consume the components or utilize the functionality provided by the Remote module, integrating it seamlessly into the Host application. This allows for the composition of different modules from multiple sources, enabling a modular and scalable frontend architecture.

It is important to note that the concept of Host, Remote, and Bidirectional Hosts is specific to Module Federation and may vary in different implementations or frameworks. However, the underlying principle remains the same, where the Host consumes and integrates modules from Remote sources to create a cohesive application experience.

Benefits of Micro-frontends with Module Federation

There are many benefits to using micro-frontends with module federation, including:

  • Scalability: Micro-frontends allow teams to scale their applications effortlessly by adding or removing modules as needed. With Module Federation, modules can be easily composed and reused across different applications, leading to a more scalable and modular architecture.
  • Independent Development and Deployment: Micro-frontends enable teams to work independently on their modules, reducing dependencies and enabling faster development cycles. With Module Federation, teams can deploy their modules separately, allowing for independent releases and reducing the risk of affecting other parts of the application.
  • Improved Team Autonomy: Micro-frontends promote team autonomy, as each team can focus on a specific module without being dependent on other teams. Module Federation further enhances this autonomy by enabling teams to choose their own technology stack, making it easier to adopt new technologies or frameworks within their modules.
  • Enhanced Performance: Module Federation enables lazy-loading of modules, which improves initial load times by loading only the necessary modules on demand. Additionally, shared dependencies are loaded once and cached, reducing duplicate code and improving overall performance.
  • Code Reusability: With Module Federation, modules can expose components, functions, or assets that can be consumed by other modules. This promotes code reusability, reduces duplication, and encourages the creation of shared libraries or UI components that can be utilized across multiple micro-frontends.

Implementation with React and Vite

To demonstrate the implementation of Micro-frontends with Module Federation, we will use React, a popular JavaScript library for building user interfaces, and Vite, a fast development server and build tool.

Step 1: Set Up the Main Application

  • Create a new React application using Vite:
  • Install the necessary dependencies:

Step 2: Configure Module Federation

  • In the main application’s vite.config.js file, import the necessary dependencies:
  • Add the Module Federation plugin configuration to the plugins array:

Step 3: Create Micro-frontends

  • Create separate React applications for each micro-frontend:
    npx create-vite module1 — template react
    and
    npx create-vite module2 — template react
  • Repeat "Step 2" for each micro-frontend’s vite.config.js file, with different names and URLs.

Step 4: Load Micro-frontends in the Main Application

  • In the main application’s React component, import the necessary dependencies:
  • Import and render the micro-frontend modules:

Step 5: Build and Run the Applications

  • Build the micro-frontends: npm run build in each micro-frontend’s directory.
  • Build the main application: npm run build in the main application’s directory
  • Serve the applications: npm run serve in each application’s directory.
Micro-Frontend Architecture

Implementation Challenges

There are some challenges to using micro-frontends with Module Federation:

  • Cohesion: It can be difficult to maintain the cohesion of a micro-frontend application. This is because each micro-frontend is a separate unit, and it can be easy for them to become decoupled from each other.
  • Communication: It can be difficult to communicate between different micro-frontends. This is because each micro-frontend is a separate unit, and they may not have a shared understanding of the application’s state.
  • Deployment: It can be complex to deploy micro-frontends. This is because each micro-frontend is a separate unit, and they may need to be deployed to different servers.

Best Practices for Micro-frontends

Here are some best practices for micro-frontends with Module Federation:

  • Keep micro-frontends small and focused: Micro-frontends should be small and focused, so that they are easy to understand and maintain.
  • Use a common API: Micro-frontends should use a common API, so that they can communicate with each other.
  • Use a centralized configuration: Micro-frontends should use a centralized configuration, so that they can share settings and data.

Micro-frontends should be version controlled, so that changes can be tracked and managed.

Conclusion

Micro-frontends with Module Federation offer a scalable and modular approach to building web applications. By breaking down applications into independent modules, developers can work autonomously and leverage the power of dynamic module loading. In this article, we explored the concept of Micro-frontends with Module Federation and demonstrated its implementation using React and Vite. Start experimenting with this architecture to build flexible, maintainable, and efficient web applications.

--

--