create-pull-request/create-pull-request.py

260 lines
8.7 KiB
Python
Raw Normal View History

2019-07-18 11:37:30 +02:00
#!/usr/bin/env python3
''' Create Pull Request '''
2019-07-16 12:58:27 +02:00
import json
import os
import random
import string
import sys
2019-09-24 13:00:49 +02:00
import time
2019-07-16 12:58:27 +02:00
from git import Repo
from github import Github
2019-11-04 03:00:38 +01:00
from github import GithubException
2019-07-16 12:58:27 +02:00
def get_github_event(github_event_path):
with open(github_event_path) as f:
github_event = json.load(f)
2019-09-24 13:00:49 +02:00
if bool(os.environ.get('DEBUG_EVENT')):
print(os.environ['GITHUB_EVENT_NAME'])
2019-07-16 12:58:27 +02:00
print(json.dumps(github_event, sort_keys=True, indent=2))
return github_event
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):
2019-07-16 12:58:27 +02:00
for ref in repo.remotes.origin.refs:
if ref.name == ("origin/%s" % branch):
return True
return False
def get_author_default(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']
2019-07-16 12:58:27 +02:00
return email, name
def set_git_config(git, email, name):
git.config('--global', 'user.email', '"%s"' % email)
git.config('--global', 'user.name', '"%s"' % name)
2019-08-13 11:14:43 +02:00
def set_git_remote_url(git, token, github_repository):
2019-10-13 18:39:53 +02:00
git.remote(
'set-url', 'origin', "https://x-access-token:%s@github.com/%s" %
(token, github_repository))
2019-08-13 11:14:43 +02:00
def checkout_branch(git, remote_exists, branch):
if remote_exists:
git.stash('--include-untracked')
git.checkout(branch)
try:
git.stash('pop')
2019-10-13 18:39:53 +02:00
except BaseException:
git.checkout('--theirs', '.')
git.reset()
else:
git.checkout('HEAD', b=branch)
def push_changes(git, branch, commit_message):
git.add('-A')
git.commit(m=commit_message)
return git.push('-f', '--set-upstream', 'origin', branch)
2019-07-16 12:58:27 +02:00
2019-09-26 10:48:53 +02:00
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))
2019-07-16 12:58:27 +02:00
2019-11-04 03:00:38 +01:00
def process_event(github_token, github_repository, repo, branch, base):
2019-09-26 10:48:53 +02:00
# Fetch optional environment variables with default values
2019-07-16 12:58:27 +02:00
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")
2019-09-26 10:48:53 +02:00
# 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')
2019-07-16 12:58:27 +02:00
# Push the local changes to the remote branch
print("Pushing changes.")
push_result = push_changes(repo.git, branch, commit_message)
print(push_result)
2019-07-16 12:58:27 +02:00
# Create the pull request
2019-09-26 10:48:53 +02:00
github_repo = Github(github_token).get_repo(github_repository)
2019-11-04 03:00:38 +01:00
try:
pull_request = github_repo.create_pull(
title=title,
body=body,
base=base,
head=branch)
print("Created pull request #%d (%s => %s)" %
(pull_request.number, branch, base))
except GithubException as e:
if e.status == 422:
pull_request = github_repo.get_pulls(
state='open',
base=base,
head=branch)[1]
print("Updated pull request #%d (%s => %s)" %
(pull_request.number, branch, base))
else:
print(str(e))
sys.exit(1)
# Set the output variable
2019-10-13 18:39:53 +02:00
os.system(
'echo ::set-env name=PULL_REQUEST_NUMBER::%d' %
pull_request.number)
2019-07-16 12:58:27 +02:00
2019-09-26 10:48:53 +02:00
# 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)
2019-11-04 03:00:38 +01:00
# Set pull request reviewers
2019-09-26 10:48:53 +02:00
if pull_request_reviewers is not None:
print("Requesting reviewers")
2019-11-04 03:00:38 +01:00
try:
pull_request.create_review_request(
reviewers=cs_string_to_list(pull_request_reviewers))
except GithubException as e:
# Likely caused by "Review cannot be requested from pull request author."
if e.status == 422:
print("Requesting reviewers failed - %s" % e.data["message"])
# Set pull request team reviewers
2019-09-26 10:48:53 +02:00
if pull_request_team_reviewers is not None:
print("Requesting team reviewers")
2019-10-13 18:39:53 +02:00
pull_request.create_review_request(
team_reviewers=cs_string_to_list(pull_request_team_reviewers))
2019-09-26 10:48:53 +02:00
2019-07-16 12:58:27 +02:00
2019-10-30 07:27:54 +01:00
# Fetch environment variables
github_token = os.environ['GITHUB_TOKEN']
github_repository = os.environ['GITHUB_REPOSITORY']
github_ref = os.environ['GITHUB_REF']
event_name = os.environ['GITHUB_EVENT_NAME']
2019-10-30 07:27:54 +01:00
# Get the JSON event data
event_data = get_github_event(os.environ['GITHUB_EVENT_PATH'])
2019-08-13 11:14:43 +02:00
2019-10-30 07:27:54 +01:00
# Set the repo to the working directory
repo = Repo(os.getcwd())
# Get the default for author email and name
author_email, author_name = get_author_default(event_name, event_data)
# Set commit author overrides
author_email = os.getenv('COMMIT_AUTHOR_EMAIL', author_email)
author_name = os.getenv('COMMIT_AUTHOR_NAME', author_name)
# 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)
# Fetch/Set the branch name
branch_prefix = os.getenv(
'PULL_REQUEST_BRANCH',
'create-pull-request/patch')
# Fetch an optional base branch override
base_override = os.environ.get('PULL_REQUEST_BASE')
# Set the base branch
if base_override is not None:
base = base_override
checkout_branch(repo.git, True, base)
elif github_ref.startswith('refs/pull/'):
# Switch to the merging branch instead of the merge commit
base = os.environ['GITHUB_HEAD_REF']
repo.git.checkout(base)
else:
base = github_ref[11:]
# Skip if the current branch is a PR branch created by this action.
# This may occur when using a PAT instead of GITHUB_TOKEN.
if base.startswith(branch_prefix):
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_prefix, get_head_short_sha1(repo))
elif branch_suffix == "timestamp":
# Suffix with the current timestamp
branch = "%s-%s" % (branch_prefix, int(time.time()))
elif branch_suffix == "random":
# Suffix with the current timestamp
branch = "%s-%s" % (branch_prefix, get_random_suffix())
elif branch_suffix == "none":
# Fixed branch name
branch = branch_prefix
else:
print(
"Branch suffix '%s' is not a valid value." %
branch_suffix)
sys.exit(1)
# Check if the remote branch exists
remote_exists = remote_branch_exists(repo, branch)
if remote_exists:
if branch_suffix == 'short-commit-hash':
# A remote branch already exists for the HEAD commit
print(
"Pull request branch '%s' already exists for this commit. Skipping." %
branch)
sys.exit()
2019-10-30 07:27:54 +01:00
elif branch_suffix in ['timestamp', 'random']:
# Generated branch name clash with an existing branch
2019-10-22 03:55:33 +02:00
print(
2019-10-30 07:27:54 +01:00
"Pull request branch '%s' already exists. Please re-run." %
branch)
2019-10-22 03:55:33 +02:00
sys.exit(1)
2019-10-30 07:27:54 +01:00
# Checkout branch
checkout_branch(repo.git, remote_exists, branch)
# 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(
github_token,
github_repository,
repo,
branch,
2019-11-04 03:00:38 +01:00
base)
2019-10-30 07:27:54 +01:00
else:
print("Repository has no modified or untracked files. Skipping.")