mirror of
https://gitlab.com/TECHNOFAB/nix-gitlab-ci.git
synced 2026-06-19 00:29:25 +02:00
Merge branch 'easy-flake-checks' into 'main'
feat: auto-generate CI jobs from flake checks See merge request TECHNOFAB/nix-gitlab-ci!17
This commit is contained in:
commit
9b6ff90e15
8 changed files with 388 additions and 5 deletions
|
|
@ -67,6 +67,86 @@ in
|
||||||
# exposes `soonix` for the soonix hook and `packages` which contain the configs, jobs etc.
|
# exposes `soonix` for the soonix hook and `packages` which contain the configs, jobs etc.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Auto-generate jobs from flake checks
|
||||||
|
|
||||||
|
You can auto-generate CI jobs from your flake checks with the `autoChecks` option.
|
||||||
|
Each check becomes a job that runs `nix build .#checks.<system>.<name>`.
|
||||||
|
Auto-generated jobs are skipped if a user-defined job with the same name exists.
|
||||||
|
|
||||||
|
### With flake-parts
|
||||||
|
|
||||||
|
```nix
|
||||||
|
{
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:NixOS/nixpkgs/nixpkgs-unstable";
|
||||||
|
flake-parts.url = "github:hercules-ci/flake-parts";
|
||||||
|
systems.url = "github:nix-systems/default-linux";
|
||||||
|
nix-gitlab-ci.url = "gitlab:TECHNOFAB/nix-gitlab-ci?dir=lib";
|
||||||
|
};
|
||||||
|
|
||||||
|
outputs =
|
||||||
|
{ flake-parts, systems, ... } @ inputs:
|
||||||
|
flake-parts.lib.mkFlake { inherit inputs; } {
|
||||||
|
imports = [ inputs.nix-gitlab-ci.flakeModule ];
|
||||||
|
systems = import systems;
|
||||||
|
perSystem = { pkgs, system, ... }: {
|
||||||
|
checks = {
|
||||||
|
lint = pkgs.runCommand "lint" { } ''
|
||||||
|
${pkgs.nodePackages.prettier}/bin/prettier --check .
|
||||||
|
touch $out
|
||||||
|
'';
|
||||||
|
test = pkgs.runCommand "test" { } ''
|
||||||
|
${pkgs.python3}/bin/python -m pytest
|
||||||
|
touch $out
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
ci = {
|
||||||
|
config.autoChecks.enable = true;
|
||||||
|
pipelines.default.stages = [ "test" "build" ];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Without flake-parts
|
||||||
|
|
||||||
|
```nix
|
||||||
|
{
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:NixOS/nixpkgs/nixpkgs-unstable";
|
||||||
|
nix-gitlab-ci.url = "gitlab:TECHNOFAB/nix-gitlab-ci?dir=lib";
|
||||||
|
};
|
||||||
|
|
||||||
|
outputs =
|
||||||
|
{ self, nixpkgs, nix-gitlab-ci, ... }:
|
||||||
|
let
|
||||||
|
system = "x86_64-linux";
|
||||||
|
pkgs = import nixpkgs { inherit system; };
|
||||||
|
cilib = nix-gitlab-ci.lib { inherit pkgs; };
|
||||||
|
in
|
||||||
|
{
|
||||||
|
checks.${system} = {
|
||||||
|
lint = pkgs.runCommand "lint" { } ''
|
||||||
|
${pkgs.nodePackages.prettier}/bin/prettier --check .
|
||||||
|
touch $out
|
||||||
|
'';
|
||||||
|
test = pkgs.runCommand "test" { } ''
|
||||||
|
${pkgs.python3}/bin/python -m pytest
|
||||||
|
touch $out
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
packages.${system} =
|
||||||
|
(cilib.mkCI {
|
||||||
|
checks = self.checks.${system};
|
||||||
|
config.autoChecks.enable = true;
|
||||||
|
pipelines.default.stages = [ "test" "build" ];
|
||||||
|
}).packages;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
______________________________________________________________________
|
______________________________________________________________________
|
||||||
|
|
||||||
Since V2 multiple pipelines are supported.
|
Since V2 multiple pipelines are supported.
|
||||||
|
|
|
||||||
|
|
@ -20,7 +20,7 @@ args: let
|
||||||
modules = [
|
modules = [
|
||||||
cilib.modules.nixCiSubmodule
|
cilib.modules.nixCiSubmodule
|
||||||
{
|
{
|
||||||
inherit config;
|
config = config // {inherit (pkgs) system;};
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
}).config;
|
}).config;
|
||||||
|
|
|
||||||
|
|
@ -18,7 +18,13 @@
|
||||||
default = {};
|
default = {};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
config.legacyPackages = config.ci.packages;
|
config = {
|
||||||
|
ci = {
|
||||||
|
inherit (config) checks;
|
||||||
|
inherit (pkgs) system;
|
||||||
|
};
|
||||||
|
legacyPackages = config.ci.packages;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -24,6 +24,9 @@
|
||||||
name,
|
name,
|
||||||
config,
|
config,
|
||||||
rootConfig,
|
rootConfig,
|
||||||
|
rootChecks ? {},
|
||||||
|
rootAutoChecks ? {},
|
||||||
|
system,
|
||||||
...
|
...
|
||||||
}: let
|
}: let
|
||||||
#
|
#
|
||||||
|
|
@ -211,10 +214,31 @@
|
||||||
// gitlabOptions;
|
// gitlabOptions;
|
||||||
config = let
|
config = let
|
||||||
attrsToKeep = builtins.attrNames gitlabOptions;
|
attrsToKeep = builtins.attrNames gitlabOptions;
|
||||||
|
autoNixConfig = {
|
||||||
|
enable = config.nix.nixJobsByDefault;
|
||||||
|
enableRunnerCache = false;
|
||||||
|
};
|
||||||
|
autoJobs = lib.optionalAttrs (rootAutoChecks.enable && name == rootAutoChecks.pipeline) (
|
||||||
|
builtins.mapAttrs (checkName: _:
|
||||||
|
cilib.mkJobPatched {
|
||||||
|
key = "auto:${checkName}";
|
||||||
|
pipelineName = name;
|
||||||
|
nixConfig = autoNixConfig;
|
||||||
|
job = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = rootAutoChecks.stage;
|
||||||
|
script = ["nix build .#checks.\"${system}\".${checkName}"];
|
||||||
|
} // lib.optionalAttrs (rootAutoChecks.tags != []) {tags = rootAutoChecks.tags;};
|
||||||
|
}
|
||||||
|
) (
|
||||||
|
builtins.removeAttrs rootChecks (builtins.attrNames config.jobs)
|
||||||
|
)
|
||||||
|
);
|
||||||
in {
|
in {
|
||||||
finalConfig =
|
finalConfig =
|
||||||
(filterUnset (filterAttrs (n: _v: builtins.elem n attrsToKeep) config))
|
(filterUnset (filterAttrs (n: _v: builtins.elem n attrsToKeep) config))
|
||||||
// mapAttrs (_name: value: value.finalConfig) config.jobs;
|
// mapAttrs (_name: value: value.finalConfig) config.jobs
|
||||||
|
// autoJobs;
|
||||||
packages =
|
packages =
|
||||||
{
|
{
|
||||||
"gitlab-ci:pipeline:${name}" = toYaml "gitlab-ci-config.json" config.finalConfig;
|
"gitlab-ci:pipeline:${name}" = toYaml "gitlab-ci-config.json" config.finalConfig;
|
||||||
|
|
|
||||||
|
|
@ -6,6 +6,41 @@
|
||||||
}: let
|
}: let
|
||||||
inherit (lib) mkOption types foldr;
|
inherit (lib) mkOption types foldr;
|
||||||
in rec {
|
in rec {
|
||||||
|
autoChecksSubmodule = {
|
||||||
|
options = {
|
||||||
|
enable = mkOption {
|
||||||
|
description = ''
|
||||||
|
Auto-generate CI jobs from [`checks`](#checks).
|
||||||
|
|
||||||
|
Each check becomes a job that runs `nix build .#checks.<system>.<name>`.
|
||||||
|
'';
|
||||||
|
type = types.bool;
|
||||||
|
default = false;
|
||||||
|
};
|
||||||
|
stage = mkOption {
|
||||||
|
description = ''
|
||||||
|
Stage to assign to auto-generated check jobs.
|
||||||
|
'';
|
||||||
|
type = types.str;
|
||||||
|
default = "test";
|
||||||
|
};
|
||||||
|
tags = mkOption {
|
||||||
|
description = ''
|
||||||
|
Tags to assign to auto-generated check jobs.
|
||||||
|
'';
|
||||||
|
type = types.listOf types.str;
|
||||||
|
default = [];
|
||||||
|
};
|
||||||
|
pipeline = mkOption {
|
||||||
|
description = ''
|
||||||
|
Name of the pipeline to add auto-generated check jobs to.
|
||||||
|
'';
|
||||||
|
type = types.str;
|
||||||
|
default = "default";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
configSubmodule = {
|
configSubmodule = {
|
||||||
options = {
|
options = {
|
||||||
soonix = mkOption {
|
soonix = mkOption {
|
||||||
|
|
@ -24,11 +59,40 @@ in rec {
|
||||||
type = types.bool;
|
type = types.bool;
|
||||||
default = true;
|
default = true;
|
||||||
};
|
};
|
||||||
|
autoChecks = mkOption {
|
||||||
|
description = ''
|
||||||
|
Auto-generate CI jobs from flake checks.
|
||||||
|
|
||||||
|
See [`config.autoChecks`](#configautochecks) for configuration options.
|
||||||
|
'';
|
||||||
|
type = types.submodule autoChecksSubmodule;
|
||||||
|
default = {};
|
||||||
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
nixCiSubmodule = {config, ...}: {
|
nixCiSubmodule = {config, ...}: let
|
||||||
|
system = config.system;
|
||||||
|
in {
|
||||||
options = {
|
options = {
|
||||||
|
system = mkOption {
|
||||||
|
description = ''
|
||||||
|
System identifier (e.g. `"x86_64-linux"`).
|
||||||
|
Used when building checks with `nix build .#checks.<system>.<name>`.
|
||||||
|
Set automatically by `flakeModule` or `cilib.mkCI`.
|
||||||
|
'';
|
||||||
|
type = types.str;
|
||||||
|
};
|
||||||
|
checks = mkOption {
|
||||||
|
description = ''
|
||||||
|
Nix flake checks to optionally auto-generate CI jobs from.
|
||||||
|
|
||||||
|
Each attribute must be a derivation. The attribute name becomes the CI job name.
|
||||||
|
Requires [`config.autoChecks.enable`](#configautochecksenable) to take effect.
|
||||||
|
'';
|
||||||
|
type = types.attrsOf types.package;
|
||||||
|
default = {};
|
||||||
|
};
|
||||||
config = mkOption {
|
config = mkOption {
|
||||||
description = ''
|
description = ''
|
||||||
Configuration of Nix-GitLab-CI itself.
|
Configuration of Nix-GitLab-CI itself.
|
||||||
|
|
@ -42,7 +106,12 @@ in rec {
|
||||||
'';
|
'';
|
||||||
type = types.attrsOf (types.submoduleWith {
|
type = types.attrsOf (types.submoduleWith {
|
||||||
modules = [pipelineSubmodule];
|
modules = [pipelineSubmodule];
|
||||||
specialArgs.rootConfig = config.config;
|
specialArgs = {
|
||||||
|
rootConfig = config.config;
|
||||||
|
rootChecks = config.checks;
|
||||||
|
rootAutoChecks = config.config.autoChecks;
|
||||||
|
inherit system;
|
||||||
|
};
|
||||||
});
|
});
|
||||||
default = {};
|
default = {};
|
||||||
};
|
};
|
||||||
|
|
|
||||||
65
tests/auto_checks_test.nix
Normal file
65
tests/auto_checks_test.nix
Normal file
|
|
@ -0,0 +1,65 @@
|
||||||
|
{
|
||||||
|
pkgs,
|
||||||
|
ntlib,
|
||||||
|
cilib,
|
||||||
|
...
|
||||||
|
}: {
|
||||||
|
suites."autoChecks" = {
|
||||||
|
pos = __curPos;
|
||||||
|
tests = [
|
||||||
|
{
|
||||||
|
name = "flakeModule";
|
||||||
|
type = "script";
|
||||||
|
script =
|
||||||
|
# sh
|
||||||
|
''
|
||||||
|
${ntlib.helpers.scriptHelpers}
|
||||||
|
${ntlib.helpers.path (with pkgs; [coreutils nix gnused gnugrep jq])}
|
||||||
|
export SSL_CERT_FILE=${pkgs.cacert}/etc/ssl/certs/ca-bundle.crt
|
||||||
|
export NIX_SSL_CERT_FILE=${pkgs.cacert}/etc/ssl/certs/ca-bundle.crt
|
||||||
|
repo_path=${../.}
|
||||||
|
|
||||||
|
cp ${./fixtures/flake_parts_autochecks}/* .
|
||||||
|
sed -i -e "s|@repo_path@|$repo_path|" flake.nix
|
||||||
|
|
||||||
|
nix build --impure .#gitlab-ci:pipeline:default
|
||||||
|
assert "-f result" "should exist"
|
||||||
|
|
||||||
|
system=$(nix eval --impure --expr "builtins.currentSystem" --raw)
|
||||||
|
jq -e '."my-check" | .stage == "test"' result > /dev/null
|
||||||
|
jq -e '."my-check" | .script[0] | contains("nix build")' result > /dev/null
|
||||||
|
jq -e '."my-check" | .script[0] | contains("'"$system"'")' result > /dev/null
|
||||||
|
jq -e '."my-check" | has("before_script")' result > /dev/null
|
||||||
|
jq -e '."my-check" | has("after_script")' result > /dev/null
|
||||||
|
'';
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "direct";
|
||||||
|
type = "script";
|
||||||
|
script = let
|
||||||
|
system = pkgs.system;
|
||||||
|
fakeCheck = pkgs.runCommand "fake-check" {} "touch $out";
|
||||||
|
ci = cilib.mkCI {
|
||||||
|
checks.my-check = fakeCheck;
|
||||||
|
config.autoChecks.enable = true;
|
||||||
|
pipelines.default.stages = [ "test" ];
|
||||||
|
};
|
||||||
|
pkg = ci.packages."gitlab-ci:pipeline:default";
|
||||||
|
in
|
||||||
|
# sh
|
||||||
|
''
|
||||||
|
${ntlib.helpers.scriptHelpers}
|
||||||
|
${ntlib.helpers.path (with pkgs; [coreutils gnugrep jq])}
|
||||||
|
|
||||||
|
json=$(cat ${pkg})
|
||||||
|
echo "$json" | jq -e '.stages == [".pre", "test", ".post"]' > /dev/null
|
||||||
|
echo "$json" | jq -e '."my-check" | .stage == "test"' > /dev/null
|
||||||
|
echo "$json" | jq -e '."my-check" | .script[0] | contains("nix build")' > /dev/null
|
||||||
|
echo "$json" | jq -e '."my-check" | .script[0] | contains("${system}")' > /dev/null
|
||||||
|
echo "$json" | jq -e '."my-check" | has("before_script")' > /dev/null
|
||||||
|
echo "$json" | jq -e '."my-check" | has("after_script")' > /dev/null
|
||||||
|
'';
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
}
|
||||||
33
tests/fixtures/flake_parts_autochecks/flake.nix
vendored
Normal file
33
tests/fixtures/flake_parts_autochecks/flake.nix
vendored
Normal file
|
|
@ -0,0 +1,33 @@
|
||||||
|
{
|
||||||
|
outputs = {
|
||||||
|
flake-parts,
|
||||||
|
systems,
|
||||||
|
...
|
||||||
|
} @ inputs:
|
||||||
|
flake-parts.lib.mkFlake {inherit inputs;} {
|
||||||
|
imports = [
|
||||||
|
"@repo_path@/lib/flakeModule.nix"
|
||||||
|
];
|
||||||
|
systems = import systems;
|
||||||
|
flake = {};
|
||||||
|
perSystem = {pkgs, ...}: let
|
||||||
|
fakeCheck = pkgs.runCommand "fake-check" {} "touch $out";
|
||||||
|
in {
|
||||||
|
checks = {
|
||||||
|
my-check = fakeCheck;
|
||||||
|
};
|
||||||
|
ci = {
|
||||||
|
config.autoChecks.enable = true;
|
||||||
|
pipelines.default = {
|
||||||
|
stages = ["test"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
inputs = {
|
||||||
|
nixpkgs.url = "github:NixOS/nixpkgs/nixpkgs-unstable";
|
||||||
|
flake-parts.url = "github:hercules-ci/flake-parts";
|
||||||
|
systems.url = "github:nix-systems/default-linux";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
@ -98,6 +98,112 @@
|
||||||
assert_file_contains ${package} 'export EXAMPLE="/nix/store/.*-hello-.*"'
|
assert_file_contains ${package} 'export EXAMPLE="/nix/store/.*-hello-.*"'
|
||||||
'';
|
'';
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
name = "autoChecks with no checks set";
|
||||||
|
expected = {
|
||||||
|
stages = [".pre" "test" ".post"];
|
||||||
|
};
|
||||||
|
actual =
|
||||||
|
(cilib.mkCI {
|
||||||
|
config.autoChecks.enable = true;
|
||||||
|
pipelines."default".stages = ["test"];
|
||||||
|
}).pipelines."default".finalConfig;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "autoChecks generates check jobs";
|
||||||
|
expected = {
|
||||||
|
stages = [".pre" "check" ".post"];
|
||||||
|
unit = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = "test";
|
||||||
|
script = ["nix build .#checks.\"${pkgs.system}\".unit"];
|
||||||
|
before_script = ["source setup_nix_ci \"gitlab-ci:pipeline:default:job-deps:auto:unit\""];
|
||||||
|
after_script = ["finalize_nix_ci"];
|
||||||
|
};
|
||||||
|
lint = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = "test";
|
||||||
|
script = ["nix build .#checks.\"${pkgs.system}\".lint"];
|
||||||
|
before_script = ["source setup_nix_ci \"gitlab-ci:pipeline:default:job-deps:auto:lint\""];
|
||||||
|
after_script = ["finalize_nix_ci"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
actual = let
|
||||||
|
fakeCheck = pkgs.runCommand "fake-check" {} "touch $out";
|
||||||
|
in
|
||||||
|
(cilib.mkCI {
|
||||||
|
config.autoChecks = {
|
||||||
|
enable = true;
|
||||||
|
};
|
||||||
|
checks = {
|
||||||
|
unit = fakeCheck;
|
||||||
|
lint = fakeCheck;
|
||||||
|
};
|
||||||
|
pipelines."default".stages = ["check"];
|
||||||
|
}).pipelines."default".finalConfig;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "autoChecks skips existing jobs";
|
||||||
|
expected = {
|
||||||
|
stages = [".pre" "check" "build" ".post"];
|
||||||
|
unit = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = "test";
|
||||||
|
script = ["nix build .#checks.\"${pkgs.system}\".unit"];
|
||||||
|
};
|
||||||
|
custom = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = "build";
|
||||||
|
script = ["echo custom"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
actual = let
|
||||||
|
fakeCheck = pkgs.runCommand "fake-check" {} "touch $out";
|
||||||
|
in
|
||||||
|
(cilib.mkCI {
|
||||||
|
config.autoChecks = {
|
||||||
|
enable = true;
|
||||||
|
};
|
||||||
|
checks = {
|
||||||
|
unit = fakeCheck;
|
||||||
|
custom = fakeCheck;
|
||||||
|
};
|
||||||
|
pipelines."default" = {
|
||||||
|
nix.nixJobsByDefault = false;
|
||||||
|
stages = ["check" "build"];
|
||||||
|
jobs."custom" = {
|
||||||
|
stage = "build";
|
||||||
|
script = ["echo custom"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}).pipelines."default".finalConfig;
|
||||||
|
}
|
||||||
|
{
|
||||||
|
name = "autoChecks uses custom stage and tags";
|
||||||
|
expected = {
|
||||||
|
stages = [".pre" "quality" ".post"];
|
||||||
|
lint = {
|
||||||
|
image = "$NIX_CI_IMAGE";
|
||||||
|
stage = "quality";
|
||||||
|
tags = ["docker" "nix"];
|
||||||
|
script = ["nix build .#checks.\"${pkgs.system}\".lint"];
|
||||||
|
before_script = ["source setup_nix_ci \"gitlab-ci:pipeline:default:job-deps:auto:lint\""];
|
||||||
|
after_script = ["finalize_nix_ci"];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
actual = let
|
||||||
|
fakeCheck = pkgs.runCommand "fake-check" {} "touch $out";
|
||||||
|
in
|
||||||
|
(cilib.mkCI {
|
||||||
|
config.autoChecks = {
|
||||||
|
enable = true;
|
||||||
|
stage = "quality";
|
||||||
|
tags = ["docker" "nix"];
|
||||||
|
};
|
||||||
|
checks.lint = fakeCheck;
|
||||||
|
pipelines."default".stages = ["quality"];
|
||||||
|
}).pipelines."default".finalConfig;
|
||||||
|
}
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue