refactor(web/blog): Move atom feed creation logic to //web/blog
This was previously all inside of my personal homepage configuration, but that's not really where it belongs. This moves the blog post -> feed entry logic to //web/blog and moves some other minor logic (like entry order) into the atom feed implementation itself. Change-Id: Idde0241c48e979580de73f2b9afd04e6ca7f4c9a Reviewed-on: https://cl.tvl.fyi/c/depot/+/3770 Tested-by: BuildkiteCI Reviewed-by: sterni <sternenseemann@systemli.org>
This commit is contained in:
parent
00ae396eeb
commit
f360bbdcf0
3 changed files with 38 additions and 27 deletions
|
@ -4,25 +4,11 @@
|
||||||
with depot.nix.yants;
|
with depot.nix.yants;
|
||||||
|
|
||||||
let
|
let
|
||||||
inherit (builtins) map readFile sort foldl';
|
inherit (builtins) map readFile;
|
||||||
inherit (lib) max singleton;
|
inherit (lib) max singleton;
|
||||||
inherit (pkgs) writeText;
|
inherit (pkgs) writeText;
|
||||||
inherit (depot.nix) renderMarkdown;
|
|
||||||
inherit (depot.web) blog atom-feed;
|
inherit (depot.web) blog atom-feed;
|
||||||
|
|
||||||
postToEntry = defun [ blog.post atom-feed.entry ] (post: rec {
|
|
||||||
id = "https://tazj.in/blog/${post.key}";
|
|
||||||
title = post.title;
|
|
||||||
content = readFile (renderMarkdown post.content);
|
|
||||||
published = post.date;
|
|
||||||
updated = post.updated or post.date;
|
|
||||||
|
|
||||||
links = singleton {
|
|
||||||
rel = "alternate";
|
|
||||||
href = id;
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
pageEntryToEntry = defun [ entry atom-feed.entry ] (e: {
|
pageEntryToEntry = defun [ entry atom-feed.entry ] (e: {
|
||||||
id = "tazjin:${e.class}:${toString e.date}";
|
id = "tazjin:${e.class}:${toString e.date}";
|
||||||
updated = e.date;
|
updated = e.date;
|
||||||
|
@ -36,16 +22,13 @@ let
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
allEntries = (map postToEntry depot.users.tazjin.blog.posts)
|
allEntries = (map blog.toFeedEntry depot.users.tazjin.blog.posts)
|
||||||
++ (map pageEntryToEntry pageEntries);
|
++ (map pageEntryToEntry pageEntries);
|
||||||
|
|
||||||
mostRecentlyUpdated = foldl' max 0 (map (e: e.updated) allEntries);
|
|
||||||
|
|
||||||
feed = {
|
feed = {
|
||||||
id = "https://tazj.in/";
|
id = "https://tazj.in/";
|
||||||
title = "tazjin's interblag";
|
title = "tazjin's interblag";
|
||||||
subtitle = "my posts, projects and other interesting things";
|
subtitle = "my posts, projects and other interesting things";
|
||||||
updated = mostRecentlyUpdated;
|
|
||||||
rights = "© 2020 tazjin";
|
rights = "© 2020 tazjin";
|
||||||
authors = [ "tazjin" ];
|
authors = [ "tazjin" ];
|
||||||
|
|
||||||
|
@ -54,6 +37,6 @@ let
|
||||||
href = "https://tazjin/feed.atom";
|
href = "https://tazjin/feed.atom";
|
||||||
};
|
};
|
||||||
|
|
||||||
entries = sort (a: b: a.published > b.published) allEntries;
|
entries = allEntries;
|
||||||
};
|
};
|
||||||
in writeText "feed.atom" (atom-feed.renderFeed feed)
|
in writeText "feed.atom" (atom-feed.renderFeed feed)
|
||||||
|
|
|
@ -5,8 +5,8 @@
|
||||||
with depot.nix.yants;
|
with depot.nix.yants;
|
||||||
|
|
||||||
let
|
let
|
||||||
inherit (builtins) map readFile replaceStrings;
|
inherit (builtins) foldl' map readFile replaceStrings sort;
|
||||||
inherit (lib) concatStrings concatStringsSep removeSuffix;
|
inherit (lib) concatStrings concatStringsSep max removeSuffix;
|
||||||
inherit (pkgs) runCommandNoCC;
|
inherit (pkgs) runCommandNoCC;
|
||||||
|
|
||||||
# 'link' describes a related link to a feed, or feed element.
|
# 'link' describes a related link to a feed, or feed element.
|
||||||
|
@ -67,8 +67,9 @@ let
|
||||||
title = string;
|
title = string;
|
||||||
|
|
||||||
# Indicates the last time the feed was modified in a significant
|
# Indicates the last time the feed was modified in a significant
|
||||||
# way (in seconds since epoch). Recommended element.
|
# way (in seconds since epoch). Will be calculated based on most
|
||||||
updated = int;
|
# recently updated entry if unset.
|
||||||
|
updated = option int;
|
||||||
|
|
||||||
# Entries contained within the feed.
|
# Entries contained within the feed.
|
||||||
entries = list entry;
|
entries = list entry;
|
||||||
|
@ -127,17 +128,23 @@ let
|
||||||
</entry>
|
</entry>
|
||||||
'');
|
'');
|
||||||
|
|
||||||
|
mostRecentlyUpdated = defun [ (list entry) int ] (entries:
|
||||||
|
foldl' max 0 (map (e: e.updated) entries)
|
||||||
|
);
|
||||||
|
|
||||||
|
sortEntries = sort (a: b: a.published > b.published);
|
||||||
|
|
||||||
renderFeed = defun [ feed string ] (f: ''
|
renderFeed = defun [ feed string ] (f: ''
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<feed xmlns="http://www.w3.org/2005/Atom">
|
<feed xmlns="http://www.w3.org/2005/Atom">
|
||||||
${elem "id" f.id}
|
${elem "id" f.id}
|
||||||
${elem "title" f.title}
|
${elem "title" f.title}
|
||||||
${elem "updated" (renderEpoch f.updated)}
|
${elem "updated" (renderEpoch (f.updated or (mostRecentlyUpdated f.entries)))}
|
||||||
${concatStringsSep "\n" (map renderAuthor (f.authors or []))}
|
${concatStringsSep "\n" (map renderAuthor (f.authors or []))}
|
||||||
${if f ? subtitle then elem "subtitle" f.subtitle else ""}
|
${if f ? subtitle then elem "subtitle" f.subtitle else ""}
|
||||||
${if f ? rights then elem "rights" f.rights else ""}
|
${if f ? rights then elem "rights" f.rights else ""}
|
||||||
${concatStrings (map renderLink (f.links or []))}
|
${concatStrings (map renderLink (f.links or []))}
|
||||||
${concatStrings (map renderEntry f.entries)}
|
${concatStrings (map renderEntry (sortEntries f.entries))}
|
||||||
</feed>
|
</feed>
|
||||||
'');
|
'');
|
||||||
in {
|
in {
|
||||||
|
|
|
@ -7,6 +7,11 @@
|
||||||
with depot.nix.yants;
|
with depot.nix.yants;
|
||||||
|
|
||||||
let
|
let
|
||||||
|
inherit (builtins) readFile;
|
||||||
|
inherit (depot.nix) renderMarkdown;
|
||||||
|
inherit (depot.web) atom-feed;
|
||||||
|
inherit (lib) singleton;
|
||||||
|
|
||||||
# Type definition for a single blog post.
|
# Type definition for a single blog post.
|
||||||
post = struct "blog-post" {
|
post = struct "blog-post" {
|
||||||
key = string;
|
key = string;
|
||||||
|
@ -31,9 +36,25 @@ let
|
||||||
oldKey = option string;
|
oldKey = option string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
# Rendering fragments for the HTML version of the blog.
|
||||||
fragments = import ./fragments.nix args;
|
fragments = import ./fragments.nix args;
|
||||||
|
|
||||||
|
# Functions for generating feeds for these blogs using //web/atom-feed.
|
||||||
|
toFeedEntry = defun [ post atom-feed.entry ] (post: rec {
|
||||||
|
id = "https://tazj.in/blog/${post.key}";
|
||||||
|
title = post.title;
|
||||||
|
content = readFile (renderMarkdown post.content);
|
||||||
|
published = post.date;
|
||||||
|
updated = post.updated or post.date;
|
||||||
|
|
||||||
|
links = singleton {
|
||||||
|
rel = "alternate";
|
||||||
|
href = id;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
in {
|
in {
|
||||||
inherit post;
|
inherit post toFeedEntry;
|
||||||
inherit (fragments) renderPost;
|
inherit (fragments) renderPost;
|
||||||
includePost = post: !(fragments.isDraft post) && !(fragments.isUnlisted post);
|
includePost = post: !(fragments.isDraft post) && !(fragments.isUnlisted post);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue