require "digest/sha1" require "net/http/post/multipart" require "rest_client" require "thor" require "tmpdir" require "uri" require "vulcan" require "yaml" class Vulcan::CLI < Thor desc "build", <<-DESC build a piece of software for the heroku cloud using COMMAND as a build command if no COMMAND is specified, a sensible default will be chosen for you DESC method_option :command, :aliases => "-c", :desc => "the command to run for compilation" method_option :name, :aliases => "-n", :desc => "the name of the library (defaults ot the directory name)" method_option :output, :aliases => "-o", :desc => "output build artifacts to this file" method_option :prefix, :aliases => "-p", :desc => "the build/install --prefix of the software" method_option :source, :aliases => "-s", :desc => "the source directory to build from" method_option :verbose, :aliases => "-v", :desc => "show the full build output", :type => :boolean def build app = read_config[:app] || "need a server first, use vulcan create" source = options[:source] || Dir.pwd name = options[:name] || File.basename(source) output = options[:output] || "/tmp/#{name}.tgz" prefix = options[:prefix] || "/app/vendor/#{name}" command = options[:command] || "./configure --prefix #{prefix} && make install" server = URI.parse(ENV["MAKE_SERVER"] || "http://#{app}.herokuapp.com") Dir.mktmpdir do |dir| puts ">> Packaging local directory" %x{ cd #{source} && tar czvf #{dir}/input.tgz . 2>&1 } puts ">> Uploading code for build" File.open("#{dir}/input.tgz", "r") do |input| request = Net::HTTP::Post::Multipart.new "/make", "code" => UploadIO.new(input, "application/octet-stream", "input.tgz"), "command" => command, "prefix" => prefix, "secret" => config[:secret] puts ">> Building with: #{command}" response = Net::HTTP.start(server.host, server.port) do |http| http.request(request) do |response| response.read_body do |chunk| print chunk if options[:verbose] end end end error "Unknown error, no build output given" unless response["X-Make-Id"] puts ">> Downloading build artifacts to: #{output}" output_url = "#{server}/output/#{response["X-Make-Id"]}" puts " (available at #{output_url})" File.open(output, "w") do |output| begin output.print RestClient.get(output_url) rescue Exception => ex puts ex.inspect end end end end rescue Interrupt error "Aborted by user" rescue Errno::EPIPE error "Could not connect to build server: #{server}" end desc "create APP_NAME", <<-DESC create a build server on Heroku DESC def create(name) secret = Digest::SHA1.hexdigest("--#{rand(10000)}--#{Time.now}--") Dir.mktmpdir do |dir| Dir.chdir(dir) do system "env BUNDLE_GEMFILE= heroku create #{name} -s cedar" end end write_config :app => name, :host => "#{name}.herokuapp.com", :secret => secret update end desc "update", <<-DESC update the build server DESC def update error "no app yet, create first" unless config[:app] FileUtils.mkdir_p File.expand_path("~/.heroku/plugins/heroku-credentials") File.open(File.expand_path("~/.heroku/plugins/heroku-credentials/init.rb"), "w") do |file| file.puts <<-CONTENTS class Heroku::Auth def self.api_key Heroku::Client.auth(user, password, host)["api_key"] end end class Heroku::Command::Credentials < Heroku::Command::Base # credentials # # list your api key # def index puts Heroku::Auth.api_key end end CONTENTS end Dir.mktmpdir do |dir| Dir.chdir(dir) do api_key = %x{ env BUNDLE_GEMFILE= heroku credentials 2>&1 }.chomp error "invalid api key detected, try running `heroku credentials`" if api_key =~ / / %x{ env BUNDLE_GEMFILE= heroku config:add BUILDPACK_URL=https://github.com/heroku/heroku-buildpack-nodejs.git#versions 2>&1 } system "git init" system "git remote add heroku git@heroku.com:#{config[:app]}.git" FileUtils.cp_r "#{server_path}/.", "." File.open(".gitignore", "w") do |file| file.puts ".env" file.puts "node_modules" end system "git add . >/dev/null" system "git commit -m commit >/dev/null" system "git push heroku -f master" %x{ env BUNDLE_GEMFILE= heroku config:add SECRET=#{config[:secret]} SPAWN_ENV=heroku HEROKU_APP=#{config[:app]} HEROKU_API_KEY=#{api_key} NODE_PATH=lib 2>&1 } %x{ env BUNDLE_GEMFILE= heroku addons:add cloudant:oxygen } end end end private def config_file File.expand_path("~/.vulcan") end def config read_config end def read_config return {} unless File.exists?(config_file) config = YAML.load_file(config_file) config.is_a?(Hash) ? config : {} end def write_config(config) full_config = read_config.merge(config) File.open(config_file, "w") do |file| file.puts YAML.dump(full_config) end end def error(message) puts "!! #{message}" exit 1 end def server_path File.expand_path("../../../server", __FILE__) end end