Hakyll on Netlify

Published: September 1, 2019, updated: January 4, 2025

Why I never write any posts

If you are like me, you are more busy switching between static site generators than actually writing posts for your blog. If you take a closer look at the commit history for this website, you may notice that I have switched site generators many times (including to and away from my own site generator) But who can resist the appeal of yet another static site generator that promises some combination of

This time I have chosen a fabulous site generator toolkit called Hakyll. Hakyll is a static site generator implemented in Haskell. It uses a DSL for configuration and allows its user to build their site in almost any way they like. Entirely custom URL schema? No problem. Compile the same file using many compilers, including pandoc? No problem. Compile to one easy to deploy static executable, including all your configuration? No problem.

Now recalling the advantages that every shiny static site generator out there promises to have, we quickly notice that Hakyll has

As you can see, there are a lot of places where Hakyll can only shine once you spend a considerable amount of time configuring everything. But what might sound like a big drawback is actually Hakyll’s biggest advantage. Few site generators give you the ability to make something tailored exactly to what you are doing. The next level of course would be to make your own site generator.

And, this is what I’ve done before. My blog once was a Flask site turned into a bundle of static HTML using Frozen Flask. If Jekyll is on one end of the site generator spectrum – thanks to its batteries-included approach, then building your own static site generator in Python is on the other end of the spectrum, even with the advantage of using powerful Jinja 2 templates inside Flask. You are forced to do everything yourself. That includes the boring stuff such as watching your Markdown files and other assets and performing live rebuilds for local testing.

Of course, creating your own site generator is a valuable experience. It can teach you a lot about how to design a good build system. At the same time, we often need some more complicated features that just take a lot of time to write from scratch by yourself. For example, one of those advanced features is tight integration with external tools and using those tools to provide build versions of the same blog post that is written in Markdown. Exactly this is what ultimately led me to choose Hakyll for my personal website.

My ideal workflow for blog posts and other pages on this site would look as follows

  1. Write a rough draft of what I want to say in Markdown.
  2. Start a live build version and check how my initial draft feels on screen.
  3. Create illustrations and charts as needed to illustrate my point, while
  4. continuing to edit and write my post.
  5. Publish by doing a git commit followed by git push.

With Jekyll or any other popular static site generator especially the last step is simple. Netlify support Jekyll well and can be set up within 10 minutes. Here, I am not willing to compromise and I would like to keep things simple and reliable. Where I always felt Jekyll was lacking, was in external tooling support. I am a big fan of pandoc, Graphviz and MscGen. Pandoc allows me to use the best of LaTeX, while pandoc filters allow me to easily add external tools, including Graphviz and MscGen.

For a while I used jekyll-graphviz, and it worked fine. Furthermore, there is a second plugin for Jekyll, that allows you to use pandoc. But it doesn’t go so well with jekyll-graphviz, and I really want to avoid using any non-pandoc-syntax. And Jekyll plugins work by creating custom liquid tags, so that I would need to type

{% graph some graph title %}
a -- b
b -- c
c -- a
{% endgraph %}

every time. The preferred way for pandoc filters would be more like this:

~~~
digraph G {Hello->World}
~~~

(Source)

Enter Haskell and Hakyll

At some point when you go down the customization road, you may discover that you need something that was made with extreme customizability in mind. That means, no implicit configuration, a better configuration language, and ideally something that blends configuring the site with writing your own customizations.

Haskell’s approach to this has always been using Domain Specific Languages (DSLs). On the one hand, you have a library of words that can do a few things really well, but since you are writing plain Haskell, you can drop into your own custom code anytime you want. Jekyll, which is based on Ruby, would have you drop your plugin files in certain folders, and that would work fine, but the degree of implicitness was always nagging me.

Then, we should also consider that pandoc just happens to be written in Haskell, and there is this wonderful static site generation toolkit called Hakyll. After some careful pondering, I decided to venture into static site generator customization land for the fifth time or so.

And here we are, roughly 2000 lines (delta) of fresh code later. I started off using the sample Hakyll configuration and I’ve added a few things:

What does all this buy me? Fast site building speeds. The static site generator code for Hakyll and pandoc are compiled into one executable, including all the filters that I’ve defined. Whereas in Jekyll, using pandoc with filters would have meant Jekyll spawning a pandoc process per document, and each pandoc process spawning a filter configured via the --filter option. Here, the main cost is not forking itself, but the startup time for each child process.

Then, in the Hakyll version I’ve just created one big switch statement that checks what to do when it encounters a ``` block:

renderAll :: Block -> IO Block
renderAll cblock@(CodeBlock (id, classes, attrs) content)
  | "msc" `elem` classes =
    let dest = fileName4Code "mscgen" (T.pack content) (Just "msc")
     in do ensureFile dest >> writeFile dest content
           % Do mscgen things here
  | "graphviz" `elem` classes =
    let dest = fileName4Code "graphviz" (T.pack content) (Just "dot")
     in do ensureFile dest >> writeFile dest content
           % Do Graphviz things here
  | otherwise = return cblock
  where
    image img =
      Para
        % Insert the image here
renderAll x = return x

The full code can be found here. If I ever want to add any new diagramming tools, I can just extend the renderAll function.

Then, the question remains: why bother with all this?

Writing in freedom

Every few years or so, a blogging platform announces itself and promises to be the one place where writers can gather and exchange stories, while unburdened with paywalls, dubious content licensing, sudden changes in content policies, and so on.

Thus far, every single blogging platform has disappointed in some way. Tumblr has suddenly decided to strictly forbid all adult content. Medium started adding paywalls and pesters users even when reading free articles.

When I write text and publish it, then I express my opinions and views. I take some time off my day to sit down and share my thoughts with others, hoping that I can find someone out there with whom I can engage in a deep conversation. I am convinced that engaging in a meaningful conversation requires freedom. Freedom from any platform agenda, freedom from the pressures of capital, and the freedom of knowing that your opinions and that what you express can reach others exactly as you intend.

If I can influence how my content is rendered, what the impression people have when they reach my site, and ultimately what ideas stick with my audience, then I feel I have freedom. This freedom of course also comes with the downside that if I’m clumsy at designing my site, not optimizing for usability or accessibility, if my site goes down, then my ideas have been expressed in vain. The upside though, makes it worth it. If I can make my site work and reach someone, I have reached my goal.

Tags

I would be thrilled to hear from you! Please share your thoughts and ideas with me via email.

Back to Index