require 'yaml'
require 'utilrb/kernel/options'
require 'nokogiri'
require 'set'
module Autoproj
@build_system_dependencies = Set.new
# Declare OS packages that are required to import and build the packages
#
# It is used by autoproj itself to install the importers and/or the build
# systems for the packages.
def self.add_build_system_dependency(*names)
@build_system_dependencies |= names.to_set
end
class << self
# Returns the set of OS packages that are needed to build and/or import
# the packages
#
# See Autoproj.add_build_system_dependency
attr_reader :build_system_dependencies
end
# Expand build options in +value+.
#
# The method will expand in +value+ patterns of the form $NAME, replacing
# them with the corresponding build option.
def self.expand_environment(value)
# Perform constant expansion on the defined environment variables,
# including the option set
options = Autoproj.option_set
options.each_key do |k|
options[k] = options[k].to_s
end
loop do
new_value = Autoproj.single_expansion(value, options)
if new_value == value
break
else
value = new_value
end
end
value
end
@env_inherit = 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.
#
# See Autoproj.env_inherit
def self.env_inherit?(name)
@env_inherit.include?(name)
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.
#
# See Autoproj.env_inherit?
def self.env_inherit(*names)
@env_inherit |= names
end
# Resets the value of the given environment variable to the given
def self.env_set(name, *value)
Autobuild.env_clear(name)
env_add(name, *value)
end
def self.env_add(name, *value)
value = value.map { |v| expand_environment(v) }
Autobuild.env_add(name, *value)
end
def self.env_set_path(name, *value)
Autobuild.env_clear(name)
env_add_path(name, *value)
end
def self.env_add_path(name, *value)
value = value.map { |v| expand_environment(v) }
Autobuild.env_add_path(name, *value)
end
class VCSDefinition
attr_reader :type
attr_reader :url
attr_reader :options
def initialize(type, url, options)
@type, @url, @options = type, url, options
if type != "none" && type != "local" && !Autobuild.respond_to?(type)
raise ConfigError.new, "version control #{type} is unknown to autoproj"
end
end
def local?
@type == 'local'
end
def ==(other_vcs)
return false if !other_vcs.kind_of?(VCSDefinition)
if local?
other_vcs.local? && url == other.url
elsif !other_vcs.local?
this_importer = create_autobuild_importer
other_importer = other_vcs.create_autobuild_importer
this_importer.repository_id == other_importer.repository_id
end
end
def self.to_absolute_url(url, root_dir = nil)
# NOTE: we MUST use nil as default argument of root_dir as we don't
# want to call Autoproj.root_dir unless completely necessary
# (to_absolute_url might be called on installations that are being
# bootstrapped, and as such don't have a root dir yet).
url = Autoproj.single_expansion(url, 'HOME' => ENV['HOME'])
if url && url !~ /^(\w+:\/)?\/|^\w+\@|^(\w+\@)?[\w\.-]+:/
url = File.expand_path(url, root_dir || Autoproj.root_dir)
end
url
end
def create_autobuild_importer
return if type == "none"
url = VCSDefinition.to_absolute_url(self.url)
Autobuild.send(type, url, options)
end
def to_s
if type == "none"
"none"
else
desc = "#{type}:#{url}"
if !options.empty?
desc = "#{desc} #{options.to_a.sort_by { |key, _| key.to_s }.map { |key, value| "#{key}=#{value}" }.join(" ")}"
end
desc
end
end
end
def self.vcs_definition_to_hash(spec)
options = Hash.new
if spec.size == 1 && spec.keys.first =~ /auto_imports$/
# The user probably wrote
# - string
# auto_imports: false
options['auto_imports'] = spec.values.first
spec = spec.keys.first.split(" ").first
end
if spec.respond_to?(:to_str)
vcs, *url = spec.to_str.split ':'
spec = if url.empty?
source_dir = File.expand_path(File.join(Autoproj.config_dir, spec))
if !File.directory?(source_dir)
raise ConfigError.new, "'#{spec.inspect}' is neither a remote source specification, nor a local source definition"
end
Hash[:type => 'local', :url => source_dir]
else
Hash[:type => vcs.to_str, :url => url.join(":").to_str]
end
end
spec, vcs_options = Kernel.filter_options spec, :type => nil, :url => nil
return spec.merge(vcs_options).merge(options)
end
# Autoproj configuration files accept VCS definitions in three forms:
# * as a plain string, which is a relative/absolute path
# * as a plain string, which is a vcs_type:url string
# * as a hash
#
# This method normalizes the three forms into a VCSDefinition object
def self.normalize_vcs_definition(spec)
spec = vcs_definition_to_hash(spec)
if !(spec[:type] && (spec[:type] == 'none' || spec[:url]))
raise ConfigError.new, "the source specification #{spec.inspect} misses either the VCS type or an URL"
end
spec, vcs_options = Kernel.filter_options spec, :type => nil, :url => nil
return VCSDefinition.new(spec[:type], spec[:url], vcs_options)
end
def self.single_expansion(data, definitions)
if !data.respond_to?(:to_str)
return data
end
definitions = { 'HOME' => ENV['HOME'] }.merge(definitions)
data = data.gsub /\$(\w+)/ do |constant_name|
constant_name = constant_name[1..-1]
if !(value = definitions[constant_name])
if !(value = Autoproj.user_config(constant_name))
if !block_given? || !(value = yield(constant_name))
raise ArgumentError, "cannot find a definition for $#{constant_name}"
end
end
end
value
end
data
end
def self.expand(value, definitions = Hash.new)
if value.respond_to?(:to_hash)
value.dup.each do |name, definition|
value[name] = expand(definition, definitions)
end
value
else
value = single_expansion(value, definitions)
if contains_expansion?(value)
raise ConfigError.new, "some expansions are not defined in #{value.inspect}"
end
value
end
end
# True if the given string contains expansions
def self.contains_expansion?(string); string =~ /\$/ end
def self.resolve_one_constant(name, value, result, definitions)
result[name] = single_expansion(value, result) do |missing_name|
result[missing_name] = resolve_one_constant(missing_name, definitions.delete(missing_name), result, definitions)
end
end
def self.resolve_constant_definitions(constants)
constants = constants.dup
constants['HOME'] = ENV['HOME']
result = Hash.new
while !constants.empty?
name = constants.keys.first
value = constants.delete(name)
resolve_one_constant(name, value, result, constants)
end
result
end
# A package set is a version control repository which contains general
# information with package version control information (source.yml file),
# package definitions (.autobuild files), and finally definition of
# dependencies that are provided by the operating system (.osdeps file).
class PackageSet
attr_reader :manifest
# The VCSDefinition object that defines the version control holding
# information for this source. Local package sets (i.e. the ones that are not
# under version control) use the 'local' version control name. For them,
# local? returns true.
attr_accessor :vcs
# The set of OSDependencies object that represent the osdeps files
# available in this package set
attr_reader :all_osdeps
# The OSDependencies which is a merged version of all OSdeps in
# #all_osdeps
attr_reader :osdeps
# If this package set has been imported from another package set, this
# is the other package set object
attr_accessor :imported_from
# If true, this package set has been loaded because another set imports
# it. If false, it is loaded explicitely by the user
def explicit?; !@imported_from end
attr_reader :source_definition
attr_reader :constants_definitions
# Sets the auto_imports flag. See #auto_imports?
attr_writer :auto_imports
# If true (the default), imports listed in this package set will be
# automatically loaded by autoproj
def auto_imports?; !!@auto_imports end
# Create this source from a VCSDefinition object
def initialize(manifest, vcs)
@manifest = manifest
@vcs = vcs
@osdeps = OSDependencies.new
@all_osdeps = []
@provides = Set.new
@imports = Array.new
@auto_imports = true
end
# Load a new osdeps file for this package set
def load_osdeps(file)
new_osdeps = OSDependencies.load(file)
@all_osdeps << new_osdeps
@osdeps.merge(@all_osdeps.last)
new_osdeps
end
# Enumerate all osdeps package names from this package set
def each_osdep(&block)
@osdeps.definitions.each_key(&block)
end
# True if this source has already been checked out on the local autoproj
# installation
def present?; File.directory?(raw_local_dir) end
# True if this source is local, i.e. is not under a version control
def local?; vcs.local? end
# True if this source defines nothing
def empty?
!source_definition['version_control'] && !source_definition['overrides']
!each_package.find { true } &&
!File.exists?(File.join(raw_local_dir, "overrides.rb")) &&
!File.exists?(File.join(raw_local_dir, "init.rb"))
end
# Create a PackageSet instance from its description as found in YAML
# configuration files
def self.from_spec(manifest, spec, load_description)
spec = Autoproj.vcs_definition_to_hash(spec)
options, vcs_spec = Kernel.filter_options spec, :auto_imports => true
# Look up for short notation (i.e. not an explicit hash). It is
# either vcs_type:url or just url. In the latter case, we expect
# 'url' to be a path to a local directory
vcs_spec = Autoproj.expand(vcs_spec, manifest.constant_definitions)
vcs_def = Autoproj.normalize_vcs_definition(vcs_spec)
source = PackageSet.new(manifest, vcs_def)
source.auto_imports = options[:auto_imports]
if load_description
if source.present?
source.load_description_file
else
raise InternalError, "cannot load description file as it has not been checked out yet"
end
else
# Try to load just the name from the source.yml file
source.load_minimal
end
source
end
def repository_id
if local?
local_dir
else
importer = vcs.create_autobuild_importer
if importer.respond_to?(:repository_id)
importer.repository_id
else
vcs.to_s
end
end
end
# Remote sources can be accessed through a hidden directory in
# $AUTOPROJ_ROOT/.remotes, or through a symbolic link in
# autoproj/remotes/
#
# This returns the former. See #user_local_dir for the latter.
#
# For local sources, is simply returns the path to the source directory.
def raw_local_dir
if local?
File.expand_path(vcs.url)
else
File.expand_path(File.join(Autoproj.remotes_dir, vcs.to_s.gsub(/[^\w]/, '_')))
end
end
# Remote sources can be accessed through a hidden directory in
# $AUTOPROJ_ROOT/.remotes, or through a symbolic link in
# autoproj/remotes/
#
# This returns the latter. See #raw_local_dir for the former.
#
# For local sources, is simply returns the path to the source directory.
def user_local_dir
if local?
return vcs.url
else
File.join(Autoproj.config_dir, 'remotes', name)
end
end
# The directory in which data for this source will be checked out
def local_dir
ugly_dir = raw_local_dir
pretty_dir = user_local_dir
if File.symlink?(pretty_dir) && File.readlink(pretty_dir) == ugly_dir
pretty_dir
else
ugly_dir
end
end
# Returns the source name
def name
if @name
@name
else
vcs.to_s
end
end
# Loads the source.yml file, validates it and returns it as a hash
#
# Raises InternalError if the source has not been checked out yet (it
# should have), and ConfigError if the source.yml file is not valid.
def raw_description_file
if !present?
raise InternalError, "source #{vcs} has not been fetched yet, cannot load description for it"
end
source_file = File.join(raw_local_dir, "source.yml")
if !File.exists?(source_file)
raise ConfigError.new, "source #{vcs.type}:#{vcs.url} should have a source.yml file, but does not"
end
source_definition = Autoproj.in_file(source_file, ArgumentError) do
YAML.load(File.read(source_file))
end
if !source_definition || !source_definition['name']
raise ConfigError.new(source_file), "in #{source_file}: missing a 'name' field"
end
source_definition
end
# Load and validate the self-contained information from the YAML hash
def load_minimal
# If @source_definition is set, it means that load_description_file
# has been called and that therefore all information has already
# been parsed
definition = @source_definition || raw_description_file
@name = definition['name']
if @name !~ /^[\w_\.-]+$/
raise ConfigError.new(source_file),
"in #{source_file}: invalid source name '#{@name}': source names can only contain alphanumeric characters, and .-_"
elsif @name == "local"
raise ConfigError.new(source_file),
"in #{source_file}: the name 'local' is a reserved name"
end
@provides = (definition['provides'] || Set.new).to_set
@imports = (definition['imports'] || Array.new).map do |set_def|
pkg_set = Autoproj.in_file(source_file) do
PackageSet.from_spec(manifest, set_def, false)
end
pkg_set.imported_from = self
pkg_set
end
rescue InternalError
# This ignores raw_description_file error if the package set is not
# checked out yet
end
# Yields the imports this package set declares, as PackageSet instances
def each_imported_set(&block)
@imports.each(&block)
end
# Path to the source.yml file
def source_file
File.join(local_dir, 'source.yml')
end
# Load the source.yml file and resolves all information it contains.
#
# This for instance requires configuration options to be defined. Use
# PackageSet#load_minimal to load only self-contained information
def load_description_file
if @source_definition
return
end
@source_definition = raw_description_file
load_minimal
# Compute the definition of constants
Autoproj.in_file(source_file) do
constants = source_definition['constants'] || Hash.new
@constants_definitions = Autoproj.resolve_constant_definitions(constants)
end
end
def single_expansion(data, additional_expansions = Hash.new)
if !source_definition
load_description_file
end
Autoproj.single_expansion(data, additional_expansions.merge(constants_definitions))
end
# Expands the given string as much as possible using the expansions
# listed in the source.yml file, and returns it. Raises if not all
# variables can be expanded.
def expand(data, additional_expansions = Hash.new)
if !source_definition
load_description_file
end
Autoproj.expand(data, additional_expansions.merge(constants_definitions))
end
# Returns the default importer definition for this package set, as a
# VCSDefinition instance
def default_importer
importer_definition_for('default')
end
# Returns an importer definition for the given package, if one is
# available. Otherwise returns nil.
#
# The returned value is a VCSDefinition object.
def version_control_field(package_name, section_name, validate = true)
urls = source_definition['urls'] || Hash.new
urls['HOME'] = ENV['HOME']
all_vcs = source_definition[section_name]
if all_vcs
if all_vcs.kind_of?(Hash)
raise ConfigError.new, "wrong format for the #{section_name} section, you forgot the '-' in front of the package names"
elsif !all_vcs.kind_of?(Array)
raise ConfigError.new, "wrong format for the #{section_name} section"
end
end
vcs_spec = Hash.new
if all_vcs
all_vcs.each do |spec|
spec = spec.dup
if spec.values.size != 1
# Maybe the user wrote the spec like
# - package_name:
# type: git
# url: blah
#
# or as
# - package_name
# type: git
# url: blah
#
# In that case, we should have the package name as
# "name => nil". Check that.
name, _ = spec.find { |n, v| v.nil? }
if name
spec.delete(name)
else
name, _ = spec.find { |n, v| n =~ / \w+$/ }
name =~ / (\w+)$/
spec[$1] = spec.delete(name)
name = name.gsub(/ \w+$/, '')
end
else
name, spec = spec.to_a.first
if name =~ / (\w+)/
spec = { $1 => spec }
name = name.gsub(/ \w+$/, '')
end
if spec.respond_to?(:to_str)
if spec == "none"
spec = { :type => "none" }
else
raise ConfigError.new, "invalid VCS specification '#{name}: #{spec}'"
end
end
end
if Regexp.new("^" + name) =~ package_name
vcs_spec = vcs_spec.merge(spec)
end
end
end
if !vcs_spec.empty?
expansions = Hash["PACKAGE" => package_name,
"PACKAGE_BASENAME" => File.basename(package_name),
"AUTOPROJ_ROOT" => Autoproj.root_dir,
"AUTOPROJ_CONFIG" => Autoproj.config_dir,
"AUTOPROJ_SOURCE_DIR" => local_dir]
vcs_spec = expand(vcs_spec, expansions)
vcs_spec = Autoproj.vcs_definition_to_hash(vcs_spec)
vcs_spec.dup.each do |name, value|
vcs_spec[name] = expand(value, expansions)
end
# If required, verify that the configuration is a valid VCS
# configuration
if validate
Autoproj.normalize_vcs_definition(vcs_spec)
end
vcs_spec
end
end
# Returns the VCS definition for +package_name+ as defined in this
# source, or nil if the source does not have any.
#
# The definition is an instance of VCSDefinition
def importer_definition_for(package_name)
vcs_spec = version_control_field(package_name, 'version_control')
if vcs_spec
Autoproj.normalize_vcs_definition(vcs_spec)
end
end
# Enumerates the Autobuild::Package instances that are defined in this
# source
def each_package
if !block_given?
return enum_for(:each_package)
end
Autoproj.manifest.packages.each_value do |pkg|
if pkg.package_set.name == name
yield(pkg.autobuild)
end
end
end
# True if this package set provides the given package set name. I.e. if
# it has this name or the name is listed in the "replaces" field of
# source.yml
def provides?(name)
name == self.name ||
provides.include?(name)
end
end
# Specialization of the PackageSet class for the overrides listed in autoproj/
class LocalPackageSet < PackageSet
def initialize(manifest)
super(manifest, Autoproj.normalize_vcs_definition(:type => 'local', :url => Autoproj.config_dir))
end
def name
'local'
end
def load_minimal
end
def repository_id
'local'
end
def source_file
File.join(Autoproj.config_dir, "overrides.yml")
end
# Returns the default importer for this package set
def default_importer
importer_definition_for('default') ||
Autoproj.normalize_vcs_definition(:type => 'none')
end
def raw_description_file
path = source_file
if File.file?(path)
data = Autoproj.in_file(path, ArgumentError) do
YAML.load(File.read(path)) || Hash.new
end
data['name'] = 'local'
data
else
{ 'name' => 'local' }
end
end
end
# DEPRECATED. For backward-compatibility only.
Source = PackageSet
# DEPRECATED. For backward-compatibility only.
LocalSource = LocalPackageSet
PackageDefinition = Struct.new :autobuild, :user_block, :package_set, :file
# The Manifest class represents the information included in the main
# manifest file, and allows to manipulate it
class Manifest
# Data structure used to use autobuild importers without a package, to
# import configuration data.
#
# It has to match the interface of Autobuild::Package that is relevant
# for importers
class FakePackage
attr_reader :text_name
attr_reader :name
attr_reader :srcdir
attr_reader :importer
# Used by the autobuild importers
attr_accessor :updated
def initialize(text_name, srcdir, importer = nil)
@text_name = text_name
@name = text_name.gsub /[^\w]/, '_'
@srcdir = srcdir
@importer = importer
end
def import
importer.import(self)
end
def progress(msg)
Autobuild.progress(msg % [text_name])
end
# Display a progress message, and later on update it with a progress
# value. %s in the string is replaced by the package name
def progress_with_value(msg)
Autobuild.progress_with_value(msg % [text_name])
end
def progress_value(value)
Autobuild.progress_value(value)
end
end
# The set of packages that are selected by the user, either through the
# manifest file or through the command line, as a set of package names
attr_accessor :explicit_selection
# Returns true if +pkg_name+ has been explicitely selected
def explicitly_selected_package?(pkg_name)
explicit_selection && explicit_selection.include?(pkg_name)
end
# Loads the manifest file located at +file+ and returns the Manifest
# instance that represents it
def self.load(file)
manifest = Manifest.new
manifest.load(file)
manifest
end
# Load the manifest data contained in +file+
def load(file)
if !File.exists?(file)
raise ConfigError.new(File.dirname(file)), "expected an autoproj configuration in #{File.dirname(file)}, but #{file} does not exist"
end
data = Autoproj.in_file(file, ArgumentError) do
YAML.load(File.read(file))
end
@file = file
@data = data
if data['constants']
@constant_definitions = Autoproj.resolve_constant_definitions(data['constants'])
end
end
# The manifest data as a Hash
attr_reader :data
# The set of packages defined so far as a mapping from package name to
# [Autobuild::Package, package_set, file] tuple
attr_reader :packages
# A mapping from package names into PackageManifest objects
attr_reader :package_manifests
# The path to the manifest file that has been loaded
attr_reader :file
# True if osdeps should be handled in update and build, or left to the
# osdeps command
def auto_osdeps?
if data.has_key?('auto_osdeps')
!!data['auto_osdeps']
else true
end
end
# True if autoproj should run an update automatically when the user
# uses" build"
def auto_update?
!!data['auto_update']
end
attr_reader :constant_definitions
def initialize
@file = nil
@data = nil
@packages = Hash.new
@package_manifests = Hash.new
@automatic_exclusions = Hash.new
@constants_definitions = Hash.new
@disabled_imports = Set.new
@moved_packages = Hash.new
@constant_definitions = Hash.new
if Autoproj.has_config_key?('manifest_source')
@vcs = Autoproj.normalize_vcs_definition(Autoproj.user_config('manifest_source'))
end
end
# True if the given package should not be built, with the packages that
# depend on him have this dependency met.
#
# This is useful if the packages are already installed on this system.
def ignored?(package_name)
if data['ignore_packages']
data['ignore_packages'].any? { |l| Regexp.new(l) =~ package_name }
else
false
end
end
# The set of package names that are listed in the excluded_packages
# section of the manifest
def manifest_exclusions
data['exclude_packages'] || Set.new
end
# A package_name => reason map of the exclusions added with #add_exclusion.
# Exclusions listed in the manifest file are returned by #manifest_exclusions
attr_reader :automatic_exclusions
# Exclude +package_name+ from the build. +reason+ is a string describing
# why the package is to be excluded.
def add_exclusion(package_name, reason)
automatic_exclusions[package_name] = reason
end
# If +package_name+ is excluded from the build, returns a string that
# tells why. Otherwise, returns nil
#
# Packages can either be excluded because their name is listed in the
# excluded_packages section of the manifest, or because they are
# disabled on this particular operating system.
def exclusion_reason(package_name)
if manifest_exclusions.any? { |l| Regexp.new(l) =~ package_name }
"#{package_name} is listed in the excluded_packages section of the manifest"
else
automatic_exclusions[package_name]
end
end
# True if the given package should not be built and its dependencies
# should be considered as met.
#
# This is useful to avoid building packages that are of no use for the
# user.
def excluded?(package_name)
if manifest_exclusions.any? { |l| Regexp.new(l) =~ package_name }
true
elsif automatic_exclusions.any? { |pkg_name, | pkg_name == package_name }
true
else
false
end
end
# Lists the autobuild files that are in the package sets we know of
def each_autobuild_file(source_name = nil, &block)
if !block_given?
return enum_for(:each_source_file, source_name)
end
# This looks very inefficient, but it is because source names are
# contained in source.yml and we must therefore load that file to
# check the package set name ...
#
# And honestly I don't think someone will have 20 000 package sets
done_something = false
each_source(false) do |source|
next if source_name && source.name != source_name
done_something = true
Dir.glob(File.join(source.local_dir, "*.autobuild")).each do |file|
yield(source, file)
end
end
if source_name && !done_something
raise ConfigError.new(file), "in #{file}: source '#{source_name}' does not exist"
end
end
# Yields each osdeps definition files that are present in our sources
def each_osdeps_file
if !block_given?
return enum_for(:each_source_file)
end
each_source(false) do |source|
Dir.glob(File.join(source.local_dir, "*.osdeps")).each do |file|
yield(source, file)
end
end
end
# True if some of the sources are remote sources
def has_remote_sources?
each_remote_source(false).any? { true }
end
# True if calling update_remote_sources will actually do anything
def should_update_remote_sources
if Autobuild.do_update
return true
end
each_remote_source(false) do |source|
if !File.directory?(source.local_dir)
return true
end
end
false
end
# Like #each_source, but filters out local package sets
def each_remote_package_set(load_description = true)
if !block_given?
enum_for(:each_remote_package_set, load_description)
else
each_package_set(load_description) do |source|
if !source.local?
yield(source)
end
end
end
end
def each_remote_source(*args, &block)
each_remote_package_set(*args, &block)
end
# Helper method for #each_package_set
def enumerate_package_set(pkg_set, explicit_sets, all_sets) # :nodoc:
if @disabled_imports.include?(pkg_set.name)
pkg_set.auto_imports = false
end
result = []
if pkg_set.auto_imports?
pkg_set.each_imported_set do |imported_set|
if explicit_sets.any? { |src| src.vcs == imported_set.vcs } ||
all_sets.any? { |src| src.vcs == imported_set.vcs }
next
end
all_sets << imported_set
result.concat(enumerate_package_set(imported_set, explicit_sets, all_sets))
end
end
result << pkg_set
result
end
# call-seq:
# each_package_set { |pkg_set| ... }
#
# Lists all package sets defined in this manifest, by yielding a
# PackageSet object that describes it.
def each_package_set(load_description = true, &block)
if !block_given?
return enum_for(:each_package_set, load_description)
end
if @package_sets
if load_description
@package_sets.each do |src|
if !src.source_definition
src.load_description_file
end
end
end
return @package_sets.each(&block)
end
explicit_sets = (data['package_sets'] || []).map do |spec|
Autoproj.in_file(self.file) do
PackageSet.from_spec(self, spec, load_description)
end
end
all_sets = Array.new
explicit_sets.each do |pkg_set|
all_sets.concat(enumerate_package_set(pkg_set, explicit_sets, all_sets + [pkg_set]))
end
# Now load the local source
local = LocalPackageSet.new(self)
if load_description
local.load_description_file
else
local.load_minimal
end
if !load_description || !local.empty?
all_sets << local
end
if load_description
all_sets.each(&:load_description_file)
else
all_sets.each(&:load_minimal)
end
all_sets.each(&block)
end
# DEPRECATED. For backward-compatibility only
#
# use #each_package_set instead
def each_source(*args, &block)
each_package_set(*args, &block)
end
def local_package_set
each_package_set.find { |s| s.kind_of?(LocalPackageSet) }
end
# Save the currently known package sets. After this call,
# #each_package_set will always return the same set regardless of
# changes on the manifest's data structures
def cache_package_sets
@package_sets = each_package_set(false).to_a
end
# Register a new package
def register_package(package, block, source, file)
@packages[package.name] = PackageDefinition.new(package, block, source, file)
end
def definition_source(package_name)
@packages[package_name].package_set
end
def definition_file(package_name)
@packages[package_name].file
end
def package(name)
packages[name]
end
# Lists all defined packages and where they have been defined
def each_package
if !block_given?
return enum_for(:each_package)
end
packages.each_value { |pkg| yield(pkg.autobuild) }
end
# The VCS object for the main configuration itself
attr_reader :vcs
def each_configuration_source
if !block_given?
return enum_for(:each_configuration_source)
end
if vcs
yield(vcs, "autoproj main configuration", Autoproj.config_dir)
end
each_remote_source(false) do |source|
yield(source.vcs, source.name || source.vcs.url, source.local_dir)
end
self
end
# Creates an autobuild package whose job is to allow the import of a
# specific repository into a given directory.
#
# +vcs+ is the VCSDefinition file describing the repository, +text_name+
# the name used when displaying the import progress, +pkg_name+ the
# internal name used to represent the package and +into+ the directory
# in which the package should be checked out.
def self.create_autobuild_package(vcs, text_name, into)
importer = vcs.create_autobuild_importer
return if !importer # updates have been disabled by using the 'none' type
FakePackage.new(text_name, into, importer)
rescue Autobuild::ConfigException => e
raise ConfigError.new, "cannot import #{name}: #{e.message}", e.backtrace
end
# Imports or updates a source (remote or otherwise).
#
# See create_autobuild_package for informations about the arguments.
def self.update_package_set(vcs, text_name, into)
fake_package = create_autobuild_package(vcs, text_name, into)
fake_package.import
rescue Autobuild::ConfigException => e
raise ConfigError.new, "cannot import #{name}: #{e.message}", e.backtrace
end
# Updates the main autoproj configuration
def update_yourself
Manifest.update_package_set(vcs, "autoproj main configuration", Autoproj.config_dir)
end
def update_remote_set(pkg_set, remotes_symlinks_dir = nil)
Manifest.update_package_set(
pkg_set.vcs,
pkg_set.name,
pkg_set.raw_local_dir)
if remotes_symlinks_dir
pkg_set.load_minimal
symlink_dest = File.join(remotes_symlinks_dir, pkg_set.name)
# Check if the current symlink is valid, and recreate it if it
# is not
if File.symlink?(symlink_dest)
dest = File.readlink(symlink_dest)
if dest != pkg_set.raw_local_dir
FileUtils.rm_f symlink_dest
FileUtils.ln_sf pkg_set.raw_local_dir, symlink_dest
end
else
FileUtils.rm_f symlink_dest
FileUtils.ln_sf pkg_set.raw_local_dir, symlink_dest
end
symlink_dest
end
end
# Updates all the remote sources in ROOT_DIR/.remotes, as well as the
# symbolic links in ROOT_DIR/autoproj/remotes
def update_remote_package_sets
remotes_symlinks_dir = File.join(Autoproj.config_dir, 'remotes')
FileUtils.mkdir_p remotes_symlinks_dir
# Iterate on the remote sources, without loading the source.yml
# file (we're not ready for that yet)
#
# Do it iteratively to properly take imports into account, but we
# first unconditionally update all the existing sets to properly
# handle imports that have been removed
updated_sets = Hash.new
known_remotes = []
each_remote_package_set(false) do |pkg_set|
next if !pkg_set.explicit?
if pkg_set.present?
known_remotes << update_remote_set(pkg_set, remotes_symlinks_dir)
updated_sets[pkg_set.repository_id] = pkg_set
end
end
old_updated_sets = nil
while old_updated_sets != updated_sets
old_updated_sets = updated_sets.dup
each_remote_package_set(false) do |pkg_set|
next if updated_sets.has_key?(pkg_set.repository_id)
if !pkg_set.explicit?
Autoproj.progress " #{pkg_set.imported_from.name}: auto-importing #{pkg_set.name}"
end
known_remotes << update_remote_set(pkg_set, remotes_symlinks_dir)
updated_sets[pkg_set.repository_id] = pkg_set
end
end
# Check for directories in ROOT_DIR/.remotes that do not map to a
# source repository, and remove them
Dir.glob(File.join(Autoproj.remotes_dir, '*')).each do |dir|
dir = File.expand_path(dir)
if File.directory?(dir) && !updated_sets.values.find { |pkg| pkg.raw_local_dir == dir }
FileUtils.rm_rf dir
end
end
# Now remove obsolete symlinks
Dir.glob(File.join(remotes_symlinks_dir, '*')).each do |file|
if File.symlink?(file) && !known_remotes.include?(file)
FileUtils.rm_f file
end
end
end
# DEPRECATED. For backward-compatibility only
def update_remote_sources(*args, &block)
update_remote_package_sets(*args, &block)
end
def importer_definition_for(package_name, package_source = nil)
if !package_source
package_source = packages.values.
find { |pkg| pkg.autobuild.name == package_name }.
package_set
end
sources = each_source.to_a.dup
# Remove sources listed before the package source
while !sources.empty? && sources[0].name != package_source.name
sources.shift
end
package_source = sources.shift
if !package_source
raise InternalError, "cannot find the package set that defines #{package_name}"
end
# Get the version control information from the package source. There
# must be one
vcs_spec = package_source.version_control_field(package_name, 'version_control')
return if !vcs_spec
sources.each do |src|
overrides_spec = src.version_control_field(package_name, 'overrides', false)
if overrides_spec
vcs_spec.merge!(overrides_spec)
end
end
Autoproj.normalize_vcs_definition(vcs_spec)
end
# Sets up the package importers based on the information listed in
# the source's source.yml
#
# The priority logic is that we take the package sets one by one in the
# order listed in the autoproj main manifest, and first come first used.
#
# A set that defines a particular package in its autobuild file
# *must* provide the corresponding VCS line in its source.yml file.
# However, it is possible for a source that does *not* define a package
# to override the VCS
#
# In other words: if package P is defined by source S1, and source S0
# imports S1, then
# * S1 must have a VCS line for P
# * S0 can have a VCS line for P, which would override the one defined
# by S1
def load_importers
packages.each_value do |pkg|
vcs = importer_definition_for(pkg.autobuild.name, pkg.package_set) ||
pkg.package_set.default_importer
if vcs
Autoproj.add_build_system_dependency vcs.type
pkg.autobuild.importer = vcs.create_autobuild_importer
else
raise ConfigError.new, "source #{pkg.package_set.name} defines #{pkg.autobuild.name}, but does not provide a version control definition for it"
end
end
end
# Returns true if +name+ is the name of a package set known to this
# autoproj installation
def has_package_set?(name)
each_package_set(false).find { |set| set.name == name }
end
# +name+ can either be the name of a source or the name of a package. In
# the first case, we return all packages defined by that source. In the
# latter case, we return the singleton array [name]
def resolve_package_set(name)
if Autobuild::Package[name]
[name]
else
pkg_set = each_package_set(false).find { |set| set.name == name }
if !pkg_set
raise ConfigError.new, "#{name} is neither a package nor a source"
end
packages.values.
find_all { |pkg| pkg.package_set.name == pkg_set.name }.
map { |pkg| pkg.autobuild.name }.
find_all { |pkg_name| !Autoproj.osdeps || !Autoproj.osdeps.has?(pkg_name) }
end
end
# Returns the packages contained in the provided layout definition
#
# If recursive is false, yields only the packages at this level.
# Otherwise, return all packages.
def layout_packages(result, validate)
normalized_layout.each_key do |pkg_or_set|
begin
resolve_package_set(pkg_or_set).each do |pkg_name|
result << pkg_name
Autobuild::Package[pkg_name].all_dependencies(result)
end
rescue ConfigError
raise if validate
end
end
result
end
# Enumerates the sublayouts defined in +layout_def+.
def each_sublayout(layout_def)
layout_def.each do |value|
if value.kind_of?(Hash)
name, layout = value.find { true }
yield(name, layout)
end
end
end
# Returns the set of package names that are explicitely listed in the
# layout, minus the excluded and ignored ones
def all_layout_packages(validate = true)
default_packages(validate)
end
# Returns all defined package names, minus the excluded and ignored ones
def all_package_names
Autobuild::Package.each.map { |name, _| name }.to_set
end
# Returns all the packages that can be built in this installation
def all_packages
packages.values.
map { |pkg| pkg.autobuild.name }.
find_all { |pkg_name| !Autoproj.osdeps || !Autoproj.osdeps.has?(pkg_name) }
end
# Returns true if +name+ is a valid package and is included in the build
#
# If +validate+ is true, the method will raise ArgumentError if the
# package does not exists.
#
# If it is false, the method will simply return false on non-defined
# packages
def package_enabled?(name, validate = true)
if !Autobuild::Package[name]
if validate
raise ArgumentError, "package #{name} does not exist"
end
return false
end
!excluded?(name)
end
# Returns true if +name+ is a valid package and is neither excluded from
# the build, nor ignored from the build
#
# If +validate+ is true, the method will raise ArgumentError if the
# package does not exists.
#
# If it is false, the method will simply return false on non-defined
# packages
def package_selected?(name, validate = true)
if package_enabled?(name)
!ignored?(name)
end
end
# Returns the set of packages that are selected by the layout
def all_selected_packages
result = default_packages.to_set
result.each do |pkg_name|
Autobuild::Package[pkg_name].all_dependencies(result)
end
result
end
# Returns the set of packages that should be built if the user does not
# specify any on the command line
def default_packages(validate = true)
names = if layout = data['layout']
layout_packages(Set.new, validate)
else
# No layout, all packages are selected
all_packages
end
names.delete_if { |pkg_name| excluded?(pkg_name) || ignored?(pkg_name) }
names.to_set
end
def normalized_layout(result = Hash.new, layout_level = '/', layout_data = (data['layout'] || Hash.new))
layout_data.each do |value|
if value.kind_of?(Hash)
subname, subdef = value.find { true }
normalized_layout(result, "#{layout_level}#{subname}/", subdef)
else
result[value] = layout_level
end
end
result
end
# Returns the package directory for the given package name
def whereis(package_name)
Autoproj.in_file(self.file) do
set_name = definition_source(package_name).name
actual_layout = normalized_layout
return actual_layout[package_name] || actual_layout[set_name] || '/'
end
end
def resolve_optional_dependencies
packages.each_value do |pkg|
pkg.autobuild.resolve_optional_dependencies
end
end
# Loads the package's manifest.xml file for the current package
#
# Right now, the absence of a manifest makes autoproj only issue a
# warning. This will later be changed into an error.
def load_package_manifest(pkg_name)
pkg = packages.values.
find { |pkg| pkg.autobuild.name == pkg_name }
package, source, file = pkg.autobuild, pkg.package_set, pkg.file
if !pkg_name
raise ArgumentError, "package #{pkg_name} is not defined"
end
manifest_path = File.join(package.srcdir, "manifest.xml")
if !File.file?(manifest_path)
Autoproj.warn "#{package.name} from #{source.name} does not have a manifest"
return
end
manifest = PackageManifest.load(package, manifest_path)
pkg.autobuild.description = manifest
package_manifests[package.name] = manifest
manifest.each_dependency do |name, is_optional|
begin
if is_optional
package.optional_dependency name
else
package.depends_on name
end
rescue Autobuild::ConfigException => e
raise ConfigError.new(manifest_path),
"manifest #{manifest_path} of #{package.name} from #{source.name} lists '#{name}' as dependency, which is listed in the layout of #{file} but has no autobuild definition", e.backtrace
rescue ConfigError => e
raise ConfigError.new(manifest_path),
"manifest #{manifest_path} of #{package.name} from #{source.name} lists '#{name}' as dependency, but it is neither a normal package nor an osdeps package. osdeps reports: #{e.message}", e.backtrace
end
end
end
# Loads the manifests for all packages known to this project.
#
# See #load_package_manifest
def load_package_manifests(selected_packages)
selected_packages.each(&:load_package_manifest)
end
# Disable all automatic imports from the given package set name
def disable_imports_from(pkg_set_name)
@disabled_imports << pkg_set_name
end
# call-seq:
# list_os_dependencies(packages) => required_packages, ospkg_to_pkg
#
# Returns the set of dependencies required by the listed packages.
#
# +required_packages+ is the set of osdeps names that are required for
# +packages+ and +ospkg_to_pkg+ a mapping from the osdeps name to the
# set of packages that require this OS package.
def list_os_dependencies(packages)
required_os_packages = Set.new
package_os_deps = Hash.new { |h, k| h[k] = Array.new }
packages.each do |pkg_name|
pkg = Autobuild::Package[pkg_name]
if !pkg
raise InternalError, "internal error: #{pkg_name} is not a package"
end
pkg.os_packages.each do |osdep_name|
package_os_deps[osdep_name] << pkg_name
required_os_packages << osdep_name
end
end
return required_os_packages, package_os_deps
end
# Installs the OS dependencies that are required by the given packages
def install_os_dependencies(packages)
required_os_packages, package_os_deps = list_os_dependencies(packages)
Autoproj.osdeps.install(required_os_packages, package_os_deps)
end
# Package selection can be done in three ways:
# * as a subdirectory in the layout
# * as a on-disk directory
# * as a package name
#
# This method converts the first two directories into the third one
def expand_package_selection(selection)
base_dir = Autoproj.root_dir
# The expanded selection
expanded_packages = Set.new
# All the packages that are available on this installation
all_layout_packages = self.all_selected_packages
# First, remove packages that are directly referenced by name or by
# package set names
selection.each do |sel|
sel = Regexp.new(Regexp.quote(sel))
packages = all_layout_packages.
find_all { |pkg_name| pkg_name =~ sel }.
to_set
expanded_packages |= packages
sources = each_source.find_all { |source| source.name =~ sel }
sources.each do |source|
packages = resolve_package_set(source.name).to_set
expanded_packages |= (packages & all_layout_packages)
end
!packages.empty? || !sources.empty?
end
# Finally, check for package source directories
all_packages = self.all_package_names
selection.each do |sel|
match_pkg_name = Regexp.new(Regexp.quote(sel))
all_packages.each do |pkg_name|
pkg = Autobuild::Package[pkg_name]
if pkg_name =~ match_pkg_name || sel =~ Regexp.new("^#{Regexp.quote(pkg.srcdir)}") || pkg.srcdir =~ Regexp.new("^#{Regexp.quote(sel)}")
# Check-out packages that are not in the manifest only
# if they are explicitely selected
if pkg_name != sel && pkg.srcdir != sel && !all_layout_packages.include?(pkg.name)
next
end
expanded_packages << pkg_name
end
end
end
# Remove packages that are explicitely excluded and/or ignored
expanded_packages.delete_if { |pkg_name| excluded?(pkg_name) || ignored?(pkg_name) }
expanded_packages.to_set
end
attr_reader :moved_packages
# Moves the given package name from its current subdirectory to the
# provided one.
#
# For instance, for a package called drivers/xsens_imu
#
# move("drivers/xsens_imu", "data_acquisition")
#
# will move the package into data_acquisition/xsens_imu
def move_package(package_name, new_dir)
moved_packages[package_name] = File.join(new_dir, File.basename(package_name))
end
end
class << self
# The singleton manifest object on which the current run works
attr_accessor :manifest
# The operating system package definitions
attr_accessor :osdeps
end
def self.load_osdeps_from_package_sets
manifest.each_osdeps_file do |source, file|
osdeps.merge(source.load_osdeps(file))
end
osdeps
end
class PackageManifest
def self.load(package, file)
doc = Nokogiri::XML(File.read(file)) do |c|
c.noblanks
end
PackageManifest.new(package, doc)
end
# The Autobuild::Package instance this manifest applies on
attr_reader :package
# The raw XML data as a Nokogiri document
attr_reader :xml
# The list of tags defined for this package
#
# Tags are defined as multiple blocks, each of which can
# contain multiple comma-separated tags
def tags
result = []
xml.xpath('//tags').each do |node|
result.concat(node.content.strip.split(','))
end
result
end
def documentation
xml.xpath('//description').each do |node|
doc = node.content.strip
if doc.empty?
if doc = short_documentation
return doc
end
return "no documentation available for #{package.name} in its manifest.xml file"
else
return doc
end
end
nil
end
def short_documentation
xml.xpath('//description').each do |node|
doc = node['brief']
if doc
doc = doc.to_s.strip
end
if doc && !doc.empty?
return doc.to_s
end
end
nil
end
def initialize(package, doc)
@package = package
@xml = doc
end
def each_dependency(&block)
if block_given?
each_os_dependency(&block)
each_package_dependency(&block)
else
enum_for(:each_dependency, &block)
end
end
def each_os_dependency
if block_given?
xml.xpath('//rosdep').each do |node|
yield(node['name'], false)
end
package.os_packages.each do |name|
yield(name, false)
end
else
enum_for :each_os_dependency
end
end
def each_package_dependency
if block_given?
xml.xpath('//depend').each do |node|
dependency = node['package']
optional =
if node['optional'].to_s == '1'
true
else false
end
if dependency
yield(dependency, optional)
else
raise ConfigError.new, "manifest of #{package.name} has a tag without a 'package' attribute"
end
end
else
enum_for :each_package_dependency
end
end
end
end