The entry list is now much more condensed. It's maybe a little *too* condensed, but already closer to what I'm looking for. Note: A new "note" post type has snuck in and can now be used for random musings or comments on previous entries. Notes do not show up in the Atom feed. Change-Id: I920c0c7650937474b8a5f30cba78416554d523ce Reviewed-on: https://cl.tvl.fyi/c/depot/+/8806 Reviewed-by: tazjin <tazjin@tvl.su> Autosubmit: tazjin <tazjin@tvl.su> Tested-by: BuildkiteCI
		
			
				
	
	
		
			97 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			Nix
		
	
	
	
	
	
			
		
		
	
	
			97 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			Nix
		
	
	
	
	
	
| # Assembles the website index and configures an nginx instance to
 | |
| # serve it.
 | |
| #
 | |
| # The website is made up of a simple header&footer and content
 | |
| # elements for things such as blog posts and projects.
 | |
| #
 | |
| # Content for the blog is in //users/tazjin/blog instead of here.
 | |
| { depot, lib, pkgs, ... }@args:
 | |
| 
 | |
| with depot;
 | |
| with nix.yants;
 | |
| 
 | |
| let
 | |
|   inherit (builtins) readFile replaceStrings sort;
 | |
|   inherit (pkgs) writeFile runCommand;
 | |
| 
 | |
|   # The different types of entries on the homepage.
 | |
|   entryClass = enum "entryClass" [
 | |
|     "blog"
 | |
|     "project"
 | |
|     "note"
 | |
|     "misc"
 | |
|   ];
 | |
| 
 | |
|   # The definition of a single entry.
 | |
|   entry = struct "entry" {
 | |
|     class = entryClass;
 | |
|     title = option string;
 | |
|     url = option string;
 | |
|     date = int; # epoch
 | |
|     description = option string;
 | |
|   };
 | |
| 
 | |
|   escape = replaceStrings [ "<" ">" "&" "'" ] [ "<" ">" "&" "'" ];
 | |
| 
 | |
|   postToEntry = defun [ web.blog.post entry ] (post: {
 | |
|     class = "blog";
 | |
|     title = post.title;
 | |
|     url = "/blog/${post.key}";
 | |
|     date = post.date;
 | |
|     description = post.description or "Blog post from ${formatDate post.date}";
 | |
|   });
 | |
| 
 | |
|   formatDate = defun [ int string ] (date: readFile (runCommand "date" { } ''
 | |
|     date --date='@${toString date}' '+%Y-%m-%d' | tr -d '\n' > $out
 | |
|   ''));
 | |
| 
 | |
|   entryUrl = defun [ entry string ] (entry:
 | |
|     if entry.class == "note"
 | |
|     then "#${toString entry.date}"
 | |
|     else entry.url
 | |
|   );
 | |
| 
 | |
|   hasDescription = defun [ entry bool ] (entry:
 | |
|     ((entry ? description) && (entry.description != null))
 | |
|   );
 | |
| 
 | |
|   entryTitle = defun [ entry string ] (entry:
 | |
|     let
 | |
|       optionalColon = lib.optionalString (hasDescription entry) ":";
 | |
|       titleText =
 | |
|         if (!(entry ? title) && (entry.class == "note"))
 | |
|         then "[${formatDate entry.date}]"
 | |
|         else lib.optionalString (entry ? title) ((escape entry.title) + optionalColon);
 | |
|     in
 | |
|     lib.optionalString (titleText != "")
 | |
|       ''<span class="entry-title ${entry.class}">${titleText}</span>''
 | |
|   );
 | |
| 
 | |
|   entryToDiv = defun [ entry string ] (entry: ''
 | |
|     <a href="${entryUrl entry}" id="${toString entry.date}" class="entry">
 | |
|       ${entryTitle entry}
 | |
|       ${
 | |
|         lib.optionalString (hasDescription entry)
 | |
|         "<span class=\"entry-description\">${escape entry.description}</span>"
 | |
|       }
 | |
|     </a>
 | |
|   '');
 | |
| 
 | |
|   index = entries: pkgs.writeText "index.html" (lib.concatStrings (
 | |
|     [ (builtins.readFile ./header.html) ]
 | |
|     ++ (map entryToDiv (sort (a: b: a.date > b.date) entries))
 | |
|     ++ [ (builtins.readFile ./footer.html) ]
 | |
|   ));
 | |
| 
 | |
|   pageEntries = import ./entries.nix;
 | |
|   homepage = index ((map postToEntry users.tazjin.blog.posts) ++ pageEntries);
 | |
|   atomFeed = import ./feed.nix (args // { inherit entry pageEntries; });
 | |
| in
 | |
| runCommand "website" { } ''
 | |
|   mkdir $out
 | |
|   cp ${homepage} $out/index.html
 | |
|   cp ${atomFeed} $out/feed.atom
 | |
|   mkdir $out/static
 | |
|   cp -r ${depot.web.static}/* $out/static
 | |
|   cp -rf ${./static}/* $out/static
 | |
| ''
 |