improved blog tutorial
This commit is contained in:
parent
47f4f8ed7c
commit
d286f603a6
|
@ -4,9 +4,13 @@ title: Making a blog from scratch
|
||||||
|
|
||||||
# Making a blog from scratch
|
# Making a blog from scratch
|
||||||
|
|
||||||
Let's see how to use **achille** for making a static site generator for a blog.
|
In this tutorial we'll see how to use **achille** for a simple blog generator.
|
||||||
First we decide what will be the structure of our source directory.
|
|
||||||
We choose the following:
|
## 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.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
content
|
content
|
||||||
|
@ -16,87 +20,165 @@ content
|
||||||
└── 2020-05-21-some-more.md
|
└── 2020-05-21-some-more.md
|
||||||
```
|
```
|
||||||
|
|
||||||
We define the kind of metadata we want to allow in the frontmatter header
|
We require every article to have a title in their Front Matter header.
|
||||||
of our markdown files:
|
We also give them the possibility to specify a summary.
|
||||||
|
|
||||||
```haskell
|
```md
|
||||||
{-# LANGUAGE DeriveGeneric #-}
|
---
|
||||||
{-# LANGUAGE DeriveAny #-}
|
title: Hello World!
|
||||||
|
draft: true
|
||||||
|
---
|
||||||
|
|
||||||
import GHC.Generics
|
This is my first article on *this* blog.
|
||||||
import Data.Aeson
|
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.
|
||||||
|
|
||||||
|
```hs
|
||||||
|
{-# LANGUAGE BlockArguments, OverloadedStrings #-}
|
||||||
|
|
||||||
|
import Lucid
|
||||||
import Data.Text (Text)
|
import Data.Text (Text)
|
||||||
|
|
||||||
data Meta = Meta
|
template :: Html () -> Html ()
|
||||||
{ title :: Text
|
template content = doctypehtml_ do
|
||||||
} deriving (Generic, FromJSON)
|
|
||||||
```
|
|
||||||
|
|
||||||
This way we enfore correct metadata when retrieving the content of our files.
|
|
||||||
Every markdown file will have to begin with the following header for our
|
|
||||||
generator to proceed:
|
|
||||||
|
|
||||||
```markdown
|
|
||||||
---
|
|
||||||
title: My first blogpost!
|
|
||||||
---
|
|
||||||
```
|
|
||||||
|
|
||||||
Then we create a generic template for displaying a page, thanks to lucid:
|
|
||||||
|
|
||||||
```haskell
|
|
||||||
{-# LANGUAGE OverloadedStrings #-}
|
|
||||||
{-# LANGUAGE BlockArguments #-}
|
|
||||||
|
|
||||||
import Lucid.Html5
|
|
||||||
|
|
||||||
renderPost :: Text -> Text -> Html ()
|
|
||||||
renderPost title content = wrapContent do
|
|
||||||
h1_ $ toHtml title
|
|
||||||
toHtmlRaw content
|
|
||||||
|
|
||||||
renderIndex :: [(Text, FilePath)] -> Html ()
|
|
||||||
renderIndex = wrapContent .
|
|
||||||
ul_ . mconcat . map \(title, path) ->
|
|
||||||
li_ $ a_ [href_ path] $ toHtml title
|
|
||||||
|
|
||||||
wrapContent :: Html () -> Html ()
|
|
||||||
wrapContent content = doctypehtml_ do
|
|
||||||
head_ do
|
head_ do
|
||||||
meta_ [charset_ "utf-8"]
|
meta_ [charset_ "utf-8"]
|
||||||
title_ "my very first blog"
|
title_ "Jenna's Weblog"
|
||||||
|
|
||||||
body_ do
|
body_ do
|
||||||
header_ $ h1_ "BLOG"
|
header_ "Jenna's Weblog"
|
||||||
content_
|
content
|
||||||
|
footer_ "© Jenna 2020"
|
||||||
```
|
```
|
||||||
|
|
||||||
We define a recipe for rendering every post:
|
We also tell achille how `Html ()` should be written on disk.
|
||||||
|
|
||||||
```haskell
|
```hs
|
||||||
buildPosts :: Task IO [(String, FilePath)]
|
import Achille.Writable as Writable (Writable)
|
||||||
buildPosts =
|
|
||||||
match "posts/*.md" do
|
instance Writable (Html ()) where
|
||||||
(Meta title, text) <- compilePandocMetadata
|
write to = Writable.write to . renderBS
|
||||||
saveFileAs (-<.> "html") (renderPost title text)
|
|
||||||
<&> (title,)
|
|
||||||
```
|
```
|
||||||
|
|
||||||
We can define a simple recipe for rendering the index, given a list of posts:
|
## Processing articles
|
||||||
|
|
||||||
```haskell
|
Now let's define what kind of information is associated with an article.
|
||||||
buildIndex :: [(Text, FilePath)] -> Task IO FilePath
|
|
||||||
buildIndex posts =
|
```hs
|
||||||
save (renderIndex posts) "index.html"
|
{-# LANGUAGE DeriveGeneric, DeriveAny #-}
|
||||||
|
|
||||||
|
import GHC.Generics (Generic)
|
||||||
|
import Data.Aeson (FromJSON)
|
||||||
|
import Data.Binary (Binary)
|
||||||
|
import Data.Time (UTCTime)
|
||||||
|
|
||||||
|
data PostMeta = PostMeta
|
||||||
|
{ title :: Text
|
||||||
|
, summary :: Maybe Text
|
||||||
|
, draft :: Maybe Bool
|
||||||
|
} deriving (Generic, Eq, FromJSON)
|
||||||
|
|
||||||
|
data Post = Post
|
||||||
|
{ postTitle :: Text
|
||||||
|
, postSummary :: Maybe Text
|
||||||
|
, postIsDraft :: Bool
|
||||||
|
, postPath :: FilePath
|
||||||
|
} deriving (Generic, Eq, Binary)
|
||||||
|
|
||||||
|
renderPost :: Post -> Text -> Html ()
|
||||||
|
renderPost post content = template $
|
||||||
|
article_ do
|
||||||
|
header_ $ h1 (toHtml $ postTitle post)
|
||||||
|
toHtmlRaw content
|
||||||
```
|
```
|
||||||
|
|
||||||
Then, it's only a matter of composing the recipes and giving them to **achille**:
|
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.
|
||||||
|
|
||||||
```haskell
|
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
|
||||||
|
```
|
||||||
|
|
||||||
|
## Rendering the index
|
||||||
|
|
||||||
|
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
|
||||||
|
```
|
||||||
|
|
||||||
|
## Wrapping things up
|
||||||
|
|
||||||
|
Finally, we can forward this task to the top-level achille runner:
|
||||||
|
|
||||||
|
```hs
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
main = achille do
|
main = achille processPosts
|
||||||
posts <- buildPosts
|
|
||||||
buildIndex posts
|
|
||||||
```
|
```
|
||||||
|
|
||||||
And that's it, you now have a very minimalist incremental blog generator!
|
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
|
||||||
|
```
|
||||||
|
|
||||||
|
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!
|
||||||
|
|
Loading…
Reference in New Issue