Migrate to Netlify Today

Netlify announces the next evolution of Gatsby Cloud. Learn more

ContactSign Up
Community Plugin
View plugin on GitHub

Description

A Gatsby plugin to transform MDX files so they can be rendered without requiring a page to be created. An example use case is using MDX files as a basic CMS.

The plugin creates a Data<Type> node for every Mdx node that is created for files in the provided data directories in addition to registering an mdx resolver for these nodes to render markdown. This allows for the Data<Type> nodes to be queried as a basic CMS.

For example, using gatsby-transformer-mdx-data you could create a src/data/products folder to store product data in .mdx format and then query the products using allDataProduct and dataProduct.

Why?

Following some changes to gatsby-plugin-mdx, in order to support MDX v2 the html node was removed and it was no longer possible to render markdown outside of a page (see the v2 discussion, question about including HTML in feeds and complaint about not being able to render multiple markdown items).

I recently rewrote my website to use the latest version of Gatsby with Typescript and wanted to use .mdx files as a rudimentary CMS. Following a suggestion on one of the Gatsby GitHub issues, I managed to get it working for my own purposes and decided to make it a plugin both to help others and for my own learning.

DISCLAIMER: This is the first Gatsby plugin I have created and I know very little about the inner workings of bundlers and module resolution. This approach may have worked for me by sheer fluke!

Install

npm

npm i gatsby-transformer-mdx-data @mdx-js/react gatsby-plugin-mdx react

yarn

yarn add gatsby-transformer-mdx-data @mdx-js/react gatsby-plugin-mdx react

How to use

Changes to gatsby config

First, make sure you have gatsby-plugin-mdx and gatsby-source-filesystem correctly configured.

module.exports = {
  // ... more config
  plugins: [
    // ... more plugins
    'gatsby-plugin-mdx',
    {
        resolve: 'gatsby-source-filesystem',
        options: {
            name: 'products',
            path: './src/data/products',
        },
        __key: 'products',
    },
  ],
}

You should be able to see the nodes returned with the following GraphQL query.

{
  allMdx {
    nodes {
      internal {
        contentFilePath
      }
    }
  }
}

Then add the plugin to gatsby-config.js (or gatsby-config.ts) and set jsxRuntime:

const path = require('path')
module.exports = {
  // ... more config
  plugins: [
    // ... more plugins
    {
      resolve: `gatsby-transformer-mdx-data`,
      options: {
        // Provide a list of absolute file paths of where data files live
        dataDirs: [
            path.resolve(__dirname, 'src/data/products'),
            path.resolve(__dirname, 'src/data/stores'),
        ],
      },
    },
  ],
  jsxRuntime: 'automatic',
}

Create data files

e.g. src/data/products/product1.mdx

---
title: Some product
price: 1234
---

Some _markdown_ about my **product**

Query data

Once configured the data nodes can be queried using allData<TYPE> and data<TYPE> - e.g. using the above example you would have the additional queries of allDataProduct, dataProduct, allDataStore and dataStore. The standard filter and sorting arguments work.

Each data node has an mdx field to return the markdown ready to pass to the MDX component. Any data in “frontmatter” will be copied directly onto the node.

{
    products: allDataProduct(limit: 10) {
        nodes {
            id
            title
            mdx
            price
        }
    }
    bestProduct: dataProduct(name: { eq: "BestProduct" }) {
        id
        title
        mdx
        price
    }
}

Render data

Once you have the mdx data, pass the result to the MDX component to render markdown.

import Mdx from 'gatsby-transformer-mdx-data';

const ProductPage = ({ mdx }) => <Mdx>{mdx}<Mdx>;

Use the MdxProvider to provide components for HTML elements rendered by Markdown. gatsby-transformer-mdx-data exports it as a convenience (import { MDXProvider } from 'gatsby-transformer-mdx-data';) or import from @mdx-js/react.

Available options

Option Type Description Default Required
dataDirs string array A list of absolute file paths of where data MDX files are stored [] Yes

Examples of usage

There are 2 examples in the examples directory, one using JavaScript and one using TypeScript - the usage is essentially the same.

An example page using queried data might look like:

import { graphql } from 'gatsby';
import Mdx, { MDXProvider } from 'gatsby-transformer-mdx-data';

const IndexPage = ({ data }) => {
    return (
        <MDXProvider components={{
            SomeTag: ({ children }) => <div style={{ color: 'red'}}>{children}</div>
        }}>
            {data.products.nodes.map(({ id, title, mdx }) => (
                <div key={id}>
                    <h1>{title}</h1>
                    <Mdx>{mdx}</Mdx>
                </div>
            ))}
        </MDXProvider>
    );
};

export default IndexPage;

export const query = graphql`
    {
        products: allDataProduct {
            nodes {
                id
                title
                mdx
            }
        }
    }
`;

How to run tests

// TODO

How to develop locally

// TODO

How to contribute

// TODO

© 2025 Gatsby, Inc.