require 'set' require 'rbconfig' module Autobuild @inherited_environment = Hash.new @environment = Hash.new @env_source_before = Set.new @env_source_after = Set.new @windows = RbConfig::CONFIG["host_os"] =~%r!(msdos|mswin|djgpp|mingw|[Ww]indows)! def self.windows? @windows end @macos = RbConfig::CONFIG["host_os"] =~%r!([Dd]arwin)! def self.macos? @macos end SYSTEM_ENV = Hash.new ORIGINAL_ENV = Hash.new ENV.each do |k, v| ORIGINAL_ENV[k] = v end SHELL_VAR_EXPANSION = if windows? then "%%%s%%" else "$%s" end SHELL_SET_COMMAND = if windows? then "set %s=%s" else "%s=\"%s\"" end SHELL_CONDITIONAL_SET_COMMAND = if windows? then "set %s=%s" else "if test -z \"$%1$s\"; then\n %1$s=\"%3$s\"\nelse\n %1$s=\"%2$s\"\nfi" end SHELL_UNSET_COMMAND = if windows? then "unset %s" else "unset %s" end SHELL_EXPORT_COMMAND = if windows? then "set %s" else "export %s" end SHELL_SOURCE_SCRIPT = if windows? then "%s" else ". \"%s\"" end class << self # List of the environment that should be set before calling a subcommand # # It is a map from environment variable name to the corresponding value. # If the value is an array, it is joined using the operating system's # path separator (File::PATH_SEPARATOR) attr_reader :environment # In generated environment update shell files, indicates whether an # environment variable should be overriden by the shell script, or # simply updated # # If inherited_environment[varname] is true, the generated shell script # will contain # # export VARNAME=new_value:new_value:$VARNAME # # otherwise # # export VARNAME=new_value:new_value attr_reader :inherited_environment end # Resets the value of +name+ to its original value. If it is inherited from # the def self.env_reset(name = nil) if name environment.delete(name) inherited_environment.delete(name) env_init_from_env(name) else environment.keys.each do |env_key| env_reset(env_key) end end end # Unsets any value on the environment variable +name+, including inherited # value. # # In a bourne shell, this would be equivalent to doing # # unset name # def self.env_clear(name = nil) if name environment[name] = nil inherited_environment[name] = nil env_update_var(name) else environment.keys.each do |env_key| env_clear(env_key) end end end # Set a new environment variable def self.env_set(name, *values) environment.delete(name) env_add(name, *values) end @env_inherit = true @env_inherited_variables = Set.new # Returns true if the given environment variable must not be reset by the # env.sh script, but that new values should simply be prepended to it. # # @param [String,nil] name the environment variable that we want to check # for inheritance. If nil, the global setting is returned. # # @see env_inherit env_inherit= def self.env_inherit?(name = nil) if @env_inherit if name @env_inherited_variables.include?(name) else true end end end # If true (the default), the environment variables that are marked as # inherited will be inherited from the global environment (during the # build as well as in the generated env.sh files) # # Otherwise, only the environment that is explicitely set in autobuild # will be passed on to subcommands, and saved in the environment # scripts. # # @see env_inherit? env_inherit def self.env_inherit=(value) @env_inherit = value inherited_environment.keys.each do |env_name| env_init_from_env(env_name) end end # Declare that the given environment variable must not be reset by the # env.sh script, but that new values should simply be prepended to it. # # @return [Boolean] true if environment inheritance is globally enabled and # false otherwise. This is controlled by {env_inherit=} # # @see env_inherit? env_inherit= def self.env_inherit(*names) flag = if !names.last.respond_to?(:to_str) names.pop else true end if flag @env_inherited_variables |= names names.each do |env_name| env_init_from_env(env_name) end else names.each do |n| if @env_inherited_variables.include?(n) @env_inherited_variables.delete(n) env_init_from_env(n) end end end @env_inherit end def self.env_init_from_env(name) if env_inherit?(name) && (parent_env = ORIGINAL_ENV[name]) inherited_environment[name] = parent_env.split(File::PATH_SEPARATOR) else inherited_environment[name] = Array.new end env_update_var(name) end def self.env_push(name, *values) if current = environment[name] current = current.dup env_set(name, *values) env_add(name, *current) else env_add(name, *values) end end # Adds a new value to an environment variable def self.env_add(name, *values) set = if environment.has_key?(name) environment[name] end if !inherited_environment.has_key?(name) env_init_from_env(name) end if !set set = Array.new elsif !set.respond_to?(:to_ary) set = [set] end values.concat(set) @environment[name] = values env_update_var(name) end # Returns an environment variable value # # @param [String] name the environment variable name # @option options [Symbol] inheritance_mode (:expand) controls how # environment variable inheritance should be done. If :expand, the current # envvar value is inserted in the generated value. If :keep, the name of # the envvar is inserted (as e.g. $NAME). If :ignore, inheritance is # disabled in the generated value. Not that this applies only for the # environment variables for which inheritance has been enabled with # #env_inherit, other variables always behave as if :ignore was selected. # @return [nil,Array] either nil if this environment variable is not # set, or an array of values. How the values should be joined to form the # actual value is OS-specific, and not handled by this method def self.env_value(name, options = Hash.new) # For backward compatibility only if !options.respond_to?(:to_hash) if options options = Hash[:inheritance_mode => :expand] else options = Hash[:inheritance_mode => :keep] end end options = Kernel.validate_options options, inheritance_mode: :expand inheritance_mode = options[:inheritance_mode] if !environment[name] && !inherited_environment[name] && !SYSTEM_ENV[name] nil else inherited = if inheritance_mode == :expand inherited_environment[name] || [] elsif inheritance_mode == :keep && env_inherit?(name) ["$#{name}"] else [] end value = [] [environment[name], inherited, SYSTEM_ENV[name]].each do |paths| (paths || []).each do |p| if !value.include?(p) value << p end end end value end end def self.env_update_var(name) if value = env_value(name) ENV[name] = value.join(File::PATH_SEPARATOR) else ENV.delete(name) end end def self.env_add_path(name, *paths) oldpath = (environment[name] ||= Array.new) paths.reverse.each do |path| next if oldpath.include?(path) env_add(name, path) oldpath << path if name == 'RUBYLIB' $LOAD_PATH.unshift path end end end def self.env_remove_path(name, *paths) paths.each do |p| environment[name].delete(p) end env_update_var(name) end def self.env_push_path(name, *values) if current = environment.delete(name) current = current.dup env_add_path(name, *values) env_add_path(name, *current) else env_add_path(name, *values) end end # Require that generated environment variable scripts source the given shell # script def self.env_source_file(file) env_source_after(file) end # @overload env_source_before # List of scripts that should be sourced at the top of env.sh # # @return [Array] a list of paths that should be sourced at the # beginning of the shell script generated by {export_env_sh} # # @overload env_source_before(path) # @param [String] path a path that should be added to env_source_before # def self.env_source_before(file = nil) if file @env_source_before << file end end # @overload env_source_after # List of scripts that should be sourced at the end of env.sh # # @return [Array] a list of paths that should be sourced at the # end of the shell script generated by {export_env_sh} # # @overload env_source_after(path) # @param [String] path a path that should be added to env_source_after # def self.env_source_after(file = nil) if file @env_source_after << file end end # Generates a shell script that sets the environment variable listed in # Autobuild.environment, following the inheritance setting listed in # Autobuild.inherited_environment. # # It also sources the files added by Autobuild.env_source_file def self.export_env_sh(io) @env_source_before.each do |path| io.puts SHELL_SOURCE_SCRIPT % path end variables = [] Autobuild.environment.each do |name, _| variables << name value_with_inheritance = env_value(name, inheritance_mode: :keep) value_without_inheritance = env_value(name, inheritance_mode: :ignore) if !value_with_inheritance shell_line = SHELL_UNSET_COMMAND % [name] elsif value_with_inheritance == value_without_inheritance # no inheritance shell_line = SHELL_SET_COMMAND % [name, value_with_inheritance.join(File::PATH_SEPARATOR)] else shell_line = SHELL_CONDITIONAL_SET_COMMAND % [name, value_with_inheritance.join(File::PATH_SEPARATOR), value_without_inheritance.join(File::PATH_SEPARATOR)] end io.puts shell_line end variables.each do |var| io.puts SHELL_EXPORT_COMMAND % [var] end @env_source_after.each do |path| io.puts SHELL_SOURCE_SCRIPT % [path] end end # DEPRECATED: use env_add_path instead def self.pathvar(path, varname) if File.directory?(path) if block_given? return unless yield(path) end env_add_path(varname, path) end end def self.each_env_search_path(prefix, patterns) arch_names = self.arch_names arch_size = self.arch_size seen = Set.new patterns.each do |base_path| paths = [] if base_path =~ /ARCHSIZE/ base_path = base_path.gsub('ARCHSIZE', arch_size.to_s) end if base_path =~ /ARCH/ arch_names.each do |arch| paths << base_path.gsub('ARCH', arch) end else paths << base_path end paths.each do |p| p = File.join(prefix, *p.split('/')) if !seen.include?(p) && File.directory?(p) yield(p) seen << p end end end end def self.arch_size if @arch_size return @arch_size end @arch_size = if RbConfig::CONFIG['host_cpu'] =~ /64/ 64 else 32 end end def self.arch_names if @arch_names return @arch_names end result = Set.new if File.file?('/usr/bin/dpkg-architecture') arch = `/usr/bin/dpkg-architecture`.split.grep(/DEB_BUILD_MULTIARCH/).first if arch result << arch.chomp.split('=').last end end @arch_names = result end # Updates the environment when a new prefix has been added def self.update_environment(newprefix, includes = nil) if !includes || includes.include?('PATH') if File.directory?("#{newprefix}/bin") env_add_path('PATH', "#{newprefix}/bin") end end if !includes || includes.include?('PKG_CONFIG_PATH') pkg_config_search = ['lib/pkgconfig', 'lib/ARCH/pkgconfig', 'libARCHSIZE/pkgconfig'] each_env_search_path(newprefix, pkg_config_search) do |path| env_add_path('PKG_CONFIG_PATH', path) end end if !includes || includes.include?('LD_LIBRARY_PATH') ld_library_search = ['lib', 'lib/ARCH', 'libARCHSIZE'] each_env_search_path(newprefix, ld_library_search) do |path| if !Dir.glob(File.join(path, "lib*.so")).empty? env_add_path('LD_LIBRARY_PATH', path) end end end # Validate the new rubylib path if !includes || includes.include?('RUBYLIB') new_rubylib = "#{newprefix}/lib" if File.directory?(new_rubylib) && !File.directory?(File.join(new_rubylib, "ruby")) && !Dir["#{new_rubylib}/**/*.rb"].empty? env_add_path('RUBYLIB', new_rubylib) end require 'rbconfig' %w{rubylibdir archdir sitelibdir sitearchdir vendorlibdir vendorarchdir}. map { |key| RbConfig::CONFIG[key] }. map { |path| path.gsub(/.*lib(?:32|64)?\//, '\\1') }. each do |subdir| if File.directory?("#{newprefix}/lib/#{subdir}") env_add_path("RUBYLIB", "#{newprefix}/lib/#{subdir}") end end end end end