2020-09-25 23:45:58 +00:00
|
|
|
|
---
|
|
|
|
|
title: Making a blog from scratch
|
|
|
|
|
---
|
|
|
|
|
|
2024-02-15 17:56:40 +00:00
|
|
|
|
> **achille** is currently undergoing a *full rewrite*, that you can keep track of
|
|
|
|
|
> on [github](https://github.com/flupe/achille). I just figured out the last missing
|
|
|
|
|
> bits needed to make it *truly* easy to use, and way more powerful. Stay tuned!
|
|
|
|
|
>
|
|
|
|
|
> The following page is largely outdated, as the syntax and internals *will change*.
|
|
|
|
|
|
2020-09-25 23:45:58 +00:00
|
|
|
|
# Making a blog from scratch
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
In this tutorial we'll see how to use **achille** for a simple blog generator.
|
|
|
|
|
|
|
|
|
|
## Content structure
|
|
|
|
|
|
|
|
|
|
The first step is to settle on the content structure.
|
|
|
|
|
For a blog, we will simply store each article in a separate markdown file,
|
|
|
|
|
inside the `posts/` folder.
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
content
|
|
|
|
|
└── posts
|
|
|
|
|
├── 2020-04-13-hello-world.md
|
|
|
|
|
├── 2020-04-14-another-article.md
|
|
|
|
|
└── 2020-05-21-some-more.md
|
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
We require every article to have a title in their Front Matter header.
|
|
|
|
|
We also give them the possibility to specify a summary.
|
|
|
|
|
|
|
|
|
|
```md
|
|
|
|
|
---
|
|
|
|
|
title: Hello World!
|
|
|
|
|
draft: true
|
|
|
|
|
---
|
|
|
|
|
|
|
|
|
|
This is my first article on *this* blog.
|
|
|
|
|
It is powered by [achille](https://acatalepsie.fr/projects/achille).
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## HTML template
|
|
|
|
|
|
|
|
|
|
Then we figure out how we want to render our blog posts. We need to produce
|
|
|
|
|
HTML, and achille doesn't care how we do it, so we are free to use any library
|
|
|
|
|
that suits us. My personal favorite is `lucid` so we'll use that here.
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
```hs
|
|
|
|
|
{-# LANGUAGE BlockArguments, OverloadedStrings #-}
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
import Lucid
|
2020-09-25 23:45:58 +00:00
|
|
|
|
import Data.Text (Text)
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
template :: Html () -> Html ()
|
|
|
|
|
template content = doctypehtml_ do
|
|
|
|
|
head_ do
|
|
|
|
|
meta_ [charset_ "utf-8"]
|
|
|
|
|
title_ "Jenna's Weblog"
|
|
|
|
|
body_ do
|
|
|
|
|
header_ "Jenna's Weblog"
|
|
|
|
|
content
|
|
|
|
|
footer_ "© Jenna 2020"
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
We also tell achille how `Html ()` should be written on disk.
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
```hs
|
|
|
|
|
import Achille.Writable as Writable (Writable)
|
|
|
|
|
|
|
|
|
|
instance Writable (Html ()) where
|
|
|
|
|
write to = Writable.write to . renderBS
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
## Processing articles
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
Now let's define what kind of information is associated with an article.
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
```hs
|
|
|
|
|
{-# LANGUAGE DeriveGeneric, DeriveAny #-}
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
import GHC.Generics (Generic)
|
|
|
|
|
import Data.Aeson (FromJSON)
|
|
|
|
|
import Data.Binary (Binary)
|
|
|
|
|
import Data.Time (UTCTime)
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
data PostMeta = PostMeta
|
|
|
|
|
{ title :: Text
|
|
|
|
|
, summary :: Maybe Text
|
|
|
|
|
, draft :: Maybe Bool
|
|
|
|
|
} deriving (Generic, Eq, FromJSON)
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
data Post = Post
|
|
|
|
|
{ postTitle :: Text
|
|
|
|
|
, postSummary :: Maybe Text
|
|
|
|
|
, postIsDraft :: Bool
|
|
|
|
|
, postPath :: FilePath
|
|
|
|
|
} deriving (Generic, Eq, Binary)
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
renderPost :: Post -> Text -> Html ()
|
|
|
|
|
renderPost post content = template $
|
|
|
|
|
article_ do
|
|
|
|
|
header_ $ h1 (toHtml $ postTitle post)
|
|
|
|
|
toHtmlRaw content
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
Because we derive `FromJSON` for `PostMeta`, we are now able to load a Front
|
|
|
|
|
Matter header and convert it directly to a value of type `PostMeta`. **Correct
|
|
|
|
|
metadata is thus enforced**. `Post` is a datatype containing all the processed
|
|
|
|
|
information about an article, and we derive `Binary` so that we can cache it later.
|
|
|
|
|
Notice how we do not store the article content here, there's no use.
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
Now we simply need a recipe for reading an article, processing it and rendering
|
|
|
|
|
the appropriate HTML output file:
|
|
|
|
|
|
|
|
|
|
```hs
|
|
|
|
|
import System.FilePath ((-<.>))
|
|
|
|
|
import Data.Maybe (fromMaybe)
|
|
|
|
|
|
|
|
|
|
processPost :: Recipe IO FilePath Post
|
|
|
|
|
processPost = do
|
|
|
|
|
outputPath <- (-<.> "html") <$> getInput
|
|
|
|
|
(meta, doc) <- readPandoc
|
|
|
|
|
|
|
|
|
|
let post = Post { postTitle = title meta
|
|
|
|
|
, postSummary = summary meta
|
|
|
|
|
, postIsDraft = fromMaybe False (draft meta)
|
|
|
|
|
, postPath = outputPath
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
renderPandoc doc <&> renderPost post >>= write outputPath
|
|
|
|
|
|
|
|
|
|
return post
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
## Rendering the index
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
Ok, we now have a recipe for building individual articles, however we want to
|
|
|
|
|
be able to display them on the index. We need to filter out drafts. And we need
|
|
|
|
|
to sort them from most recent to oldest.
|
|
|
|
|
|
|
|
|
|
```hs
|
|
|
|
|
import Control.Monad (forM_, mapM_)
|
|
|
|
|
|
|
|
|
|
processPosts :: Task IO ()
|
|
|
|
|
processPosts = do
|
|
|
|
|
posts <- match "posts/*.md" processPost
|
|
|
|
|
let visible = filter (not postIsDraft) posts
|
|
|
|
|
watch visible $ write "index.html" (renderIndex visible)
|
|
|
|
|
|
|
|
|
|
renderIndex :: [Post] -> Html ()
|
|
|
|
|
renderIndex posts = do
|
|
|
|
|
h2_ "Latest articles"
|
|
|
|
|
ul_ $ forM_ posts \ post -> li_ do
|
|
|
|
|
a_ [href_ (fromString $ postPath post)] (toHtml $ postTitle post)
|
|
|
|
|
forM_ (postSummary post) \summary -> p_ $ toHtml summary
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
## Wrapping things up
|
|
|
|
|
|
|
|
|
|
Finally, we can forward this task to the top-level achille runner:
|
2020-09-25 23:45:58 +00:00
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
```hs
|
2020-09-25 23:45:58 +00:00
|
|
|
|
main :: IO ()
|
2020-09-28 00:58:01 +00:00
|
|
|
|
main = achille processPosts
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Compile into an executable `blog` and run the command `blog build`. Hold and
|
|
|
|
|
behold, if you look into the `_site/` folder, there should be:
|
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
_site
|
|
|
|
|
├── index.html
|
|
|
|
|
└── posts
|
|
|
|
|
├── 2020-04-13-hello-world.html
|
|
|
|
|
├── 2020-04-14-another-article.html
|
|
|
|
|
└── 2020-05-21-some-more.html
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
What's more, your generator is **incremental**. Modify a single article, and
|
|
|
|
|
trigger a rebuild.
|
|
|
|
|
|
|
|
|
|
```bash
|
|
|
|
|
$ touch -m content/posts/2020-04-13-hello-world.md
|
|
|
|
|
$ blog build
|
2020-09-25 23:45:58 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-09-28 00:58:01 +00:00
|
|
|
|
You should see that only this article is rebuilt. And because we haven't
|
|
|
|
|
actually changed neither the title nor the summary, the index hasn't been rebuilt.
|
|
|
|
|
Magic!
|