Nix that looks like Bazel

Published 2025-04-02 on Farid Zakaria's Blog

This is an idea 💡 that came from PlanetNix. I did not originate the idea.

At the 2025 North American NixCon (PlanetNix), one of the interesting lightning talks was from someone from Groq who demo’d what I originally thought to be a terrible idea but within a few minutes thought it was so evil it was good. 😈

What if we redesigned building software in Nix to look like Bazel?

What got me thinking about this? Well a blog post was published about bonanza a potential “next-gen” incarnation of Bazel. Nix already solves many of the challenges bonanza seeks to fix.

Follow me while I try to rebuild a Nix build-framework to build software, specifically Java, such that it looks like Bazel. 👇

If you are unfamiliar with Bazel, it’s a large-scale monorepo-centric build system open-sourced by Google. It has inspired many clones such as Buck, Pants, Please and so forth.

It uses a “python-like language to define build targets. The surface area is much smaller than something like Nix which lets you run arbitrary bash – although Bazel does have a “generic bash rule” as well.

Here is what a typical Bazel build definition for a Java program may look like. One key distinction are that dependencies are referenced by label and targets within the same file (package), can be defined starting after the colon.

If you are confused, that’s ok. This is not meant to be a great tutorial on Bazel. 🤔

java_binary(
    name = "ProjectRunner",
    srcs = ["src/main/java/com/example/ProjectRunner.java"],
    main_class = "com.example.ProjectRunner",
    deps = [":greeter"],
)

java_library(
    name = "greeter",
    srcs = ["src/main/java/com/example/Greeting.java"],
)

Traditionally in Nix, you would replace these rules with something like mkDerivation and build the single final application.

Here is something similar we can write in pure Nix.

# com/example/lib_b/default.nix
{java_library}:
java_library {
  name = "lib_b";
  srcs = [
    ./LibraryB.java
  ];
  deps = [
    "//com/example/lib_a"
  ];
}
# com/example/default.nix
{java_binary}:
java_binary {
  name = "main";
  mainClass = "com.example.Main";
  srcs = [
    ./Main.java
  ];
  deps = [
    "//com/example/lib_b"
  ];
}

Wow, that looks surprisingly similar. 😮

Getting this to work is surprisingly easy. We only need two function definitions for java_library and java_binary.

First in order to build anything in Java we need “libraries” (JARs). Nixpkgs already has this great concept that any JAR placed in share/java gets automatically added to the CLASSPATH during compilation in a mkDerivation.

{
  stdenv,
  lib,
  jdk,
  pkgs,
}: let
  fs = lib.fileset;
in
  {
    name,
    srcs,
    deps ? [],
  }:
    stdenv.mkDerivation {
      inherit name;
      srcs = fs.toSource {
        root = ./.;
        fileset = fs.unions srcs;
      };
      buildInputs = map (d: pkgs.${d}) deps;
      nativeBuildInputs = [jdk];
      buildPhase = ''
        find $srcs -name "*.java" | xargs javac -d .
        jar -cvf ${name}.jar -C . .
      '';
      installPhase = ''
        mkdir -p $out/share/java
        mv ${name}.jar $out/share/java/${name}.jar
      '';
    }

That makes compiling individal libraries pretty straightforward.

What about running them? In that case, we need the full transitive-closure of all compile dependencies to be present at runtime.

Recursion! In this case it is safe to do since we aren’t using any infinite lazy lists. 😏

Our java_binary definition now becomes straightforward. It is a java_library & a writeTextFile that sets the CLASSPATH before calling our main class.

{
  writeTextFile,
  java_library,
  jdk,
  lib,
  pkgs,
}: {
  name,
  mainClass,
  srcs,
  deps ? [],
}: let
  # get all deps transitively
  java_lib = java_library {
    name = "lib_${name}";
    inherit srcs;
    inherit deps;
  };
  # Recursively collect buildInputs from a list of derivations
  collectBuildInputs = inputs:
    builtins.concatMap (
      drv: let
        deps = drv.buildInputs or [];
      in
        [drv] ++ collectBuildInputs deps
    )
    inputs;
  depsAsPkgs = map (d: pkgs.${d}) deps;
  classpath = lib.concatStringsSep ":" (map (x: "${x}/share/java/${x.name}.jar") (collectBuildInputs (depsAsPkgs ++ [java_lib])));
in
  writeTextFile {
    inherit name;
    text = ''
      ${jdk}/bin/java -cp ${classpath} ${mainClass}
    '';
    executable = true;
    destination = "/bin/${name}";
  }

collectBuildInputs is the function that recursively walks all the dependencies and collects them to produce the necessary CLASSPATH.

I create now my top-level default.nix to define the targets possible

This step could likely be done at evaluation time and traverse the filesystem, but I’m keeping it simple for the purpose of understanding. 💪

let
  pkgs =
    import (fetchTarball "https://github.com/NixOS/nixpkgs/archive/5ef6c425980847c78a80d759abc476e941a9bf42.tar.gz") {
      overlays = [
        (self: super: rec {
          java_library = super.callPackage ./java_library.nix {};
          java_binary = super.callPackage ./java_binary.nix {};
          "//com/example/lib_a" = super.callPackage ./com/example/lib_a {};
          "//com/example/lib_b"= super.callPackage ./com/example/lib_b {};
          "//com/example:main"= super.callPackage ./com/example {};
        })
      ];
    };
in
{
  "//com/example/lib_a" = pkgs."//com/example/lib_a";
  "//com/example/lib_b" = pkgs."//com/example/lib_b";
  "//com/example:main" = pkgs."//com/example:main";
}

Now all that’s left to do is build & run the program to validate it works.

> nix-build -A "//com/example:main"
/nix/store/ry72i3ha3jrcpbz6yn4yna2wsx532gv8-main

> cat /nix/store/ry72i3ha3jrcpbz6yn4yna2wsx532gv8-main/bin/main 
/nix/store/1frnfh27i5pqk9xqahrjchlwyfzqgs1y-openjdk-21.0.5+11/bin/java -cp /nix/store/566jmxk1f8slkmp3mvrg4q0d8lbng5xx-lib_b/share/java/lib_b.jar:/nix/store/30lvqr3sc75yf9afzcl7l6j8phhw0xzv-lib_a/share/java/lib_a.jar:/nix/store/4zdhqm0ld93cqiv811brk5i6pyrcdvlg-lib_main/share/java/lib_main.jar:/nix/store/566jmxk1f8slkmp3mvrg4q0d8lbng5xx-lib_b/share/java/lib_b.jar:/nix/store/30lvqr3sc75yf9afzcl7l6j8phhw0xzv-lib_a/share/java/lib_a.jar com.example.Main

> ./result/bin/main 
Hello from Library A! and Library B!

Nice! 🔥

What is the appeal of all this?

Well, having a smaller API surface to build packages for a particular language is nice. You limit the opportunity for esoteric setups to creep in.

Finally, it’s likely my familiarity to Bazel, but I find reading the build definitions for the languages relatively straightforward as they all follow the same format.

By defining all the build targets individually at the language level, the code is also better set up to do incremental & parallel builds.

n.b. Specifically for Java, doing incremental builds would necessitate something like ijar.


Improve this page @ 2c7b6e1
The content for this site is CC-BY-SA.