Configuring Next.js plugins

Next.js plugins are configured in the project's next.config.js file. Nx adds a its own plugin (withNx) to help the Next.js application understand workspace libraries, and other Nx-specific features. See below for an example of the withNx plugin that Nx adds by default.

// next.config.js // ... module.exports = withNx({ // Nx configuration goes here nx: { svgr: false, }, // Add Next.js configuration goes here });

This guide contains information on how to compose the Nx plugin with other plugins, such as @next/mdx. Note that Nx prior to version 16 is missing the compose utility from the @nx/next package, and a workaround will be provided for Nx 15 and prior.

Avoid next-compose-plugins

There is a popular package called next-compose-plugins that has not been maintained for over two years. This package does not correctly combine plugins in all situations. If you do use it, replace the package with Nx 16's composePlugins utility (see below).

Composing plugins using composePlugins utility (Nx 16 and later)

Since Nx 16, we provide a composePlugins utility function that helps users combine multiple Next.js plugins together.

// next.config.js const { composePlugins, withNx } = require('@nx/next'); /** * @type {import('@nx/next/plugins/with-nx').WithNxOptions} **/ const nextConfig = { nx: { // Set this to true if you would like to to use SVGR // See: https://github.com/gregberge/svgr svgr: false, }, // Add Next.js configuration here }; const plugins = [ // Add more Next.js plugins to this list if needed. withNx, ]; module.exports = composePlugins(...plugins)(nextConfig);

If you want to add additional plugins, say @next/mdx, you can add it to the plugins list.

const plugins = [ // Add more Next.js plugins to this list if needed. require('@next/mdx')(), withNx, ]; module.exports = composePlugins(...plugins)(nextConfig);

This the exported configuration will correctly call each plugin in order and apply their configuration changes to the final object. Note that composePlugins function will return an async function, so if you need to debug the configuration you can add a debug plugin as follows.

module.exports = composePlugins(...plugins, function debug(config) { // The debug plugin will be called last console.log({ config }); return config; })(nextConfig);

Manually composing plugins (Nx 15 and prior)

If you are not on Nx 16 and later versions, the composePlugins utility is not available. However, you can use the workaround below to use multiple plugins.

// next.config.js // ... /** * @type {import('@nx/next/plugins/with-nx').WithNxOptions} **/ const nextConfig = { // ... }; const plugins = [ // Your plugins exlcuding withNx ]; module.exports = async (phase, context) => { let updatedConfig = plugins.reduce((acc, fn) => fn(acc), nextConfig); // Apply the async function that `withNx` returns. updatedConfig = await withNx(updatedConfig)(phase, context); // If you have plugins that has to be added after Nx you can do that here. // For example, Sentry needs to be added last. updatedConfig = require('@sentry/nextjs')(updatedConfig, { silent: true }); return updatedConfig; };