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
#
# Sixth, specifying custom configure, build, and install commands:
#
# package :mysql_build do
# source 'http://dev.mysql.com/get/Downloads/MySQL-5.5/mysql-5.5.25a.tar.gz/from/http://cdn.mysql.com/' do
# custom_archive 'mysql-5.5.25a.tar.gz'
# configure_command 'cmake .'
# build_command 'make' # This is actually the default command but could be set to something else here.
# install_command 'make install' # This is actually the default command but could be set to something else here.
# 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
multi_attributes :enable, :disable, :with, :without, :option,
:custom_install
attributes :configure_command, :build_command, :install_command, :custom_archive, :custom_dir
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]}",
"mkdir -p #{@options[:builds]}",
"mkdir -p #{@options[:archives]}" ]
end
def download_commands #:nodoc:
[ "wget -cq -O '#{@options[:archives]}/#{archive_name}' #{@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 = "#{configure_command || './configure'} --prefix=#{@options[:prefix]} "
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 }
[ in_build_dir(with_log(command,:configure)) ]
end
def build_commands #:nodoc:
return [] if custom_install?
[ in_build_dir(with_log("#{build_command || "make"}",:build)) ]
end
def install_commands #:nodoc:
return custom_install_commands if custom_install?
[ in_build_dir(with_log("#{install_command || "make install"}",:install)) ]
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], nil, :install
end
protected
def with_log(cmd, stage)
"#{cmd} > #{@package.name}-#{stage}.log 2>&1"
end
def in_build_dir(cmd)
"bash -c 'cd #{build_dir} && #{cmd}'"
end
def pre_commands(stage) #:nodoc:
dress @pre[stage] || [], :pre, stage
end
def post_commands(stage) #:nodoc:
dress @post[stage] || [], :post, stage
end
# 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, pre_or_post, stage)
chdir = "cd #{build_dir} && "
chdir = "" if [:prepare, :download].include?(stage)
chdir = "" if stage == :extract and pre_or_post == :pre
flatten(commands).collect { |command| "bash -c '#{chdir}#{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 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 = options[:custom_archive] || @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 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