Test harnesses

This commit is contained in:
Judson 2017-05-01 09:07:42 -07:00
parent b2065a2790
commit 2b414e1c15
No known key found for this signature in database
GPG Key ID: 1817B08954BF0B7D
6 changed files with 189 additions and 36 deletions

View File

@ -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

View File

@ -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;
};
})

View File

@ -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" \

View 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}
'';
}

View 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)

View 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; }