module Autoproj module Ops # Operations that modify the source of the main configuration (bootstrap # and switch-config) class MainConfigSwitcher attr_reader :root_dir def initialize(root_dir) @root_dir = root_dir end # Set of directory entries that are expected to be present in the # directory in which the user bootstraps # # @see check_root_dir_empty EXPECTED_ROOT_ENTRIES = [".", "..", "autoproj_bootstrap", ".gems", "bootstrap.sh", ENV_FILENAME].to_set # Verifies that {#root_dir} contains only expected entries, to make # sure that the user bootstraps into a new directory # # If the environment variable AUTOPROJ_BOOTSTRAP_IGNORE_NONEMPTY_DIR # is set to 1, the check is skipped def check_root_dir_empty return if ENV['AUTOPROJ_BOOTSTRAP_IGNORE_NONEMPTY_DIR'] == '1' require 'set' curdir_entries = Dir.entries('.').to_set - EXPECTED_ROOT_ENTRIES return if curdir_entries.empty? while true print "The current directory is not empty, continue bootstrapping anyway ? [yes] " STDOUT.flush answer = STDIN.readline.chomp if answer == "no" raise Interrupt, "Interrupted by user" end if answer == "" || answer == "yes" # Set this environment variable since we might restart # autoproj later on. ENV['AUTOPROJ_BOOTSTRAP_IGNORE_NONEMPTY_DIR'] = '1' return else STDOUT.puts "invalid answer. Please answer 'yes' or 'no'" STDOUT.flush end end end # Validates the environment variable AUTOPROJ_CURRENT_ROOT during a # bootstrap # # AUTOPROJ_CURRENT_ROOT must be set to either the new root # ({root_dir}) or a root that we are reusing # # @param [Array] reuse set of autoproj roots that are being reused # @raise ConfigError def validate_autoproj_current_root(reuse) if current_root = ENV['AUTOPROJ_CURRENT_ROOT'] # Allow having a current root only if it is being reused if (current_root != root_dir) && !reuse.include?(current_root) Autoproj.error "the env.sh from #{ENV['AUTOPROJ_CURRENT_ROOT']} seem to already be sourced" Autoproj.error "start a new shell and try to bootstrap again" Autoproj.error Autoproj.error "you are allowed to boostrap from another autoproj installation" Autoproj.error "only if you reuse it with the --reuse flag" raise ConfigError end end end def handle_bootstrap_options(args) reuse = [] parser = OptionParser.new do |opt| opt.on '--reuse [DIR]', "reuse the given autoproj installation (can be given multiple times). If given without arguments, reuse the currently active install (#{ENV['AUTOPROJ_CURRENT_ROOT']})" do |path| path ||= ENV['AUTOPROJ_CURRENT_ROOT'] path = File.expand_path(path) if !File.directory?(path) || !File.directory?(File.join(path, 'autoproj')) raise ConfigError.new, "#{path} does not look like an autoproj installation" end reuse << path end end Tools.common_options(parser) args = parser.parse(args) return args, reuse end def bootstrap(*args) check_root_dir_empty args, reuse = handle_bootstrap_options(args) validate_autoproj_current_root(reuse) Autoproj.root_dir = root_dir Autobuild.prefix = Autoproj.build_dir Autobuild.srcdir = Autoproj.root_dir Autobuild.logdir = File.join(Autobuild.prefix, 'log') Autoproj.manifest = Manifest.new Tools.load_autoprojrc Autoproj.prepare_environment Autoproj::OSDependencies.define_osdeps_mode_option osdeps = Autoproj::OSDependencies.load_default osdeps.osdeps_mode Autoproj.osdeps = osdeps CmdLine.update_myself :force => true, :restart_on_update => false Autoproj.change_option 'reused_autoproj_installations', reuse, true Autoproj.export_env_sh # If we are not getting the installation setup from a VCS, copy the template # files if args.empty? || args.size == 1 sample_dir = File.expand_path(File.join("..", "..", "..", "samples"), File.dirname(__FILE__)) FileUtils.cp_r File.join(sample_dir, "autoproj"), "autoproj" end if args.size == 1 # the user asks us to download a manifest manifest_url = args.first Autoproj.message("autoproj: downloading manifest file #{manifest_url}", :bold) manifest_data = begin open(manifest_url) { |file| file.read } rescue # Delete the autoproj directory FileUtils.rm_rf 'autoproj' raise ConfigError.new, "cannot read file / URL #{manifest_url}, did you mean 'autoproj bootstrap VCSTYPE #{manifest_url}' ?" end File.open(File.join(Autoproj.config_dir, "manifest"), "w") do |io| io.write(manifest_data) end elsif args.size >= 2 # is a VCS definition for the manifest itself ... type, url, *options = *args url = VCSDefinition.to_absolute_url(url, Dir.pwd) do_switch_config(false, type, url, *options) end Autoproj.save_config end def switch_config(*args) Autoproj.load_config if Autoproj.has_config_key?('manifest_source') vcs = VCSDefinition.from_raw(Autoproj.user_config('manifest_source')) end if args.first =~ /^(\w+)=/ # First argument is an option string, we are simply setting the # options without changing the type/url type, url = vcs.type, vcs.url else type, url = args.shift, args.shift end options = args url = VCSDefinition.to_absolute_url(url) if vcs && (vcs.type == type && vcs.url == url) # Don't need to do much: simply change the options and save the config # file, the VCS handler will take care of the actual switching vcs_def = Autoproj.user_config('manifest_source') options.each do |opt| opt_name, opt_value = opt.split('=') vcs_def[opt_name] = opt_value end # Validate the VCS definition, but save the hash as-is VCSDefinition.from_raw(vcs_def) Autoproj.change_option "manifest_source", vcs_def.dup, true Autoproj.save_config true else # We will have to delete the current autoproj directory. Ask the user. opt = Autoproj::BuildOption.new("delete current config", "boolean", Hash[:default => "false", :doc => "delete the current configuration ? (required to switch)"], nil) return if !opt.ask(nil) Dir.chdir(Autoproj.root_dir) do do_switch_config(true, type, url, *options) end false end end def do_switch_config(delete_current, type, url, *options) vcs_def = Hash.new vcs_def[:type] = type vcs_def[:url] = VCSDefinition.to_absolute_url(url) options.each do |opt| name, value = opt.split("=") if value =~ /^\d+$/ value = Integer(value) end vcs_def[name] = value end vcs = VCSDefinition.from_raw(vcs_def) # Install the OS dependencies required for this VCS Autoproj::OSDependencies.define_osdeps_mode_option osdeps = Autoproj::OSDependencies.load_default osdeps.osdeps_mode osdeps.install([vcs.type]) # Now check out the actual configuration config_dir = File.join(Dir.pwd, "autoproj") if delete_current # Find a backup name for it backup_base_name = backup_name = "#{config_dir}.bak" index = 0 while File.directory?(backup_name) backup_name = "#{backup_base_name}-#{index}.bak" index += 1 end FileUtils.mv config_dir, backup_name end Ops::Configuration.update_configuration_repository( vcs, "autoproj main configuration", config_dir) # If the new tree has a configuration file, load it and set # manifest_source Autoproj.load_config # And now save the options: note that we keep the current option set even # though we switched configuration. This is not a problem as undefined # options will not be reused # # TODO: cleanup the options to only keep the relevant ones vcs_def = Hash['type' => type, 'url' => url] options.each do |opt| opt_name, opt_val = opt.split '=' vcs_def[opt_name] = opt_val end # Validate the option hash, just in case VCSDefinition.from_raw(vcs_def) # Save the new options Autoproj.change_option "manifest_source", vcs_def.dup, true Autoproj.save_config rescue Exception => e Autoproj.error "switching configuration failed: #{e.message}" if backup_name Autoproj.error "restoring old configuration" FileUtils.rm_rf config_dir if config_dir FileUtils.mv backup_name, config_dir end raise ensure if backup_name FileUtils.rm_rf backup_name end end end end end