This.name = "Errors2Html" This.synopsis = "<%= form_for @post do %> <%= @post.errors.to_html %>" This.rubyforge_project = 'codeforpeople' This.author = "Ara T. Howard" This.email = "ara.t.howard@gmail.com" This.homepage = "https://github.com/ahoward/#{ This.lib }" This.setup! task :default do puts((Rake::Task.tasks.map{|task| task.name.gsub(/::/,':')} - ['default']).sort) end task :test do This.run_tests! end namespace :test do task(:unit){ This.run_tests!(:unit) } task(:functional){ This.run_tests!(:functional) } task(:integration){ This.run_tests!(:integration) } end def This.run_tests!(which = nil) which ||= '**' test_dir = File.join(This.dir, "test") test_glob ||= File.join(test_dir, "#{ which }/**_test.rb") test_rbs = Dir.glob(test_glob).sort div = ('=' * 119) line = ('-' * 119) test_rbs.each_with_index do |test_rb, index| testno = index + 1 command = "#{ File.basename(This.ruby) } -I ./lib -I ./test/lib #{ test_rb }" puts This.say(div, :color => :cyan, :bold => true) This.say("@#{ testno } => ", :bold => true, :method => :print) This.say(command, :color => :cyan, :bold => true) This.say(line, :color => :cyan, :bold => true) system(command) This.say(line, :color => :cyan, :bold => true) status = $?.exitstatus if status.zero? This.say("@#{ testno } <= ", :bold => true, :color => :white, :method => :print) This.say("SUCCESS", :color => :green, :bold => true) else This.say("@#{ testno } <= ", :bold => true, :color => :white, :method => :print) This.say("FAILURE", :color => :red, :bold => true) end This.say(line, :color => :cyan, :bold => true) exit(status) unless status.zero? end end task :gemspec do ignore_extensions = ['git', 'svn', 'tmp', /sw./, 'bak', 'gem'] ignore_directories = ['pkg', 'db'] ignore_files = ['test/log', 'test/db.yml', 'a.rb', 'b.rb'] + Dir['db/*'] + %w'db' shiteless = lambda do |list| list.delete_if do |entry| next unless test(?e, entry) extension = File.basename(entry).split(%r/[.]/).last ignore_extensions.any?{|ext| ext === extension} end list.delete_if do |entry| next unless test(?d, entry) dirname = File.expand_path(entry) ignore_directories.any?{|dir| File.expand_path(dir) == dirname} end list.delete_if do |entry| next unless test(?f, entry) filename = File.expand_path(entry) ignore_files.any?{|file| File.expand_path(file) == filename} end end lib = This.lib object = This.object version = This.version files = shiteless[Dir::glob("**/**")] executables = shiteless[Dir::glob("bin/*")].map{|exe| File.basename(exe)} #has_rdoc = true #File.exist?('doc') test_files = test(?e, "test/#{ lib }.rb") ? "test/#{ lib }.rb" : nil summary = This.summary || This.synopsis || "#{ lib } kicks the ass" description = This.description || summary if This.extensions.nil? This.extensions = [] extensions = This.extensions %w( Makefile configure extconf.rb ).each do |ext| extensions << ext if File.exists?(ext) end end extensions = [extensions].flatten.compact # TODO if This.dependencies.nil? dependencies = [] else case This.dependencies when Hash dependencies = This.dependencies.values when Array dependencies = This.dependencies end end template = if test(?e, 'gemspec.erb') This.template_for{ IO.read('gemspec.erb') } else This.template_for { <<-__ ## <%= lib %>.gemspec # Gem::Specification::new do |spec| spec.name = <%= lib.inspect %> spec.version = <%= version.inspect %> spec.platform = Gem::Platform::RUBY spec.summary = <%= lib.inspect %> spec.description = <%= description.inspect %> spec.files =\n<%= files.sort.pretty_inspect %> spec.executables = <%= executables.inspect %> spec.require_path = "lib" spec.test_files = <%= test_files.inspect %> <% dependencies.each do |lib_version| %> spec.add_dependency(*<%= Array(lib_version).flatten.inspect %>) <% end %> spec.extensions.push(*<%= extensions.inspect %>) spec.rubyforge_project = <%= This.rubyforge_project.inspect %> spec.author = <%= This.author.inspect %> spec.email = <%= This.email.inspect %> spec.homepage = <%= This.homepage.inspect %> end __ } end FileUtils.mkdir_p(This.pkgdir) gemspec = "#{ lib }.gemspec" open(gemspec, "w"){|fd| fd.puts(template)} This.gemspec = gemspec end task :gem => [:clean, :gemspec] do FileUtils.mkdir_p(This.pkgdir) before = Dir['*.gem'] cmd = "gem build #{ This.gemspec }" `#{ cmd }` after = Dir['*.gem'] gem = ((after - before).first || after.first) or abort('no gem!') FileUtils.mv(gem, This.pkgdir) This.gem = File.join(This.pkgdir, File.basename(gem)) end task :readme do samples = '' prompt = '~ > ' lib = This.lib version = This.version Dir['sample*/*'].sort.each do |sample| samples << "\n" << " <========< #{ sample } >========>" << "\n\n" cmd = "cat #{ sample }" samples << This.util.indent(prompt + cmd, 2) << "\n\n" samples << This.util.indent(`#{ cmd }`, 4) << "\n" cmd = "ruby #{ sample }" samples << This.util.indent(prompt + cmd, 2) << "\n\n" cmd = "ruby -e'STDOUT.sync=true; exec %(ruby -I ./lib #{ sample })'" samples << This.util.indent(`#{ cmd } 2>&1`, 4) << "\n" end template = if test(?e, 'readme.erb') This.template_for{ IO.read('readme.erb') } else This.template_for { <<-__ NAME #{ lib } DESCRIPTION INSTALL gem install #{ lib } SAMPLES #{ samples } __ } end open("README", "w"){|fd| fd.puts template} end task :clean do Dir[File.join(This.pkgdir, '**/**')].each{|entry| FileUtils.rm_rf(entry)} end task :release => [:clean, :gemspec, :gem] do gems = Dir[File.join(This.pkgdir, '*.gem')].flatten raise "which one? : #{ gems.inspect }" if gems.size > 1 raise "no gems?" if gems.size < 1 cmd = "gem push #{ This.gem }" puts cmd puts system(cmd) abort("cmd(#{ cmd }) failed with (#{ $?.inspect })") unless $?.exitstatus.zero? #cmd = "rubyforge login && rubyforge add_release #{ This.rubyforge_project } #{ This.lib } #{ This.version } #{ This.gem }" #puts cmd #puts #system(cmd) #abort("cmd(#{ cmd }) failed with (#{ $?.inspect })") unless $?.exitstatus.zero? end BEGIN { # support for this rakefile # $VERBOSE = nil require 'erb' require 'fileutils' require 'rbconfig' require 'pp' # cache a bunch of stuff about this rakefile/environment # This = Class.new(Hash) do def method_missing(method, *args, &block) if method.to_s =~ /=/ key = method.to_s.chomp('=') value = block ? block : args.shift self[key] = value else key = method.to_s if block value = block self[key] = value else value = self[key] if value.respond_to?(:call) self[key] = value.call() else value end end end end def inspect expand! PP.pp(self, '') end def expand! keys.each do |key| value = self[key] if value.respond_to?(:call) self[key] = value.call() end end end end.new() This.file = File.expand_path(__FILE__) This.dir = File.dirname(This.file) This.pkgdir = File.join(This.dir, 'pkg') # defaults # This.lib do File.basename(Dir.pwd) end def This.setup! begin require "./lib/#{ This.lib }" rescue LoadError abort("could not load #{ This.lib }") end end This.name do This.name = This.lib.capitalize end This.object do begin This.object = eval(This.name) rescue Object abort("could not determine object from #{ This.name }") end end This.version do This.object.send(:version) end This.dependencies do if This.object.respond_to?(:dependencies) This.object.dependencies end end This.ruby do c = Config::CONFIG bindir = c["bindir"] || c['BINDIR'] ruby_install_name = c['ruby_install_name'] || c['RUBY_INSTALL_NAME'] || 'ruby' ruby_ext = c['EXEEXT'] || '' File.join(bindir, (ruby_install_name + ruby_ext)) end # some utils # This.util = Module.new do def indent(s, n = 2) s = unindent(s) ws = ' ' * n s.gsub(%r/^/, ws) end def unindent(s) indent = nil s.each_line do |line| next if line =~ %r/^\s*$/ indent = line[%r/^\s*/] and break end indent ? s.gsub(%r/^#{ indent }/, "") : s end extend self end # template support # This.template = Class.new do def initialize(&block) @block = block @template = block.call.to_s end def expand(b=nil) ERB.new(This.util.unindent(@template)).result((b||@block).binding) end alias_method 'to_s', 'expand' end def This.template_for(*args, &block) This.template.new(*args, &block) end # colored console output support # This.ansi = { :clear => "\e[0m", :reset => "\e[0m", :erase_line => "\e[K", :erase_char => "\e[P", :bold => "\e[1m", :dark => "\e[2m", :underline => "\e[4m", :underscore => "\e[4m", :blink => "\e[5m", :reverse => "\e[7m", :concealed => "\e[8m", :black => "\e[30m", :red => "\e[31m", :green => "\e[32m", :yellow => "\e[33m", :blue => "\e[34m", :magenta => "\e[35m", :cyan => "\e[36m", :white => "\e[37m", :on_black => "\e[40m", :on_red => "\e[41m", :on_green => "\e[42m", :on_yellow => "\e[43m", :on_blue => "\e[44m", :on_magenta => "\e[45m", :on_cyan => "\e[46m", :on_white => "\e[47m" } def This.say(something, *args) options = args.last.is_a?(Hash) ? args.pop : {} options[:color] = args.shift.to_s.to_sym unless args.empty? keys = options.keys keys.each{|key| options[key.to_s.to_sym] = options.delete(key)} color = options[:color] bold = options.has_key?(:bold) parts = [something] parts.unshift(This.ansi[color]) if color parts.unshift(This.ansi[:bold]) if bold parts.push(This.ansi[:clear]) if parts.size > 1 method = options[:method] || :puts Kernel.send(method, parts.join) end # always run out of the project dir # Dir.chdir(This.dir) }