mirror of
https://github.com/ilyakooo0/haskell.nix.git
synced 2024-11-10 06:47:48 +03:00
0de60e8b0c
- reinstallable lib:ghc - adds global options and option inheritance.
184 lines
5.9 KiB
Nix
184 lines
5.9 KiB
Nix
# The plan (that is, a package set description like an LTS set or a
|
|
# plan.nix (derived from plan.json)) will producde a structure that
|
|
# looks like, which is stored in config.plan.pkg-def:
|
|
#
|
|
# { packages = { "package" = { revision = hackageConfigs.$package.$version.revisions.default;
|
|
# flags = { flag1 = true; flag2 = false; ... }; };
|
|
# ... };
|
|
# compiler = { version = "X.Y.Z"; nix-name ="ghcXYZ";
|
|
# # packages that come bundled with the compiler
|
|
# packages = { "bytestring" = "a.b.c.d"; ... }; };
|
|
# }
|
|
|
|
{ lib, config, pkgs, pkgconfPkgs, haskellLib, ... }:
|
|
|
|
with lib;
|
|
with types;
|
|
|
|
|
|
let
|
|
# This is just like listOf, except that it filters out all null elements.
|
|
listOfFilteringNulls = elemType: listOf elemType // {
|
|
# Mostly copied from nixpkgs/lib/types.nix
|
|
merge = loc: defs:
|
|
map (x: x.value) (filter (x: x ? value && x.value != null) (concatLists (imap1 (n: def:
|
|
if isList def.value then
|
|
imap1 (m: def':
|
|
(mergeDefinitions
|
|
(loc ++ ["[definition ${toString n}-entry ${toString m}]"])
|
|
elemType
|
|
[{ inherit (def) file; value = def'; }]
|
|
).optionalValue
|
|
) def.value
|
|
else
|
|
throw "The option value `${showOption loc}` in `${def.file}` is not a list.") defs)));
|
|
};
|
|
|
|
componentOptions = def: {
|
|
configureFlags = mkOption {
|
|
type = listOfFilteringNulls str;
|
|
default = (def.configureFlags or []);
|
|
};
|
|
setupBuildFlags = mkOption {
|
|
type = listOfFilteringNulls str;
|
|
default = (def.setupBuildFlags or []);
|
|
};
|
|
setupTestFlags = mkOption {
|
|
type = listOfFilteringNulls str;
|
|
default = (def.setupTestFlags or []);
|
|
};
|
|
setupInstallFlags = mkOption {
|
|
type = listOfFilteringNulls str;
|
|
default = (def.setupInstallFlags or []);
|
|
};
|
|
setupHaddockFlags = mkOption {
|
|
type = listOfFilteringNulls str;
|
|
default = (def.setupHaddockFlags or []);
|
|
};
|
|
doExactConfig = mkOption {
|
|
type = bool;
|
|
default = (def.doExactConfig or false);
|
|
};
|
|
doCheck = mkOption {
|
|
type = bool;
|
|
default = (def.doCheck or false);
|
|
};
|
|
doCrossCheck = mkOption {
|
|
description = "Run doCheck also in cross compilation settings. This can be tricky as the test logic must know how to run the tests on the target.";
|
|
type = bool;
|
|
default = (def.doCrossCheck or false);
|
|
};
|
|
doHaddock = mkOption {
|
|
description = "Enable building of the Haddock documentation from the annotated Haskell source code.";
|
|
type = bool;
|
|
default = (def.doHaddock or true);
|
|
};
|
|
};
|
|
packageOptions = def: componentOptions def // {
|
|
preUnpack = mkOption {
|
|
type = nullOr lines;
|
|
default = (def.preUnpack or null);
|
|
};
|
|
postUnpack = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postUnpack or null);
|
|
};
|
|
preConfigure = mkOption {
|
|
type = nullOr string;
|
|
default = (def.preConfigure or null);
|
|
};
|
|
postConfigure = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postConfigure or null);
|
|
};
|
|
preBuild = mkOption {
|
|
type = nullOr string;
|
|
default = (def.preBuild or null);
|
|
};
|
|
postBuild = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postBuild or null);
|
|
};
|
|
preCheck = mkOption {
|
|
type = nullOr string;
|
|
default = (def.preCheck or null);
|
|
};
|
|
postCheck = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postCheck or null);
|
|
};
|
|
preInstall = mkOption {
|
|
type = nullOr string;
|
|
default = (def.preInstall or null);
|
|
};
|
|
postInstall = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postInstall or null);
|
|
};
|
|
preHaddock = mkOption {
|
|
type = nullOr string;
|
|
default = (def.preHaddock or null);
|
|
};
|
|
postHaddock = mkOption {
|
|
type = nullOr string;
|
|
default = (def.postHaddock or null);
|
|
};
|
|
shellHook = mkOption {
|
|
type = nullOr string;
|
|
default = (def.shellHook or null);
|
|
};
|
|
};
|
|
|
|
|
|
in {
|
|
# Global options. These are passed down to the package level, and from there to the
|
|
# component level, unless specifically overridden. Depending on the flag flags are
|
|
# combined or replaced. We seed the package Options with an empty set forcing the
|
|
# default values.
|
|
options = (packageOptions {}) // {
|
|
packages = mkOption {
|
|
type =
|
|
let mod_args = {
|
|
inherit pkgs pkgconfPkgs haskellLib;
|
|
inherit (config) hsPkgs;
|
|
inherit (config.cabal) system compiler;
|
|
}; in
|
|
attrsOf (submodule (import ./package.nix {
|
|
inherit mod_args listOfFilteringNulls;
|
|
inherit componentOptions packageOptions;
|
|
parentConfig = config;
|
|
}));
|
|
};
|
|
|
|
compiler = {
|
|
version = mkOption {
|
|
type = str;
|
|
};
|
|
nix-name = mkOption {
|
|
type = str;
|
|
};
|
|
packages = mkOption {
|
|
type = attrsOf str;
|
|
};
|
|
};
|
|
|
|
plan.pkg-def = mkOption {
|
|
type = unspecified;
|
|
visible = false;
|
|
internal = true;
|
|
};
|
|
};
|
|
|
|
config = let module = config.plan.pkg-def config.hackage.configs; in {
|
|
inherit (module) compiler;
|
|
packages = lib.mapAttrs (name: { revision, ... }@revArgs: { system, compiler, flags, pkgs, hsPkgs, pkgconfPkgs, ... }@modArgs:
|
|
|
|
let m = if revision == null
|
|
then (abort "${name} has no revision!")
|
|
else revision (modArgs // { hsPkgs = hsPkgs // (mapAttrs (l: _: hsPkgs.${name}.components.sublibs.${l}) (m.components.sublibs or {})); });
|
|
in
|
|
m // { flags = lib.mapAttrs (_: lib.mkDefault) (m.flags // revArgs.flags or {}); }
|
|
) (lib.filterAttrs (n: v: v == null || v.revision != null ) module.packages);
|
|
};
|
|
}
|