Building a Simple Static Blog with JavaScript and Markdown
Building a Simple Static Blog with JavaScript and Markdown
As a developer who values simplicity and performance, I've always been drawn to static site generators. While there are many excellent options like Jekyll, Hugo, and Eleventy, sometimes you want something tailored precisely to your needs without the overhead of learning a new framework. That's why I built a custom, lightweight blogging system for my personal website using JavaScript, Markdown, and HTML templates.
Why Build a Custom Blog System?
When redesigning my personal website, I had a few specific requirements:
- Content in Markdown - I wanted to write in Markdown for ease of use and portability
- Simple file structure - No complex database, just files and folders
- SEO optimization - Generated pages with proper meta tags, structured data, and sitemaps
- Minimal dependencies - Just a few core npm packages
- Full control - The ability to customize every aspect of the build process
After evaluating several existing solutions, I decided to build my own system that would tick all these boxes while being extremely lightweight.
How It Works
The system consists of a few key components:
- Content files - Markdown files with YAML frontmatter stored in a content directory
- HTML templates - Template files with placeholders for dynamic content
- Build script - A Node.js script that processes the markdown files and generates the static HTML
The build process is straightforward:
- Read all markdown files from the content directory
- Parse the frontmatter and markdown content
- Convert markdown to HTML
- Insert the HTML and metadata into the template
- Generate an index page listing all posts
- Create a sitemap and RSS feed for SEO
The Build Script
The heart of the system is the build script, which handles the entire process of converting markdown to HTML and generating the blog pages. Here's a breakdown of how it works:
Setting Up
The script starts by importing dependencies and defining paths:
const fs = require('fs');
const path = require('path');
const marked = require('marked');
const matter = require('gray-matter');
// Configure paths
const CONTENT_DIR = path.join(__dirname, '../content/posts');
const OUTPUT_DIR = path.join(__dirname, '../blog/posts');
const TEMPLATE_PATH = path.join(__dirname, '../blog/template.html');
Processing Markdown Files
For each markdown file, the script:
- Reads the file content
- Parses the frontmatter and markdown using gray-matter
- Generates a slug from the filename
- Converts markdown to HTML using marked
- Extracts metadata like date, title, description
mdFiles.forEach(mdFile => {
const filePath = path.join(CONTENT_DIR, mdFile);
const content = fs.readFileSync(filePath, 'utf8');
// Parse frontmatter and content
const { data: frontmatter, content: markdownContent } = matter(content);
// Generate slug from filename
const slug = mdFile.replace('.md', '');
// Convert markdown to HTML
const htmlContent = marked.parse(markdownContent);
// Create post object with metadata
const post = {
slug,
title: frontmatter.title || 'Untitled Post',
description: frontmatter.description || '',
date: postDate,
// Additional metadata...
};
// Generate HTML from template
let postHtml = template
.replace(/{{title}}/g, post.title)
.replace(/{{date}}/g, formatDate(post.date))
// Replace other placeholders...
// Write to file
fs.writeFileSync(outputPath, postHtml);
});
Generating the Blog Index
The index page lists all blog posts in reverse chronological order. The script:
- Sorts all posts by date
- Generates HTML for each blog card
- Inserts the HTML into the index template
The index template includes placeholders for the blog grid:
<div class="blog-grid">
<!-- Blog posts will be inserted here -->
</div>
The script replaces this with dynamically generated HTML for each post:
const blogCardsHtml = posts.map(post => {
return `
<article class="blog-card">
<div class="blog-thumbnail">
${post.img ? `<img src="${post.img}" alt="${post.title}">` : '<span>Featured Image</span>'}
</div>
<div class="blog-content">
<h3 class="blog-title">${post.title}</h3>
<div class="blog-meta">
<span class="blog-date">${formatDate(post.date)}</span>
<span class="blog-read-time">${readTime} min read</span>
</div>
<div class="blog-excerpt">
<p>${post.description}</p>
</div>
<a href="posts/${post.slug}.html" class="button">Read More</a>
</div>
</article>
`;
}).join('\n');
SEO Optimization
One of the benefits of this approach is the ability to add comprehensive SEO features:
Meta Tags
The template includes meta tags for social sharing and SEO:
<!-- Primary Meta Tags -->
<meta name="description" content="{{description}}">
<meta name="author" content="Matt Segar">
<meta name="keywords" content="markdown, blog, content, static, system">
<!-- Open Graph / Facebook -->
<meta property="og:type" content="article">
<meta property="og:url" content="https://segar.me/blog/posts/static_blog.html">
<meta property="og:title" content="{{title}} - Matt Segar">
<meta property="og:description" content="{{description}}">
<meta property="og:image" content="../assets/images/static-blog.png">
<!-- Twitter -->
<meta property="twitter:card" content="summary_large_image">
<!-- Additional Twitter tags... -->
Structured Data
The system also includes structured data for rich search results:
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "BlogPosting",
"headline": "{{title}}",
"description": "{{description}}",
"image": "../assets/images/static-blog.png",
"author": {
"@type": "Person",
"name": "Matt Segar"
},
<!-- Additional structured data... -->
}
</script>
Sitemap and RSS Feed
The script also generates a sitemap.xml and RSS feed to improve search engine indexing:
function generateSitemap(posts) {
const baseUrl = 'https://segar.me';
let sitemap = '<?xml version="1.0" encoding="UTF-8"?>\n';
sitemap += '<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">\n';
// Add homepage and other main pages...
// Add blog posts
posts.forEach(post => {
sitemap += ` <url>
<loc>${baseUrl}/blog/posts/${post.slug}.html</loc>
<lastmod>${postDate}</lastmod>
<changefreq>monthly</changefreq>
<priority>0.7</priority>
</url>\n`;
});
sitemap += '</urlset>';
// Write sitemap to file
fs.writeFileSync(path.join(__dirname, '../sitemap.xml'), sitemap);
}
Benefits of This Approach
Using this custom static blog system offers several advantages:
- Performance - Static HTML pages are extremely fast to load
- Security - No server-side processing means fewer security vulnerabilities
- Simplicity - The entire system is just a few files
- Version control - Content can be easily tracked in Git
- Flexibility - Add new features or modify the build process as needed
- Low cost - Host on GitHub Pages, Netlify, or any static hosting service for free or low cost
Adding Content
With this system, adding a new blog post is as simple as:
- Create a new markdown file in the content directory
- Add frontmatter with metadata (title, description, date, etc.)
- Write the post content in markdown
- Run the build script to generate HTML
- Deploy the updated files
---
title: My New Blog Post
description: This is an example blog post
date: 2025-03-24
img: ../assets/images/example.png
categories: [Example, Blog]
---
# My New Blog Post
This is an example of how easy it is to create content with this system.
Conclusion
Building a custom static blog system might seem like reinventing the wheel, but it offers a level of control and simplicity that's hard to match with off-the-shelf solutions. This approach is perfect for developers who want a lightweight, customizable blogging platform without the overhead of a full CMS or complex static site generator.
The entire system I've described is less than 300 lines of code, yet it provides all the essential features of a modern blog: markdown support, responsive design, SEO optimization, and easy content management.
If you're looking to create a personal blog or simple content website, consider this approach. It's lightweight, flexible, and puts you in complete control of your content.
Reach out to me if you'd like to see the full source code for this system. It's also hosted in my GitHub.