Test harnesses
This commit is contained in:
parent
b2065a2790
commit
2b414e1c15
@ -5,11 +5,11 @@
|
|||||||
}@defs:
|
}@defs:
|
||||||
|
|
||||||
{
|
{
|
||||||
drvName
|
pname
|
||||||
, pname
|
|
||||||
, gemfile
|
, gemfile
|
||||||
, lockfile
|
, lockfile
|
||||||
, gemset
|
, gemset
|
||||||
|
, gemdir
|
||||||
, ruby ? defs.ruby
|
, ruby ? defs.ruby
|
||||||
, gemConfig ? defaultGemConfig
|
, gemConfig ? defaultGemConfig
|
||||||
, postBuild ? null
|
, postBuild ? null
|
||||||
@ -20,14 +20,13 @@
|
|||||||
, ...
|
, ...
|
||||||
}@args:
|
}@args:
|
||||||
|
|
||||||
with (import ./functions.nix);
|
with (import ./functions.nix { inherit lib ruby gemConfig groups; });
|
||||||
|
|
||||||
let
|
let
|
||||||
mainGem = gems."${pname}" or (throw "bundlerEnv: gem ${pname} not found");
|
|
||||||
|
|
||||||
importedGemset = import gemset;
|
importedGemset = import gemset;
|
||||||
|
|
||||||
filteredGemset = lib.filterAttrs (name: attrs: platformMatches attrs && groupMatches attrs) importedGemset;
|
filteredGemset = filterGemset importedGemset;
|
||||||
|
|
||||||
configuredGemset = lib.flip lib.mapAttrs filteredGemset (name: attrs:
|
configuredGemset = lib.flip lib.mapAttrs filteredGemset (name: attrs:
|
||||||
applyGemConfigs (attrs // { inherit ruby; gemName = name; })
|
applyGemConfigs (attrs // { inherit ruby; gemName = name; })
|
||||||
@ -47,14 +46,18 @@ let
|
|||||||
'' else ""
|
'' else ""
|
||||||
);
|
);
|
||||||
|
|
||||||
maybeCopyAll = main: if main == null then "" else copyIfBundledByPath main;
|
maybeCopyAll = pname: if pname == null then "" else
|
||||||
|
let
|
||||||
|
mainGem = gems."${pname}" or (throw "bundlerEnv: gem ${pname} not found");
|
||||||
|
in
|
||||||
|
copyIfBundledByPath mainGem;
|
||||||
|
|
||||||
# We have to normalize the Gemfile.lock, otherwise bundler tries to be
|
# We have to normalize the Gemfile.lock, otherwise bundler tries to be
|
||||||
# helpful by doing so at run time, causing executables to immediately bail
|
# helpful by doing so at run time, causing executables to immediately bail
|
||||||
# out. Yes, I'm serious.
|
# out. Yes, I'm serious.
|
||||||
confFiles = runCommand "gemfile-and-lockfile" {} ''
|
confFiles = runCommand "gemfile-and-lockfile" {} ''
|
||||||
mkdir -p $out
|
mkdir -p $out
|
||||||
${maybeCopyAll mainGem}
|
${maybeCopyAll pname}
|
||||||
cp ${gemfile} $out/Gemfile || ls -l $out/Gemfile
|
cp ${gemfile} $out/Gemfile || ls -l $out/Gemfile
|
||||||
cp ${lockfile} $out/Gemfile.lock || ls -l $out/Gemfile.lock
|
cp ${lockfile} $out/Gemfile.lock || ls -l $out/Gemfile.lock
|
||||||
'';
|
'';
|
||||||
@ -71,13 +74,10 @@ let
|
|||||||
|
|
||||||
envPaths = lib.attrValues gems ++ lib.optional (!hasBundler) bundler;
|
envPaths = lib.attrValues gems ++ lib.optional (!hasBundler) bundler;
|
||||||
|
|
||||||
# binPaths = if mainGem != null then [ mainGem ] else envPaths;
|
basicEnv = buildEnv {
|
||||||
|
|
||||||
in
|
|
||||||
buildEnv {
|
|
||||||
inherit ignoreCollisions;
|
inherit ignoreCollisions;
|
||||||
|
|
||||||
name = drvName;
|
name = pname;
|
||||||
|
|
||||||
paths = envPaths;
|
paths = envPaths;
|
||||||
pathsToLink = [ "/lib" ];
|
pathsToLink = [ "/lib" ];
|
||||||
@ -90,20 +90,20 @@ in
|
|||||||
meta = { platforms = ruby.meta.platforms; } // meta;
|
meta = { platforms = ruby.meta.platforms; } // meta;
|
||||||
|
|
||||||
passthru = rec {
|
passthru = rec {
|
||||||
inherit ruby bundler gems;
|
inherit ruby bundler gems; # drvName;
|
||||||
|
|
||||||
wrappedRuby = stdenv.mkDerivation {
|
wrappedRuby = stdenv.mkDerivation {
|
||||||
name = "wrapped-ruby-${drvName}";
|
name = "wrapped-ruby-${pname}";
|
||||||
nativeBuildInputs = [ makeWrapper ];
|
nativeBuildInputs = [ makeWrapper ];
|
||||||
buildCommand = ''
|
buildCommand = ''
|
||||||
mkdir -p $out/bin
|
mkdir -p $out/bin
|
||||||
for i in ${ruby}/bin/*; do
|
for i in ${ruby}/bin/*; do
|
||||||
makeWrapper "$i" $out/bin/$(basename "$i") \
|
makeWrapper "$i" $out/bin/$(basename "$i") \
|
||||||
--set BUNDLE_GEMFILE ${confFiles}/Gemfile \
|
--set BUNDLE_GEMFILE ${confFiles}/Gemfile \
|
||||||
--set BUNDLE_PATH ${bundlerEnv}/${ruby.gemPath} \
|
--set BUNDLE_PATH ${basicEnv}/${ruby.gemPath} \
|
||||||
--set BUNDLE_FROZEN 1 \
|
--set BUNDLE_FROZEN 1 \
|
||||||
--set GEM_HOME ${bundlerEnv}/${ruby.gemPath} \
|
--set GEM_HOME ${basicEnv}/${ruby.gemPath} \
|
||||||
--set GEM_PATH ${bundlerEnv}/${ruby.gemPath}
|
--set GEM_PATH ${basicEnv}/${ruby.gemPath}
|
||||||
done
|
done
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
@ -117,8 +117,8 @@ in
|
|||||||
require 'bundler/setup'
|
require 'bundler/setup'
|
||||||
'';
|
'';
|
||||||
in stdenv.mkDerivation {
|
in stdenv.mkDerivation {
|
||||||
name = "${drvName}-interactive-environment";
|
name = "${pname}-interactive-environment";
|
||||||
nativeBuildInputs = [ wrappedRuby bundlerEnv ];
|
nativeBuildInputs = [ wrappedRuby basicEnv ];
|
||||||
shellHook = ''
|
shellHook = ''
|
||||||
export OLD_IRBRC="$IRBRC"
|
export OLD_IRBRC="$IRBRC"
|
||||||
export IRBRC=${irbrc}
|
export IRBRC=${irbrc}
|
||||||
@ -131,4 +131,6 @@ in
|
|||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
};
|
||||||
|
in
|
||||||
|
basicEnv
|
||||||
|
@ -24,15 +24,13 @@
|
|||||||
}@args:
|
}@args:
|
||||||
|
|
||||||
let
|
let
|
||||||
|
inherit (import ./functions.nix (defs // args)) genStubsScript;
|
||||||
|
|
||||||
drvName =
|
drvName =
|
||||||
if name != null then name
|
if name != null then name
|
||||||
else if pname != null then "${toString pname}-${mainGem.version}"
|
else if pname != null then "${toString pname}-${basicEnv.gems."${pname}".version}"
|
||||||
else throw "bundlerEnv: either pname or name must be set";
|
else throw "bundlerEnv: either pname or name must be set";
|
||||||
|
|
||||||
mainGem =
|
|
||||||
if pname == null then null
|
|
||||||
else gems."${pname}" or (throw "bundlerEnv: gem ${pname} not found");
|
|
||||||
|
|
||||||
gemfile' =
|
gemfile' =
|
||||||
if gemfile == null then gemdir + "/Gemfile"
|
if gemfile == null then gemdir + "/Gemfile"
|
||||||
else gemfile;
|
else gemfile;
|
||||||
@ -45,12 +43,13 @@ let
|
|||||||
if gemset == null then gemdir + "/gemset.nix"
|
if gemset == null then gemdir + "/gemset.nix"
|
||||||
else gemset;
|
else gemset;
|
||||||
|
|
||||||
envPaths = lib.attrValues gems ++ lib.optional (!hasBundler) bundler;
|
basicEnv = (callPackage ./basic.nix {}) (args // { inherit pname gemdir;
|
||||||
|
gemfile = gemfile';
|
||||||
binPaths = if mainGem != null then [ mainGem ] else envPaths;
|
lockfile = lockfile';
|
||||||
|
gemset = gemset';
|
||||||
basicEnv = import ./basic args // { inherit drvName pname gemfile lockfile gemset; };
|
});
|
||||||
|
|
||||||
|
inherit (basicEnv) envPaths;
|
||||||
# Idea here is a mkDerivation that gen-bin-stubs new stubs "as specified" -
|
# Idea here is a mkDerivation that gen-bin-stubs new stubs "as specified" -
|
||||||
# either specific executables or the bin/ for certain gem(s), but
|
# either specific executables or the bin/ for certain gem(s), but
|
||||||
# incorporates the basicEnv as a requirement so that its $out is in our path.
|
# incorporates the basicEnv as a requirement so that its $out is in our path.
|
||||||
@ -63,8 +62,26 @@ let
|
|||||||
|
|
||||||
# The basicEnv should be put into passthru so that e.g. nix-shell can use it.
|
# The basicEnv should be put into passthru so that e.g. nix-shell can use it.
|
||||||
in
|
in
|
||||||
(linkFarm drvName entries) // {
|
if builtins.trace "pname: ${toString pname}" pname == null then
|
||||||
passthru = {
|
basicEnv // { inherit name; }
|
||||||
inherit basicEnv;
|
else
|
||||||
};
|
(buildEnv {
|
||||||
}
|
inherit ignoreCollisions;
|
||||||
|
|
||||||
|
name = builtins.trace "name: ${toString drvName}" drvName;
|
||||||
|
|
||||||
|
paths = envPaths;
|
||||||
|
pathsToLink = [ "/lib" ];
|
||||||
|
|
||||||
|
postBuild = genStubsScript defs // args // {
|
||||||
|
inherit bundler;
|
||||||
|
confFiles = basicEnv.confFiles;
|
||||||
|
binPaths = [ basicEnv.mainGem ];
|
||||||
|
} + lib.optionalString (postBuild != null) postBuild;
|
||||||
|
|
||||||
|
meta = { platforms = ruby.meta.platforms; } // meta;
|
||||||
|
passthru = basicEnv.passthru // {
|
||||||
|
inherit basicEnv;
|
||||||
|
inherit (basicEnv) env;
|
||||||
|
};
|
||||||
|
})
|
||||||
|
@ -1,4 +1,7 @@
|
|||||||
|
{ lib, ruby, groups, gemConfig, ... }:
|
||||||
rec {
|
rec {
|
||||||
|
filterGemset = gemset: lib.filterAttrs (name: attrs: platformMatches attrs && groupMatches attrs) gemset;
|
||||||
|
|
||||||
platformMatches = attrs: (
|
platformMatches = attrs: (
|
||||||
!(attrs ? "platforms") ||
|
!(attrs ? "platforms") ||
|
||||||
builtins.any (platform:
|
builtins.any (platform:
|
||||||
@ -17,7 +20,7 @@ rec {
|
|||||||
then attrs // gemConfig."${attrs.gemName}" attrs
|
then attrs // gemConfig."${attrs.gemName}" attrs
|
||||||
else attrs);
|
else attrs);
|
||||||
|
|
||||||
genStubsScript = { lib, ruby, confFile, bundler, groups, binPaths }@args: ''
|
genStubsScript = { lib, ruby, confFiles, bundler, groups, binPaths }: ''
|
||||||
${ruby}/bin/ruby ${./gen-bin-stubs.rb} \
|
${ruby}/bin/ruby ${./gen-bin-stubs.rb} \
|
||||||
"${ruby}/bin/ruby" \
|
"${ruby}/bin/ruby" \
|
||||||
"${confFiles}/Gemfile" \
|
"${confFiles}/Gemfile" \
|
||||||
|
20
pkgs/development/ruby-modules/bundler-env/tap-support.nix
Normal file
20
pkgs/development/ruby-modules/bundler-env/tap-support.nix
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
with builtins;
|
||||||
|
let
|
||||||
|
withIndexes = list: genList (idx: (elemAt list idx) // {index = idx;}) (length list);
|
||||||
|
|
||||||
|
testLine = report: "${okStr report} ${toString report.index} ${report.description}" + testDirective report + testYaml report;
|
||||||
|
|
||||||
|
testDirective = report: "";
|
||||||
|
|
||||||
|
testYaml = report: "";
|
||||||
|
|
||||||
|
okStr = { result, ...}: if result == "pass" then "ok" else "not ok";
|
||||||
|
in
|
||||||
|
{
|
||||||
|
output = reports: ''
|
||||||
|
TAP version 13
|
||||||
|
1..${toString (length reports)}'' + (foldl' (l: r: l + "\n" + r) "" (map testLine (withIndexes reports))) + ''
|
||||||
|
|
||||||
|
# Finished at ${toString currentTime}
|
||||||
|
'';
|
||||||
|
}
|
49
pkgs/development/ruby-modules/bundler-env/test.nix
Normal file
49
pkgs/development/ruby-modules/bundler-env/test.nix
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
{ writeText, lib, ruby, defaultGemConfig, callPackage }:
|
||||||
|
let
|
||||||
|
test = import ./testing.nix;
|
||||||
|
tap = import ./tap-support.nix;
|
||||||
|
|
||||||
|
bundlerEnv = callPackage ./default.nix {};
|
||||||
|
|
||||||
|
testConfigs = {
|
||||||
|
groups = ["default"];
|
||||||
|
gemConfig = defaultGemConfig;
|
||||||
|
confFiles = "./testConfs";
|
||||||
|
};
|
||||||
|
functions = (import ./functions.nix ({ inherit lib ruby; } // testConfigs));
|
||||||
|
|
||||||
|
should = {
|
||||||
|
equal = expected: actual:
|
||||||
|
if actual == expected then
|
||||||
|
(test.passed "= ${toString expected}") else
|
||||||
|
(test.failed "'${toString actual}'(${builtins.typeOf actual}) != '${toString expected}'(${builtins.typeOf expected})");
|
||||||
|
|
||||||
|
beASet = actual:
|
||||||
|
if builtins.isAttrs actual then
|
||||||
|
(test.passed "is a set") else
|
||||||
|
(test.failed "is not a set, was ${builtins.typeOf actual}: ${toString actual}");
|
||||||
|
};
|
||||||
|
|
||||||
|
justName = bundlerEnv {
|
||||||
|
name = "test";
|
||||||
|
gemset = ./test/gemset.nix;
|
||||||
|
};
|
||||||
|
|
||||||
|
pnamed = bundlerEnv {
|
||||||
|
pname = "test";
|
||||||
|
gemset = ./test/gemset.nix;
|
||||||
|
};
|
||||||
|
|
||||||
|
results = builtins.concatLists [
|
||||||
|
(test.run "Filter empty gemset" {} (set: functions.filterGemset set == {}))
|
||||||
|
(test.run "bundlerEnv { name }" justName {
|
||||||
|
name = should.equal "test";
|
||||||
|
})
|
||||||
|
(test.run "bundlerEnv { pname }" pnamed
|
||||||
|
{
|
||||||
|
name = should.equal "test-0.1.2";
|
||||||
|
env = should.beASet;
|
||||||
|
})
|
||||||
|
];
|
||||||
|
in
|
||||||
|
writeText "test-results.tap" (tap.output results)
|
62
pkgs/development/ruby-modules/bundler-env/testing.nix
Normal file
62
pkgs/development/ruby-modules/bundler-env/testing.nix
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
with builtins;
|
||||||
|
let
|
||||||
|
/*
|
||||||
|
underTest = {
|
||||||
|
x = {
|
||||||
|
a = 1;
|
||||||
|
b = "2";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
tests = [
|
||||||
|
(root: false)
|
||||||
|
{
|
||||||
|
x = [
|
||||||
|
(set: true)
|
||||||
|
{
|
||||||
|
a = (a: a > 1);
|
||||||
|
b = (b: b == "3");
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
results = run "Examples" underTest tests;
|
||||||
|
*/
|
||||||
|
|
||||||
|
passed = desc: {
|
||||||
|
result = "pass";
|
||||||
|
description = desc;
|
||||||
|
};
|
||||||
|
|
||||||
|
failed = desc: {
|
||||||
|
result = "failed";
|
||||||
|
description = desc;
|
||||||
|
};
|
||||||
|
|
||||||
|
prefixName = name: res: {
|
||||||
|
inherit (res) result;
|
||||||
|
description = "${name}: ${res.description}";
|
||||||
|
};
|
||||||
|
|
||||||
|
run = name: under: tests: if isList tests then
|
||||||
|
(concatLists (map (run name under) tests))
|
||||||
|
else if isAttrs tests then
|
||||||
|
(concatLists (map (
|
||||||
|
subName: run (name + "." + subName) (if hasAttr subName under then getAttr subName under else "<MISSING!>") (getAttr subName tests)
|
||||||
|
) (attrNames tests)))
|
||||||
|
else if isFunction tests then
|
||||||
|
let
|
||||||
|
res = tests under;
|
||||||
|
in
|
||||||
|
if isBool res then
|
||||||
|
[
|
||||||
|
(prefixName name (if tests under then passed "passed" else failed "failed"))
|
||||||
|
]
|
||||||
|
else
|
||||||
|
[ (prefixName name res) ]
|
||||||
|
else [
|
||||||
|
failed (name ": not a function, list or set")
|
||||||
|
];
|
||||||
|
in
|
||||||
|
{ inherit run passed failed; }
|
Loading…
x
Reference in New Issue
Block a user