Merge pull request #19180 from zimbatm/bundler-env-bins

Bundler env bins
This commit is contained in:
zimbatm 2016-10-14 11:51:23 +01:00 committed by GitHub
commit 6351c4d24b
9 changed files with 89 additions and 116 deletions

View File

@ -6,7 +6,14 @@
, tree , tree
}@defs: }@defs:
{ name, gemset, gemfile, lockfile, ruby ? defs.ruby, gemConfig ? defaultGemConfig { name ? null
, pname ? null
, gemdir ? null
, gemfile ? null
, lockfile ? null
, gemset ? null
, ruby ? defs.ruby
, gemConfig ? defaultGemConfig
, postBuild ? null , postBuild ? null
, document ? [] , document ? []
, meta ? {} , meta ? {}
@ -16,54 +23,95 @@
}@args: }@args:
let let
importedGemset = import gemset; drvName =
if name != null then name
else if pname != null then "${toString pname}-${mainGem.version}"
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' =
if gemfile == null then gemdir + "/Gemfile"
else gemfile;
lockfile' =
if lockfile == null then gemdir + "/Gemfile.lock"
else lockfile;
gemset' =
if gemset == null then gemdir + "/gemset.nix"
else gemset;
importedGemset = import gemset';
filteredGemset = (lib.filterAttrs (name: attrs: filteredGemset = (lib.filterAttrs (name: attrs:
if (builtins.hasAttr "groups" attrs) if (builtins.hasAttr "groups" attrs)
then (builtins.any (gemGroup: builtins.any (group: group == gemGroup) groups) attrs.groups) then (builtins.any (gemGroup: builtins.any (group: group == gemGroup) groups) attrs.groups)
else true else true
) importedGemset); ) importedGemset);
applyGemConfigs = attrs: applyGemConfigs = attrs:
(if gemConfig ? "${attrs.gemName}" (if gemConfig ? "${attrs.gemName}"
then attrs // gemConfig."${attrs.gemName}" attrs then attrs // gemConfig."${attrs.gemName}" attrs
else attrs); else attrs);
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; })
); );
hasBundler = builtins.hasAttr "bundler" filteredGemset; hasBundler = builtins.hasAttr "bundler" filteredGemset;
bundler = if hasBundler then gems.bundler else defs.bundler.override (attrs: { inherit ruby; });
bundler =
if hasBundler then gems.bundler
else defs.bundler.override (attrs: { inherit ruby; });
gems = lib.flip lib.mapAttrs configuredGemset (name: attrs: gems = lib.flip lib.mapAttrs configuredGemset (name: attrs:
buildRubyGem ((removeAttrs attrs ["source"]) // attrs.source // { buildRubyGem ((removeAttrs attrs ["source"]) // attrs.source // {
inherit ruby; inherit ruby;
gemName = name; gemName = name;
gemPath = map (gemName: gems."${gemName}") (attrs.dependencies or []); gemPath = map (gemName: gems."${gemName}") (attrs.dependencies or []);
})); }));
# 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
cp ${gemfile} $out/Gemfile cp ${gemfile'} $out/Gemfile
cp ${lockfile} $out/Gemfile.lock cp ${lockfile'} $out/Gemfile.lock
''; '';
envPaths = lib.attrValues gems ++ lib.optional (!hasBundler) bundler; envPaths = lib.attrValues gems ++ lib.optional (!hasBundler) bundler;
binPaths = if mainGem != null then [ mainGem ] else envPaths;
bundlerEnv = buildEnv { bundlerEnv = buildEnv {
inherit name ignoreCollisions; inherit ignoreCollisions;
name = drvName;
paths = envPaths; paths = envPaths;
pathsToLink = [ "/lib" ]; pathsToLink = [ "/lib" ];
postBuild = '' postBuild = ''
${ruby}/bin/ruby ${./gen-bin-stubs.rb} \ ${ruby}/bin/ruby ${./gen-bin-stubs.rb} \
"${ruby}/bin/ruby" \ "${ruby}/bin/ruby" \
"${confFiles}/Gemfile" \ "${confFiles}/Gemfile" \
"$out/${ruby.gemPath}" \ "$out/${ruby.gemPath}" \
"${bundler}/${ruby.gemPath}" \ "${bundler}/${ruby.gemPath}" \
${lib.escapeShellArg envPaths} \ ${lib.escapeShellArg binPaths} \
${lib.escapeShellArg groups} ${lib.escapeShellArg groups}
'' + lib.optionalString (postBuild != null) postBuild; '' + lib.optionalString (postBuild != null) postBuild;
meta = { platforms = ruby.meta.platforms; } // meta;
passthru = rec { passthru = rec {
inherit ruby bundler meta gems; inherit ruby bundler gems;
wrappedRuby = stdenv.mkDerivation { wrappedRuby = stdenv.mkDerivation {
name = "wrapped-ruby-${name}"; name = "wrapped-ruby-${drvName}";
nativeBuildInputs = [ makeWrapper ]; nativeBuildInputs = [ makeWrapper ];
buildCommand = '' buildCommand = ''
mkdir -p $out/bin mkdir -p $out/bin
@ -87,7 +135,7 @@ let
require 'bundler/setup' require 'bundler/setup'
''; '';
in stdenv.mkDerivation { in stdenv.mkDerivation {
name = "interactive-${name}-environment"; name = "interactive-${drvName}-environment";
nativeBuildInputs = [ wrappedRuby bundlerEnv ]; nativeBuildInputs = [ wrappedRuby bundlerEnv ];
shellHook = '' shellHook = ''
export OLD_IRBRC="$IRBRC" export OLD_IRBRC="$IRBRC"
@ -102,7 +150,5 @@ let
}; };
}; };
}; };
in in
bundlerEnv
bundlerEnv

View File

@ -1,29 +1,14 @@
{ stdenv, lib, bundlerEnv, ruby }: { lib, bundlerEnv, ruby }:
stdenv.mkDerivation rec { bundlerEnv {
name = "travis-${version}";
version = env.gems.travis.version;
env = bundlerEnv {
inherit ruby; inherit ruby;
name = "${name}-gems"; pName = "travis";
gemset = ./gemset.nix; gemdir = ./.;
gemfile = ./Gemfile;
lockfile = ./Gemfile.lock;
};
phases = ["installPhase"];
installPhase = ''
mkdir -p $out/bin
ln -s ${env}/bin/travis $out/bin/travis
'';
meta = with lib; { meta = with lib; {
description = "CLI and Ruby client library for Travis CI"; description = "CLI and Ruby client library for Travis CI";
homepage = https://github.com/travis-ci/travis.rb; homepage = https://github.com/travis-ci/travis.rb;
license = licenses.mit; license = licenses.mit;
maintainers = with maintainers; [ zimbatm ]; maintainers = with maintainers; [ zimbatm ];
platforms = ruby.meta.platforms;
}; };
} }

View File

@ -1,21 +1,9 @@
{ bundlerEnv, lib, stdenv }: { bundlerEnv, lib, ruby }:
let bundlerEnv {
name = "riemann-dash-${env.gems.riemann-dash.version}"; inherit ruby;
pName = "riemann-dash";
env = bundlerEnv { gemdir = ./.;
inherit name;
gemfile = ./Gemfile;
lockfile = ./Gemfile.lock;
gemset = ./gemset.nix;
};
in stdenv.mkDerivation {
inherit name;
buildCommand = ''
mkdir -p $out/bin
ln -s ${env}/bin/riemann-dash $out/bin/riemann-dash
'';
meta = with lib; { meta = with lib; {
description = "A javascript, websockets-powered dashboard for Riemann"; description = "A javascript, websockets-powered dashboard for Riemann";

View File

@ -1,12 +1,10 @@
{ stdenv, lib, bundlerEnv, ruby, curl }: { stdenv, lib, bundlerEnv, ruby, curl }:
bundlerEnv { bundlerEnv {
name = "fluentd-0.14.0";
inherit ruby; inherit ruby;
gemfile = ./Gemfile;
lockfile = ./Gemfile.lock; pname = "fluentd";
gemset = ./gemset.nix; gemdir = ./.;
meta = with lib; { meta = with lib; {
description = "A data collector"; description = "A data collector";

View File

@ -2,12 +2,10 @@
, pkgconfig, which }: , pkgconfig, which }:
bundlerEnv { bundlerEnv {
name = "lolcat-42.1.0";
inherit ruby; inherit ruby;
gemfile = ./Gemfile;
lockfile = ./Gemfile.lock; pname = "lolcat";
gemset = ./gemset.nix; gemdir = ./.;
meta = with lib; { meta = with lib; {
description = "A rainbow version of cat"; description = "A rainbow version of cat";

View File

@ -1,13 +1,9 @@
{ lib, bundlerEnv, ruby }: { lib, bundlerEnv, ruby }:
bundlerEnv rec { bundlerEnv rec {
name = "fpm-${version}";
version = (import gemset).fpm.version;
inherit ruby; inherit ruby;
gemfile = ./Gemfile; pname = "fpm";
lockfile = ./Gemfile.lock; gemdir = ./.;
gemset = ./gemset.nix;
meta = with lib; { meta = with lib; {
description = "Tool to build packages for multiple platforms with ease"; description = "Tool to build packages for multiple platforms with ease";

View File

@ -1,24 +1,9 @@
{ stdenv, lib, ruby, bundlerEnv, makeWrapper }: { bundlerEnv, lib, ruby }:
stdenv.mkDerivation rec { bundlerEnv {
name = "foreman-${env.gems.foreman.version}";
env = bundlerEnv {
inherit ruby; inherit ruby;
name = "${name}-gems"; pName = "foreman";
gemfile = ./Gemfile; gemdir = ./.;
lockfile = ./Gemfile.lock;
gemset = ./gemset.nix;
};
phases = ["installPhase"];
nativeBuildInputs = [ makeWrapper ];
installPhase = ''
mkdir -p $out/bin
makeWrapper ${env}/bin/foreman $out/bin/foreman
'';
meta = with lib; { meta = with lib; {
description = "Process manager for applications with multiple components"; description = "Process manager for applications with multiple components";

View File

@ -1,22 +1,9 @@
{ lib, bundlerEnv, stdenv }: { lib, bundlerEnv, ruby }:
let bundlerEnv {
name = "hiera-eyaml-${env.gems.hiera-eyaml.version}"; inherit ruby;
pName = "hiera-eyaml";
env = bundlerEnv { gemdir = ./.;
inherit name;
gemfile = ./Gemfile;
lockfile = ./Gemfile.lock;
gemset = ./gemset.nix;
};
in stdenv.mkDerivation {
inherit name;
buildCommand = ''
mkdir -p $out/bin
ln -s ${env}/bin/eyaml $out/bin/eyaml
'';
meta = with lib; { meta = with lib; {
description = "Per-value asymmetric encryption of sensitive data for Hiera"; description = "Per-value asymmetric encryption of sensitive data for Hiera";

View File

@ -1,19 +1,9 @@
{ stdenv, lib, bundlerEnv, ruby_2_2, curl }: { stdenv, lib, bundlerEnv, ruby_2_2, curl }:
bundlerEnv rec { bundlerEnv rec {
name = "asciidoctor-${version}"; pname = "asciidoctor";
version = "1.5.4";
ruby = ruby_2_2; ruby = ruby_2_2;
gemfile = ./Gemfile; gemdir = ./.;
lockfile = ./Gemfile.lock;
gemset = ./gemset.nix;
# Delete dependencies' executables
postBuild = ''
find $out/bin -type f -not -wholename '*bin/asciidoctor*' -print0 \
| xargs -0 rm
'';
meta = with lib; { meta = with lib; {
description = "A faster Asciidoc processor written in Ruby"; description = "A faster Asciidoc processor written in Ruby";