What is Nix?

Nix is confusingly both a language & a package manager, among other things.

The language & the package manager can work together to create reproducible, declarative, shareable, & isolated environments for software development.

Nix develop

The nix develop command is a tool in the Nix ecosystem that allows the creation of development environments defined by a local file called a “flake”.

Running this command builds & installs all dependencies defined in the flake hermetically, meaning that exiting the environment removes the installed dependencies from the environment.

Nix flakes

Nix flakes, currently an experimental feature, offers an enhanced way to manage Nix configurations & dependencies in a declarative fashion.

By using Nix flakes, developers can define reusable configurations, making it easier to share & maintain complex environments across teams & projects.

Create flake.nix

The basis & entry point for Nix flakes is a text file named flake.nix written in the Nix functional language. Nix can quickly create a default flake based on the current packages in the Nix package repositories:

$ nix flake init
wrote: /your/current/directory/flake.nix

Creating one manually is often more efficient. In theory, the flake.nix file is simple & only requires two things, inputs & outputs.

Inputs

inputs are an attribute set (a.k.a. map, hashmap, dictionary, associative array, etc.) that defines our source code repositories. This is almost always going to have a git-style url pointing to the nixpkgs github. Here the nixos-unstable branch of nixpkgs is specifically referenced.

# flake.nix
{
  inputs = {
    nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
  };
}

Outputs

outputs is a function that outputs your “derivation”. A derivation can be anything: an executable file, a Dockerfile; or in this case, a developer environment. This function should take in the self parameter & our nixpkgs input in order to get access to the nix packages hosted on github.

# flake.nix
{
  inputs = {
    nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
  };

  outputs = {self, nixpkgs}:
}

Because nix is a purely functional language, it is common to assign variables scoped to our outputs function in a let/in clause before the function body.

This is useful to specify the system for which our software dependencies should be built. Here x86_64-linux is specified, but will be different for MacOS or various other systems. These will be taken into account later in the article.

# flake.nix
{
  inputs = {
    nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
  };

  outputs = {self, nixpkgs}:
  let
    system = "x86_64-linux";
    pkgs = import nixpkgs { inherit system; };
  in {
}

Now that the system for the dependencies has been defined, devShells can be correctly called with the defined system. mkShell, similarly, is now defined for the system & the list of packages to be installed will use the correct system architecture.

# flake.nix
{
  inputs = {
    nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
  };

  outputs = {self, nixpkgs}:
  let
    system = "x86_64-linux";
    pkgs = import nixpkgs { inherit system; };
  in {
    devShells.${system}.default = pkgs.mkShell {
      packages = with pkgs; [
        go
        hugo
      ];
    };
  };
}

shellHook can be optionally defined to run any shell command. Defining environment variables or printing important info to the terminal when entering the environment are common usages:

# flake.nix
{
  inputs = {
    nixpkgs.url = "github:nixos/nixpkgs?ref=nixos-unstable";
  };

  outputs = {self, nixpkgs}:
  let
    system = "x86_64-linux";
    pkgs = import nixpkgs { inherit system; };
  in {
    devShells.${system}.default = pkgs.mkShell {
      packages = with pkgs; [
        go
        hugo
      ];

      shellHook = ''
        export HELPFUL_ENV_VAR="a helpful environment variable"
        echo "Happy hacking!"
        echo "$(go version)"
        echo "$(hugo version)"
      '';

    };
  };
}
$ nix develop
Happy hacking!
go version go1.22.8 linux/amd64
hugo v0.126.1+extended linux/amd64 BuildDate=unknown VendorInfo=nixpkgs
(nix:nix-shell-env) bash-5.2$ echo "$HELPFUL_ENV_VAR"
a helpful environment variable
(nix:nix-shell-env) bash-5.2$

Validate flake.nix

In order to validate the syntax & configuration of our flake.nix file, Nix understands common development tools & flake.nix needs to be checked in to a version control system. Add Explanation Here. Let’s use git.

$ git init
Initialized empty Git repository in /your/current/directory/here.git/
$ git add flake.nix

Now check the flake.nix for syntax errors & other problems:

$ nix flake check

Manage Nix dependencies with flake.lock

Running any nix command on flake.nix such as nix flake check will create or update a flake.lock file that is similar to other languages dependency files such as packages.json in Node for Javascript or go.sum for Golang.

This file references the sha hash of a specific git commit in the nixpkgs repo (or any repo defined in the inputs) & gives the ability to ensure completely accurate & reproducible builds.

Dynamic System Builds

Development teams often have a varied number of workstations, & therefore systems, that need to be taken into account when building or installing dependencies.

There are numerous ways of doing so; flake-utils, developed by numtide, is a common approach. Creating a function that takes in systems you want to support & loops over them can also be a more simple method:

# flake.nix
  outputs = { self, nixpkgs }:
    let
      systems = ["x86_64-linux" "x86_64-darwin"];
      forEachSystem = f: nixpkgs.lib.genAttrs systems (system: f {
        pkgs = import nixpkgs { inherit system; };
      });
    in { ...

This function, named forEachSystem, uses genAttrs & loops over the array of systems to be installed for each system listed in the array.

To finish, call the forEachSystem function when creating the development shells:

    in {
      devShells = forEachSystem ({ pkgs }: {
        default = pkgs.mkShell {
          packages = with pkgs; [
            go
            hugo
          ];

          shellHook = ''
            export HELPFUL_ENV_VAR="a helpful environment variable"
            echo "Happy hacking!"
            echo "$(go version)"
            echo "$(hugo version)"
          '';
        };
      });
    };

Real World Example

This site itself uses nix flakes in order to create a development environment that contains Go & Hugo so that creating, building, & collaborating on the site is easy.