require 'thor' require 'rest_client' require 'socialcast-git-extensions' require 'socialcast' require 'socialcast/command_line/message' require 'highline/import' module Socialcast module Gitx class CLI < Thor include Socialcast::Gitx include Socialcast::Gitx::Git include Socialcast::Gitx::Github PULL_REQUEST_DESCRIPTION = "\n\n" + <<-EOS.dedent # Use GitHub flavored Markdown http://github.github.com/github-flavored-markdown/ # Links to screencasts or screenshots with a desciption of what this is showcasing. For architectual changes please include diagrams that will make it easier for the reviewer to understand the change. Format is ![title](url). # Link to ticket describing feature/bug (plantain, JIRA, bugzilla). Format is [title](url). # Brief description of the change, and how it accomplishes the task they set out to do. EOS method_option :quiet, :type => :boolean, :aliases => '-q' method_option :trace, :type => :boolean, :aliases => '-v' def initialize(*args) super(*args) RestClient.proxy = ENV['HTTPS_PROXY'] if ENV.has_key?('HTTPS_PROXY') RestClient.log = Logger.new(STDOUT) if options[:trace] end desc "reviewrequest", "Create a pull request on github" method_option :description, :type => :string, :aliases => '-d', :desc => 'pull request description' method_option :additional_reviewers, :type => :string, :aliases => '-a', :desc => 'add additional reviewers to mention automatically, and skips the prompt' method_option :skip_additional_reviewers, :type => :string, :aliases => '-s', :desc => 'Skips adding additional reviewers' # @see http://developer.github.com/v3/pulls/ def reviewrequest(*additional_reviewers) token = authorization_token update review_mention = if buddy = socialcast_review_buddy(current_user) "Assigned to @#{buddy}" end if !specialty_reviewers.empty? && !options.key?('skip_additional_reviewers') additional_reviewers = options[:additional_reviewers] || additional_reviewers if additional_reviewers.empty? prompt_text = "#{specialty_reviewers.map { |_,v| v['command'] }.join(", ")} or (or hit enter to continue): " additional_reviewers = $terminal.ask("Notify additional people? #{prompt_text} ") end additional_reviewers = additional_reviewers.is_a?(String) ? additional_reviewers.split(" ") : additional_reviewers (specialty_reviewers.keys & additional_reviewers).each do |command| reviewer = specialty_reviewers[command] review_mention = review_mention || '' review_mention += "\nAssigned additionally to @#{reviewer['socialcast_username']} for #{reviewer['label']} review" end end assignee = github_review_buddy(current_user) description = options[:description] || editor_input(PULL_REQUEST_DESCRIPTION) branch = current_branch repo = current_repo url = create_pull_request token, branch, repo, description, assignee say "Pull request created: #{url}" short_description = description.split("\n").first(5).join("\n") review_message = ["#reviewrequest for #{branch} #scgitx", "/cc @SocialcastDevelopers", review_mention, short_description, changelog_summary(branch)].compact.join("\n\n") post review_message, :url => url, :message_type => 'review_request' end desc "findpr", "Find pull requests including a given commit" def findpr(commit_hash) token = authorization_token repo = current_repo data = pull_requests_for_commit(token, repo, commit_hash) if data['items'] data['items'].each do |entry| say "\n" << [entry['html_url'], entry['title'], "#{entry['user'] && entry['user']['login']} #{entry['created_at']}"].join("\n\t") end else say "No results found", :yellow end end # TODO: use --no-edit to skip merge messages # TODO: use pull --rebase to skip merge commit desc 'update', 'Update the current branch with latest changes from the remote feature branch and master' def update branch = current_branch say 'updating ' say "#{branch} ", :green say "to have most recent changes from " say base_branch, :green run_cmd "git pull origin #{branch}" rescue nil run_cmd "git pull origin #{base_branch}" run_cmd 'git push origin HEAD' end desc 'cleanup', 'Cleanup branches that have been merged into master from the repo' def cleanup run_cmd "git checkout #{base_branch}" run_cmd "git pull" run_cmd 'git remote prune origin' say "Deleting branches that have been merged into " say base_branch, :green branches(:merged => true, :remote => true).each do |branch| run_cmd "git push origin --delete #{branch}" unless reserved_branch?(branch) end branches(:merged => true).each do |branch| run_cmd "git branch -d #{branch}" unless reserved_branch?(branch) end end desc 'track', 'set the current branch to track the remote branch with the same name' def track track_branch current_branch end desc 'start', 'start a new git branch with latest changes from master' def start(branch_name = nil) unless branch_name example_branch = %w{ cpr-3922-api-fix-invalid-auth red-212-desktop-cleanup-avatar-markup red-3212-share-form-add-edit-link }.sample repo = Grit::Repo.new(Dir.pwd) remote_branches = repo.remotes.collect {|b| b.name.split('/').last } ## Explicitly use Highline.ask branch_name = $terminal.ask("What would you like to name your branch? (ex: #{example_branch})") do |q| q.validate = lambda { |branch| branch =~ /^[A-Za-z0-9\-_]+$/ && !remote_branches.include?(branch) } q.responses[:not_valid] = "This branch name is either already taken, or is not a valid branch name" end end run_cmd "git checkout #{base_branch}" run_cmd 'git pull' run_cmd "git checkout -b #{branch_name}" post "#worklog starting work on #{branch_name} #scgitx" end desc 'share', 'Share the current branch in the remote repository' def share share_branch current_branch end desc 'integrate', 'integrate the current branch into one of the aggregate development branches' def integrate(target_branch = prototype_branch) branch = current_branch update integrate_branch(branch, target_branch) integrate_branch(target_branch, prototype_branch) if target_branch == staging_branch run_cmd "git checkout #{branch}" post "#worklog integrating #{branch} into #{target_branch} #scgitx" end desc 'promote', '(DEPRECATED) promote the current branch into staging' def promote say "DEPRECATED: Use `git integrate #{staging_branch}` instead", :red integrate staging_branch end desc 'nuke', 'nuke the specified aggregate branch and reset it to a known good state' method_option :destination, :type => :string, :aliases => '-d', :desc => 'destination branch to reset to' def nuke(bad_branch) default_good_branch = "last_known_good_#{bad_branch}" good_branch = options[:destination] || ask("What branch do you want to reset #{bad_branch} to? (default: #{default_good_branch})") good_branch = default_good_branch if good_branch.length == 0 good_branch = "last_known_good_#{good_branch}" unless good_branch.starts_with?('last_known_good_') removed_branches = nuke_branch(bad_branch, good_branch) nuke_branch("last_known_good_#{bad_branch}", good_branch) message_parts = [] message_parts << "#worklog resetting #{bad_branch} branch to #{good_branch} #scgitx" message_parts << "/cc @SocialcastDevelopers" if removed_branches.any? message_parts << "" message_parts << "the following branches were affected:" message_parts += removed_branches.map{|b| ['*', b].join(' ')} end post message_parts.join("\n") end desc 'release', 'release the current branch to production' def release branch = current_branch assert_not_protected_branch!(branch, 'release') return unless yes?("Release #{branch} to production? (y/n)", :green) update run_cmd "git checkout #{base_branch}" run_cmd "git pull origin #{base_branch}" run_cmd "git pull . #{branch}" run_cmd "git push origin HEAD" integrate_branch(base_branch, staging_branch) cleanup post "#worklog releasing #{branch} to #{base_branch} #scgitx" end private # post a message in socialcast # skip sharing message if CLI quiet option is present def post(message, params = {}) return if options[:quiet] ActiveResource::Base.logger = Logger.new(STDOUT) if options[:trace] Socialcast::CommandLine::Message.configure_from_credentials response = Socialcast::CommandLine::Message.create params.merge(:body => message) say "Message has been posted: #{response.permalink_url}" end end end end