2015-01-20 02:02:47 +01:00
|
|
|
{ stdenv, runCommand, writeText, writeScript, writeScriptBin, ruby, lib
|
|
|
|
, callPackage , gemFixes, fetchurl, fetchgit, buildRubyGem
|
|
|
|
#, bundler_PATCHED
|
|
|
|
, bundler_HEAD
|
|
|
|
, git
|
2015-01-19 00:47:28 +01:00
|
|
|
}@defs:
|
|
|
|
|
|
|
|
# This is a work-in-progress.
|
2015-01-20 02:02:47 +01:00
|
|
|
# The idea is that his will replace load-ruby-env.nix.
|
2015-01-19 00:47:28 +01:00
|
|
|
|
|
|
|
{ name, gemset, gemfile, lockfile, ruby ? defs.ruby, fixes ? gemFixes }@args:
|
|
|
|
|
|
|
|
let
|
|
|
|
const = x: y: x;
|
2015-01-20 02:02:47 +01:00
|
|
|
#bundler = bundler_PATCHED;
|
|
|
|
bundler = bundler_HEAD.override { inherit ruby; };
|
|
|
|
inherit (builtins) attrValues;
|
2015-01-19 00:47:28 +01:00
|
|
|
|
|
|
|
fetchers.path = attrs: attrs.src.path;
|
2015-01-20 02:02:47 +01:00
|
|
|
fetchers.gem = attrs:
|
|
|
|
let fname = "${attrs.name}-${attrs.version}.gem";
|
|
|
|
in toString (runCommand fname {
|
|
|
|
gem = fetchurl {
|
|
|
|
url = "${attrs.src.source or "https://rubygems.org"}/downloads/${fname}";
|
|
|
|
inherit (attrs.src) sha256;
|
|
|
|
};
|
|
|
|
} ''
|
|
|
|
mkdir $out
|
|
|
|
cp $gem $out/${fname}
|
|
|
|
'') + "/${fname}";
|
|
|
|
|
2015-01-19 00:47:28 +01:00
|
|
|
fetchers.git = attrs: fetchgit {
|
|
|
|
inherit (attrs.src) url rev sha256 fetchSubmodules;
|
|
|
|
leaveDotGit = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
fixSpec = attrs:
|
|
|
|
attrs // (fixes."${attrs.name}" or (const {})) attrs;
|
|
|
|
|
|
|
|
instantiate = (attrs:
|
|
|
|
let
|
|
|
|
withFixes = fixSpec attrs;
|
|
|
|
withSource = withFixes //
|
|
|
|
(if (lib.isDerivation withFixes.src || builtins.isString withFixes.src)
|
|
|
|
then { source = attrs.src; }
|
|
|
|
else { source = attrs.src; src = (fetchers."${attrs.src.type}" attrs); });
|
|
|
|
|
|
|
|
in
|
|
|
|
withSource
|
|
|
|
);
|
|
|
|
|
|
|
|
instantiated = lib.flip lib.mapAttrs (import gemset) (name: attrs:
|
|
|
|
instantiate (attrs // { inherit name; })
|
|
|
|
);
|
|
|
|
|
2015-01-20 02:02:47 +01:00
|
|
|
# only the *.gem files.
|
|
|
|
gems = lib.fold (next: acc:
|
|
|
|
if next.source.type == "gem"
|
|
|
|
then acc ++ [next.src]
|
|
|
|
else acc
|
|
|
|
) [] (attrValues instantiated);
|
|
|
|
|
2015-01-19 00:47:28 +01:00
|
|
|
runRuby = name: env: command:
|
|
|
|
runCommand name env ''
|
|
|
|
${ruby}/bin/ruby ${writeText name command}
|
|
|
|
'';
|
|
|
|
|
|
|
|
# TODO: include json_pure, so the version of ruby doesn't matter.
|
|
|
|
# not all rubies have support for JSON built-in,
|
|
|
|
# so we'll convert JSON to ruby expressions.
|
|
|
|
json2rb = writeScriptBin "json2rb" ''
|
|
|
|
#!${ruby}/bin/ruby
|
|
|
|
begin
|
|
|
|
require 'json'
|
|
|
|
rescue LoadError => ex
|
|
|
|
require 'json_pure'
|
|
|
|
end
|
|
|
|
|
|
|
|
puts JSON.parse(STDIN.read).inspect
|
|
|
|
'';
|
|
|
|
|
|
|
|
# dump the instantiated gemset as a ruby expression.
|
|
|
|
serializedGemset = runCommand "gemset.rb" { json = builtins.toJSON instantiated; } ''
|
|
|
|
printf '%s' "$json" | ${json2rb}/bin/json2rb > $out
|
|
|
|
'';
|
|
|
|
|
|
|
|
# this is a mapping from a source type and identifier (uri/path/etc)
|
|
|
|
# to the pure store path.
|
|
|
|
# we'll use this from the patched bundler to make fetching sources pure.
|
|
|
|
sources = runRuby "sources.rb" { gemset = serializedGemset; } ''
|
|
|
|
out = ENV['out']
|
|
|
|
gemset = eval(File.read(ENV['gemset']))
|
|
|
|
|
|
|
|
sources = {
|
|
|
|
"git" => { },
|
|
|
|
"path" => { },
|
|
|
|
"gem" => { },
|
|
|
|
"svn" => { }
|
|
|
|
}
|
|
|
|
|
|
|
|
gemset.each_value do |spec|
|
|
|
|
type = spec["source"]["type"]
|
|
|
|
val = spec["src"]
|
|
|
|
key =
|
|
|
|
case type
|
|
|
|
when "gem"
|
|
|
|
spec["name"]
|
|
|
|
when "git"
|
|
|
|
spec["source"]["url"]
|
|
|
|
when "path"
|
|
|
|
spec["source"]["originalPath"]
|
|
|
|
when "svn"
|
|
|
|
nil # TODO
|
|
|
|
end
|
|
|
|
|
|
|
|
sources[type][key] = val if key
|
|
|
|
end
|
|
|
|
|
|
|
|
File.open(out, "wb") do |f|
|
|
|
|
f.print sources.inspect
|
|
|
|
end
|
|
|
|
'';
|
|
|
|
|
|
|
|
# rewrite PATH sources to point into the nix store.
|
2015-01-20 02:02:47 +01:00
|
|
|
purifyLockfile = writeScript "purifyLockfile" ''
|
|
|
|
#!${ruby}/bin/ruby
|
|
|
|
|
|
|
|
out = ENV['out']
|
|
|
|
sources = eval(File.read("${sources}"))
|
|
|
|
paths = sources["path"]
|
2015-01-19 00:47:28 +01:00
|
|
|
|
2015-01-20 02:02:47 +01:00
|
|
|
lockfile = STDIN.read
|
2015-01-19 00:47:28 +01:00
|
|
|
|
|
|
|
paths.each_pair do |impure, pure|
|
|
|
|
lockfile.gsub!(/^ remote: #{Regexp.escape(impure)}/, " remote: #{pure}")
|
|
|
|
end
|
|
|
|
|
2015-01-20 02:02:47 +01:00
|
|
|
print lockfile
|
2015-01-19 00:47:28 +01:00
|
|
|
'';
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
stdenv.mkDerivation {
|
|
|
|
inherit name;
|
2015-01-20 02:02:47 +01:00
|
|
|
buildInputs = [
|
|
|
|
ruby
|
|
|
|
bundler
|
|
|
|
git
|
|
|
|
];
|
|
|
|
phases = [ "installPhase" "fixupPhase" ];
|
2015-01-19 00:47:28 +01:00
|
|
|
outputs = [
|
|
|
|
"out" # the installed libs/bins
|
|
|
|
"bundler" # supporting files for bundler
|
|
|
|
];
|
|
|
|
installPhase = ''
|
|
|
|
# Copy the Gemfile and Gemfile.lock
|
|
|
|
mkdir -p $bundler
|
2015-01-20 02:02:47 +01:00
|
|
|
export BUNDLE_GEMFILE=$bundler/Gemfile
|
2015-01-19 00:47:28 +01:00
|
|
|
cp ${gemfile} $BUNDLE_GEMFILE
|
2015-01-20 02:02:47 +01:00
|
|
|
cat ${lockfile} | ${purifyLockfile} > $BUNDLE_GEMFILE.lock
|
2015-01-19 00:47:28 +01:00
|
|
|
|
|
|
|
export NIX_GEM_SOURCES=${sources}
|
2015-01-20 02:02:47 +01:00
|
|
|
export NIX_BUNDLER_GEMPATH=${bundler}/${ruby.gemPath}
|
2015-01-19 00:47:28 +01:00
|
|
|
|
|
|
|
export GEM_HOME=$out/${ruby.gemPath}
|
|
|
|
export GEM_PATH=$GEM_HOME
|
|
|
|
mkdir -p $GEM_HOME
|
|
|
|
|
2015-01-20 02:02:47 +01:00
|
|
|
mkdir gems
|
|
|
|
for gem in ${toString gems}; do
|
|
|
|
ln -s $gem gems
|
|
|
|
done
|
|
|
|
|
|
|
|
cp ${./monkey_patches.rb} monkey_patches.rb
|
|
|
|
export RUBYOPT="-rmonkey_patches.rb -I $(pwd -P)"
|
|
|
|
bundler install --frozen
|
2015-01-19 00:47:28 +01:00
|
|
|
'';
|
|
|
|
passthru = {
|
|
|
|
inherit ruby;
|
|
|
|
};
|
|
|
|
}
|