Vite plugin for Module Federation

April 23, 2026 ยท View on GitHub

npm

Reason why ๐Ÿค”

Microservices nowadays is a well-known concept and maybe you are using it in your current company. Do you know that now you can apply similar ideas on the Frontend? With Module Federation you can load separately compiled and deployed code into a unique application. This plugin makes Module Federation work together with Vite.

Working implementations

Examples live in gioboa/module-federation-vite-examples:

ExampleHostRemoteFramework
Alpinealpine-hostalpine-remoteAlpine.js
Angularangular-hostangular-remoteAngular
Litlit-hostlit-remoteLit
Nuxtnuxt-hostnuxt-remoteNuxt 4
Preactpreact-hostpreact-remotePreact 10
Reactreact-hostreact-remoteReact 19
Solidsolid-hostsolid-remoteSolid
Sveltesvelte-hostsvelte-remoteSvelte 5
TanStacktanstack-hosttanstack-remoteTanStack Router + React 19
Vinextvinext-hostvinext-remoteVinext + Next 16 + React 19
Vuevue-hostvue-remoteVue 3

Try this crazy example with all these bundlers together

pnpm install
pnpm run build
pnpm run multi-example

Getting started ๐Ÿš€

https://module-federation.io/guide/basic/webpack.html

With @module-federation/vite, the process becomes delightfully simple, you will only find the differences from a normal Vite configuration.

This example is with Vue.js
The @module-federation/vite configuration remains the same for different frameworks.

The Remote Application configuration

file: remote/vite.config.ts

import { defineConfig } from 'vite';
import { federation } from '@module-federation/vite'; ๐Ÿ‘ˆ

export default defineConfig({
  [...]
  plugins: [
    [...]
    federation({ ๐Ÿ‘ˆ
      name: "remote",
      filename: "remoteEntry.js",
      // optional: additional "var" remoteEntry file
      // needed only for legacy hosts with "var" usage (remote.type = 'var')
      varFilename: "varRemoteEntry.js",
      exposes: {
        "./remote-app": "./src/App.vue",
      },
      shared: ["vue"],
    }),
  ],
  server: {
    origin: "http://localhost:{Your port}"
  },
  [...]
});

In this remote app configuration, we define a remoteEntry.js file that will expose the App component. The shared property ensures that both host and remote applications use the same vue library.

The Host Application configuration

file host/vite.config.ts

import { defineConfig } from 'vite';
import { federation } from '@module-federation/vite'; ๐Ÿ‘ˆ

export default defineConfig({
  [...]
  plugins: [
    [...]
    federation({ ๐Ÿ‘ˆ
      name: "host",
      remotes: {
        remote: {
          type: "module", // type "var" (default) for vite remote is supported with remote's `varFilename` option
          name: "remote",
          entry: "https://[...]/remoteEntry.js",
          entryGlobalName: "remote",
          shareScope: "default",
        },
      },
      filename: "remoteEntry.js",
      shared: ["vue"],
      // Optional parameter that controls where the host initialization script is injected.
      // By default, it is injected into the index.html file.
      // You can set this to "entry" to inject it into the entry script instead.
      // Useful if your application does not load from index.html.
      hostInitInjectLocation: "html", // or "entry"
      // Controls whether all CSS assets from the bundle should be added to every exposed module.
      // When false (default), the plugin will not process any CSS assets.
      // When true, all CSS assets are bundled into every exposed module.
      bundleAllCSS: false, // or true
      // Timeout for parsing modules in seconds.
      // Defaults to 10 seconds.
      moduleParseTimeout: 10,
      // Idle timeout for parsing modules in seconds. When set, the timeout
      // resets on every parsed module and only fires when there has been no
      // module activity for the configured duration. Prefer this over
      // moduleParseTimeout for large codebases where total build time may
      // exceed the fixed timeout value.
      moduleParseIdleTimeout: 10,
      // Controls whether module federation manifest artifacts are generated.
      // Type: boolean | object
      // - false/undefined: no manifest generated
      // - true: generates mf-manifest.json + mf-stats.json (default names)
      // - object: overrides fileName/filePath and asset analysis behavior
      manifest: {
        // Optional output file name for runtime manifest.
        // Default: "mf-manifest.json"
        fileName: "mf-manifest.json",
        // Optional output directory/path for both artifacts.
        // Example: "dist/" -> dist/mf-manifest.json + dist/mf-stats.json
        filePath: "dist/",
        // If true, skips asset analysis.
        // Effect: shared/exposes are omitted from manifest and assetAnalysis is omitted from stats.
        // It also disables the preload-helper patch used for remotes.
        // In serve for consumer-only apps, this defaults to true unless explicitly set.
        disableAssetsAnalyze: false,
      },
    }),
  ],
  server: {
    origin: "http://localhost:{Your port}"
  },
  [...]
});

The host app configuration specifies its name, the filename of its exposed remote entry remoteEntry.js, and importantly, the configuration of the remote application to load. You can specify the place the host initialization file is injected with the hostInitInjectLocation option, which is described in the example code above. The moduleParseTimeout option allows you to configure the maximum time to wait for module parsing during the build process. The moduleParseIdleTimeout option is an alternative that resets the timer on every parsed module. It only fires when there has been no module activity for the configured duration, making it suitable for large codebases where the total build time exceeds the fixed timeout.

Load the Remote App

In your host app, you can now import and use the remote app with defineAsyncComponent

file host/src/App.vue

<script setup lang="ts">
import { defineAsyncComponent } from "vue";
const RemoteMFE = defineAsyncComponent( ๐Ÿ‘ˆ
  () => import("remote/remote-app")
);
</script>

<template>
  <RemoteMFE v-if="!!RemoteMFE" /> ๐Ÿ‘ˆ
</template>

โš ๏ธ codeSplitting settings are controlled by the plugin

Do not set either build.rollupOptions.output.codeSplitting or build.rolldownOptions.output.codeSplitting to false with this plugin โ€” it will be automatically ignored.

codeSplitting.groups is also ignored because grouping shared-runtime chunks can break MF init order. Module Federation needs loadShare and runtimeInitStatus isolated into separate chunks for correct bootstrap behavior.

โš ๏ธ manualChunks is not supported

Do not use build.rollupOptions.output.manualChunks or build.rolldownOptions.output.manualChunks with this plugin โ€” it will be automatically ignored. The plugin manages the runtime chunk graph itself, and forcing custom chunk grouping can break Module Federation bootstrap order. The plugin injects the splits it needs so runtimeInitStatus and loadShare stay isolated.

So far so good ๐ŸŽ‰

Now you are ready to use Module Federation in Vite!