#!/usr/bin/env python3 ''' Create Pull Request ''' import json import os import random import string import sys import time from git import Repo from github import Github def get_github_event(github_event_path): with open(github_event_path) as f: github_event = json.load(f) if bool(os.environ.get('DEBUG_EVENT')): print(os.environ['GITHUB_EVENT_NAME']) print(json.dumps(github_event, sort_keys=True, indent=2)) return github_event def ignore_event(event_name, event_data): if event_name == "push": # Ignore push events on deleted branches # The event we want to ignore occurs when a PR is created but the repository owner decides # not to commit the changes. They close the PR and delete the branch. This creates a # "push" event that we want to ignore, otherwise it will create another branch and PR on # the same commit. deleted = "{deleted}".format(**event_data) if deleted == "True": print("Ignoring delete branch event.") return True ref = "{ref}".format(**event_data) if not ref.startswith('refs/heads/'): print("Ignoring events for tags and remotes.") return True return False def get_head_short_sha1(repo): return repo.git.rev_parse('--short', 'HEAD') def get_random_suffix(size=7, chars=string.ascii_lowercase + string.digits): return ''.join(random.choice(chars) for _ in range(size)) def remote_branch_exists(repo, branch): for ref in repo.remotes.origin.refs: if ref.name == ("origin/%s" % branch): return True return False def get_head_author(event_name, event_data): if event_name == "push": email = "{head_commit[author][email]}".format(**event_data) name = "{head_commit[author][name]}".format(**event_data) else: email = os.environ['GITHUB_ACTOR'] + '@users.noreply.github.com' name = os.environ['GITHUB_ACTOR'] return email, name def set_git_config(git, email, name): git.config('--global', 'user.email', '"%s"' % email) git.config('--global', 'user.name', '"%s"' % name) def set_git_remote_url(git, token, github_repository): git.remote('set-url', 'origin', "https://x-access-token:%s@github.com/%s" % (token, github_repository)) def push_changes(git, remote_exists, branch, commit_message): git.add('-A') git.commit(m=commit_message) if remote_exists: git.checkout(branch) git.rebase('-Xtheirs', '-') else: git.checkout('HEAD', b=branch) return git.push('-f', '--set-upstream', 'origin', branch) def cs_string_to_list(str): # Split the comma separated string into a list l = [i.strip() for i in str.split(',')] # Remove empty strings return list(filter(None, l)) def process_event(event_name, event_data, repo, branch, base, remote_exists): # Fetch required environment variables github_token = os.environ['GITHUB_TOKEN'] github_repository = os.environ['GITHUB_REPOSITORY'] # Fetch optional environment variables with default values commit_message = os.getenv( 'COMMIT_MESSAGE', "Auto-committed changes by create-pull-request action") title = os.getenv( 'PULL_REQUEST_TITLE', "Auto-generated by create-pull-request action") body = os.getenv( 'PULL_REQUEST_BODY', "Auto-generated pull request by " "[create-pull-request](https://github.com/peter-evans/create-pull-request) GitHub Action") # Fetch optional environment variables with no default values pull_request_labels = os.environ.get('PULL_REQUEST_LABELS') pull_request_assignees = os.environ.get('PULL_REQUEST_ASSIGNEES') pull_request_milestone = os.environ.get('PULL_REQUEST_MILESTONE') pull_request_reviewers = os.environ.get('PULL_REQUEST_REVIEWERS') pull_request_team_reviewers = os.environ.get('PULL_REQUEST_TEAM_REVIEWERS') # Get the HEAD committer's email and name author_email, author_name = get_head_author(event_name, event_data) # Set git configuration set_git_config(repo.git, author_email, author_name) # Update URL for the 'origin' remote set_git_remote_url(repo.git, github_token, github_repository) # Push the local changes to the remote branch print("Pushing changes.") push_result = push_changes(repo.git, remote_exists, branch, commit_message) print(push_result) # If the remote existed then a PR likely exists and we can finish here if remote_exists: print("Updated pull request branch %s." % branch) sys.exit() # Create the pull request print("Creating a request to pull %s into %s." % (branch, base)) github_repo = Github(github_token).get_repo(github_repository) pull_request = github_repo.create_pull( title=title, body=body, base=base, head=branch) print("Created pull request %d." % pull_request.number) os.system('echo ::set-env name=PULL_REQUEST_NUMBER::%d' % pull_request.number) # Set labels, assignees and milestone if pull_request_labels is not None: print("Applying labels") pull_request.as_issue().edit(labels=cs_string_to_list(pull_request_labels)) if pull_request_assignees is not None: print("Applying assignees") pull_request.as_issue().edit(assignees=cs_string_to_list(pull_request_assignees)) if pull_request_milestone is not None: print("Applying milestone") milestone = github_repo.get_milestone(int(pull_request_milestone)) pull_request.as_issue().edit(milestone=milestone) # Set pull request reviewers and team reviewers if pull_request_reviewers is not None: print("Requesting reviewers") pull_request.create_review_request(reviewers=cs_string_to_list(pull_request_reviewers)) if pull_request_team_reviewers is not None: print("Requesting team reviewers") pull_request.create_review_request(team_reviewers=cs_string_to_list(pull_request_team_reviewers)) # Get the JSON event data event_name = os.environ['GITHUB_EVENT_NAME'] event_data = get_github_event(os.environ['GITHUB_EVENT_PATH']) # Check if this event should be ignored skip_ignore_event = bool(os.environ.get('SKIP_IGNORE')) if skip_ignore_event or not ignore_event(event_name, event_data): # Set the repo to the working directory repo = Repo(os.getcwd()) # Fetch/Set the branch name branch = os.getenv('PULL_REQUEST_BRANCH', 'create-pull-request/patch') # Set the current branch as the target base branch base = os.environ['GITHUB_REF'][11:] # Skip if the current branch is a PR branch created by this action if base.startswith(branch): print("Branch '%s' was created by this action. Skipping." % base) sys.exit() # Fetch an optional environment variable to determine the branch suffix branch_suffix = os.getenv('BRANCH_SUFFIX', 'short-commit-hash') if branch_suffix == "short-commit-hash": # Suffix with the short SHA1 hash branch = "%s-%s" % (branch, get_head_short_sha1(repo)) elif branch_suffix == "timestamp": # Suffix with the current timestamp branch = "%s-%s" % (branch, int(time.time())) elif branch_suffix == "random": # Suffix with the current timestamp branch = "%s-%s" % (branch, get_random_suffix()) # Check if the remote branch exists remote_exists = remote_branch_exists(repo, branch) # If using short commit hash prefixes, check if a remote # branch already exists for this HEAD commit if branch_suffix == 'short-commit-hash' and remote_exists: print("Pull request branch '%s' already exists for this commit. Skipping." % branch) sys.exit() # Check if there are changes to pull request if repo.is_dirty() or len(repo.untracked_files) > 0: print("Repository has modified or untracked files.") process_event(event_name, event_data, repo, branch, base, remote_exists) else: print("Repository has no modified or untracked files. Skipping.")