2022-04-06 15:14:42 +01:00
|
|
|
{ pkgs }:
|
|
|
|
rec {
|
|
|
|
/* Prepare a derivation for local builds.
|
|
|
|
*
|
2022-12-20 12:53:03 +00:00
|
|
|
* This function prepares checkpoint builds by provinding,
|
2022-06-15 10:10:47 +01:00
|
|
|
* containing the build output and the sources for cross checking.
|
2022-12-20 12:53:03 +00:00
|
|
|
* The build output can be used later to allow checkpoint builds
|
|
|
|
* by passing the derivation output to the `mkCheckpointBuild` function.
|
2022-04-07 13:47:56 +01:00
|
|
|
*
|
2022-12-20 12:53:03 +00:00
|
|
|
* To build a project with checkpoints follow these steps:
|
2022-04-07 13:47:56 +01:00
|
|
|
* - run prepareIncrementalBuild on the desired derivation
|
2022-12-20 12:53:03 +00:00
|
|
|
* e.G `incrementalBuildArtifacts = (pkgs.checkpointBuildTools.prepareCheckpointBuild pkgs.virtualbox);`
|
2022-04-07 13:47:56 +01:00
|
|
|
* - change something you want in the sources of the package( e.G using source override)
|
|
|
|
* changedVBox = pkgs.virtuabox.overrideAttrs (old: {
|
|
|
|
* src = path/to/vbox/sources;
|
|
|
|
* }
|
2022-12-20 12:53:03 +00:00
|
|
|
* - use `mkCheckpointedBuild changedVBox buildOutput`
|
2022-04-07 13:47:56 +01:00
|
|
|
* - enjoy shorter build times
|
2022-04-06 15:14:42 +01:00
|
|
|
*/
|
2022-12-20 12:53:03 +00:00
|
|
|
prepareCheckpointBuild = drv: drv.overrideAttrs (old: {
|
2022-06-15 10:10:47 +01:00
|
|
|
outputs = [ "out" ];
|
2022-12-20 12:53:03 +00:00
|
|
|
name = drv.name + "-checkpointArtifacts";
|
2022-12-20 09:43:15 +00:00
|
|
|
# To determine differences between the state of the build directory
|
|
|
|
# from an earlier build and a later one we store the state of the build
|
|
|
|
# directory before build, but after patch phases.
|
|
|
|
# This way, the same derivation can be used multiple times and only changes are detected.
|
|
|
|
# Additionally Removed files are handled correctly in later builds.
|
2022-06-15 10:10:47 +01:00
|
|
|
preBuild = (old.preBuild or "") + ''
|
|
|
|
mkdir -p $out/sources
|
|
|
|
cp -r ./* $out/sources/
|
|
|
|
'';
|
|
|
|
|
2022-12-20 09:43:15 +00:00
|
|
|
# After the build the build directory is copied again
|
|
|
|
# to get the output files.
|
|
|
|
# We copy the complete build folder, to take care for
|
|
|
|
# Build tools, building in the source directory, instead of
|
|
|
|
# having a build root directory, e.G the Linux kernel.
|
2022-06-15 10:10:47 +01:00
|
|
|
installPhase = ''
|
|
|
|
mkdir -p $out/outputs
|
|
|
|
cp -r ./* $out/outputs/
|
2022-04-06 15:14:42 +01:00
|
|
|
'';
|
|
|
|
});
|
|
|
|
|
2022-12-20 12:53:03 +00:00
|
|
|
/* Build a derivation based on the checkpoint output generated by
|
|
|
|
* the `prepareCheckpointBuild function.
|
2022-04-06 15:14:42 +01:00
|
|
|
*
|
|
|
|
* Usage:
|
|
|
|
* let
|
2022-12-20 12:53:03 +00:00
|
|
|
* checkpointArtifacts = prepareCheckpointBuild drv
|
|
|
|
* in mkCheckpointedBuild drv checkpointArtifacts
|
2022-04-06 15:14:42 +01:00
|
|
|
*/
|
2022-12-20 12:53:03 +00:00
|
|
|
mkCheckpointedBuild = drv: previousBuildArtifacts: drv.overrideAttrs (old: {
|
|
|
|
# The actual checkpoint build phase.
|
2022-12-20 09:43:15 +00:00
|
|
|
# We compare the changed sources from a previous build with the current and create a patch
|
|
|
|
# Afterwards we clean the build directory to copy the previous output files (Including the sources)
|
|
|
|
# The source difference patch is applied to get the latest changes again to allow short build times.
|
2022-06-15 10:10:47 +01:00
|
|
|
preBuild = (old.preBuild or "") + ''
|
|
|
|
set +e
|
|
|
|
diff -ur ${previousBuildArtifacts}/sources ./ > sourceDifference.patch
|
|
|
|
set -e
|
2022-12-20 09:29:27 +00:00
|
|
|
shopt -s extglob dotglob
|
2022-06-15 10:10:47 +01:00
|
|
|
rm -r !("sourceDifference.patch")
|
|
|
|
${pkgs.rsync}/bin/rsync -cutU --chown=$USER:$USER --chmod=+w -r ${previousBuildArtifacts}/outputs/* .
|
|
|
|
patch -p 1 -i sourceDifference.patch
|
|
|
|
'';
|
2022-04-06 15:14:42 +01:00
|
|
|
});
|
|
|
|
}
|