module Sprinkle module Installers # = Source Package Installer # # The source package installer installs software from source. # It handles downloading, extracting, configuring, building, # and installing software. # # == Configuration Options # # The source installer has many configuration options: # * prefix - The prefix directory that is configured to. # * archives - The location all the files are downloaded to. # * builds - The directory the package is extracted to to configure and install # # == Pre/Post Hooks # # The source installer defines a myriad of new stages which can be hooked into: # * prepare - Prepare is the stage which all the prefix, archives, and build directories are made. # * download - Download is the stage which the software package is downloaded. # * extract - Extract is the stage which the software package is extracted. # * configure - Configure is the stage which the ./configure script is run. # * build - Build is the stage in which `make` is called. # * install - Install is the stage which `make install` is called. # # == Example Usage # # First, a simple package, no configuration: # # package :magic_beans do # source 'http://magicbeansland.com/latest-1.1.1.tar.gz' # end # # Second, specifying exactly where I want my files: # # package :magic_beans do # source 'http://magicbeansland.com/latest-1.1.1.tar.gz' do # prefix '/usr/local' # archives '/tmp' # builds '/tmp/builds' # end # end # # Third, specifying some hooks: # # package :magic_beans do # source 'http://magicbeansland.com/latest-1.1.1.tar.gz' do # prefix '/usr/local' # # pre :prepare { 'echo "Here we go folks."' } # post :extract { 'echo "I believe..."' } # pre :build { 'echo "Cross your fingers!"' } # end # end # # As you can see, setting options is as simple as creating a # block and calling the option as a method with the value as # its parameter. class Source < Installer attr_accessor :source #:nodoc: def initialize(parent, source, options = {}, &block) #:nodoc: @source = source super parent, options, &block end protected def install_sequence #:nodoc: prepare + download + extract + configure + build + install end %w( prepare download extract configure build install ).each do |stage| define_method stage do pre_commands(stage.to_sym) + self.send("#{stage}_commands") + post_commands(stage.to_sym) end end def prepare_commands #:nodoc: raise 'No installation area defined' unless @options[:prefix] raise 'No build area defined' unless @options[:builds] raise 'No source download area defined' unless @options[:archives] [ "mkdir -p #{@options[:prefix]}", "mkdir -p #{@options[:builds]}", "mkdir -p #{@options[:archives]}" ] end def download_commands #:nodoc: [ "wget -cq --directory-prefix='#{@options[:archives]}' #{@source}" ] end def extract_commands #:nodoc: [ "bash -c 'cd #{@options[:builds]} && #{extract_command} #{@options[:archives]}/#{archive_name}'" ] end def configure_commands #:nodoc: return [] if custom_install? command = "bash -c 'cd #{build_dir} && ./configure --prefix=#{@options[:prefix]} " extras = { :enable => '--enable', :disable => '--disable', :with => '--with', :without => '--without' } extras.inject(command) { |m, (k, v)| m << create_options(k, v) if options[k]; m } [ command << " > #{@package.name}-configure.log 2>&1'" ] end def build_commands #:nodoc: return [] if custom_install? [ "bash -c 'cd #{build_dir} && make > #{@package.name}-build.log 2>&1'" ] end def install_commands #:nodoc: return custom_install_commands if custom_install? [ "bash -c 'cd #{build_dir} && make install > #{@package.name}-install.log 2>&1'" ] end def custom_install? #:nodoc: !! @options[:custom_install] end # REVISIT: must be better processing of custom install commands somehow? use splat operator? def custom_install_commands #:nodoc: dress @options[:custom_install], :install end protected # dress is overriden from the base Sprinkle::Installers::Installer class so that the command changes # directory to the build directory first. Also, the result of the command is logged. def dress(commands, stage) commands.collect { |command| "bash -c 'cd #{build_dir} && #{command} >> #{@package.name}-#{stage}.log 2>&1'" } end private def create_options(key, prefix) #:nodoc: @options[key].inject(' ') { |m, option| m << "#{prefix}-#{option} "; m } end def extract_command #:nodoc: case @source when /(tar.gz)|(tgz)$/ 'tar xzf' when /(tar.bz2)|(tb2)$/ 'tar xjf' when /tar$/ 'tar xf' when /zip$/ 'unzip' else raise "Unknown source archive format: #{archive_name}" end end def archive_name #:nodoc: name = @source.split('/').last raise "Unable to determine archive name for source: #{source}, please update code knowledge" unless name name end def build_dir #:nodoc: "#{@options[:builds]}/#{options[:custom_dir] || base_dir}" end def base_dir #:nodoc: if @source.split('/').last =~ /(.*)\.(tar\.gz|tgz|tar\.bz2|tb2)/ return $1 end raise "Unknown base path for source archive: #{@source}, please update code knowledge" end end end end