This repository implements a program to check Nixpkgs' pkgs/by-name
directory as part of RFC 140.
See CONTRIBUTING.md
for contributor documentation. Below is the user documentation.
Currently the only intended user for this program is Nixpkgs. So the interface may be changed in breaking ways as long as Nixpkgs is adjusted to deal with it. See the pkgs/by-name
Readme for how it's used in Nixpkgs.
The source code contains a default.nix
file, which defines a Nix function.
The function takes an attribute set with at least these attributes as its argument:
system
(String, defaults tobuiltins.currentSystem
): Thesystem
to build the resulting derivation with.
The function returns an attribute set with at least these attributes:
build
(Package attribute set): A derivation that can be built with the givensystem
.
There is no guarantee that the derivation succeeds on systems that don't have prebuilt store paths, but it can be attempted with
nix-build https://github.com/NixOS/nixpkgs-check-by-name/tarball/master -A build
The GitHub releases contain a gzip-compressed Nix Archive of the build closure of the Nix derivation with x86_64-linux
as the system
.
This release artifact is named x86_64-linux.nar.gz
and can be imported into a local Nix store using:
storePath=$(gzip -cd x86_64-linux.nar.gz | nix-store --import | tail -1)
# To prevent it from being garbage-collected
nix-store --realise "$storePath" --add-root result
Compared to building the Nix derivations, this has the benefit that no Nix evaluation needs to take place and is therefore much faster and less storage intensive.
The store path acquired from the above methods contains a system
-specific binary under $storePath/bin/nixpkgs-check-by-name
.
The public interface of this binary is printed by calling
result/bin/nixpkgs-check-by-name --help
The following checks are performed when calling the binary:
pkgs/by-name
must only contain subdirectories of the form${shard}/${name}
, called package directories.- The
name
's of package directories must be unique when lowercased. name
is a string only consisting of the ASCII charactersa-z
,A-Z
,0-9
,-
or_
.shard
is the lowercased first two letters ofname
, expressed in Nix:shard = toLower (substring 0 2 name)
.- Each package directory must contain a
package.nix
file and may contain arbitrary other files.
- Each package directory must not refer to files outside itself using symlinks or Nix path expressions.
Evaluate Nixpkgs with system
set to x86_64-linux
and check that:
- For each package directory, the
pkgs.${name}
attribute must be defined ascallPackage pkgs/by-name/${shard}/${name}/package.nix args
for someargs
. - For each package directory,
pkgs.lib.isDerivation pkgs.${name}
must betrue
.
Furthermore, this tool implements certain ratchet checks. This allows gradually phasing out deprecated patterns without breaking the base branch or having to migrate it all at once. It works by not allowing new instances of the pattern to be introduced, but allowing already existing instances. The existing instances are coming from <BASE_NIXPKGS>
, which is then checked against <NIXPKGS>
for new instances. Ratchets should be removed eventually once the pattern is not used anymore.
The current ratchets are:
- New manual definitions of
pkgs.${name}
(e.g. inpkgs/top-level/all-packages.nix
) withargs = { }
(see nix evaluation checks) must not be introduced. - New top-level packages defined using
pkgs.callPackage
must be defined with a package directory.- Once a top-level package uses
pkgs/by-name
, it also can't be moved back out of it.
- Once a top-level package uses