Edit and public Nix Blog article
This commit is contained in:
		
							parent
							
								
									fe1c05fe46
								
							
						
					
					
						commit
						6145d9e804
					
				| @ -1,7 +1,6 @@ | ||||
| --- | ||||
| title: "Declaratively Deploying Multiple Blog Versions with NixOS and Flakes" | ||||
| date: 2022-04-10T00:24:58-07:00 | ||||
| draft: true | ||||
| tags: ["Hugo", "Nix"] | ||||
| --- | ||||
| 
 | ||||
| @ -40,21 +39,21 @@ I wanted to be able to represent all of this complexity in my NixOS configuratio | ||||
| that's what this post is about! | ||||
| 
 | ||||
| ### Why Flakes | ||||
| This article is about using Nix flakes to manage my configuration. But what is it that made | ||||
| me use flakes? Well, two things: | ||||
| I decided to use Nix flakes to manage my configuration. But what is it that made | ||||
| me do so? Well, two things: | ||||
| 
 | ||||
| * __Adding custom packages__. The Nix code for my blog provides a package / derivation for each | ||||
|   version of my website, and I want to use these packages in my `configuration.nix` so that | ||||
|   I can point various Nginx virtual hosts to each of them. This is typically done using | ||||
|   version of my website, and I want to use these packages in my `configuration.nix`. Adding | ||||
|   custom packages is typically done using | ||||
|   overlays; however, how should my system configuration get my overlay Nix expression? I would | ||||
|   like to be able to separate my build-the-blog code from my describe-the-server code, and | ||||
|   so I need a clean way to let my system access the former from the latter. | ||||
|   flakes solve this issue by letting me specify a blog flake, and pull it in as one | ||||
|   of the inputs. | ||||
| * __Versioning__. My process for deploying new versions of the site prior to flakes boiled down to fethcing | ||||
|   Flakes solve this issue by letting me specify a blog flake, and pull it in as one | ||||
|   of the system configuration's inputs. | ||||
| * __Versioning__. My process for deploying new versions of the site prior to flakes boiled down to fetching | ||||
|   the latest commit from the `master` branch of my blog repository, and updating the `default.nix` | ||||
|   file with that commit. This way, I could reliably fetch the version of my site that | ||||
|   I want published. Flakes do the same thing: the `flake.lock` file | ||||
|   file with the corresponding hash. This way, I could reliably fetch the version of my site that | ||||
|   I wanted published. Flakes do the same thing: the `flake.lock` file | ||||
|   contains the hashes of the Git-based dependencies of a flake, and thus prevents builds from | ||||
|   accidentally pulling in something else. However, unlike my approach, which relies on custom | ||||
|   scripts and extra tools such as `jq`, the locking mechanism used by flakes is provided with | ||||
| @ -71,7 +70,7 @@ it's very clear from the configuration what I want from my server: | ||||
| three virtual hosts, one with HTTPS, one with drafts, and one with drafts and | ||||
| _in Russian_. Second, there's plenty of code reuse. I'm using two builder functions, | ||||
| `english` and `russian`, but under the hood, the exact same code is being | ||||
| used to run Hugo and all the necessary post-processing. Finally, all of this can be | ||||
| used to run Hugo and perform all the necessary post-processing. Finally, all of this can be | ||||
| used pretty much immediately given my blog flake, which reduces the amount of glue | ||||
| code I have to write. | ||||
| 
 | ||||
| @ -90,7 +89,7 @@ There are a few things here: | ||||
| * On line 7, the settings `src`, `ssl`, and `host` are inherited into the derivation. | ||||
|   The `src` setting provides a handle on the source code of the blog. I haven't had | ||||
|   much time to test and fine-tune the changes enabling multi-language support on the site, | ||||
|   so they reside on a separate branch. It's up to the caller to provide which version of the | ||||
|   so they reside on a separate branch. It's up to the caller to specify which version of the | ||||
|   source code should be used for building. The `host` and `ssl` settings are interesting | ||||
|   because __they don't actually matter for the derivation itself__ -- they just aren't used | ||||
|   in the builder. However, attributes given to a derivation are accessible from "outside", | ||||
| @ -98,15 +97,15 @@ There are a few things here: | ||||
| * Lines 10 through 14 deal with setting the base URL of the site. Hugo | ||||
|   does not know how to interpret | ||||
|   the `--baseURL` option when a blog has multiple languages. What this means is that in the end, | ||||
|   it is impossible to configure the base URL used in links from the command line, | ||||
|   and I need to apply some manual changes to the configuration file. I need to be able to adjust | ||||
|   the base URL becasue each version of my website is hosted in a different place: the default (english) | ||||
|   it is impossible to configure the base URL used in links from the command line. | ||||
|   I need to apply some manual changes to the configuration file. It's necessary to adjust | ||||
|   the base URL because each version of my website is hosted in a different place: the default (English) | ||||
|   website is hosted on `danilafe.com`, the version with drafts on `drafts.danilafe.com`, and so on. | ||||
|   However, the configuration file only knows one base URL per language, and so it _doesn't_ know | ||||
|   when or when not to use the `drafts.` prefix. The `urlSub` variable is used in the builder. | ||||
| * On line 15, the `publicPath` variable is set; while single-language Hugo puts all the generated | ||||
|   HTML into the `public` folder, the multi-language configuration places it into `public/[language-code]`. | ||||
|   Thus, depending on the configuration, the builer needs to look in a different place for final output. | ||||
|   Thus, depending on the configuration, the builder needs to look in a different place for final output. | ||||
| 
 | ||||
| This new `website` function is general enough to represent all my blog versions, but it's too low-level. | ||||
| Do I really want to specify the `publicPath` each time I want to describe a version of the site? | ||||
| @ -161,7 +160,7 @@ is the same as `replaceUrl.to` (since I'd want the Nginx virtual host for a blog | ||||
| version to handle links within that version). The `ssl` ghost parameter corresponds | ||||
| precisely to whether or not a virtual host will need SSL (and thus ACME, and thus | ||||
| the `systemd` setting). For each derivation built using `website`, I can access | ||||
| the attributes like `ssl` or `host` to generate the corresponding piece of the Nginx configuation. | ||||
| the attributes like `ssl` or `host` to generate the corresponding piece of the Nginx configuration. | ||||
| 
 | ||||
| To make this _really_ nice, I wanted all of this to be "just another section of my | ||||
| configuration file". That is, I wanted to control my site deployment via regular | ||||
| @ -220,7 +219,7 @@ We now have two "things" that handle the deployment of the blog: | ||||
| the builder functions `english` and `russian` which help describe various | ||||
| blog versions, and the NixOS module that configures the server's Nginx to | ||||
| serve said versions. We now want to expose these to the NixOS system configuration, | ||||
| which describes the entire server. I decided to do this using a flake. | ||||
| which describes the entire server. This is where flakes finally come in. | ||||
| [Yanik Sander](https://blog.ysndr.de/posts/internals/2021-01-01-flake-ification/index.html) wrote up | ||||
| a pretty comprehensive explanation of how their blog is deployed using flakes, which I often consulted | ||||
| while getting started -- check it out if you are looking for more details. | ||||
| @ -263,8 +262,8 @@ The flake `output` schema provides a standard option for exposing modules, `nixo | ||||
| exposing my `module.nix` file from the flake is simply a matter of importing it, as on line 31. | ||||
| There is, however, no standard way for exposing a _function_. The good news is that any | ||||
| attribute defined on a flake is accessible from code that imports that flake. Thus, I simply | ||||
| added a `buildersFor` function, which, given an operating system, fetches the `nixpkgs` collection | ||||
| and LaTeX builder script for that system, and feeds them to the file that defines the `english` | ||||
| added a `buildersFor` function, which fetches the `nixpkgs` collection | ||||
| and LaTeX builder script for a given system, and feeds them to the file that defines the `english` | ||||
| and `russian` builders. This `buildersFor` function also provides the builders with the two | ||||
| different blog sources they reference, `blog-source` and `blog-source-localized`. | ||||
| 
 | ||||
| @ -276,7 +275,7 @@ Finally, the last little bit on lines 32 through 34 defines a default package fo | ||||
| is the package that is built if a user runs `nix build .#`. This isn't strictly necessary for my purposes, | ||||
| but it's nice to be able to test that the builders still work by running a test build. The | ||||
| `eachDefaultSystem` function generates a `defaultPackage` attribute for each of the "default" | ||||
| operating systems, so that the package is buildable on more than just my server architecture. | ||||
| systems, so that the package is buildable on more than just my server architecture. | ||||
| 
 | ||||
| And that's it for the blog flake! I simply push it to Git, and move on to actually _using_ it from elsewhere. | ||||
| 
 | ||||
| @ -293,12 +292,12 @@ this can be done using the `specialArgs` attribute. The whole `flake.nix` file i | ||||
| 
 | ||||
| {{< codeblock "Nix" "server-config/flake.nix" >}} | ||||
| 
 | ||||
| Finally, in `configuration.nix`, taking `builders` as one of the inputs, I finally write what you saw above: | ||||
| Finally, in `configuration.nix`, taking `builders` as one of the inputs, I write what you saw above: | ||||
| 
 | ||||
| {{< codelines "Nix" "server-config/configuration.nix" 42 59 >}} | ||||
| 
 | ||||
| ### Wrapping Up | ||||
| So there you have it, a flake-based blog deployment written in a declarative style. You can | ||||
| So there you have it, a flake-based multi-version blog deployment written in a declarative style. You can | ||||
| check out both my [system configuration flake](https://dev.danilafe.com/Nix-Configs/server-config) | ||||
| and my [blog flake](https://dev.danilafe.com/Nix-Configs/blog-static-flake) on my Git server. | ||||
| If you want more, check out the articles by Xe and Yannik linked above. Thanks for reading! | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user