Skip to main content

A lightweight, drop-in Markdown CMS for FastAPI with hot reloading

Project description

Moosey CMS ๐ŸซŽ

A lightweight, drop-in Markdown CMS for FastAPI.

Moosey CMS transforms your FastAPI application into a content-driven website without the need for a database. It bridges the gap between static site generators and dynamic web servers, offering hot-reloading, intelligent caching, SEO management, and a powerful templating hierarchy.

Example Screenshot

Example Screenshot

Check out the /example for templating and content samples used to generate the images above.


๐Ÿš€ Features

  • No Database Required: Content is managed via Markdown files with YAML Frontmatter.
  • Intelligent Routing: URL paths automatically map to your content directory structure.
  • Smart Templating: "Waterfall" inheritance logic (Singular/Plural) to automatically find the best layout for every page.
  • Hot Reloading: Instant browser refresh when Content or Templates change (Development mode only).
  • High Performance: Built-in caching (TTL-based) that auto-clears on file changes.
  • SEO Ready: Automatic OpenGraph, Twitter Cards, JSON-LD, and Meta tags generation.
  • Site Management: Built-in sitemap.xml, robots.txt, RSS feeds, and a reusable content index.
  • Rich Markdown: Supports tables, emojis, task lists, and syntax highlighting out of the box.
  • Jinja2 Power: Use Jinja2 logic directly inside your Markdown files (Securely Sandboxed).

๐Ÿ“ฆ Installation

Using UV (Recommended)

uv add moosey-cms

Using Pip

pip install moosey-cms

โšก Quick Start

Integrate Moosey CMS into your existing FastAPI app in just a few lines.

from fastapi import FastAPI
from fastapi.staticfiles import StaticFiles
from pathlib import Path
from moosey_cms import init_cms

app = FastAPI()

# 1. Define your paths
BASE_DIR = Path(__file__).resolve().parent
CONTENT_DIR = BASE_DIR / "content"
TEMPLATES_DIR = BASE_DIR / "templates"

# 2. Mount static files (Optional, but recommended for CSS/Images)
app.mount("/static", StaticFiles(directory="static"), name="static")

# 3. Initialize the CMS
init_cms(
    app,
    host="localhost",
    port=8000,
    dirs={
        "content": CONTENT_DIR, 
        "templates": TEMPLATES_DIR
    },
    mode="development",  # Enables hot-reloading
    site_data={
        "name": "My Awesome Site",
        "description": "A site built with Moosey CMS",
        "author": "Jane Doe",
        "keywords": ["fastapi", "cms", "python"],
        "open_graph": {
             "og_image": "/static/cover.jpg"
        },
        "social": {
            "twitter": "https://x.com/myhandle",
            "github": "https://github.com/myhandle"
        },
        "web": {
            "site_url": "https://example.com",
            "feed": {
                "collection": "/blog",
                "title": "My Awesome Site Feed"
            }
        }
    },
    reload_delay=2.5 # Triggers hot-reload after this duration
)

๐Ÿ“‚ Directory Structure

Moosey CMS relies on a convention-over-configuration file structure.

.
โ”œโ”€โ”€ main.py
โ”œโ”€โ”€ content/               <-- Your Markdown Files
โ”‚   โ”œโ”€โ”€ index.md           <-- Homepage (/)
โ”‚   โ”œโ”€โ”€ about.md           <-- About Page (/about)
โ”‚   โ””โ”€โ”€ blog/
โ”‚       โ”œโ”€โ”€ index.md       <-- Blog Listing (/blog)
โ”‚       โ”œโ”€โ”€ post-1.md      <-- Blog Post (/blog/post-1)
โ”‚       โ””โ”€โ”€ post-2.md
โ””โ”€โ”€ templates/ 
    โ”œโ”€โ”€ layout          
        โ”œโ”€โ”€ base.html          <-- Base layout
    โ”œโ”€โ”€ index.html         <-- Home Page layout
    โ”œโ”€โ”€ page.html          <-- Default fallback
    โ”œโ”€โ”€ blog.html          <-- Layout for /blog (Listing)
    โ””โ”€โ”€ post.html          <-- Layout for /blog/post-1 (Single Item)

๐ŸŽจ Templating Logic (The Waterfall)

When a user visits a URL, Moosey CMS searches for templates in a specific cascading order. This allows you to set global defaults while retaining the ability to customize specific pages or sections.

Example Scenario: A user visits /posts/post-1.

Directory Structure:

.
โ”œโ”€โ”€ content/
โ”‚   โ””โ”€โ”€ posts/
โ”‚       โ”œโ”€โ”€ index.md        <-- Required for the '/posts' listing page to work
โ”‚       โ”œโ”€โ”€ post-1.md       <-- The article being requested
โ”‚       โ””โ”€โ”€ post-2.md
โ””โ”€โ”€ templates/
    โ”œโ”€โ”€ posts/
    โ”‚   โ””โ”€โ”€ post-1.html     <-- 1. Specific Override
    โ”œโ”€โ”€ post.html           <-- 2. Singular (Item) Layout
    โ”œโ”€โ”€ posts.html          <-- 3. Plural (Section) Layout
    โ””โ”€โ”€ page.html           <-- 4. Global Fallback

Resolution Order:

  1. Frontmatter Override: If post-1.md contains template: special.html, that template is used immediately.
  2. Exact Match: templates/posts/post-1.html.
  3. Singular Parent: templates/post.html (Perfect for generic blog posts).
  4. Plural Parent: templates/posts.html (Perfect for section indexes).
  5. Fallback: templates/page.html.

๐Ÿ“ Frontmatter Configuration

You can control routing, visibility, and layout directly from the Markdown file YAML frontmatter.

Basic Metadata

title: My Amazing Post
date: 2024-01-01
description: A short summary for SEO.

Organization & Navigation

Key Type Description
order int Sort order in sidebars. Lower numbers appear first. Default: 9999.
nav_title str Short title to display in sidebars (if different from title).
visible bool Set to false to hide from sidebars/menus (page remains accessible via URL).
draft bool If true, the page is only visible in development mode.
group str Group sidebar items under a heading (requires template support).

Advanced Routing

Key Type Description
template str Force a specific template file (e.g., template: landing.html).
external_link str The sidebar link will point to this external URL instead of the page itself.
redirect str Alias for external_link.

Publishing, SEO & Feeds

Key Type Description
canonical / canonical_url str Canonical URL used by {{ seo() }}.
noindex bool Adds noindex, nofollow via {{ seo() }} and excludes the page from sitemap/feed output.
sitemap bool or dict Set false to exclude from /sitemap.xml, or provide changefreq / priority.
feed / rss bool Set false to exclude a page from RSS feeds.
date.published date Preferred publish date for sorting and RSS pubDate.

Example:

---
title: API Documentation
nav_title: API Docs
order: 1
group: "Developer Tools"
external_link: "https://api.mysite.com"
---

๐Ÿ•ธ๏ธ Built-in Website Routes

Moosey automatically registers everyday site-management routes before the content catch-all route:

Route Purpose
/sitemap.xml Autogenerated XML sitemap from publishable Markdown pages.
/robots.txt Environment-aware robots rules with a sitemap pointer.
/feed.xml RSS 2.0 feed generated from your content index.
/rss.xml Alias for /feed.xml unless disabled.

Configure them in site_data.web:

site_data = {
    "name": "My Site",
    "web": {
        "site_url": "https://example.com",
        "sitemap": {
            "default_changefreq": "weekly",
            "default_priority": "0.5",
        },
        "robots": {
            "production": {"allow": ["/"], "disallow": []},
            "staging": {"disallow": ["/"]},
            "testing": {"disallow": ["/"]},
        },
        "feed": {
            "collection": "/blog",
            "limit": 50,
            "title": "My Site Blog",
            "description": "Latest articles from My Site",
        },
    },
}

Set any feature to false to disable it, for example "feed": false.


๐Ÿงฉ Custom Filters & Logic

Moosey CMS comes packed with a comprehensive library of Jinja2 filters to help you format your data effortlessly.

Date & Time

Filter Usage Output
fancy_date {{ date | fancy_date }} 13th Jan, 2026 at 6:00 PM
short_date {{ date | short_date }} Jan 13, 2026
iso_date {{ date | iso_date }} 2026-01-13
time_only {{ date | time_only }} 6:00 PM
relative_time {{ date | relative_time }} 2 hours ago / yesterday
rfc822_date {{ date | rfc822_date }} Thu, 15 Jan 2026 00:00:00 GMT

Currency & Numbers

Filter Usage Output
currency {{ 1234.5 | currency('USD') }} $1,234.50
compact_currency {{ 1500000 | compact_currency }} $1.5M
currency_name {{ 'KES' | currency_name }} Kenyan Shilling
number_format {{ 1000 | number_format }} 1,000
percentage {{ 50.5 | percentage }} 50.5%
ordinal {{ 3 | ordinal }} 3rd

Geography & Locale

Filter Usage Output
country_flag {{ 'US' | country_flag }} ๐Ÿ‡บ๐Ÿ‡ธ
country_name {{ 'DE' | country_name }} Germany
language_name {{ 'fr' | language_name }} French

Text Formatting

Filter Usage Output
truncate_words {{ text | truncate_words(10) }} Truncates text to 10 words...
excerpt {{ text | excerpt(150) }} Smart excerpt breaking at sentences.
read_time {{ content | read_time }} 5 min read
slugify {{ 'Hello World' | slugify }} hello-world
title_case {{ 'a tale of two cities' | title_case }} A Tale of Two Cities
smart_quotes {{ '"Hello"' | smart_quotes }} โ€œHelloโ€
strip_html {{ content | strip_html }} Plain text without HTML tags

Utilities

Filter Usage Output
filesize {{ 1024 | filesize }} 1.0 KB
yesno {{ True | yesno }} Yes
default_if_none {{ val | default_if_none('N/A') }} Returns default if None
absolute_url {{ '/about' | absolute_url }} Absolute URL using site_data.web.site_url or the request base URL

More On Filters and how to use some interesting ones such as stripping comments.

Advanced Features and how to use some interesting ones such as stripping comments.

โš™๏ธ Configuration Reference

The init_cms function accepts the following parameters:

Parameter Type Description
app FastAPI Your FastAPI application instance.
host str Server host (used for hot-reload script injection).
port int Server port.
dirs dict Dictionary containing content and templates Paths.
mode str "development" (enables hot reload/no cache), "production", "staging", or "testing".
site_data dict Global data (name, author, social links, optional web config for sitemap/robots/RSS).
reload_delay float Seconds to delay hot-reload broadcast after a file change. Useful when a build step runs post-save. Default: 0 (immediate). Development mode only.

๐Ÿ›ก๏ธ Security & Mitigation

Moosey CMS takes security seriously. We have implemented several layers of protection to ensure your site remains safe:

  1. Path Traversal Protection: All URL requests are securely resolved against the content root using strict pathlib checks. It is impossible to access files outside the content directory (e.g., ../../etc/passwd).
  2. SSTI Sandbox: While we allow Jinja2 logic inside Markdown files, this is executed in a Sandboxed Environment. Dangerous attributes (like __class__, __subclasses__) are stripped, preventing Remote Code Execution (RCE) attacks.
  3. DoS Prevention: The Hot-Reload middleware includes size checks to prevent memory exhaustion attacks from large file uploads/downloads.

๐Ÿ› Bug Reporting

Security is an ongoing process. If you discover a vulnerability, bug, or potential risk, please open an issue on our GitHub repository immediately. We appreciate community feedback to keep Moosey secure for everyone.


Gratitude

This project is inspired by fastapi-blog by Daniel. Initially, I wanted to use fastapi-blog and it worked really well till I needed features like hot-reloading.

License

MIT License. Copyright (c) 2026 Anthony Mugendi.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

moosey_cms-0.9.2.tar.gz (212.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

moosey_cms-0.9.2-py3-none-any.whl (34.3 kB view details)

Uploaded Python 3

File details

Details for the file moosey_cms-0.9.2.tar.gz.

File metadata

  • Download URL: moosey_cms-0.9.2.tar.gz
  • Upload date:
  • Size: 212.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for moosey_cms-0.9.2.tar.gz
Algorithm Hash digest
SHA256 c679b894080cbea4d7220d9b6021ce5b52baaa5ee06d560141c4371376d38a7b
MD5 d5b6e9d48a1bfb73ab2b3f9d2f0a1929
BLAKE2b-256 8e31b2dbeaff51720de32d80fefa7a68f3ad4f77dccf97b01437fe9fe070442b

See more details on using hashes here.

File details

Details for the file moosey_cms-0.9.2-py3-none-any.whl.

File metadata

  • Download URL: moosey_cms-0.9.2-py3-none-any.whl
  • Upload date:
  • Size: 34.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for moosey_cms-0.9.2-py3-none-any.whl
Algorithm Hash digest
SHA256 d482574164ded1fa6afd2549ebac7ee48d78b31437ad1c4bf838a60e6f576ed0
MD5 8eccf37cd9cdc52be5151a12d4983a9c
BLAKE2b-256 5d7560ffacfb0e9a227a71d673ea1057d336a1f69f4e7c26c538757c57cbbb8a

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page