#!/usr/bin/python3 # # merge-pr - Rebase, merge, and close a github pull request # # Copyright (C) 2015 Red Hat, Inc. # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published # by the Free Software Foundation; either version 2.1 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with this program. If not, see . # # Author: David Shea # A quick note on logging in to github: # This script uses git's credentials API for retrieving and possibly storing # your github user name and password. Git comes with credential-cache, which # stores information in memory, and credential-store, which stores information # in a file in your home directory. Use `git config credential.helper ' # if you want to use one of these. See the git help pages for credential-cache # and credential-store for more information, and also see the help pages for # 'credentials' for information on how to change the thing that asks for your # password. # # You can use an OAuth token in place of your username and password. Create a # personal access token using https://github.com/settings/tokens/new, input # "token" as your username and put in the token as your password. Github only # shows you these tokens once, so OAuth makes the most sense when combined with # a credential helper that permanently stores your passwords. The token only # needs access to the public_repo scope. # # If you use 2-factor authentication and do not use an OAuth token, you will # be asked for your 2-factor code three times. # # Use the --nosavepw option if you don't want to try saving your credentials # with git. # This script expects there to be local branches that match the names of # the remote branches the pull request is against. So if there's a pull # request against f22-branch but you don't have f22-branch locally, it won't # work. import argparse import subprocess import sys import os import re import atexit import json from tempfile import NamedTemporaryFile import requests import six def talk_to_github(request): # Send a requests.Request to github, handle 2-factor auth request.headers.update({'User-Agent': 'merge-pr'}) prep = request.prepare() session = requests.Session() response = session.send(prep) # github sometimes uses 404 in response to unauthenticated API calls if response.status_code in (401, 404) and \ response.headers.get('X-GitHub-OTP', '').startswith('required'): try: twofactor = input("Input 2-factor authentication code: ") except EOFError: twofactor = "" request.headers.update({'X-GitHub-OTP': twofactor}) prep = request.prepare() response = session.send(prep) if response.status_code not in (200, 201): print("Error communicating with github: %s\n%s" % (response.status_code, response.text)) sys.exit(1) return response def main(): parser = argparse.ArgumentParser(description='Github pull request merger') parser.add_argument('--nosavepw', action='store_true', default=False, help='Do not attempt to save the github user name and password') parser.add_argument('pr_url', metavar='URL', help='Pull request URL to merge') args = parser.parse_args() # SANITY CHECK: are we in a git repo? We need to be in a git repo. # Git's error message is probably good enough if something went wrong. if subprocess.call(['git', 'rev-parse']) != 0: sys.exit(1) # Parse the web URL into something that can be used for an API call, make # sure all the pieces are there. pr_url = six.moves.urllib.parse.urlparse(args.pr_url) # The path should be /:owner/:repo/pull/:numbero # The first part of the split will be empty since path starts with / pr_path = pr_url[2].split('/') if len(pr_path) != 5 or pr_path[3] != 'pull' or not pr_path[4].isdigit(): print("Unable to parse pull request URL") sys.exit(1) pr_owner = pr_path[1] pr_repo = pr_path[2] pr_number = pr_path[4] # Figure out where we are in the current repo so we can go back to it try: current_head = subprocess.check_output(['git', 'symbolic-ref', '-q', '--short', 'HEAD']) except subprocess.CalledProcessError: try: current_head = subprocess.check_output(['git', 'rev-parse', 'HEAD']) except subprocess.CalledProcessError: # There's probably a ton of error output from git by now, so just exit sys.exit(1) current_head = current_head.rstrip('\n') atexit.register(lambda: subprocess.call(['git', 'checkout', '-q', current_head])) # Time to get a password so we can start talking to github github_cred = 'protocol=https\nhost=api.github.com\n' try: p = subprocess.Popen(['git', 'credential', 'fill'], stdin=subprocess.PIPE, stdout=subprocess.PIPE) p.stdin.write(github_cred) (stdin, _stderr) = p.communicate() except (OSError, IOError) as e: print("Unable to get github credentials: %s" % e) sys.exit(1) # Parse the username and password m = re.search('^username=(.*)$', stdin, flags=re.MULTILINE) if not m: print("Unable to determine github username") sys.exit(1) username = m.group(1) m = re.search('^password=(.*)$', stdin, flags=re.MULTILINE) if not m: print("Unable to determine github password") sys.exit(1) password = m.group(1) # If using a OAuth token, rearrange the auth data to let github know we're # sending a token as a basic http auth if username == "token": username = password password = "x-oauth-basic" # Save the username and password back to git if not args.nosavepw: try: p = subprocess.Popen(['git', 'credential', 'approve'], stdin=subprocess.PIPE) p.stdin.write(stdin) p.communicate() except (OSError, IOError) as e: print("Unable to save github credentials: %s" % e) sys.exit(1) # Now to talk to github. First, get the PR object pr_req = requests.Request(method='GET', url='https://api.github.com/repos/%s/%s/pulls/%s' % (pr_owner, pr_repo, pr_number), auth=(username, password)) pr = talk_to_github(pr_req).json() # If github reports the PR is not mergeable, give up now if not pr['mergeable']: print("Pull request is not mergeable, exiting") sys.exit(1) # Start messing with the git repo # Try checking out the base of the PR. If it isn't available, do a fetch of # the base repo and try again. try: subprocess.check_call(['git', 'checkout', '-q', pr['base']['sha']], stderr=subprocess.DEVNULL) except subprocess.CalledProcessError: try: subprocess.check_call(['git', 'fetch', '-q', pr['base']['repo']['clone_url'], pr['base']['sha']]) subprocess.check_call(['git', 'checkout', '-q', pr['base']['sha']]) except subprocess.CalledProcessError: sys.exit(1) try: branch_name = 'merge-pr-%s-%s' % (pr['head']['user']['login'], pr['head']['ref']) # Create a branch for the PR and pull the data into it subprocess.check_call(['git', 'checkout', '-q', '-b', branch_name]) subprocess.check_call(['git', 'pull', '-q', '--ff-only', pr['head']['repo']['clone_url'], pr['head']['sha']]) # Rebase the PR to the current state of the target branch subprocess.check_call(['git', 'rebase', '-q', pr['base']['ref']]) # Merge the PR onto the target branch and delete the PR branch subprocess.check_call(['git', 'checkout', '-q', pr['base']['ref']]) subprocess.check_call(['git', 'merge', '-q', '--ff-only', branch_name]) subprocess.check_call(['git', 'branch', '-q', '-d', branch_name]) # Before we push, launch an editor for the message to use when closing # the pull request try: editor = subprocess.check_call(['git', 'config', '--get', 'core.editor']) except subprocess.CalledProcessError: if 'VISUAL' in os.environ: editor = os.environ['VISUAL'] elif 'EDITOR' in os.environ: editor = os.environ['EDITOR'] else: editor = 'vi' with NamedTemporaryFile() as pr_msg_file: # Display some information about what the merge being pushed will # be, and list the commits commit_list = subprocess.check_output(['git', 'log', pr['base']['ref'], '--not', '--remotes=*/%s' % pr['base']['ref'], '--pretty=format:%h %s']) commit_list = '\n'.join('# %s' % line for line in commit_list.splitlines()) pr_msg_file.write(""" # Please enter the message with which to close the pull request. Lines # starting with '#' will be ignored, and an empty message aborts the merge. # The merged pull request will remain in your working copy under the # '%s' branch. # # Pull request to merge: %s/%s/%s (%s) # into -> %s # %s """ % (pr['base']['ref'], pr_owner, pr_repo, pr_number, pr['head']['label'], subprocess.check_output(['git', 'rev-parse', '--abbrev-ref', '@{upstream}']).rstrip('\n'), commit_list)) pr_msg_file.flush() subprocess.check_call([editor, pr_msg_file.name]) pr_msg_file.seek(0) # Strip comments and whitespace pr_msg = ''.join(line for line in pr_msg_file if not line.startswith('#')).strip() # Done with the message file at this point # If the message is empty, abort if not pr_msg: print("Empty pull request message, aborting") sys.exit(1) # Push the commits subprocess.check_call(['git', 'push', '-q']) # Add a comment to the PR pr_comment = requests.Request(method='POST', url='https://api.github.com/repos/%s/%s/issues/%s/comments' % (pr_owner, pr_repo, pr_number), data=json.dumps({'body': pr_msg}), auth=(username, password)) talk_to_github(pr_comment) # Close the PR pr_close = requests.Request(method='PATCH', url='https://api.github.com/repos/%s/%s/pulls/%s' % (pr_owner, pr_repo, pr_number), data=json.dumps({'state': 'closed'}), auth=(username, password)) talk_to_github(pr_close) except subprocess.CalledProcessError: sys.exit(1) if __name__ == "__main__": try: main() except KeyboardInterrupt: print("Exiting on user interrupt") sys.exit(1)