Convert Figma logo to code with AI

numtide logoflake-utils

Pure Nix flake utility functions [maintainer=@zimbatm]

1,226
80
1,226
22

Top Related Projects

Simplified nix packaging for various programming language ecosystems [maintainer=@DavHau]

Quick Overview

Flake-utils is a Nix library that provides a set of utility functions and helpers for working with Nix flakes. It aims to simplify common patterns and reduce boilerplate code when defining flake outputs, making it easier to create and maintain Nix-based projects.

Pros

  • Simplifies the creation of flake outputs for multiple systems
  • Provides helper functions for common tasks, reducing code duplication
  • Improves readability and maintainability of flake.nix files
  • Actively maintained and widely used in the Nix community

Cons

  • Adds an additional dependency to your project
  • May introduce a learning curve for those new to Nix flakes
  • Some users might prefer more explicit control over their flake configurations

Code Examples

  1. Creating a simple flake output for multiple systems:
{
  description = "My project";

  inputs.flake-utils.url = "github:numtide/flake-utils";

  outputs = { self, nixpkgs, flake-utils }:
    flake-utils.lib.eachDefaultSystem (system:
      let pkgs = nixpkgs.legacyPackages.${system};
      in {
        packages.default = pkgs.hello;
      }
    );
}
  1. Using mkApp to create a flake app:
flake-utils.lib.eachDefaultSystem (system:
  let pkgs = nixpkgs.legacyPackages.${system};
  in {
    apps.default = flake-utils.lib.mkApp {
      drv = pkgs.hello;
    };
  }
)
  1. Combining multiple system-dependent outputs:
flake-utils.lib.eachDefaultSystem (system:
  let pkgs = nixpkgs.legacyPackages.${system};
  in {
    packages.default = pkgs.hello;
    devShell = pkgs.mkShell {
      buildInputs = [ pkgs.hello ];
    };
  }
)

Getting Started

To use flake-utils in your project:

  1. Add flake-utils as an input in your flake.nix:
{
  inputs.flake-utils.url = "github:numtide/flake-utils";

  outputs = { self, nixpkgs, flake-utils }:
    flake-utils.lib.eachDefaultSystem (system:
      let pkgs = nixpkgs.legacyPackages.${system};
      in {
        # Your flake outputs here
      }
    );
}
  1. Use the provided functions and helpers in your flake outputs.

  2. Run nix flake update to fetch the flake-utils dependency.

Competitor Comparisons

Simplified nix packaging for various programming language ecosystems [maintainer=@DavHau]

Pros of dream2nix

  • More comprehensive solution for packaging various ecosystems (Python, Node.js, etc.)
  • Provides automated dependency resolution and fetching
  • Offers a higher level of abstraction for complex packaging scenarios

Cons of dream2nix

  • Steeper learning curve due to its more complex nature
  • May be overkill for simpler projects or single-language environments
  • Potentially slower build times for large projects due to its comprehensive approach

Code Comparison

flake-utils:

{
  inputs.flake-utils.url = "github:numtide/flake-utils";
  outputs = { self, nixpkgs, flake-utils }:
    flake-utils.lib.eachDefaultSystem (system: {
      # Your system-specific outputs here
    });
}

dream2nix:

{
  inputs.dream2nix.url = "github:nix-community/dream2nix";
  outputs = { self, nixpkgs, dream2nix }:
    dream2nix.lib.makeFlakeOutputs {
      systems = ["x86_64-linux" "aarch64-darwin"];
      config.projectRoot = ./.;
      source = ./.;
    };
}

The code comparison shows that dream2nix provides a more abstracted approach to creating flake outputs, while flake-utils offers a lower-level utility for iterating over systems. dream2nix's approach is more declarative and handles more complexity under the hood, potentially simplifying the configuration for complex projects spanning multiple ecosystems.

Convert Figma logo designs to code with AI

Visual Copilot

Introducing Visual Copilot: A new AI model to turn Figma designs to high quality code using your components.

Try Visual Copilot

README

flake-utils

Pure Nix flake utility functions.

A numtide project.

The goal of this project is to build a collection of pure Nix functions that don't depend on nixpkgs, and that are useful in the context of writing other Nix flakes.

Usage

system :: { system = system, ... }

A map from system to system built from allSystems:

system = {
  x86_64-linux = "x86_64-linux";
  x86_64-darwin = "x86_64-darwin";
  ...
}

It's mainly useful to detect typos and auto-complete if you use rnix-lsp.

Eg: instead of typing "x86_64-linux", use system.x86_64-linux.

allSystems :: [<system>]

A list of all systems defined in nixpkgs. For a smaller list see defaultSystems.

defaultSystems :: [<system>]

The list of systems to use in eachDefaultSystem and simpleFlake.

The default values are ["x86_64-linux" "aarch64-linux" "x86_64-darwin" "aarch64-darwin"].

It's possible to override and control that list by changing the systems input of this flake.

Eg (in your flake.nix):

{
  # 1. Defined a "systems" inputs that maps to only ["x86_64-linux"]
  inputs.systems.url = "github:nix-systems/x86_64-linux";

  inputs.flake-utils.url = "github:numtide/flake-utils";
  # 2. Override the flake-utils default to your version
  inputs.flake-utils.inputs.systems.follows = "systems";

  outputs = { self, flake-utils, ... }:
    # Now eachDefaultSystem is only using ["x86_64-linux"], but this list can also
    # further be changed by users of your flake.
    flake-utils.lib.eachDefaultSystem (system: {
      # ...
    });
}

For more details in this pattern, see: https://github.com/nix-systems/nix-systems.

eachSystem :: [<system>] -> (<system> -> attrs)

A common case is to build the same structure for each system. Instead of building the hierarchy manually or per prefix, iterate over each systems and then re-build the hierarchy.

Eg:

eachSystem [ system.x86_64-linux ] (system: { hello = 42; })
# => { hello = { x86_64-linux = 42; }; }
eachSystem allSystems (system: { hello = 42; })
# => {
   hello.aarch64-darwin = 42,
   hello.aarch64-genode = 42,
   hello.aarch64-linux = 42,
   ...
   hello.x86_64-redox = 42,
   hello.x86_64-solaris = 42,
   hello.x86_64-windows = 42
}

eachSystemPassThrough :: [<system>] -> (<system> -> attrs)

Unlike eachSystem, this function does not inject the ${system} key by merely providing the system argument to the function.

eachDefaultSystem :: (<system> -> attrs)

eachSystem pre-populated with defaultSystems.

Example

$ examples/each-system/flake.nix as nix

{
  description = "Flake utils demo";

  inputs.flake-utils.url = "github:numtide/flake-utils";

  outputs = { self, nixpkgs, flake-utils }:
    flake-utils.lib.eachDefaultSystem (system:
      let pkgs = nixpkgs.legacyPackages.${system}; in
      {
        packages = rec {
          hello = pkgs.hello;
          default = hello;
        };
        apps = rec {
          hello = flake-utils.lib.mkApp { drv = self.packages.${system}.hello; };
          default = hello;
        };
      }
    );
}

eachDefaultSystemPassThrough :: (<system> -> attrs)

eachSystemPassThrough pre-populated with defaultSystems.

Example

inputs.flake-utils.lib.eachDefaultSystem (system: {
  checks./*<SYSTEM>.*/"<CHECK>" = /* ... */;
  devShells./*<SYSTEM>.*/"<DEV_SHELL>" = /* ... */;
  packages./*<SYSTEM>.*/"<PACKAGE>" = /* ... */;
})
// inputs.flake-utils.lib.eachDefaultSystemPassThrough (system: {
  homeConfigurations."<HOME_CONFIGURATION>" = /* ... */;
  nixosConfigurations."<NIXOS_CONFIGURATION>" = /* ... */;
})

meld :: attrs -> [ path ] -> attrs

Meld merges subflakes using common inputs. Useful when you want to split up a large flake with many different components into more manageable parts.

mkApp { drv, name ? drv.pname or drv.name, exePath ? drv.passthru.exePath or "/bin/${name}"

A small utility that builds the structure expected by the special apps and defaultApp prefixes.

flattenTree :: attrs -> attrs

Nix flakes insists on having a flat attribute set of derivations in various places like the packages and checks attributes.

This function traverses a tree of attributes (by respecting recurseIntoAttrs) and only returns their derivations, with a flattened key-space.

Eg:

flattenTree { hello = pkgs.hello; gitAndTools = pkgs.gitAndTools }

Returns:

{
  hello = «derivation»;
  "gitAndTools/git" = «derivation»;
  "gitAndTools/hub" = «derivation»;
  # ...
}

simpleFlake :: attrs -> attrs

This function should be useful for most common use-cases where you have a simple flake that builds a package. It takes nixpkgs and a bunch of other parameters and outputs a value that is compatible as a flake output.

Input:

{
  # pass an instance of self
  self
, # pass an instance of the nixpkgs flake
  nixpkgs
, # we assume that the name maps to the project name, and also that the
  # overlay has an attribute with the `name` prefix that contains all of the
  # project's packages.
  name
, # nixpkgs config
  config ? { }
, # pass either a function or a file
  overlay ? null
, # use this to load other flakes overlays to supplement nixpkgs
  preOverlays ? [ ]
, # maps to the devShell output. Pass in a shell.nix file or function.
  shell ? null
, # pass the list of supported systems
  systems ? [ "x86_64-linux" "aarch64-linux" "x86_64-darwin" "aarch64-darwin" ]
}: null

Example

Here is how it looks like in practice:

$ examples/simple-flake/flake.nix as nix

{
  description = "Flake utils demo";

  inputs.flake-utils.url = "github:numtide/flake-utils";

  outputs = { self, nixpkgs, flake-utils }:
    flake-utils.lib.simpleFlake {
      inherit self nixpkgs;
      name = "simple-flake";
      overlay = ./overlay.nix;
      shell = ./shell.nix;
    };
}

Commercial support

Looking for help or customization?

Get in touch with Numtide to get a quote. We make it easy for companies to work with Open Source projects: https://numtide.com/contact