From 6a1c8dd042c2d3191e3ac69ac73d97f09ad859a0 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 10:51:39 -0700 Subject: [PATCH 01/10] Split away formula classes into separate parts, then have debian/redhat formula files include common_formula. (This makes it easier to extend brew2deb to build other types of packages) --- lib/common_formula.rb | 45 ++++++ lib/debian_formula.rb | 46 +----- lib/redhat_formula.rb | 338 ++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 384 insertions(+), 45 deletions(-) create mode 100644 lib/common_formula.rb create mode 100644 lib/redhat_formula.rb diff --git a/lib/common_formula.rb b/lib/common_formula.rb new file mode 100644 index 0000000..8d8b1f9 --- /dev/null +++ b/lib/common_formula.rb @@ -0,0 +1,45 @@ +$:.unshift File.expand_path('../../homebrew/Library/Homebrew', __FILE__) +require 'tempfile' +require 'global' +require 'formula' + +class String + # Useful for writing indented String and unindent on demand, based on the + # first line with indentation. + def unindent + find_indent = proc{ |l| l.find{|l| !l.strip.empty?}.to_s[/^(\s+)/, 1] } + + lines = self.split("\n") + space = find_indent[lines] + space = find_indent[lines.reverse] unless space + + strip.gsub(/^#{space}/, '') + end + alias ui unindent + + # Destructive variant of undindent, replacing the String + def unindent! + self.replace unindent + end + alias ui! unindent! +end + +class Formula + def self.attr_rw_list(*attrs) + attrs.each do |attr| + instance_variable_set("@#{attr}", []) + class_eval %Q{ + def self.#{attr}(*list) + @#{attr} ||= superclass.respond_to?(:#{attr}) ? superclass.#{attr} : [] + list.empty? ? @#{attr} : @#{attr} += list + @#{attr}.uniq! + @#{attr} + end + def self.#{attr}!(*list) + @#{attr} = [] + #{attr}(*list) + end + } + end + end +end diff --git a/lib/debian_formula.rb b/lib/debian_formula.rb index 028ab6d..383807c 100644 --- a/lib/debian_formula.rb +++ b/lib/debian_formula.rb @@ -1,48 +1,4 @@ -$:.unshift File.expand_path('../../homebrew/Library/Homebrew', __FILE__) -require 'tempfile' -require 'global' -require 'formula' - -class String - # Useful for writing indented String and unindent on demand, based on the - # first line with indentation. - def unindent - find_indent = proc{ |l| l.find{|l| !l.strip.empty?}.to_s[/^(\s+)/, 1] } - - lines = self.split("\n") - space = find_indent[lines] - space = find_indent[lines.reverse] unless space - - strip.gsub(/^#{space}/, '') - end - alias ui unindent - - # Destructive variant of undindent, replacing the String - def unindent! - self.replace unindent - end - alias ui! unindent! -end - -class Formula - def self.attr_rw_list(*attrs) - attrs.each do |attr| - instance_variable_set("@#{attr}", []) - class_eval %Q{ - def self.#{attr}(*list) - @#{attr} ||= superclass.respond_to?(:#{attr}) ? superclass.#{attr} : [] - list.empty? ? @#{attr} : @#{attr} += list - @#{attr}.uniq! - @#{attr} - end - def self.#{attr}!(*list) - @#{attr} = [] - #{attr}(*list) - end - } - end - end -end +require 'common_formula' class DebianFormula < Formula attr_rw :name, :description diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb new file mode 100644 index 0000000..a040649 --- /dev/null +++ b/lib/redhat_formula.rb @@ -0,0 +1,338 @@ +require 'common_formula' + +class RedHatFormula < Formula + attr_rw :name, :description + attr_rw :maintainer, :section, :arch + attr_rw :pre_install, :post_install, :pre_uninstall, :post_uninstall + + attr_rw_list :depends, :build_depends + attr_rw_list :provides, :conflicts, :replaces + attr_rw_list :config_files + + attr_accessor :skip_build + attr_writer :installing + + build_depends \ + 'gcc', + 'gcc-c++', + 'kernel-devel', + 'patch', + 'curl' + + def self.requires_user(name, opts={}) + @create_user ||= {} + @create_user.update(opts) + @create_user[:name] = name + end + + class << self + attr_reader :create_user + @extra_sources = [] + end + + def self.extra_sources + @extra_sources ||= (self == RedHatFormula ? [] : superclass.extra_sources.dup) + end + + def self.source(url, opts={}) + extra_sources << [url, opts] + end + + def download_extra_sources + self.class.extra_sources.each do |url, opts| + spec = SoftwareSpecification.new(url, opts) + downloader = spec.download_strategy.new(url, nil, spec.detect_version, opts) + downloader.fetch + chdir(builddir) do + downloader.stage + end + end + end + + def stage + if skip_build + chdir(builddir) do + @downloader.send :chdir + yield + end + else + super do + download_extra_sources + yield + end + end + end + + def patch + skip_build ? nil : super + end + + def postinst + set_instance_variable 'create_user' + + if @create_user and user = @create_user[:name] + home = @create_user[:home] || '/nonexistent' + + script = " + #!/bin/sh + set -e + + (id #{user} >/dev/null 2>&1) || + adduser --system --group --no-create-home --home #{home} #{user} + " + + paths = Array(@create_user[:chown]) + script += " + chown -R #{user}:#{user} #{paths.map(&:dump).join(' ')} + " if paths and paths.any? + + script.ui + end + end + + def postrm + if @create_user and user = @create_user[:name] and @create_user[:remove] != false + " + #!/bin/sh + set -e + + deluser --system #{user} + ".ui + end + end + + def self.package! + raise 'Missing name/version' if self == RedHatFormula and (!name or !version) + + f = new + + unless RUBY_PLATFORM =~ /darwin/ + # Check for build deps. + system '/bin/rpm', '-q', f.class.build_depends.join(' '), '/dev/null' + if $? != 0 + f.send :onoe, 'Missing build dependencies.' + exit(1) + end + end + + built_file = HOMEBREW_WORKDIR + "tmp-build/.built-#{f.name}-#{f.version.gsub(/[^\w]/,'_')}" + if File.exists?(built_file) + f.skip_build = true + f.send :ohai, 'Skipping build (`brew2deb clean` to rebuild)' + end + + env = ENV.to_hash + + f.brew do + unless f.skip_build + f.send :ohai, 'Compiling source' + f.installing = false + f.build + end + + FileUtils.touch(built_file) + + begin + f.send :ohai, 'Installing binaries' + f.installing = true + dir = f.send(:destdir) + FileUtils.rm_rf(dir) + FileUtils.mkdir_p(dir) + f.install + f.post_install if f.respond_to?(:post_install) + ensure + f.installing = false + end + + ENV.replace(env) + + f.send :ohai, 'Packaging into a .rpm' + f.package + end + end + + def package + FileUtils.mkdir_p(HOMEBREW_WORKDIR+'pkg') + Dir.chdir HOMEBREW_WORKDIR+'pkg' do + epoch, ver = self.class.version.split(':', 2) + if ver.nil? + ver, epoch = epoch, nil + end + + opts = [ + '-n', name, + '-v', ver, + '-t', 'rpm', + '-s', 'dir', + '--url', self.class.homepage || self.class.url, + '-C', destdir, + '--maintainer', maintainer, + '--category', self.class.section, + ] + + opts += [ + '--epoch', epoch + ] if epoch + + opts += [ + '--description', self.class.description.ui.strip + ] if self.class.description + + opts += [ + '--architecture', self.class.arch.to_s + ] if self.class.arch + + if self.postinst + postinst_file = Tempfile.open('postinst') + postinst_file.puts(postinst) + chmod 0755, postinst_file.path + postinst_file.close + opts += ['--post-install', postinst_file.path] + end + if self.postrm + postrm_file = Tempfile.open('postrm') + postrm_file.puts(postrm) + chmod 0755, postrm_file.path + postrm_file.close + opts += ['--post-uninstall', postrm_file.path] + end + + %w[ depends provides replaces conflicts config_files ].each do |type| + if self.class.send(type).any? + self.class.send(type).each do |dep| + opts += ["--#{type.gsub('_','-')}", dep] + end + end + end + + opts << '.' + + safe_system File.expand_path('../../fpm/bin/fpm', __FILE__), *opts + end + end + + protected + + alias :sh :system + + def make(*args) + env = args.pop if args.last.is_a?(Hash) + env ||= {} + + args += env.map{ |k,v| "#{k}=#{v}" } + args.map!{ |a| a.to_s } + + safe_system 'make', *args + end + + def skip_clean_all? + true + end + + def builddir + HOMEBREW_WORKDIR+'tmp-build' + end + + def mkbuilddir + FileUtils.mkdir_p(builddir) + raise "Couldn't create build sandbox" if not builddir.directory? + + chdir(builddir) do + yield + end + end + alias :mktemp :mkbuilddir + + def chdir(dir) + wd = Dir.pwd + Dir.chdir(dir) + yield + ensure + Dir.chdir(wd) + end + + def maintainer + @maintainer ||= self.class.maintainer || begin + username = `git config --get user.name`.strip + useremail = `git config --get user.email`.strip + raise 'Set maintainer name/email via `git config --global user.name `' if username.empty? + "#{username} <#{useremail}>" + end + end + + def workdir + HOMEBREW_WORKDIR + end + + def prefix + current_pathname_for('usr') + end + + def etc + current_pathname_for('etc') + end + + def var + current_pathname_for('var') + end + + def current_pathname_for(dir) + @installing ? destdir + dir : Pathname.new("/#{dir}") + end + + def destdir + HOMEBREW_WORKDIR+'tmp-install' + end + + def configure(*args) + if args.last.is_a?(Hash) + opts = args.pop + args += opts.map{ |k,v| + option = k.to_s.gsub('_','-') + if v == true + "--#{option}" + else + "--#{option}=#{v}" + end + } + end + + sh "./configure", *args + end + + public + + def build + configure :prefix => prefix + make + end + + def install + make :install, 'DESTDIR' => destdir + end +end + +#class RedHatSourceFormula < RedHatFormula +# build_depends \ +# 'fakeroot', +# 'devscripts', +# 'dpkg-dev' +# +# def build +# ENV['DEBEMAIL'] = maintainer +# if ver = self.class.version +# safe_system 'dch', '-v', ver, 'brew2deb package' +# end +# safe_system 'dpkg-buildpackage', '-rfakeroot', '-us', '-uc' +# end +# +# def install +# end +# +# def package +# FileUtils.mkdir_p(HOMEBREW_WORKDIR+'pkg') +# Dir[HOMEBREW_WORKDIR+'tmp-build'+'*.{dsc,gz,changes,deb,udeb}'].each do |file| +# FileUtils.cp file, HOMEBREW_WORKDIR+'pkg' +# end +# end +#end From aa56802d509092cd8d1482ed3fe8b48f4cc449d7 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:08:53 -0700 Subject: [PATCH 02/10] Add credis-rpmtest for testing RPM packaging. --- packages/credis-rpmtest/formula.rb | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) create mode 100644 packages/credis-rpmtest/formula.rb diff --git a/packages/credis-rpmtest/formula.rb b/packages/credis-rpmtest/formula.rb new file mode 100644 index 0000000..9caa6d8 --- /dev/null +++ b/packages/credis-rpmtest/formula.rb @@ -0,0 +1,22 @@ +class Credis < RedHatFormula + homepage 'http://code.google.com/p/credis/' + url 'http://credis.googlecode.com/files/credis-0.2.3.tar.gz' + md5 '338c21667ece272d9ab669738e27b191' + + name 'libcredis' + section 'devel' + version '0.2.3+github1' + description 'C client library for redis' + + def build + make + end + + def install + ['libcredis.a', 'libcredis.so'].each do |f| + lib.install f + end + include.install 'credis.h' + doc.install 'README' + end +end From e256e0a4dd226b0ffa0af1a8ba307f41ca36e8de Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:10:44 -0700 Subject: [PATCH 03/10] Require 'common_formula', 'debian_formula', and 'redhat_formula' --- bin/brew2deb | 2 ++ 1 file changed, 2 insertions(+) diff --git a/bin/brew2deb b/bin/brew2deb index 9e83995..4300064 100755 --- a/bin/brew2deb +++ b/bin/brew2deb @@ -18,7 +18,9 @@ if ARGV.include? 'clean' end $:.unshift File.expand_path('../../lib', __FILE__) +require 'common_fomrula' require 'debian_formula' +require 'redhat_formula' Object.__send__ :remove_const, :HOMEBREW_CACHE HOMEBREW_WORKDIR = Pathname.new(Dir.pwd) From 69e02563346a71567655b3e81444c75c49b42da5 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:11:52 -0700 Subject: [PATCH 04/10] Oops, typo. --- bin/brew2deb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bin/brew2deb b/bin/brew2deb index 4300064..6ce5361 100755 --- a/bin/brew2deb +++ b/bin/brew2deb @@ -18,7 +18,7 @@ if ARGV.include? 'clean' end $:.unshift File.expand_path('../../lib', __FILE__) -require 'common_fomrula' +require 'common_formula' require 'debian_formula' require 'redhat_formula' From 45adf0e1d97cd5e3d7767ba421f6623f84cbaa1e Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:13:48 -0700 Subject: [PATCH 05/10] Try redirecting /dev/null via pipe. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index a040649..44ebfde 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system '/bin/rpm', '-q', f.class.build_depends.join(' '), '/dev/null' + system '/bin/rpm', '-q', f.class.build_depends.join(' '), '> /dev/null' if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1) From 1d85e9c25dd7c6d1cbcdeca79cfeb536cd732662 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:13:48 -0700 Subject: [PATCH 06/10] Try redirecting stdout to /dev/null via pipe. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index a040649..44ebfde 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system '/bin/rpm', '-q', f.class.build_depends.join(' '), '/dev/null' + system '/bin/rpm', '-q', f.class.build_depends.join(' '), '> /dev/null' if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1) From fb910eb7d29956a8b2dd046c2a45a478c7781405 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:15:25 -0700 Subject: [PATCH 07/10] Try using no redirection. Join package array by comma, like in DebianFormula. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index 44ebfde..0626f90 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system '/bin/rpm', '-q', f.class.build_depends.join(' '), '> /dev/null' + system '/bin/rpm', '-q', f.class.build_depends.join(',') if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1) From f6c36d8dc5b54e7f41ace1d69caebaeb927839a1 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:17:27 -0700 Subject: [PATCH 08/10] Try joining by space again. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index 0626f90..4559e91 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system '/bin/rpm', '-q', f.class.build_depends.join(',') + system '/bin/rpm', '-q', f.class.build_depends.join(' ') if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1) From 4879955e59f6e33f62be412bce6ee8666c449c51 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:24:20 -0700 Subject: [PATCH 09/10] Use entire string and interpolate f.class.build_depends in that. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index 4559e91..acbe0fd 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system '/bin/rpm', '-q', f.class.build_depends.join(' ') + system "/bin/rpm -q #{f.class.build_depends.join(' ')}" if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1) From aa5bb69cc23e996087e97a625a1dd624f0860d97 Mon Sep 17 00:00:00 2001 From: Christian Paredes Date: Wed, 5 Oct 2011 12:33:34 -0700 Subject: [PATCH 10/10] Redirect to /dev/null. --- lib/redhat_formula.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/redhat_formula.rb b/lib/redhat_formula.rb index acbe0fd..17d6311 100644 --- a/lib/redhat_formula.rb +++ b/lib/redhat_formula.rb @@ -108,7 +108,7 @@ def self.package! unless RUBY_PLATFORM =~ /darwin/ # Check for build deps. - system "/bin/rpm -q #{f.class.build_depends.join(' ')}" + system "/bin/rpm -q #{f.class.build_depends.join(' ')} > /dev/null" if $? != 0 f.send :onoe, 'Missing build dependencies.' exit(1)