Class: Gem::Ext::Builder
Relationships & Source Files | |
Extension / Inclusion / Inheritance Descendants | |
Subclasses:
|
|
Super Chains via Extension / Inclusion / Inheritance | |
Instance Chain:
|
|
Inherits: | Object |
Defined in: | lib/rubygems/ext/builder.rb |
Class Method Summary
- .class_name
- .make(dest_path, results, make_dir = Dir.pwd, sitedir = nil)
-
.new(spec, build_args = spec.build_args) ⇒ Builder
constructor
Creates a new extension builder for
spec
. - .run(command, results, command_name = nil, dir = Dir.pwd, env = {}) {|status, results| ... }
Instance Attribute Summary
- #build_args rw Internal use only
::Gem::DefaultUserInteraction
- Included
Instance Method Summary
-
#build_extensions
Builds extensions.
-
#build_error(output, backtrace = nil)
Internal use only
Logs the build
output
, then raisesBuildError
. - #build_extension(extension, dest_path) Internal use only
-
#builder_for(extension)
Internal use only
Chooses the extension builder class for
extension
-
#write_gem_make_out(output)
Internal use only
Writes
output
to gem_make.out in the extension install directory.
::Gem::UserInteraction
- Included
#alert | Displays an alert |
#alert_error | Displays an error |
#alert_warning | Displays a warning |
#ask | Asks a |
#ask_for_password | Asks for a password with a |
#ask_yes_no | Asks a yes or no |
#choose_from_list | Asks the user to answer |
#say | Displays the given |
#terminate_interaction | Terminates the RubyGems process with the given |
#verbose | Calls |
::Gem::DefaultUserInteraction
- Included
::Gem::Text
- Included
#clean_text | Remove any non-printable characters and make the text suitable for printing. |
#format_text | Wraps |
#levenshtein_distance | Returns a value representing the “cost” of transforming str1 into str2 Vendored version of |
#truncate_text, #min3 |
Constructor Details
.new(spec, build_args = spec.build_args) ⇒ Builder
Creates a new extension builder for spec
. If the spec
does not yet have build arguments, saved, set #build_args which is an ARGV-style array.
# File 'lib/rubygems/ext/builder.rb', line 111
def initialize(spec, build_args = spec.build_args) @spec = spec @build_args = build_args @gem_dir = spec.full_gem_path @ran_rake = false end
Class Method Details
.class_name
[ GitHub ]# File 'lib/rubygems/ext/builder.rb', line 15
def self.class_name name =~ /Ext::(.*)Builder/ $1.downcase end
.make(dest_path, results, make_dir = Dir.pwd, sitedir = nil)
[ GitHub ]# File 'lib/rubygems/ext/builder.rb', line 20
def self.make(dest_path, results, make_dir = Dir.pwd, sitedir = nil) unless File.exist? File.join(make_dir, "Makefile") raise Gem::InstallError, "Makefile not found" end # try to find make program from Ruby configure arguments first RbConfig::CONFIG["configure_args"] =~ /with-make-prog\=(\w+)/ make_program_name = ENV["MAKE"] || ENV["make"] || $1 unless make_program_name make_program_name = (/mswin/ =~ RUBY_PLATFORM) ? "nmake" : "make" end make_program = Shellwords.split(make_program_name) # The installation of the bundled gems is failed when DESTDIR is empty in mswin platform. destdir = (/\bnmake/i !~ make_program_name || ENV["DESTDIR"] && ENV["DESTDIR"] != "") ? "DESTDIR=%s" % ENV["DESTDIR"] : "" env = [destdir] if sitedir env << "sitearchdir=%s" % sitedir env << "sitelibdir=%s" % sitedir end ["clean", "", "install"].each do |target| # Pass DESTDIR via command line to override what's in MAKEFLAGS cmd = [ *make_program, *env, target, ].reject(&:empty?) begin run(cmd, results, "make #{target}".rstrip, make_dir) rescue Gem::InstallError raise unless target == "clean" # ignore clean failure end end end
.run(command, results, command_name = nil, dir = Dir.pwd, env = {}) {|status, results| ... }
# File 'lib/rubygems/ext/builder.rb', line 58
def self.run(command, results, command_name = nil, dir = Dir.pwd, env = {}) verbose = Gem.configuration.really_verbose begin rubygems_gemdeps, ENV["RUBYGEMS_GEMDEPS"] = ENV["RUBYGEMS_GEMDEPS"], nil if verbose puts("current directory: #{dir}") p(command) end results << "current directory: #{dir}" require "shellwords" results << command.shelljoin require "open3" # Set $SOURCE_DATE_EPOCH for the subprocess. build_env = { "SOURCE_DATE_EPOCH" => Gem.source_date_epoch_string }.merge(env) output, status = begin Open3.capture2e(build_env, *command, :chdir => dir) rescue => error raise Gem::InstallError, "#{command_name || class_name} failed#{error.}" end if verbose puts output else results << output end ensure ENV["RUBYGEMS_GEMDEPS"] = rubygems_gemdeps end unless status.success? results << "Building has failed. See above output for more information on the failure." if verbose end yield(status, results) if block_given? unless status.success? exit_reason = if status.exited? ", exit code #{status.exitstatus}" elsif status.signaled? ", uncaught signal #{status.termsig}" end raise Gem::InstallError, "#{command_name || class_name} failed#{exit_reason}" end end
Instance Attribute Details
#build_args (rw)
# File 'lib/rubygems/ext/builder.rb', line 13
attr_accessor :build_args # :nodoc:
Instance Method Details
#build_error(output, backtrace = nil)
Logs the build output
, then raises BuildError
.
# File 'lib/rubygems/ext/builder.rb', line 144
def build_error(output, backtrace = nil) # :nodoc: gem_make_out = write_gem_make_out output = <<-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::Ext::BuildError, , backtrace end
#build_extension(extension, dest_path)
# File 'lib/rubygems/ext/builder.rb', line 159
def build_extension(extension, dest_path) # :nodoc: results = [] builder = builder_for(extension) extension_dir = File. File.join(@gem_dir, File.dirname(extension)) lib_dir = File.join @spec.full_gem_path, @spec.raw_require_paths.first begin FileUtils.mkdir_p dest_path results = builder.build(extension, dest_path, results, @build_args, lib_dir, extension_dir) verbose { results.join("\n") } write_gem_make_out results.join "\n" rescue => e results << e. build_error(results.join("\n"), $@) end end
#build_extensions
Builds extensions. Valid types of extensions are extconf.rb files, configure scripts and rakefiles or mkrf_conf files.
# File 'lib/rubygems/ext/builder.rb', line 187
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 = @spec.extension_dir require "fileutils" FileUtils.rm_f @spec.gem_build_complete_path @spec.extensions.each do |extension| break if @ran_rake build_extension extension, dest_path end FileUtils.touch @spec.gem_build_complete_path end
#builder_for(extension)
Chooses the extension builder class for extension
# File 'lib/rubygems/ext/builder.rb', line 122
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 when /Cargo.toml/ then # We use the spec name here to ensure we invoke the correct init function later Gem::Ext::CargoBuilder.new(@spec) else build_error("No builder for extension '#{extension}'") end end
#write_gem_make_out(output)
Writes output
to gem_make.out in the extension install directory.
# File 'lib/rubygems/ext/builder.rb', line 214
def write_gem_make_out(output) # :nodoc: destination = File.join @spec.extension_dir, "gem_make.out" FileUtils.mkdir_p @spec.extension_dir File.open destination, "wb" do |io| io.puts output end destination end