#! /usr/bin/ruby if RUBY_VERSION < "1.8.7" STDERR.puts "autoproj requires Ruby >= 1.8.7" exit 1 end needed_gem_home = "#{Dir.pwd}/.gems" if $LOADED_FEATURES.find { |str| str =~ /bygems/ } if ENV['GEM_HOME'] != needed_gem_home require 'rbconfig' RUBY = RbConfig::CONFIG['RUBY_INSTALL_NAME'] ENV['GEM_HOME'] = needed_gem_home exec RUBY, __FILE__, *ARGV end end ENV['GEM_HOME'] = needed_gem_home ENV['PATH'] = "#{ENV['GEM_HOME']}/bin:#{ENV['PATH']}" require 'yaml' require 'set' module Autoproj class ConfigError < RuntimeError; end class << self attr_reader :verbose end # Fake option management for the OSdeps class def self.has_config_key?(*args) false end def self.change_option(*args) false end end module Autobuild def self.do_update true end def self.progress(str) STDERR.puts " #{str}" end class << self attr_reader :programs end @programs = Hash.new def self.tool(name) # Let the ability to set programs[name] to nil to make sure we don't use # that program. This is used later on in this file to make sure we # aren't using the wrong rubygems binary if programs.has_key?(name) programs[name] else name end end module Subprocess def self.run(name, phase, *cmd) output = `#{cmd.join(" ")}` if $?.exitstatus != 0 STDERR.puts "ERROR: failed to run #{cmd.join(" ")}" STDERR.puts "ERROR: command output is: #{output}" exit 1 end end end end require 'tempfile' module Autoproj class OSDependencies def self.load(file) file = File.expand_path(file) begin data = YAML.load(File.read(file)) || Hash.new verify_definitions(data) rescue ArgumentError => e raise ConfigError, "error in #{file}: #{e.message}" end OSDependencies.new(data, file) end class << self attr_reader :aliases end @aliases = Hash.new def self.alias(old_name, new_name) @aliases[new_name] = old_name end def self.autodetect_ruby ruby_package = if RUBY_VERSION < "1.9.0" then "ruby18" else "ruby19" end self.alias(ruby_package, "ruby") end AUTOPROJ_OSDEPS = File.join(File.expand_path(File.dirname(__FILE__)), 'default.osdeps') def self.load_default file = ENV['AUTOPROJ_DEFAULT_OSDEPS'] || AUTOPROJ_OSDEPS if !File.file?(file) Autoproj.progress "WARN: #{file} (from AUTOPROJ_DEFAULT_OSDEPS) is not a file, falling back to #{AUTOPROJ_OSDEPS}" file = AUTOPROJ_OSDEPS end OSDependencies.load(file) end # The information contained in the OSdeps files, as a hash attr_reader :definitions # The information as to from which osdeps file the current package # information in +definitions+ originates. It is a mapping from the # package name to the osdeps file' full path attr_reader :sources # The Gem::SpecFetcher object that should be used to query RubyGems, and # install RubyGems packages def gem_fetcher @gem_fetcher ||= Gem::SpecFetcher.fetcher end def initialize(defs = Hash.new, file = nil) @definitions = defs.to_hash @sources = Hash.new if file defs.each_key do |package_name| sources[package_name] = file end end end # Returns the full path to the osdeps file from which the package # definition for +package_name+ has been taken def source_of(package_name) sources[package_name] end # Merges the osdeps information of +info+ into +self+. If packages are # defined in both OSDependencies objects, the information in +info+ # takes precedence def merge(info) root_dir = nil @definitions = definitions.merge(info.definitions) do |h, v1, v2| if v1 != v2 root_dir ||= "#{Autoproj.root_dir}/" old = source_of(h).gsub(root_dir, '') new = info.source_of(h).gsub(root_dir, '') Autoproj.warn("osdeps definition for #{h}, previously defined in #{old} overriden by #{new}") end v2 end @sources = sources.merge(info.sources) end # Perform some sanity checks on the given osdeps definitions def self.verify_definitions(hash) hash.each do |key, value| if !key.kind_of?(String) raise ArgumentError, "invalid osdeps definition: found an #{key.class}. Don't forget to put quotes around numbers" end next if !value if value.kind_of?(Array) || value.kind_of?(Hash) verify_definitions(value) else if !value.kind_of?(String) raise ArgumentError, "invalid osdeps definition: found an #{value.class}. Don't forget to put quotes around numbers" end end end end # Autodetects the operating system name and version # # +osname+ is the operating system name, all in lowercase (e.g. ubuntu, # arch, gentoo, debian) # # +versions+ is a set of names that describe the OS version. It includes # both the version number (as a string) and/or the codename if there is # one. # # Examples: ['debian', ['sid', 'unstable']] or ['ubuntu', ['lucid lynx', '10.04']] def self.operating_system if @operating_system return @operating_system elsif Autoproj.has_config_key?('operating_system') @operating_system = Autoproj.user_config('operating_system') elsif data = os_from_lsb if data[0] != "debian" # Fall back to reading debian_version, as # sid is listed as lenny by lsb-release @operating_system = data end end if !@operating_system # Need to do some heuristics unfortunately @operating_system = if File.exists?('/etc/debian_version') codename = [File.read('/etc/debian_version').strip] if codename.first =~ /sid/ codename << "unstable" << "sid" end ['debian', codename] elsif File.exists?('/etc/gentoo-release') release_string = File.read('/etc/gentoo-release').strip release_string =~ /^.*([^\s]+)$/ version = $1 ['gentoo', [version]] elsif File.exists?('/etc/arch-release') ['arch', []] else raise ConfigError, "Unknown operating system" end end # Normalize the names to lowercase @operating_system = [@operating_system[0].downcase, @operating_system[1].map(&:downcase)] Autoproj.change_option('operating_system', @operating_system, true) @operating_system end def self.os_from_lsb has_lsb_release = `which lsb_release` return unless $?.success? distributor = `lsb_release -i -s` distributor = distributor.strip.downcase codename = `lsb_release -c -s`.strip.downcase version = `lsb_release -r -s`.strip.downcase return [distributor, [codename, version]] end GAIN_ROOT_ACCESS = <<-EOSCRIPT if test `id -u` != "0"; then exec sudo /bin/bash $0 "$@" fi EOSCRIPT OS_PACKAGE_INSTALL = { 'debian' => 'export DEBIAN_FRONTEND=noninteractive; apt-get install -y %s', 'ubuntu' => 'export DEBIAN_FRONTEND=noninteractive; apt-get install -y %s', 'gentoo' => 'emerge --noreplace %s', 'arch' => 'pacman -Sy --noconfirm %s' } # Resolves the given OS dependencies into the actual packages that need # to be installed on this particular OS. # # Raises ConfigError if some packages can't be found def resolve_os_dependencies(dependencies) os_name, os_version = OSDependencies.operating_system if !OS_PACKAGE_INSTALL.has_key?(os_name) raise ConfigError, "I don't know how to install packages on #{os_name}" end os_packages = [] shell_snippets = [] dependencies.each do |name| dep_def = definitions[name] if !dep_def raise ConfigError, "I don't know how to install '#{name}'" end # Find a matching entry for the OS name os_entry = dep_def.find do |name_list, data| name_list.split(','). map(&:downcase). any? { |n| n == os_name } end if !os_entry raise ConfigError, "I don't know how to install '#{name}' on #{os_name}" end data = os_entry.last # This package does not need to be installed on this operating system (example: build tools on Gentoo) next if !data || data == "ignore" if data.kind_of?(Hash) version_entry = data.find do |version_list, data| version_list.to_s.split(','). map(&:downcase). any? do |v| os_version.any? { |osv| Regexp.new(v) =~ osv } end end if !version_entry raise ConfigError, "I don't know how to install '#{name}' on this specific version of #{os_name} (#{os_version.join(", ")})" end data = version_entry.last end if data.respond_to?(:to_ary) os_packages.concat data.to_ary elsif data.to_str =~ /\w+/ os_packages << data.to_str else shell_snippets << data.to_str end end return os_packages, shell_snippets end def generate_os_script(dependencies) os_name, os_version = OSDependencies.operating_system os_packages, shell_snippets = resolve_os_dependencies(dependencies) "#! /bin/bash\n" + GAIN_ROOT_ACCESS + "\n" + (OS_PACKAGE_INSTALL[os_name] % [os_packages.join(" ")]) + "\n" + shell_snippets.join("\n") end # Returns true if there is an operating-system package with that name, # and false otherwise def has?(name) osdeps, gemdeps = partition_packages([name].to_set) resolve_os_dependencies(osdeps) true rescue ConfigError false end # call-seq: # partition_packages(package_names) => os_packages, gem_packages # # Resolves the package names listed in +package_set+, and returns a set # of packages that have to be installed using the platform's native # package manager, and the set of packages that have to be installed # using Ruby's package manager, RubyGems. # # Raises ConfigError if no package can be found def partition_packages(package_set, package_osdeps = Hash.new) package_set = package_set. map { |name| OSDependencies.aliases[name] || name }. to_set osdeps, gems = [], [] package_set.to_set.each do |name| pkg_def = definitions[name] if !pkg_def msg = "I know nothing about a prepackaged software called '#{name}'" if pkg_names = package_osdeps[name] msg += ", it is listed as dependency of the following package(s): #{pkg_names.join(", ")}" end raise ConfigError, msg end pkg_def = pkg_def.dup if pkg_def.respond_to?(:to_str) case(pkg_def.to_str) when "ignore" then when "gem" then gems << name else raise ConfigError, "unknown OS-independent package management type #{pkg_def} for #{name}" end else pkg_def.delete_if do |distrib_name, defs| if distrib_name == "gem" gems.concat([*defs]) true end end if !pkg_def.empty? osdeps << name end end end return osdeps, gems end def guess_gem_program if Autobuild.programs['gem'] return Autobuild.programs['gem'] end ruby_bin = Config::CONFIG['RUBY_INSTALL_NAME'] if ruby_bin =~ /^ruby(.+)$/ Autobuild.programs['gem'] = "gem#{$1}" else Autobuild.programs['gem'] = "gem" end end # Requests the installation of the given set of packages def install(packages, package_osdeps = Hash.new) osdeps, gems = partition_packages(packages, package_osdeps) did_something = false # Ideally, we would feed the OS dependencies to rosdep. # Unfortunately, this is C++ code and I don't want to install the # whole ROS stack just for rosdep ... # # So, for now, reimplement rosdep by ourselves. Given how things # are done, this is actually not so hard. if !osdeps.empty? shell_script = generate_os_script(osdeps) if Autoproj.verbose Autoproj.progress "Installing non-ruby OS dependencies with" Autoproj.progress shell_script end File.open('osdeps.sh', 'w') do |file| file.write shell_script end Autobuild.progress "installing/updating OS dependencies: #{osdeps.join(", ")}" begin Autobuild::Subprocess.run 'autoproj', 'osdeps', '/bin/bash', File.expand_path('osdeps.sh') ensure FileUtils.rm_f 'osdeps.sh' end did_something ||= true end if !gems.empty? Autobuild.progress "looking for RubyGems updates" # Don't install gems that are already there ... gems.delete_if do |name| version_requirements = Gem::Requirement.default installed = Gem.source_index.find_name(name, version_requirements) if !installed.empty? && Autobuild.do_update # Look if we can update the package ... dep = Gem::Dependency.new(name, version_requirements) available = gem_fetcher.find_matching(dep) installed_version = installed.map(&:version).max available_version = available.map { |(name, v), source| v }.max needs_update = (available_version > installed_version) !needs_update else !installed.empty? end end end # Now install what is left if !gems.empty? guess_gem_program if Autoproj.verbose Autoproj.progress "Installing rubygems dependencies with" Autoproj.progress "gem install #{gems.join(" ")}" end Autobuild.progress "installing/updating RubyGems dependencies: #{gems.join(", ")}" Autobuild::Subprocess.run 'autoproj', 'osdeps', Autobuild.tool('gem'), 'install', *gems did_something ||= true end did_something end end end DEFS = < e STDERR.puts "failed: #{e.message}" exit(1) end # Now try to find out the name of the gem binary PACKAGES = %w{rdoc libxml2 libxslt zlib build-essential lsb_release} USER_PACKAGES = %w{autoproj} ENV['RUBYOPT'] = "-rubygems" require 'rubygems' packages = PACKAGES.dup if ARGV.first != "dev" packages += USER_PACKAGES end begin osdeps_management = Autoproj::OSDependencies.new(YAML.load(DEFS)) STDERR.puts "autoproj: installing autoproj and its dependencies (this can take a long time)" osdeps_management.install(packages) rescue Autoproj::ConfigError => e STDERR.puts "failed: #{e.message}" exit(1) end if ARGV.first != "dev" Autobuild::Subprocess.run('bootstrap', 'post', 'autoproj', 'bootstrap', *ARGV) end File.open('env.sh', 'w') do |io| io.write <<-EOSHELL export RUBYOPT=-rubygems export GEM_HOME=#{Dir.pwd}/.gems export PATH=$GEM_HOME/bin:$PATH EOSHELL end STDERR.puts <