How to create a Markdown blog with Next.js 15
1. Project setup
Create a new Next.js project following the official documentation or with npx create-next-app@latest
.
You can also add a blog to an existing Next.js website.
(i) This post is created for Next.js v15 using the app router and Typescript.
2. /blog and /blog/[id] pages
In the app
folder, create a subfolder blog
with a page.tsx
file.
This page will contain a list of links to all blog posts.
export default function BlogHomePage() {
return <>
<h2>Blog</h2>
</>
}
In the app/blog
folder, create a subfolder [id]
with a page.tsx
file.
(This will create a route with a dynamic id in the URL like /blog/my-first-post
where my-first-post
is the id of a blog post.)
This page will render a single blog post with the id
from the URL.
export default async function BlogPostPage({ params }: { params: Promise<{ id: string }>}) {
const { id } = await params
return <>
<h2>{id}</h2>
</>
}
3. Blog posts
In the root of your project (on the same level as your package.json
), create a folder blog-posts
and add some markdown files.
For example
/blog-posts/my-first-post.md
---
title: 'My first blog post'
date: 2024-10-26
---
This is my very first blog post!
/blog-posts/front-matter.md
---
title: 'YAML Front Matter'
date: 2024-10-26
---
The metadata at the top of this markdown file is YAML Front Matter.
4. Fetch blog posts
To fetch the blog posts from the blog-posts
folder, first create a blog.types.ts
file in /app/blog
that exports the BlogPost
type.
/app/blog/blog.types.ts
export type BlogPost = {
id: string
title: string
date: Date
content: string
}
To read the Front Matter metadata from the Markdown files, we need to install a package called gray-matter
.
npm install gray-matter
Then create a file blog.utils.ts
in /app/blog
.
/app/blog/blog.utils.ts
import fs from 'fs'
import path from 'path'
import matter from 'gray-matter'
import { BlogPost } from './blog.types'
export const blogPostsFolder = path.join(process.cwd(), 'blog-posts')
export async function readAllBlogPostFiles() {
const dirEntries = await fs.promises.readdir(blogPostsFolder, { recursive: true, withFileTypes: true })
return dirEntries.filter(entry => entry.isFile())
}
export async function getAllBlogPosts() {
const blogPostFiles = await readAllBlogPostFiles()
return Promise.all(blogPostFiles.map(mapFileToBlogPost))
}
export function sortBlogPosts(a: BlogPost, b: BlogPost) {
if (a.date > b.date) return 1
return -1
}
export async function getBlogPostById(id: string): Promise<BlogPost | undefined> {
const allBlogPostFiles = await readAllBlogPostFiles()
const blogPostFile = allBlogPostFiles.find(entry => parseFileId(entry) === id)
if (!blogPostFile) return undefined
return mapFileToBlogPost(blogPostFile)
}
async function mapFileToBlogPost(file: fs.Dirent): Promise<BlogPost> {
const fileContents = await fs.promises.readFile(getFilePath(file), { encoding: 'utf8' })
const matterData = matter(fileContents)
return {
id: parseFileId(file),
title: matterData.data.title,
date: matterData.data.date,
content: matterData.content,
}
}
function getFilePath(file: fs.Dirent): string {
return path.join(file.parentPath, file.name)
}
export function parseFileId(file: fs.Dirent): string {
return file.name.replace(/\.md$/, '') // remove the '.md' file extension
}
5. Overview of blog posts
Add the blog posts to the /blog
page
/app/blog/page.tsx
import Link from 'next/link'
import { getAllBlogPosts, sortBlogPosts } from './blog.utils'
export default async function BlogHomePage() {
const blogPosts = await getAllBlogPosts()
blogPosts.sort(sortBlogPosts).reverse()
return <>
<h2>My Blog</h2>
<ul>{blogPosts.map(blogPost => <li key={blogPost.id}><Link href={`/blog/${blogPost.id}`}>blogPost.title</Link></li>)}</ul>
</>
}
(i) Note that the function BlogHomePage() {}
is now async
.
You can now add a link to the /blog
page on the homepage or anywhere on your website.
6. Markdown to HTML
To render the Markdown blog post as HTML at /blog/[id]
, install the remark
and remark-html
packages.
npm install remark remark-html
Then get the blog post by its id
in the URL and transform the content to HTML.
/app/blog/[id]/page.tsx
import { remark } from 'remark'
import remarkHtml from 'remark-html'
import { getBlogPostById } from "../blog.utils";
export default async function BlogPostPage({ params }: { params: Promise<{ id: string }> }) {
const { id } = await params;
const blogPost = await getBlogPostById(id)
if (!blogPost) return
const htmlContent = (await remark().use(remarkHtml).process(blogPost.content)).toString()
return <>
<h3>{blogPost.title}</h3>
<p>{blogPost.date?.toString()}</p>
<p dangerouslySetInnerHTML={{ __html: htmlContent }} />
</>
}
7. Static blog post pages
We currently have a working blog, but it is not very well optimized for the web.
A big improvement is to provide Next.js with a list of all possible blog post ids so the pages can be pre-rendered at build time.
We can do this by exporting the generateStaticParams()
function in /app/blog/[id]/page.tsx
.
app/blog/[id]/page.tsx
export async function generateStaticParams() {
const entries = await readAllBlogPostFiles()
return entries.map((entry) => ({
id: parseFileId(entry),
}))
}
To prevent Next.js from trying to render a blog post for an id
not in the list of static params (static ids), add export const dynamicParams = false
to the app/blog/[id]/page.tsx
file.
8. SEO optimization
Providing a fitting title and description for your blog post immensely helps search engines with understanding the contents of your post and suggesting it to its users.
In Next.js we can dynamically generate the title
and description
meta tag by utilizing the generateMetaData()
function.
Besides the title
and description
, you can specify additional Open Graph tags to improve the shareability of your posts.
/app/blog/[id]/page.tsx
export async function generateMetadata({ params }: { params: Promise<{ id: string }>}) {
const { id } = await params
const blogPost = await getBlogPostById(id)
if (!blogPost) return {}
return {
title: blogPost.title,
description: blogPost.description,
openGraph: {
title: blogPost.title,
description: blogPost.description,
type: 'article',
publishedTime: blogPost.date?.toISOString(),
}
}
}
Since we now call getBlogPostById()
in both generateMetadata()
and the BlogPostPage()
component, it is best to cache the result of that function utilizing React's cache()
function.
/app/blog/blog.utils.ts
import { cache } from 'react'
export const getBlogPostById = cache(fetchBlogPostById)
export async function fetchBlogPostById(id: string): Promise<BlogPost | undefined> {
const allBlogPostFiles = await readAllBlogPostFiles()
const blogPostFile = allBlogPostFiles.find(entry => parseFileId(entry) === id)
if (!blogPostFile) return undefined
return mapFileToBlogPost(blogPostFile)
}