{ stdenv, runCommand, writeText, writeScript, writeScriptBin, ruby, lib , callPackage , gemFixes, fetchurl, fetchgit, buildRubyGem #, bundler_PATCHED , bundler_HEAD , git }@defs: # This is a work-in-progress. # The idea is that his will replace load-ruby-env.nix. { name, gemset, gemfile, lockfile, ruby ? defs.ruby, fixes ? gemFixes }@args: let const = x: y: x; bundler = bundler_HEAD.override { inherit ruby; }; inherit (builtins) attrValues; gemName = attrs: "${attrs.name}-${attrs.version}.gem"; fetchers.path = attrs: attrs.source.path; fetchers.gem = attrs: fetchurl { url = "${attrs.source.source or "https://rubygems.org"}/downloads/${gemName attrs}"; inherit (attrs.source) sha256; }; fetchers.git = attrs: fetchgit { inherit (attrs.source) url rev sha256 fetchSubmodules; leaveDotGit = true; }; applySrc = attrs: attrs // { src = (fetchers."${attrs.source.type}" attrs); }; applyFixes = attrs: if fixes ? "${attrs.name}" then attrs // fixes."${attrs.name}" attrs else attrs; # patch a gem or source tree. # for gems, the gem is unpacked, patched, and then repacked. # see: https://github.com/fedora-ruby/gem-patch/blob/master/lib/rubygems/patcher.rb applyPatches = { attrs }: if (!(attrs ? patches)) then attrs else attrs // { src = stdenv.mkDerivation { name = gemName attrs; phases = [ "unpackPhase" "patchPhase" "installPhase" ]; buildInputs = [ ruby ]; inherit (attrs) patches; unpackPhase = '' runHook preUnpack if [[ -f ${attrs.src} ]]; then isGem=1 gem unpack ${attrs.src} --target=contents else cp -r ${attrs.src} contents chmod -R +w contents fi cd contents runHook postUnpack ''; installPhase = '' runHook preInstall mkdir build if [[ -n "$isGem" ]]; then ${writeScript "repack.rb" '' #!${ruby}/bin/ruby require 'rubygems' require 'fileutils' out = ENV['out'] files = Dir['**/{.[^\.]*,*}'] package = Gem::Package.new("${attrs.src}") patched_package = Gem::Package.new(package.spec.file_name) patched_package.spec = package.spec.clone patched_package.spec.files = files # Change dir and build the patched gem Dir.chdir("../build") do patched_package.build false end FileUtils.cp "../build/#{package.file_name}" out ''} else cp -r . out fi runHook postInstall ''; }; }; instantiate = (attrs: applyFixes (applySrc attrs) ); instantiated = lib.flip lib.mapAttrs (import gemset) (name: attrs: instantiate (attrs // { inherit name; }) ); # copy *.gem to ./gems copyGems = lib.fold (next: acc: if next.source.type == "gem" then acc + "cp ${next.src} gems/${gemName next}\n" else acc ) "" (attrValues instantiated); 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 = writeScript "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} > $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. purifiedLockfile = runRuby "purifiedLockfile" {} '' #!${ruby}/bin/ruby out = ENV['out'] sources = eval(File.read("${sources}")) paths = sources["path"] lockfile = File.read("${lockfile}") paths.each_pair do |impure, pure| lockfile.gsub!(/^ remote: #{Regexp.escape(impure)}/, " remote: #{pure}") end File.open(out, "wb") do |f| f.print lockfile end ''; in stdenv.mkDerivation { inherit name; buildInputs = [ ruby bundler git ]; phases = [ "installPhase" "fixupPhase" ]; outputs = [ "out" # the installed libs/bins "bundle" # supporting files for bundler ]; installPhase = '' # Copy the Gemfile and Gemfile.lock mkdir -p $bundle export BUNDLE_GEMFILE=$bundle/Gemfile cp ${gemfile} $BUNDLE_GEMFILE cp ${purifiedLockfile} $BUNDLE_GEMFILE.lock export NIX_GEM_SOURCES=${sources} export NIX_BUNDLER_GEMPATH=${bundler}/${ruby.gemPath} export GEM_HOME=$out/${ruby.gemPath} export GEM_PATH=$GEM_HOME mkdir -p $GEM_HOME mkdir gems ${copyGems} mkdir $out/bin cp ${./monkey_patches.rb} monkey_patches.rb export RUBYOPT="-rmonkey_patches.rb -I $(pwd -P)" bundler install --frozen --binstubs ''; passthru = { inherit ruby; inherit bundler; }; }