2018-11-02 19:07:56 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-06-01 05:49:40 -04:00
|
|
|
require_relative "../current_ruby"
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
module Bundler
|
|
|
|
class CLI::Exec
|
|
|
|
attr_reader :options, :args, :cmd
|
|
|
|
|
|
|
|
TRAPPED_SIGNALS = %w[INT].freeze
|
|
|
|
|
|
|
|
def initialize(options, args)
|
|
|
|
@options = options
|
|
|
|
@cmd = args.shift
|
|
|
|
@args = args
|
|
|
|
|
2019-08-17 16:00:52 -04:00
|
|
|
if !Bundler.current_ruby.jruby?
|
2018-11-02 19:07:56 -04:00
|
|
|
@args << { :close_others => !options.keep_file_descriptors? }
|
|
|
|
elsif options.keep_file_descriptors?
|
|
|
|
Bundler.ui.warn "Ruby version #{RUBY_VERSION} defaults to keeping non-standard file descriptors on Kernel#exec."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def run
|
|
|
|
validate_cmd!
|
|
|
|
SharedHelpers.set_bundle_environment
|
|
|
|
if bin_path = Bundler.which(cmd)
|
|
|
|
if !Bundler.settings[:disable_exec_load] && ruby_shebang?(bin_path)
|
2019-12-18 00:53:19 -05:00
|
|
|
return kernel_load(bin_path, *args)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
2019-12-18 00:53:19 -05:00
|
|
|
kernel_exec(bin_path, *args)
|
2018-11-02 19:07:56 -04:00
|
|
|
else
|
|
|
|
# exec using the given command
|
2019-12-18 00:53:19 -05:00
|
|
|
kernel_exec(cmd, *args)
|
2018-11-02 19:07:56 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-10-15 00:20:25 -04:00
|
|
|
private
|
2018-11-02 19:07:56 -04:00
|
|
|
|
|
|
|
def validate_cmd!
|
|
|
|
return unless cmd.nil?
|
|
|
|
Bundler.ui.error "bundler: exec needs a command to run"
|
|
|
|
exit 128
|
|
|
|
end
|
|
|
|
|
|
|
|
def kernel_exec(*args)
|
|
|
|
Kernel.exec(*args)
|
|
|
|
rescue Errno::EACCES, Errno::ENOEXEC
|
|
|
|
Bundler.ui.error "bundler: not executable: #{cmd}"
|
|
|
|
exit 126
|
|
|
|
rescue Errno::ENOENT
|
|
|
|
Bundler.ui.error "bundler: command not found: #{cmd}"
|
|
|
|
Bundler.ui.warn "Install missing gem executables with `bundle install`"
|
|
|
|
exit 127
|
|
|
|
end
|
|
|
|
|
|
|
|
def kernel_load(file, *args)
|
|
|
|
args.pop if args.last.is_a?(Hash)
|
|
|
|
ARGV.replace(args)
|
|
|
|
$0 = file
|
|
|
|
Process.setproctitle(process_title(file, args)) if Process.respond_to?(:setproctitle)
|
2019-06-01 05:49:40 -04:00
|
|
|
require_relative "../setup"
|
2018-11-02 19:07:56 -04:00
|
|
|
TRAPPED_SIGNALS.each {|s| trap(s, "DEFAULT") }
|
|
|
|
Kernel.load(file)
|
|
|
|
rescue SystemExit, SignalException
|
|
|
|
raise
|
|
|
|
rescue Exception => e # rubocop:disable Lint/RescueException
|
|
|
|
Bundler.ui.error "bundler: failed to load command: #{cmd} (#{file})"
|
|
|
|
backtrace = e.backtrace ? e.backtrace.take_while {|bt| !bt.start_with?(__FILE__) } : []
|
|
|
|
abort "#{e.class}: #{e.message}\n #{backtrace.join("\n ")}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_title(file, args)
|
|
|
|
"#{file} #{args.join(" ")}".strip
|
|
|
|
end
|
|
|
|
|
|
|
|
def ruby_shebang?(file)
|
|
|
|
possibilities = [
|
|
|
|
"#!/usr/bin/env ruby\n",
|
|
|
|
"#!/usr/bin/env jruby\n",
|
|
|
|
"#!/usr/bin/env truffleruby\n",
|
|
|
|
"#!#{Gem.ruby}\n",
|
|
|
|
]
|
|
|
|
|
|
|
|
if File.zero?(file)
|
|
|
|
Bundler.ui.warn "#{file} is empty"
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
first_line = File.open(file, "rb") {|f| f.read(possibilities.map(&:size).max) }
|
|
|
|
possibilities.any? {|shebang| first_line.start_with?(shebang) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|