159 lines
5.5 KiB
Nix
159 lines
5.5 KiB
Nix
|
/* Nix expression to test for regressions in the Haskell configuration overlays.
|
||
|
|
||
|
test-configurations.nix determines all attributes touched by given Haskell
|
||
|
configuration overlays (i. e. pkgs/development/haskell-modules/configuration-*.nix)
|
||
|
and builds all derivations (or at least a reasonable subset) affected by
|
||
|
these overrides.
|
||
|
|
||
|
By default, it checks `configuration-{common,nix,ghc-8.10.x}.nix`. You can
|
||
|
invoke it like this:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix --keep-going
|
||
|
|
||
|
It is possible to specify other configurations:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--arg files '[ "configuration-ghc-9.0.x.nix" "configuration-ghc-9.2.x.nix" ]' \
|
||
|
--keep-going
|
||
|
|
||
|
You can also just supply a single string:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--argstr files "configuration-arm.nix" --keep-going
|
||
|
|
||
|
You can even supply full paths which is handy, as it allows for tab-completing
|
||
|
the configurations:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--argstr files pkgs/development/haskell-modules/configuration-arm.nix \
|
||
|
--keep-going
|
||
|
|
||
|
By default, derivation that fail to evaluate are skipped, unless they are
|
||
|
“just” marked as broken. You can check for other eval errors like this:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--arg skipEvalErrors false --keep-going
|
||
|
|
||
|
You can also disable checking broken packages by passing a nixpkgs config:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--arg config '{ allowBroken = false; }' --keep-going
|
||
|
|
||
|
By default the haskell.packages.ghc*Binary sets used for bootstrapping GHC
|
||
|
are _not_ tested. You can change this using:
|
||
|
|
||
|
nix-build maintainers/scripts/haskell/test-configurations.nix \
|
||
|
--arg skipBinaryGHCs false --keep-going
|
||
|
|
||
|
*/
|
||
|
{ files ? [
|
||
|
"configuration-common.nix"
|
||
|
"configuration-nix.nix"
|
||
|
"configuration-ghc-8.10.x.nix"
|
||
|
]
|
||
|
, nixpkgsPath ? ../../..
|
||
|
, config ? { allowBroken = true; }
|
||
|
, skipEvalErrors ? true
|
||
|
, skipBinaryGHCs ? true
|
||
|
}:
|
||
|
|
||
|
let
|
||
|
pkgs = import nixpkgsPath { inherit config; };
|
||
|
inherit (pkgs) lib;
|
||
|
|
||
|
# see usage explanation for the input format `files` allows
|
||
|
files' = builtins.map builtins.baseNameOf (
|
||
|
if !builtins.isList files then [ files ] else files
|
||
|
);
|
||
|
|
||
|
packageSetsWithVersionedHead = pkgs.haskell.packages // (
|
||
|
let
|
||
|
headSet = pkgs.haskell.packages.ghcHEAD;
|
||
|
# Determine the next GHC release version following GHC HEAD.
|
||
|
# GHC HEAD always has an uneven, tentative version number, e.g. 9.7.
|
||
|
# GHC releases always have even numbers, i.e. GHC 9.8 is branched off from
|
||
|
# GHC HEAD 9.7. Since we use the to be release number for GHC HEAD's
|
||
|
# configuration file, we need to calculate this here.
|
||
|
headVersion = lib.pipe headSet.ghc.version [
|
||
|
lib.versions.splitVersion
|
||
|
(lib.take 2)
|
||
|
lib.concatStrings
|
||
|
lib.strings.toInt
|
||
|
(builtins.add 1)
|
||
|
toString
|
||
|
];
|
||
|
in
|
||
|
{
|
||
|
"ghc${headVersion}" = headSet;
|
||
|
}
|
||
|
);
|
||
|
|
||
|
setsForFile = fileName:
|
||
|
let
|
||
|
# extract the unique part of the config's file name
|
||
|
configName = builtins.head (
|
||
|
builtins.match "configuration-(.+).nix" fileName
|
||
|
);
|
||
|
# match the major and minor version of the GHC the config is intended for, if any
|
||
|
configVersion = lib.concatStrings (
|
||
|
builtins.match "ghc-([0-9]+).([0-9]+).x" configName
|
||
|
);
|
||
|
# return all package sets under haskell.packages matching the version components
|
||
|
setsForVersion = builtins.map (name: packageSetsWithVersionedHead.${name}) (
|
||
|
builtins.filter (setName:
|
||
|
lib.hasPrefix "ghc${configVersion}" setName
|
||
|
&& (skipBinaryGHCs -> !(lib.hasInfix "Binary" setName))
|
||
|
) (
|
||
|
builtins.attrNames packageSetsWithVersionedHead
|
||
|
)
|
||
|
);
|
||
|
|
||
|
defaultSets = [ pkgs.haskellPackages ];
|
||
|
in {
|
||
|
# use plain haskellPackages for the version-agnostic files
|
||
|
# TODO(@sternenseemann): also consider currently selected versioned sets
|
||
|
"common" = defaultSets;
|
||
|
"nix" = defaultSets;
|
||
|
"arm" = defaultSets;
|
||
|
"darwin" = defaultSets;
|
||
|
}.${configName} or setsForVersion;
|
||
|
|
||
|
# attribute set that has all the attributes of haskellPackages set to null
|
||
|
availableHaskellPackages = builtins.listToAttrs (
|
||
|
builtins.map (attr: lib.nameValuePair attr null) (
|
||
|
builtins.attrNames pkgs.haskellPackages
|
||
|
)
|
||
|
);
|
||
|
|
||
|
# evaluate a configuration and only return the attributes changed by it,
|
||
|
# pass availableHaskellPackages as super in case intersectAttrs is used
|
||
|
overriddenAttrs = fileName: builtins.attrNames (
|
||
|
lib.fix (self:
|
||
|
import (nixpkgsPath + "/pkgs/development/haskell-modules/${fileName}") {
|
||
|
haskellLib = pkgs.haskell.lib.compose;
|
||
|
inherit pkgs;
|
||
|
} self availableHaskellPackages
|
||
|
)
|
||
|
);
|
||
|
|
||
|
# list of derivations that are affected by overrides in the given configuration
|
||
|
# overlays. For common, nix, darwin etc. only the derivation from the default
|
||
|
# package set will be emitted.
|
||
|
packages = builtins.filter (v:
|
||
|
lib.warnIf (v.meta.broken or false) "${v.pname} is marked as broken" (
|
||
|
v != null
|
||
|
&& (skipEvalErrors -> (builtins.tryEval (v.outPath or v)).success)
|
||
|
)
|
||
|
) (
|
||
|
lib.concatMap (fileName:
|
||
|
let
|
||
|
sets = setsForFile fileName;
|
||
|
attrs = overriddenAttrs fileName;
|
||
|
in
|
||
|
lib.concatMap (set: builtins.map (attr: set.${attr}) attrs) sets
|
||
|
) files'
|
||
|
);
|
||
|
in
|
||
|
|
||
|
packages
|