AsciiDoc, Liquid and Jekyll

A couple of days ago I was giving a little update to my website and I needed a way to create and inject custom HTML into some of my posts.

Hehe, something tells me that you are talking about me and this cool chat!

Yeah... now let me talk a bit more about the website

My website is entirely statically generated, I’m specifically using Jekyll which is a very well known static site generator. It is served through GitHub Pages and I use Cloudflare to cache and route traffic to the GitHub page.

At the time I decided to go for Jeyll because I wanted to learn its internals and because it seemed that it offered a lot of flexibility for technical blogs like mine. Jekyll has a huge community and there are a lot of useful plugins to integrate in the framework, one of my favorite is the AsciiDoc plugin.

For those of you that don’t know what AsciiDoc:

AsciiDoc is a plain text markup language for writing technical content. It’s packed with semantic elements and equipped with features to modularize and reuse content. AsciiDoc content can be composed using a text editor, managed in a version control system, and published to multiple output formats.

As you might guess, I am writing all my articles using AsciiDoc because it gives me even more flexibility on top of what Jekyll already offers. If you want to use AsciiDoc too you need to declare the plugin in the _config.yml of your root directory

plugins:
  - jekyll-asciidoc

Let’s see if Jekyll really is as flexible as I initially thought.

To create the chat section above I need to find a way to make Jekyll generate HTML code from a specific syntax, there is really no other way around it in this scenario because exploiting generated HTML classes and CSS is not really feasible here, at least for me. Since I’m going to use these chat-like sections a lot in the future, I would like to end up with a construct that is very similar to common AsciiDoc syntax, to write the professor’s chat above I would need to write something like this:

[chat, professor]
--
Hehe, something tells me that you are talking about me and this cool chat!
--

For those of you that are familiar with AsciiDoc, this snippet might be familiar. It’s an AsciiDoc Block. The above snippet would be the ideal final result, but let’s go ahead and see what Jekyll offers out-of-the-box first.

Jekyll has a special feature that enables you to create custom components called includes which could be written in HTML and you call/include them directly in your posts. To use this feature you just need to

  1. Declare your HTML component

  2. Include the component in your post with a special syntax

Every custom component that you whish to include in your post must live under the _includes directory in your root Jekyll folder. I’m going to call mine chat.html and it contains this HTML snippet

<div class="dialog {{ include.character }}" title="{{ include.character }}">
    <div class="dialog-head">
        {% include character.svg %}
    </div>
    <div class="dialog-text">
        {{ include.text | markdownify }}
    </div>
</div>

Before we continue, see those {{ include ... }} and {% include ... %}? That's Liquid templating syntax. With Jekyll you can make use of that pretty much everywhere in the project. For the moment, just consider each element in those bracket as a variable.

These components are HTML files that can contain whatever you’d like them to contain.

<div class="dialog {{ include.character }}" title="{{ include.character | capitalize }}">

This snippet is creating a <div> and also injecting {{ include.character }} as its class, along with the title {{ include.character }}

That looks intuitive, are those variable that you can pass to the component? If so, how?

Remember that each element enclosed in {% %} and {{ }} is a variable? You can pass those vars from the post in which you’d like to insert this custom component, this is how it would look like in the post

{% include chat.html character="matt" text="Hey there, here is some text in a chat!" %}

This is pretty straightforward, Jekyll is going to insert the HTML snippet in the statically generated page and it will also populate the snippet with the variables that you declared in the include construct.

This is the basic approach that you would go through to customize your Jekyll website, what I did not tell you is that this is only going to work if you are using the default Jekyll document language, which is Markdown. That’s unfortunate, because I’ve transitioned every single post of my website to AsciiDoc just a couple of months ago and I don’t plan to move back to Markdown just for this.

What happens if we use the include syntax in AsciiDoc? Maybe we're lucky and it's going to work

AsciiDoc does not support Liquid syntax by default, so what’s going to happen is that you’re going to find {% include chat.html …​ %} verbatim in your post, just like this

{% include chat.html character="matt" text="Hey there! We got a probelm here :(" %}

Not a great start, especially because there are not a lot of similar scenarios out there surprisingly, at least I couldn’t find that much by first googling the problem. My StackOverflow question didn’t receive any answers too, speak volumes since Jekyll and AsciiDoc have a very big community.

I’m alone in this, let’s see if I can find something on GitHub.

The plugin that I’m using to generate HTML from AsciiDoc is jekyll-asciidoc, maybe we can find something interesting in there. By making a project-wide search of the word liquid I immediately get to this documentation page which is just what I am looking for, lucky me.

After reading the docs, it seems like all I need to do is enable Liquid preprocessing by appending :page-liquid: at the top of my post. That is going to parse and generate Liquid code before sending the result to the AsciiDoc generator. Indeed, if we now add that tag at the top and reload the page, we’re going to be presented with this

<div class="dialog matt" title="matt"> <div class="dialog-head"> <svg>…​…​</svg> </div> <div class="dialog-text">Hey there! Finally we made it! This chat that you’re reading is the generated and injected component! </div> </div>

Almost there, now Liquid preprocessing correctly generates my custom HTML component, but it’s not really injected in the page source. That is because the Liquid preprocessor only generates the content, but when everything is passed to the AsciiDoc generator it is interpreted as content text. If we really want to inject that HTML code in the static page, we need a way to tell the AsciiDoc generator that. This is a common feature of the language luckily, AsciiDoc lets you inject raw HTML code in the page, you just need to wrap it in a ++++ block. The final syntax we reached is this

++++
{% include chat.html character="matt" text="Hey there! Finally we made it! This chat that you're reading is the generated and injected component!" %}
++++

Let’s go through the entire generation process:

  • Liquid preprocessing

++++
<div class="dialog matt" title="matt">
    <div class="dialog-head">
        <svg>......</svg>
    </div>
    <div class="dialog-text">
        Hey there! Finally we made it! This chat that you're reading is the generated and injected component!
    </div>
</div>
++++
  • AsciiDoc generation (and HTML injection)

Hey there! Finally we made it! This chat that you're reading is the generated and injected component!

Ok, we made some progress, we are now able to use the include feature Jekyll offers to render custom HTML. But we’re far from the initial AsciiDoc-like syntax that I wanted to achieve. Can we do better?

The greates feature of AsciiDoc probably is its Extension API, which makes the language extremely powerful and extensible.

An extension is a library that enriches the AsciiDoc content either by introducing new syntax or weaving additional features into the output.

This is what we need! It’s also a feature supported by the jekyll-asciidoc plugin. What we could do is create a new extension that recognizes the [chat] block by directly declaring a custom block Asciidoctor::Extension.

But wait, AsciiDoc is written in Ruby, and you don't know Ruby!

I don't, but let's see if I can write something good enough for the job

jekyll-asciidoc plugins docs will look for potential extensions by looking in the _plugins directory of the Jekyll project, so that’s where our extension is going to be saved. This is my chat-extension.rb file

require 'asciidoctor/extensions'

include Asciidoctor

Asciidoctor::Extensions.register do
  block :chat do
    process do |parent, reader, attributes|
      character = attributes.values[1]

      svg = File.read("_includes/" + character + ".svg")
      content = reader.lines.join(' ')

      html = %(
        <div class="dialog #{character}" title="#{character.capitalize}">
          <div class="dialog-head">
          #{svg}
          </div>
          <div class="dialog-text">
          <p>#{content}</p>
          </div>
        </div>
      )

      create_pass_block parent, html, {}, :content_model => :raw
    end
  end
end

As you can see, I’m not a magician with Ruby, this is mainly strings manipulation, so it’s not that difficult. Let me go through the code once more

  • I initially take the second field of the syntax block

# [chat, professor]
# --
# ...
# --
character = attributes.values[1] # <- "professor"
  • Load svg from file using the character variable we just read

svg = File.read("_includes/" + character + ".svg")
  • Put the content of the block in a string

# this contains everything that's inside the -- block
#
# [chat, professor]
# --
# Hey there!
# --
content = reader.lines.join(' ') # <- "Hey there!"
  • Inject raw html in page

create_pass_block parent, html, {}, :content_model => :raw

If I now try to replace the original include syntax with

[chat, matt]
--
Hey there! Finally we made it! This chat that you're reading is the generated
and injected component! This time using AsciiDoc Extension
--

I’m going to get

Hey there! Finally we made it! This chat that you're reading is the generated and injected component! This time using AsciiDoc Extension

That is looking really good and a lot less verbose than the initial Jekyll way, if you inspect the page source you’re going to see that AsciiDoc now generates the custom HTML code, as expected. The only issue that I have right now is that the content of the block is not getting parsed since I’m just putting it into a string variable and spitting it out in the HTML variable as-is, but that’s good enough for what I need at the moment.

Hopefully you learned a little bit more about this topic which is not really a big thing out there for some reason, it took me quite a lot of research to get to this result. I guess that Ruby is the only thing blocking me from writing more complex logic for this extension and others yet to come, but I am super satisfied with the AsciiDoc switch, you can do literally everything you want with the language if you start digging into the parsers and extensions.