This plugin enables Module Federation on Next.js
- next ^11.x.x || ^12.x.x
- Client side only, SSR has a PR open. Help needed
I highly recommend referencing this application which takes advantage of the best capabilities: https://github.com/module-federation/module-federation-examples
SSR support for federated applications is much harder, as such - it utilizes a different licensing model. If you need SSR support, consider this package instead - it does everything that nextjs-mf does, and them some. https://app.privjs.com/buy/packageDetail?pkg=@module-federation/nextjs-ssr
There is a pull request moving SSR into this repo and package - but it is not ready yet.
Under the hood we share some next internals automatically You do not need to share these packages, sharing next internals yourself will cause errors.
const DEFAULT_SHARE_SCOPE = {
react: {
singleton: true,
requiredVersion: false,
},
'react/': {
singleton: true,
requiredVersion: false,
},
'react-dom': {
singleton: true,
requiredVersion: false,
},
'next/dynamic': {
requiredVersion: false,
singleton: true,
},
'styled-jsx': {
requiredVersion: false,
singleton: true,
},
'next/link': {
requiredVersion: false,
singleton: true,
},
'next/router': {
requiredVersion: false,
singleton: true,
},
'next/script': {
requiredVersion: false,
singleton: true,
},
'next/head': {
requiredVersion: false,
singleton: true,
},
};
const SampleComponent = dynamic(() => import('next2/sampleComponent'), {
ssr: false,
});
If you want support for sync imports. It is possible in next@12 as long as there is an async boundary.
See the implementation here: https://github.com/module-federation/module-federation-examples/tree/master/nextjs/home/pages
With async boundary installed at the page level. You can then do the following
if (process.browser) {
const SomeHook = require('next2/someHook');
}
// if client only file
import SomeComponent from 'next2/someComponent';
Make sure you are using mini-css-extract-plugin@2
- version 2 supports resolving assets through publicPath:'auto'
You can see it in action here: https://github.com/module-federation/module-federation-examples/tree/master/nextjs
const SampleComponent = dynamic(() => import('next2/sampleComponent'), {
ssr: false,
});
If you want support for sync imports. It is possible in next@12 as long as there is an async boundary.
See the implementation here: https://github.com/module-federation/module-federation-examples/tree/master/nextjs/home/pages
With async boundary installed at the page level. You can then do the following
if (process.browser) {
const SomeHook = require('next2/someHook');
}
// if client only file
import SomeComponent from 'next2/someComponent';
Make sure you are using mini-css-extract-plugin@2
- version 2 supports resolving assets through publicPath:'auto'
This plugin works exactly like ModuleFederationPlugin, use it as you'd normally. Note that we already share react and next stuff for you automatically.
Also NextFederationPlugin has own optional argument extraOptions
where you can unlock additional features of this plugin:
new NextFederationPlugin({
name: ...,
filename: ...,
remotes: ...,
exposes: ...,
shared: ...,
extraOptions: {
exposePages: true, // `false` by default
enableImageLoaderFix: true, // `false` by default
enableUrlLoaderFix: true, // `false` by default
},
});
exposePages
– exposes automatically all nextjs pages for you and theirs./pages-map
.enableImageLoaderFix
– adds public hostname to all assets bundled bynextjs-image-loader
. So if you serve remoteEntry fromhttp://example.com
then all bundled assets will get this hostname in runtime. It's something like Base URL in HTML but for federated modules.enableUrlLoaderFix
– adds public hostname to all assets bundled byurl-loader
.
You can see it in action here: module-federation/module-federation-examples#2147
- Use
NextFederationPlugin
in yournext.config.js
of the app that you wish to expose modules from. We'll call this "next2".
// next.config.js
const NextFederationPlugin = require('@module-federation/nextjs-mf/NextFederationPlugin');
module.exports = {
webpack(config, options) {
if (!options.isServer) {
config.plugins.push(
new NextFederationPlugin({
name: 'next2',
remotes: {
next1: `next1@http://localhost:3001/_next/static/chunks/remoteEntry.js`,
},
filename: 'static/chunks/remoteEntry.js',
exposes: {
'./title': './components/exposedTitle.js',
'./checkout': './pages/checkout',
'./pages-map': './pages-map.js',
},
shared: {
// whatever else
},
})
);
}
return config;
},
};
// _app.js or some other file in as high up in the app (like next's new layouts)
// this ensures various parts of next.js are imported and "used" somewhere so that they wont be tree shaken out
import '@module-federation/nextjs-mf/lib/include-defaults';
- For the consuming application, we'll call it "next1", add an instance of the ModuleFederationPlugin to your webpack config, and ensure you have a custom Next.js App
pages/_app.js
(or.tsx
): Inside that _app.js or layout.js file, ensure you importinclude-defaults
file
// next.config.js
const NextFederationPlugin = require('@module-federation/nextjs-mf/NextFederationPlugin');
module.exports = {
webpack(config, options) {
if (!options.isServer) {
config.plugins.push(
new NextFederationPlugin({
name: 'next1',
remotes: {
next2: `next2@http://localhost:3000/_next/static/chunks/remoteEntry.js`,
},
})
);
}
return config;
},
};
// _app.js or some other file in as high up in the app (like next's new layouts)
// this ensures various parts of next.js are imported and "used" somewhere so that they wont be tree shaken out
import '@module-federation/nextjs-mf/lib/include-defaults';
- Use next/dynamic or low level api to import remotes.
import dynamic from 'next/dynamic';
const SampleComponent = dynamic(
() => window.next2.get('./sampleComponent').then((factory) => factory()),
{
ssr: false,
}
);
// or
const SampleComponent = dynamic(() => import('next2/sampleComponent'), {
ssr: false,
});
Ive added a util for dynamic chunk loading, in the event you need to load remote containers dynamically.
import { injectScript } from '@module-federation/nextjs-mf/lib/utils';
// if i have remotes in my federation plugin, i can pass the name of the remote
injectScript('home').then((remoteContainer) => {
remoteContainer.get('./exposedModule');
});
// if i want to load a custom remote not known at build time.
injectScript({
global: 'home',
url: 'http://somthing.com/remoteEntry.js',
}).then((remoteContainer) => {
remoteContainer.get('./exposedModule');
});
If you have any questions or need to report a bug Reach me on Twitter @ScriptedAlchemy
Or join this discussion thread: https://github.com/module-federation/module-federation-examples/discussions/978