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
#
# Fourth, specifying a custom archive name because the downloaded file name
# differs from the source URL:
#
# package :gitosis do
# source 'http://github.com/crafterm/sprinkle/tarball/master' do
# custom_archive 'crafterm-sprinkle-518e33c835986c03ec7ae8ea88c657443b006f28.tar.gz'
# end
# end
#
# Fifth, specifying a custom directory where the archive actually is extracted to:
#
# package :ruby_build do
# source 'https://github.com/sstephenson/ruby-build/archive/v20130227.tar.gz' do
# custom_dir 'ruby-build-20130227'
# custom_install './install.sh'
# 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:
api do
def source(source, options = {}, &block)
@recommends << :build_essential # Ubuntu/Debian
install Source.new(self, source, options, &block)
end
end
def initialize(parent, source, options = {}, &block) #:nodoc:
super parent, options, &block
@source = source
end
def install_sequence #:nodoc:
prepare + download + extract + configure + build + install
end
protected
%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].first}",
"mkdir -p #{@options[:builds].first}",
"mkdir -p #{@options[:archives].first}" ]
end
def download_commands #:nodoc:
[ "wget -cq -O '#{@options[:archives].first}/#{archive_name}' #{@source}" ]
end
def extract_commands #:nodoc:
[ "bash -c 'cd #{@options[:builds].first} && #{extract_command} #{@options[:archives].first}/#{archive_name}'" ]
end
def configure_commands #:nodoc:
return [] if custom_install?
command = "bash -c 'cd #{build_dir} && ./configure --prefix=#{@options[:prefix].first} "
extras = {
:enable => '--enable', :disable => '--disable',
:with => '--with', :without => '--without',
:option => '-',
}
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].first.inject('') { |m, option| m << "#{prefix}-#{option} "; m }
end
def extract_command #:nodoc:
case archive_name
when /(tar.gz)|(tgz)$/
'tar xzf'
when /(tar.bz2)|(tb2)$/
'tar xjf'
when /tar$/
'tar xf'
when /zip$/
'unzip -o'
else
raise "Unknown source archive format: #{archive_name}"
end
end
def archive_name #:nodoc:
name = @source.split('/').last
if options[:custom_archive]
name = options[:custom_archive]
name = name.join if name.is_a? Array
end
raise "Unable to determine archive name for source: #{source}, please update code knowledge" unless name
name
end
def build_dir #:nodoc:
"#{@options[:builds].first}/#{options[:custom_dir] || base_dir}"
end
def base_dir #:nodoc:
if archive_name.split('/').last =~ /(.*)\.(tar\.gz|tgz|tar\.bz2|tar|tb2|zip)/
return $1
end
raise "Unknown base path for source archive: #{@source}, please update code knowledge"
end
end
end
end