Version: Feb 22, 2025
Converting Markdown to HTML with Node.js
Converting Markdown (.md
) files into HTML is a common task when generating static web pages, documentation, or blogs. This guide explains how to build a Markdown-to-HTML converter using Node.js, Markdown-It, and Highlight.js for syntax highlighting. The
blow js file is what I use on my own sites.
Some of the key features:
- Parses Front Matter: Extracts metadata like title, description, and version.
- Syntax Highlighting: Uses highlight.js for code block styling.
- Custom HTML Template: Wraps the converted Markdown content in a structured HTML template.
- Recursive Directory Processing: Converts all Markdown files, even inside subdirectories.
Prerequisites
Before running the script, ensure you have the necessary dependencies installed.
Install Required Packages
Run the following command:
npm install fs path markdown-it gray-matter highlight.js
At the very end, I'll provide the js script I use, but I wanted to take a minute and break down the script. I usually don't like to add mindless filler text, so its just the code. It should be self explanatory.
Import Required Modules
const fs = require('fs');
const path = require('path');
const MarkdownIt = require('markdown-it');
const matter = require('gray-matter');
const hljs = require('highlight.js');
Define Input and Output Directories
const inputDir = path.join(__dirname, '../local_markdown');
const outputDir = path.join(__dirname, '../local_html');
if (!fs.existsSync(outputDir)) {
fs.mkdirSync(outputDir, { recursive: true });
}
Initialize Markdown Parser with Syntax Highlighting
const md = new MarkdownIt({
highlight: (str, lang) => {
if (lang && hljs.getLanguage(lang)) {
try {
return `<pre><code class="hljs ${lang}">` +
hljs.highlight(str, { language: lang }).value +
`</code></pre>`;
} catch (_) {}
}
return `<pre><code class="hljs">` + md.utils.escapeHtml(str) + `</code></pre>`;
}
});
Convert Markdown Files to HTML
function processDirectory(inputPath, outputPath) {
if (!fs.existsSync(outputPath)) {
fs.mkdirSync(outputPath, { recursive: true });
}
const items = fs.readdirSync(inputPath);
items.forEach(item => {
const itemPath = path.join(inputPath, item);
const outputItemPath = path.join(outputPath, item);
if (fs.lstatSync(itemPath).isDirectory()) {
processDirectory(itemPath, outputItemPath);
} else if (path.extname(item) === '.md') {
const fileContent = fs.readFileSync(itemPath, 'utf-8');
const { data: frontMatter, content } = matter(fileContent);
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta name="description" content="${frontMatter.description || 'Markdown to HTML Conversion'}">
<meta name="author" content="Derek Dreblow">
<title>${frontMatter.title || 'Markdown to HTML'}</title>
</head>
<body>
<main class="markdown-body">
<article>
<p><em class="blogVersion">Version: ${frontMatter.version || 'Unknown'}</em></p>
${md.render(content)}
</article>
</main>
</body>
</html>`;
const outputFilePath = path.join(outputPath, `${path.basename(item, '.md')}.html`);
fs.writeFileSync(outputFilePath, htmlContent, 'utf-8');
console.log(`Converted: ${itemPath} -> ${outputFilePath}`);
}
});
}
Start the Conversion Process
Throw this at the end of the script.
processDirectory(inputDir, outputDir);
Run the Script
node convert.js
Personal Code - edited
Below is my personal code, I use it still to this day.
const fs = require('fs');
const path = require('path');
const MarkdownIt = require('markdown-it');
const matter = require('gray-matter');
const hljs = require('highlight.js');
const ROOT_DIR = "../../../"
const ROOT_BLOG_DIR = "../"
// Initialize Markdown-it with highlight.js
const md = new MarkdownIt({
highlight: (str, lang) => {
if (lang && hljs.getLanguage(lang)) {
try {
return `<pre><code class="hljs ${lang}">` +
hljs.highlight(str, { language: lang }).value +
`</code></pre>`;
} catch (_) {}
}
return `<pre><code class="hljs">` + md.utils.escapeHtml(str) + `</code></pre>`;
}
});
// Define input/output directories
const inputDir = path.join(__dirname, '../local_markdown'); // Input directory for Markdown files
const outputDir = path.join(__dirname, '../local_html'); // Output directory for HTML files
// Ensure output directory exists
if (!fs.existsSync(outputDir)) {
fs.mkdirSync(outputDir, { recursive: true });
}
// Function to calculate relative path depth
function calculateRelativePath(filePath, basePath) {
const relativeDepth = path.relative(filePath, basePath).split(path.sep).length - 1;
return '../'.repeat(relativeDepth);
}
// Format the version date
function formatVersionDate(dateString) {
// Attempt to parse the date string
let date;
// Try ISO 8601 format first
if (!isNaN(Date.parse(dateString))) {
date = new Date(dateString);
} else {
// Fallback to manual parsing if standard parsing fails
const dateParts = dateString.match(
/(\w{3}) (\w{3}) (\d{1,2}) (\d{4}) (\d{2}:\d{2}:\d{2})/ // Fri Nov 29 2024 16:00:00
);
if (dateParts) {
const [, , month, day, year] = dateParts;
const months = {
Jan: 0, Feb: 1, Mar: 2, Apr: 3, May: 4, Jun: 5,
Jul: 6, Aug: 7, Sep: 8, Oct: 9, Nov: 10, Dec: 11,
};
date = new Date(year, months[month], day);
} else {
console.warn(`Invalid date provided: ${dateString}`);
return "Unknown Date"; // Fallback for invalid dates
}
}
// Format the date to "Nov 29, 2024"
const options = {
year: 'numeric',
month: 'short',
day: 'numeric'
};
return date.toLocaleDateString('en-US', options);
}
// Recursively process directories
function processDirectory(inputPath, outputPath) {
// Ensure the output directory exists
if (!fs.existsSync(outputPath)) {
fs.mkdirSync(outputPath, { recursive: true });
}
// Read all files and directories
const items = fs.readdirSync(inputPath);
items.forEach(item => {
const itemPath = path.join(inputPath, item);
const outputItemPath = path.join(outputPath, item);
// Check if item is a directory
if (fs.lstatSync(itemPath).isDirectory()) {
processDirectory(itemPath, outputItemPath); // Recurse into subdirectory
} else if (path.extname(item) === '.md') {
// Process Markdown files
const fileContent = fs.readFileSync(itemPath, 'utf-8');
const { data: frontMatter, content } = matter(fileContent);
// Calculate the relative path from the output file to the root
const relativePath = calculateRelativePath(outputItemPath, outputDir);
const version = formatVersionDate(frontMatter.version); // Default if no version is provided
const title = frontMatter.title || 'Dreblow Designs Blog';
const description = frontMatter.description || 'Discover the latest blog posts from Derek Dreblow, focusing on engineering, software development, and project insights.';
const author = frontMatter.author || "Derek Dreblow"
const keyword = frontMatter.keyword || "Dreblow Design's Blog"
const image = frontMatter.image || "https://dreblowdesigns.com/pages/blog/local_images/BlogFavicon.png"
const url = frontMatter.url || "https://dreblowdesigns.com"
// Generate HTML content
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta name="description" content="${description}">
<meta name="author" content="${author}">
<meta name="keywords" content="${keyword}">
<link rel="canonical" href="https://dreblowdesigns.com/pages/blog/local_html/blog.html">
<!-- Open Graph Metadata -->
<meta property="og:title" content="${title}">
<meta property="og:description" content="${description}">
<meta property="og:image" content="${image}">
<meta property="og:url" content="${url}">
<meta property="og:type" content="website">
<!-- Twitter Card Metadata -->
<meta name="twitter:card" content="summary_large_image">
<meta name="twitter:title" content="${title}">
<meta name="twitter:description" content="${description}">
<meta name="twitter:image" content="${image}">
<!-- Default favicon -->
<link rel="icon" type="image/svg+xml" href="https://dreblowdesigns.com/resources/images/favicon_io/favicon.svg" />
<link rel="shortcut icon" href="https://dreblowdesigns.com/resources/images/favicon_io/favicon.ico" />
<!-- PNG favicon for high-resolution displays -->
<link rel="icon" type="image/png" href="https://dreblowdesigns.com/resources/images/favicon_io/favicon-96x96.png" sizes="96x96" />
<!-- Apple Touch Icon -->
<link rel="apple-touch-icon" sizes="180x180" href="https://dreblowdesigns.com/resources/images/favicon_io/apple-touch-icon.png" />
<meta name="apple-mobile-web-app-title" content="Dreblow Designs">
<!-- Site Manifest-->
<link rel="manifest" href="https://dreblowdesigns.com/resources/images/favicon_io/site.webmanifest" />
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id="></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', '');
</script>
<link rel="stylesheet" href="${ROOT_DIR}${relativePath}resources/css/styles.css">
<link rel="stylesheet" href="${ROOT_BLOG_DIR}${relativePath}local_css/blog.css">
<link rel="stylesheet" href="${ROOT_BLOG_DIR}${relativePath}local_css/github-dark.min.css">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
</head>
<body>
<header>
<nav>
<ul>
<li><a href="/" aria-label="Home">Home</a></li>
<li><a href="${ROOT_DIR}${relativePath}pages/about.html" aria-label="About">About</a></li>
<!--<li><a href="${ROOT_DIR}${relativePath}pages/portfolio.html">Portfolio</a></li>-->
<li><a href="${ROOT_DIR}${relativePath}pages/contact/contact.html" aria-label="Contact">Contact</a></li>
<li><a href="${ROOT_DIR}${relativePath}pages/blog/local_html/blog.html" aria-label="Blog">Blog</a></li>
</ul>
</nav>
</header>
<main class="markdown-body">
<article>
<p><em class="blogVersion">Version: ${version}</em></p>
${md.render(content)}
</article>
</main>
<footer>
<div class="social-links">
<a href="https://www.linkedin.com/in/derek-dreblow-4756134b/" target="_blank">
<i class="fab fa-linkedin"></i> LinkedIn
</a>
<a href="https://github.com/Dreblow" target="_blank">
<i class="fab fa-github"></i> GitHub
</a>
<a href="https://leetcode.com/u/dreblow/" target="_blank">
<img src="${ROOT_DIR}${relativePath}resources/images/brands/LeetCode/leetcode.png" alt="LeetCode Logo" style="width: 15px; height: 15px;"> LeetCode
</a>
<a href="https://www.threads.net/@derek_dre" target="_blank">
<img src="${ROOT_DIR}${relativePath}resources/images/brands/threads/threads.png" alt="Threads Logo" style="width: 15px; height: 15px;"> Threads
</a>
</div>
<p class="copy-right">© 2024 - <span id="copyright-year"></span> Derek Dreblow, Dreblow Designs. All Rights Reserved.</p>
<script>
document.getElementById("copyright-year").textContent = new Date().getFullYear();
</script>
</footer>
</body>
</html>`;
// Write the converted file to the output directory
const outputFilePath = path.join(outputPath, `${path.basename(item, '.md')}.html`);
fs.writeFileSync(outputFilePath, htmlContent, 'utf-8');
console.log(`Converted: ${itemPath} -> ${outputFilePath}`);
}
});
}
// Start processing from the root directory
processDirectory(inputDir, outputDir);