2007-11-10 10:48:56 +03:00
|
|
|
#--
|
|
|
|
# Copyright 2006 by Chad Fowler, Rich Kilmer, Jim Weirich and others.
|
|
|
|
# All rights reserved.
|
|
|
|
# See LICENSE.txt for permissions.
|
|
|
|
#++
|
|
|
|
|
2013-07-31 02:10:21 +04:00
|
|
|
require 'rubygems/user_interaction'
|
|
|
|
require 'thread'
|
|
|
|
|
2007-11-10 10:48:56 +03:00
|
|
|
class Gem::Ext::Builder
|
|
|
|
|
2013-07-31 02:10:21 +04:00
|
|
|
include Gem::UserInteraction
|
|
|
|
|
|
|
|
##
|
|
|
|
# The builder shells-out to run various commands after changing the
|
|
|
|
# directory. This means multiple installations cannot be allowed to build
|
|
|
|
# extensions in parallel as they may change each other's directories leading
|
|
|
|
# to broken extensions or failed installations.
|
|
|
|
|
|
|
|
CHDIR_MUTEX = Mutex.new # :nodoc:
|
|
|
|
|
|
|
|
attr_accessor :build_args # :nodoc:
|
|
|
|
|
2007-11-10 10:48:56 +03:00
|
|
|
def self.class_name
|
|
|
|
name =~ /Ext::(.*)Builder/
|
|
|
|
$1.downcase
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.make(dest_path, results)
|
|
|
|
unless File.exist? 'Makefile' then
|
2011-06-01 08:05:03 +04:00
|
|
|
raise Gem::InstallError, "Makefile not found:\n\n#{results.join "\n"}"
|
2007-11-10 10:48:56 +03:00
|
|
|
end
|
|
|
|
|
2011-01-29 02:46:47 +03:00
|
|
|
# try to find make program from Ruby configure arguments first
|
2011-01-19 03:08:49 +03:00
|
|
|
RbConfig::CONFIG['configure_args'] =~ /with-make-prog\=(\w+)/
|
2013-07-10 03:21:36 +04:00
|
|
|
make_program = ENV['MAKE'] || ENV['make'] || $1
|
2007-11-10 10:48:56 +03:00
|
|
|
unless make_program then
|
|
|
|
make_program = (/mswin/ =~ RUBY_PLATFORM) ? 'nmake' : 'make'
|
|
|
|
end
|
|
|
|
|
2013-07-23 02:46:50 +04:00
|
|
|
destdir = '"DESTDIR=%s"' % ENV['DESTDIR'] if RUBY_VERSION > '2.0'
|
|
|
|
|
2013-06-25 17:28:57 +04:00
|
|
|
['', 'install'].each do |target|
|
|
|
|
# Pass DESTDIR via command line to override what's in MAKEFLAGS
|
|
|
|
cmd = [
|
|
|
|
make_program,
|
2013-07-23 02:46:50 +04:00
|
|
|
destdir,
|
2013-06-25 17:28:57 +04:00
|
|
|
target
|
|
|
|
].join(' ').rstrip
|
|
|
|
run(cmd, results, "make #{target}".rstrip)
|
2007-11-10 10:48:56 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.redirector
|
|
|
|
'2>&1'
|
|
|
|
end
|
|
|
|
|
2012-11-29 10:52:18 +04:00
|
|
|
def self.run(command, results, command_name = nil)
|
|
|
|
verbose = Gem.configuration.really_verbose
|
|
|
|
|
2013-02-05 06:37:35 +04:00
|
|
|
begin
|
|
|
|
# TODO use Process.spawn when ruby 1.8 support is dropped.
|
|
|
|
rubygems_gemdeps, ENV['RUBYGEMS_GEMDEPS'] = ENV['RUBYGEMS_GEMDEPS'], nil
|
|
|
|
if verbose
|
|
|
|
puts(command)
|
|
|
|
system(command)
|
|
|
|
else
|
|
|
|
results << command
|
|
|
|
results << `#{command} #{redirector}`
|
|
|
|
end
|
|
|
|
ensure
|
|
|
|
ENV['RUBYGEMS_GEMDEPS'] = rubygems_gemdeps
|
2012-11-29 10:52:18 +04:00
|
|
|
end
|
2007-11-10 10:48:56 +03:00
|
|
|
|
2008-07-01 10:01:15 +04:00
|
|
|
unless $?.success? then
|
2012-11-29 10:52:18 +04:00
|
|
|
results << "Building has failed. See above output for more information on the failure." if verbose
|
|
|
|
raise Gem::InstallError, "#{command_name || class_name} failed:\n\n#{results.join "\n"}"
|
2007-11-10 10:48:56 +03:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-07-31 02:10:21 +04:00
|
|
|
##
|
|
|
|
# Creates a new extension builder for +spec+ using the given +build_args+.
|
|
|
|
# The gem for +spec+ is unpacked in +gem_dir+.
|
|
|
|
|
|
|
|
def initialize spec, build_args
|
|
|
|
@spec = spec
|
|
|
|
@build_args = build_args
|
|
|
|
@gem_dir = spec.gem_dir
|
|
|
|
|
|
|
|
@ran_rake = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# Chooses the extension builder class for +extension+
|
|
|
|
|
|
|
|
def builder_for extension # :nodoc:
|
|
|
|
case extension
|
|
|
|
when /extconf/ then
|
|
|
|
Gem::Ext::ExtConfBuilder
|
|
|
|
when /configure/ then
|
|
|
|
Gem::Ext::ConfigureBuilder
|
|
|
|
when /rakefile/i, /mkrf_conf/i then
|
|
|
|
@ran_rake = true
|
|
|
|
Gem::Ext::RakeBuilder
|
|
|
|
when /CMakeLists.txt/ then
|
|
|
|
Gem::Ext::CmakeBuilder
|
|
|
|
else
|
|
|
|
extension_dir = File.join @gem_dir, File.dirname(extension)
|
|
|
|
|
|
|
|
message = "No builder for extension '#{extension}'"
|
|
|
|
build_error extension_dir, message
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# Logs the build +output+ in +build_dir+, then raises ExtensionBuildError.
|
|
|
|
|
|
|
|
def build_error build_dir, output, backtrace = nil # :nodoc:
|
|
|
|
gem_make_out = File.join build_dir, 'gem_make.out'
|
|
|
|
|
|
|
|
open gem_make_out, 'wb' do |io| io.puts output end
|
|
|
|
|
|
|
|
message = <<-EOF
|
|
|
|
ERROR: Failed to build gem native extension.
|
|
|
|
|
|
|
|
#{output}
|
|
|
|
|
|
|
|
Gem files will remain installed in #{@gem_dir} for inspection.
|
|
|
|
Results logged to #{gem_make_out}
|
|
|
|
EOF
|
|
|
|
|
|
|
|
raise Gem::Installer::ExtensionBuildError, message, backtrace
|
|
|
|
end
|
|
|
|
|
|
|
|
def build_extension extension, dest_path # :nodoc:
|
|
|
|
results = []
|
|
|
|
|
|
|
|
extension ||= '' # I wish I knew why this line existed
|
|
|
|
extension_dir = File.join @gem_dir, File.dirname(extension)
|
|
|
|
|
|
|
|
builder = builder_for extension
|
|
|
|
|
|
|
|
begin
|
|
|
|
FileUtils.mkdir_p dest_path
|
|
|
|
|
|
|
|
CHDIR_MUTEX.synchronize do
|
|
|
|
Dir.chdir extension_dir do
|
|
|
|
results = builder.build(extension, @gem_dir, dest_path,
|
|
|
|
results, @build_args)
|
|
|
|
|
|
|
|
say results.join("\n") if Gem.configuration.really_verbose
|
|
|
|
end
|
|
|
|
end
|
|
|
|
rescue
|
|
|
|
build_error extension_dir, results.join("\n"), $@
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
##
|
|
|
|
# Builds extensions. Valid types of extensions are extconf.rb files,
|
|
|
|
# configure scripts and rakefiles or mkrf_conf files.
|
|
|
|
|
|
|
|
def build_extensions
|
|
|
|
return if @spec.extensions.empty?
|
|
|
|
|
|
|
|
if @build_args.empty?
|
|
|
|
say "Building native extensions. This could take a while..."
|
|
|
|
else
|
|
|
|
say "Building native extensions with: '#{@build_args.join ' '}'"
|
|
|
|
say "This could take a while..."
|
|
|
|
end
|
|
|
|
|
|
|
|
dest_path = File.join @gem_dir, @spec.require_paths.first
|
|
|
|
|
|
|
|
@ran_rake = false # only run rake once
|
|
|
|
|
|
|
|
@spec.extensions.each do |extension|
|
|
|
|
break if @ran_rake
|
|
|
|
|
|
|
|
build_extension extension, dest_path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2007-11-10 10:48:56 +03:00
|
|
|
end
|
|
|
|
|