# Evaluates +script+ in autobuild context def Autobuild(&script) # rubocop:disable Naming/MethodName Autobuild.send(:module_eval, &script) end # Main Autobuild module. This module includes the build configuration options # (see Autobuild::DEFAULT_OPTIONS) for the default values) # nice:: the nice value at which we should spawn subprocesses # srcdir:: the base source directory. If a package defines a relative srcdir, then # it is defined relatively to Autobuild.srcdir. Defaults to the current directory. # prefix:: the base install directory. If a package defines a relative prefix, then # it is defined relatively to Autobuild.prefix. # verbose:: if true, displays all subprocesses output # debug:: more verbose than 'verbose': displays Rake's debugging output # do_update:: if we should update the packages # do_build:: if we should build the packages # do_forced_build:: if we should forcefully trigger all the packages build phases # do_rebuild:: if we should cleanly rebuild every packages # do_doc:: if we should produce the documentation # doc_errors:: if errors during the documentation generation are treated as errors # daemonize:: if the build should go into daemon mode (only if the daemons gem # is available) # clean_log:: remove all logs before starting the build # packages:: a list of packages to build specifically # default_packages:: the list of packages to build if Autobuild.packages is empty. # It this array is empty too, build all defined packages. # keep_oldlogs:: if true, new runs will be appended to existing logfiles. # Otherwise, the existing logfile contents is erased. module Autobuild class << self %w[ nice srcdir prefix verbose debug do_update do_build do_rebuild do_forced_build daemonize clean_log packages default_packages keep_oldlogs ].each do |name| attr_accessor name end # @return [{String=>Class}] the known utilities # @see {register_utility_class} attr_reader :utilities # Yields all known utility classes # # @yieldparam [String] utility name # @yieldparam [Class] utility class # @yieldparam [Hash] utility options # @return [void] def each_utility return enum_for(__method__) unless block_given? utilities.each { |name, (utl, options)| yield(name, utl, options) } end def register_utility_class(name, klass, disabled_by_default: false, **options) creation_options = { disabled_by_default: disabled_by_default } utilities[name] = [klass, creation_options, options] singleton_class.class_eval do attr_accessor "only_#{name}" attr_accessor "do_#{name}" attr_accessor "#{name}_prefix" attr_accessor "pass_#{name}_errors" end instance_variable_set "@only_#{name}", false instance_variable_set "@do_#{name}", false instance_variable_set "@pass_#{name}_errors", false instance_variable_set "@#{name}_prefix", name end def create_utility(utility_name, package) klass, creation_options, options = utilities[utility_name] if klass utility = klass.new(utility_name, package, **options) package.utilities[utility_name] = utility utility.enabled = !creation_options[:disabled_by_default] utility else raise ArgumentError, "there is no utility called #{utility_name}, "\ "available utilities are #{utilities.keys.sort.join(', ')}" end end # The directory in which logs are saved. Defaults to PREFIX/log. attr_writer :logdir # True if we build and if the build is applied on all packages def full_build? do_build && !only_doc && packages.empty? end end @utilities = Hash.new register_utility_class 'doc', Utility, disabled_by_default: false register_utility_class 'test', TestUtility, disabled_by_default: true DEFAULT_OPTIONS = { :nice => nil, :srcdir => Dir.pwd, :prefix => Dir.pwd, :logdir => nil, :verbose => false, :debug => false, :do_build => true, :do_forced_build => false, :do_rebuild => false, :do_update => true, :daemonize => false, :packages => [], :default_packages => [], :keep_oldlogs => false }.freeze DEFAULT_OPTIONS.each do |name, value| send("#{name}=", value) end @post_install_handlers = Array.new def self.post_install(*args, &block) if args.empty? @post_install_handlers << block elsif !block @post_install_handlers << args else raise ArgumentError, "cannot set both arguments and block" end end class << self attr_reader :post_install_handlers end def self.apply_post_install(pkg, info) return unless info case info when Array args = info.dup tool = Autobuild.tool(args.shift) pkg.run 'post-install', tool, *args when Proc if info.arity == 1 info.call(pkg) else info.call end end end @mail = Hash.new class << self # Mailing configuration. It is a hash with the following keys (as symbols) # [:to] the mail destination. Defaults to USER@HOSTNAME, where USER is # the username of autobuild's caller, and HOSTNAME the hostname of the # current machine. # [:from] the mail origin. Defaults to the same value than +:to+ # [:smtp] the hostname of the SMTP server, defaults to localhost # [:port] the port of the SMTP server, defauts to 22 # [:only_errors] mail only on errors. Defaults to false. attr_reader :mail # call-seq: # post_success_message => string # post_success_message "msg" => "msg" # post_success_message { } => block # # Gets or updates a message to be displayed on success. Can either be # a string or a block, in which case the block must return the message # string. def post_success_message(*args, &block) if args.empty? && !block if @post_success_message.respond_to?(:to_str) @post_success_message.to_str elsif @post_success_message @post_success_message.call end elsif block @post_success_message = block else @post_success_message = args.first.to_str end end # The directory in which logs are saved def logdir @logdir || "#{prefix}/log" end # Removes all log files def clean_log! Reporting.each_log do |file| FileUtils.rm_f file end end # Gets autobuild options from the command line and returns the # remaining elements def commandline(args) parser = OptionParser.new do |opts| opts.separator "Path specification" opts.on("--srcdir PATH", "sources are installed in PATH") do |v| Autobuild.srcdir = v end opts.on("--prefix PATH", "built packages are installed in PATH") do |v| Autobuild.prefix = v end opts.on("--logdir PATH", "logs are saved in PATH "\ "(default: /autobuild)") do |v| Autobuild.logdir = v end opts.separator "" opts.separator "General behaviour" opts.on('--nice NICE', Integer, 'nice the subprocesses to the given value') do |v| Autobuild.nice = v end opts.on("-h", "--help", "Show this message") do puts opts exit end if defined? Daemons opts.on("--[no-]daemon", "go into daemon mode") do |v| Autobuild.daemonize = v end end opts.on("--no-update", "update already checked-out sources") do |v| Autobuild.do_update = v end opts.on("--no-build", "only prepare packages, do not build them") do |v| Autobuild.do_build = v end opts.on("--forced-build", "force the trigger of "\ "all the build commands") do |v| Autobuild.do_forced_build = v end opts.on("--rebuild", "clean and rebuild") do |v| Autobuild.do_forced_build = v end opts.on("--only-doc", "only generate documentation") do |v| Autobuild.only_doc = v end opts.on("--no-doc", "don't generate documentation") do |v| Autobuild.do_doc = v end opts.on("--doc-errors", "treat documentation failure as error") do |v| Autobuild.pass_doc_errors = v end opts.separator "" opts.separator "Program output" opts.on("--[no-]verbose", "display output of commands on stdout") do |v| Autobuild.verbose = v end opts.on("--[no-]debug", "debug information") do |v| Autobuild.debug = v end opts.on("--keep-oldlogs", "old logs will be kept, "\ "new program output being appended") do |v| Autobuild.keep_oldlogs = v end opts.on('--version', "displays autobuild version and then exits") do puts "autobuild v#{Autobuild::VERSION}" exit 0 end opts.separator "" opts.separator "Mail reports" opts.on("--mail-from EMAIL", String, "From: field of the sent mails") do |from_email| mail[:from] = from_email end opts.on("--mail-to EMAILS", String, "comma-separated list of emails "\ "to which the reports should be sent") do |emails| mail[:to] ||= [] mail[:to] += emails.split(',') end opts.on("--mail-subject SUBJECT", String, "Subject: field of the sent mails") do |subject_email| mail[:subject] = subject_email end opts.on("--mail-smtp HOSTNAME", String, "address of the mail server "\ "written as hostname[:port]") do |smtp| unless smtp =~ /^([^:]+)(?::(\d+))?$/ raise "invalid SMTP specification #{smtp}" end mail[:smtp] = $1 mail[:port] = Integer($2) if $2 && !$2.empty? end opts.on("--mail-only-errors", "send mail only on errors") do mail[:only_errors] = true end end parser.parse!(args) unless args[0] puts parser exit end Rake.application.options.trace = debug args[0..-1] end end # The name of all build phases that Autobuild knows about # # @return [Array] def self.all_phases %w[import prepare build] + utilities.keys end def self.apply(packages, buildname = "autobuild", phases = [], options = Hash.new) options = Kernel.validate_options options, parallel: Autobuild.parallel_build_level if Autobuild.mail[:to] if !Autobuild::HAS_RMAIL Autobuild.warn "RMail is not available. Mail notification is disabled" else Reporting << MailReporter.new(Autobuild.mail) end end if Autobuild.do_rebuild packages.each do |pkg_name| Autobuild::Package[pkg_name].prepare_for_rebuild end # And delete the prefix ! FileUtils.rm_rf Autobuild.prefix elsif Autobuild.do_forced_build packages.each do |pkg_name| Autobuild::Package[pkg_name].prepare_for_forced_build end end if phases.empty? if Autobuild.only_doc phases = ['doc'] else phases = ['import'] phases += %w[prepare build] if Autobuild.do_build phases << 'doc' if Autobuild.do_doc end end phases.each do |phase| # We create a dummy task listing what needs to be done, and then we # call it targets = if packages.empty? phase else packages. find_all { |pkg| Rake.application.lookup("#{pkg}-#{phase}") }. map { |pkg| "#{pkg}-#{phase}" } end task "#{buildname}-#{phase}" => targets end begin invoker = Autobuild::RakeTaskParallelism.new(options[:parallel]) Autobuild.parallel_task_manager = invoker phases.each do |phase| invoker.invoke_parallel([Rake::Task["#{buildname}-#{phase}"]]) end ensure Autobuild.parallel_task_manager = nil end end end