✨ A functional static site generator so smol you won't even notice it's there ✨
WARNING: Nuzzle is in the alpha stage. Expect API breakages.
Nuzzle is a static site generator packaged as a Clojure library. It requires an astonishingly small code footprint to use. Keep all your static site data and building instructions in a single Clojure file!
- Easy to use for developers new to Clojure
- Highly flexible for advanced Clojure users
- Great development experience with excellent hot-reloadability story
- Semantically aligned with the Atom feed spec
- Small code footprint! Your Clojure code interacting with Nuzzle can be extremely succinct such that you can keep it all in a single file and run Nuzzle functions via a Clojure CLI tool alias
- Users can define their own process for loading content so it can be of any type (Markdown, Org, HTML, etc) and come from anywhere (local files, headless CMS API calls, mix of both, etc)
- Generate an Atom feed with embedded HTML content
- Generate a sitemap
- Generate tag index pages from tag information
- Define all markup in hiccup without ever worrying about converting it to HTML
- Retrieve all data for any page while creating markup
- Start dual website/nREPL servers without adding nREPL deps for creating incredible development feedback loop
- Inject a script based on livejs with configurable refresh interval to automatically refresh browser view to see content/markup changes in realtime while developing
- Transform hiccup to do any customization imaginable including statically rendering syntax highlighted code blocks with Pygments or Chroma
Want to read some code already? Check out this repo which uses Nuzzle to build my personal developer blog deployed at stel.codes.
- Java >= 11
- Clojure >= 1.11.1
Here's a minimal Nuzzle setup:
deps.edn
{:aliases
{:site {:deps {codes.stel/nuzzle {:mvn/version "0.6.498"}
org.clojure/clojure {:mvn/version "1.11.1"}}
:ns-default site}}}
site.clj
(ns site
(:require [nuzzle.core :as nuzz]))
;; Create a pages map
(defn pages []
{[]
{:nuzzle/title "Homepage"
:nuzzle/render-page (fn [{:nuzzle/keys [title] :as _page}]
[:html
[:h1 title]
[:a {:href [:about]} "About"]])}
[:about]
{:nuzzle/title "About"
:nuzzle/render-content #(-> "content/about.md" slurp nuzz/parse-md)
:nuzzle/render-page (fn [{:nuzzle/keys [render-content title] :as _page}]
[:html
[:h1 title]
(render-content)])}})
;; Start static site server + nREPL server with nuzzle.core/develop
;; Pass the pages as a var to get full hot-reloading capabilities!
;; The returned value is a function that stops both servers.
(defn develop [_]
(nuzz/develop #'pages))
;; Publish the static site to ./dist
(defn publish [_]
(nuzz/publish pages))
Call the site/develop
or site/publish
functions from the command line:
# Using Babashka (my preference)
bb clojure -T:site develop
bb clojure -T:site publish
# Using official Clojure CLI executable
clj -T:site develop
clj -T:site publish
Test it out yourself!
git clone https://github.com/stelcodes/nuzzle && cd nuzzle/examples/minimal
If you're a trainer from Pallet town, your site.clj
might look like this:
(ns site
(:require
[nuzzle.core :as nuzz]))
(defn render-page [{:nuzzle/keys [title index render-content get-pages] :as _page}]
[:html
[:head [:title title]
;; Add link to CSS file /css/main.css (must be in overlay directory)
[:link {:href "/css/main.css" :rel "stylesheet"}]]
[:body
[:h1 title]
(when index
[:ul (for [url index
:let [{:nuzzle/keys [title url] :as _indexed-page} (get-pages url)]]
;; The url is a vector of keywords, but Nuzzle will convert them
;; into relative URL strings when used for an :href value
[:li [:a {:href url} title]])])
(render-content)]])
;; Here we make a function that creates a render-content function that turns markdown files into hiccup
(defn md-content [md-path]
#(-> md-path slurp nuzz/parse-md))
;; Here we define an author for our Atom feed
(def ash {:name "Ash Ketchum"
:email "ashketchum@fastmail.com"})
(defn pages []
{[]
{:nuzzle/title "Home"
:nuzzle/render-content (md-content "content/intro.md")
:nuzzle/render-page render-page
:nuzzle/index :children}
[:about]
{:nuzzle/title "About Me"
:nuzzle/render-content (md-content "content/about.md")
:nuzzle/render-page render-page}
[:blog-posts]
{:nuzzle/title "Blog Posts"
:nuzzle/render-page render-page
:nuzzle/index :children}
[:blog-posts :catching-pikachu]
{:nuzzle/title "I Caught a Pikachu!"
:nuzzle/render-content (md-content "content/caught-pikachu.md")
:nuzzle/render-page render-page
:nuzzle/tags #{:pokedex}
:nuzzle/author ash
:nuzzle/feed true}
[:blog-posts :defeating-misty]
{:nuzzle/draft true
:nuzzle/title "Misty's Pokemon Got Wrecked by Pikachu"
:nuzzle/render-content (md-content "content/defeated-misty.md")
:nuzzle/render-page render-page
:nuzzle/tags #{:gym-battle}
:nuzzle/author ash
:nuzzle/feed true}
[:tags]
{:nuzzle/title "Tags"
:nuzzle/render-page render-page
:nuzzle/index :children}})
(defn develop [_]
(nuzz/develop #'pages {:remove-drafts false
:overlay-dir "public"}))
;; By default build this site with a sitemap and Atom feed
;; Overlay the directory containing the css/main.css file.
(defn publish [_]
(nuzz/publish pages {:base-url "https://ashketchum.com"
:atom-feed {:title "Ash Ketchum's Blog"
:subtitle "In a world we must defend"}
:overlay-dir "public"}))
Test it out yourself!
git clone https://github.com/stelcodes/nuzzle && cd nuzzle/examples/ash-ketchum-blog
Each page entry in the pages map represents a single page of the static site. Each entry must have a key that is a vector of keywords representing a relative URL. The key [:blog-posts :catching-pikachu]
translates to the URL "/blog-posts/catching-pikachu"
and will be rendered to disk as <publish-dir>/blog-posts/catching-pikachu/index.html
. The associated value must be a map with data about the page. The keys of the page are based on the Atom feed spec.
;; The key for each entry is a vector of keywords representing a relative URL
;; This URL vector represents the URL /blog/learning-clojure/
[:blog :learning-clojure]
{;; A title for the web page
;; Required, no default
:nuzzle/title "Learning Clojure"
;; A function that returns hiccup containing the whole HTML document
;; Must have one argument (the containing page map)
;; Used by nuzzle.core/serve, publish, and develop for creating each page of the static site
;; Required, no default
:nuzzle/render-page (fn [{:nuzzle/keys [title render-content] :as _page}]
[:html [:head [:title title]] [:h1 title] (render-content)])
;; A function that returns hiccup containing the page's main content
;; Must have zero or one argument (the containing page map)
;; Available to user's corresponding :nuzzle/render-page function
;; Used by nuzzle.core/publish for creating Atom feed (for embedding HTML content directly into feed)
;; Optional, defaults to (constantly nil)
:nuzzle/render-content (fn [] [:p "The first step to learning Clojure is pressing the ( key."])
;; A boolean indicating whether this page is a draft or not
;; Used by nuzzle.core/serve, develop, and publish via :remove-drafts keyword argument to remove unfinished page entries
;; Optional, defaults to nil
:nuzzle/draft false
;; A boolean indicating whether the page should be included in the optional Atom feed when publishing
;; Used by nuzzle.core/publish for creating Atom feed
;; Optional, defaults to nil
:nuzzle/feed true
;; A set of keywords where each keyword represents a tag name
;; Used by nuzzle.core/add-tag-pages for assoc-ing tag index pages
;; Optional, defaults to nil
:nuzzle/tags #{:clojure}
;; Either a set of vector URLs (vectors of keywords), or the keyword :children
;; Describes what pages should be linked to from this page
;; When set to :children, Nuzzle will replace it with set of URLs of all the pages directly "beneath" this page
;; This key is always present in tag pages when enabled with the :tag-pages keyword argument to nuzzle.core/serve, develop, publish
;; Optional, defaults to nil
:nuzzle/index :children
;; An inst representing when the page was last updated
;; Used by nuzzle.core/publish for creating sitemap and Atom feed
;; Optional, defaults to nil
:nuzzle/updated #inst "2022-09-16T12:00:00Z"
;; A string summary of the page content
;; Used by nuzzle.core/publish for creating Atom feed
;; Optional, defaults to nil
:nuzzle/summary "An in-depth guide to learning Clojure."
;; A map representing the author of the page.
;; Used by nuzzle.core/publish for creating Atom feed
;; Optional, defaults to nil
:nuzzle/author {;; A string name, required per the Atom spec
:name "Lucy Lambda"
;; A string email, optional
:email "lucylamda@email.com"
;; A string URL for the author's homepage, optional
:url "https://lucylambda.com"}
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; Automatically added keys (you don't have to add these manually)
;; The URL vector of keywords is copied from the key into the map itself by Nuzzle
;; Always present
:nuzzle/url [:blog :learning-clojure]
;; A function that can access the whole page map, takes zero or one arguments
;; Zero arguments returns a list of all pages
;; One argument returns a page map with given vector URL
;; One argument with options map argument {:children true} returns a list of all children pages for given vector URL
;; Nuzzle adds this so your render-page function doesn't have to accept more than one argument
;; Always present
:nuzzle/get-pages (fn get-pages ...)}
Each function under the :nuzzle/render-page
key must turn that page map into Hiccup. The function must take one parameter (a page entry map). It can return a vector of Hiccup (more flexible) or a string of HTML (wrapped with nuzzle.hiccup/raw-html
)).
In Nuzzle, all strings in the Hiccup are automatically escaped, so if you want to add a string of raw HTML, use the
nuzzle.hiccup/raw-html
wrapper function like so:(raw-html "<h1>Title</h1>")
.
Hiccup is a method for representing HTML using Clojure data-structures. It comes from the original Hiccup library written by James Reeves. For a quick guide to Hiccup, check out this lightning tutorial.
With many static site generators, accessing global data inside markup templates can be painful. Since Nuzzle is just playing with data and functions, this problem becomes much easier to solve.
Instead of complicating your page rendering functions with multiple required arguments, Nuzzle adds a function to each page entry map under the key :nuzzle/get-pages
. This function can access the whole pages map, making referencing other pages a breeze.
;; Get a list of every page
(get-pages)
;; Get a single page
(get-pages [:blog-posts])
;; Get a list of all the direct children of a page
(get-pages [:blog-posts] :children true)
The get-pages
function will always return pages which also have the :nuzzle/get-pages
key attached. This naturally lends itself to a convention where most Hiccup-generating functions can accept a page entry map as its first or only argument while still being able to access any data in your whole site if need be.