mirror of
https://github.com/PokeAPI/pokeapi
synced 2024-11-10 06:04:18 +00:00
338 lines
13 KiB
Bash
338 lines
13 KiB
Bash
#!/bin/bash
|
|
# Executed when the master/staging branch of PokeAPI/pokeapi gets updated
|
|
# Runs in CircleCI
|
|
# Generates new data using the latest changes of PokeAPI/pokeapi in order to open a Pull Request towards PokeAPI/api-data
|
|
|
|
set -o pipefail
|
|
|
|
org='PokeAPI'
|
|
data_repo='api-data'
|
|
engine_repo='pokeapi'
|
|
branch_name='staging'
|
|
username='pokeapi-machine-user'
|
|
email='pokeapi.co@gmail.com'
|
|
staging_environment_url='https://staging.pokeapi.co/api/v2/'
|
|
production_environment_url='https://pokeapi.co/api/v2/'
|
|
data_repo_url='https://github.com/PokeAPI/api-data'
|
|
engine_circleci_status_url='https://app.circleci.com/pipelines/github/PokeAPI/pokeapi'
|
|
deploy_circleci_status_url='https://app.circleci.com/pipelines/github/PokeAPI/deploy'
|
|
auth_header="Authorization: token $MACHINE_USER_GITHUB_API_TOKEN"
|
|
|
|
# Exit the script notifying the user about its success
|
|
cleanexit() {
|
|
echo "Exiting"
|
|
echo "$2"
|
|
if [ "$1" = "success" ]; then
|
|
notify_engine_pr "end_success"
|
|
exit 0
|
|
elif [ "$1" = "no-deploy" ]; then
|
|
notify_engine_pr "end_no_deploy"
|
|
exit 0
|
|
elif [ "$1" = "no-new-data" ]; then
|
|
notify_engine_pr "end_no_new_data"
|
|
exit 0
|
|
else
|
|
notify_engine_pr "end_failed"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# Checks if the PR carries a special `no-deploy` label
|
|
# If so it aborts the deployment
|
|
assert_no_deploy_label() {
|
|
engine_repo_pr_number=$(get_invokator_pr_number)
|
|
if [ "$engine_repo_pr_number" != "null" ]; then
|
|
no_deploy_label=$(curl -f -H "$auth_header" -X GET "https://api.github.com/repos/$org/$engine_repo/pulls/$engine_repo_pr_number" | jq --raw-output '.labels | .[] | select(.name == "no-deploy") | .name')
|
|
if [ "$no_deploy_label" == 'no-deploy' ]; then
|
|
cleanexit 'no-deploy' 'No-deploy label is present thus no need to deploy the project'
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Create and use a personal folder
|
|
prepare() {
|
|
mkdir -p ./repositories
|
|
cd repositories || cleanexit 'fail' "Failed to cd"
|
|
}
|
|
|
|
# GraphQL query to retrieve a PR's number based on a commit hash
|
|
pr_associated_with_sha_graphql_query_content() {
|
|
cat <<EOF
|
|
query associatedPRs {
|
|
repository(name: \"pokeapi\", owner: \"PokeAPI\") {
|
|
commit: object(expression: \"$1\") {
|
|
... on Commit {
|
|
associatedPullRequests(first: 1) {
|
|
edges {
|
|
node {
|
|
number
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# Get the PR's numer associated with the last commit
|
|
get_invokator_pr_number_from_graphql() {
|
|
last_commit_sha="$(git rev-parse HEAD)"
|
|
query="$(pr_associated_with_sha_graphql_query_content "$last_commit_sha")"
|
|
query=$(echo $query) # echo strips all IFS characters (newline, space)
|
|
pr_number=$(curl -s -H "Content-Type: application/json" -H "$auth_header" -X POST --data "{\"query\": \"$query\"}" "https://api.github.com/graphql" | jq ".data.repository.commit.associatedPullRequests.edges[0].node.number" )
|
|
echo "$pr_number"
|
|
}
|
|
|
|
# Check and return the number of the Pull Request that started this job
|
|
get_invokator_pr_number() {
|
|
commit_msg_regex="Merge pull request #([0-9]+) from PokeAPI/(.*)"
|
|
last_commit_message=$(git log --oneline --format=%B -n 1 HEAD | head -n 1)
|
|
invokator_pr_number_from_graphql="$(get_invokator_pr_number_from_graphql)"
|
|
if [ "$invokator_pr_number_from_graphql" != 'null' ]; then
|
|
echo "$invokator_pr_number_from_graphql"
|
|
elif [ -n "$CIRCLE_PULL_REQUEST" ]; then
|
|
echo "${CIRCLE_PULL_REQUEST##*/}"
|
|
elif [[ $last_commit_message =~ $commit_msg_regex ]]; then
|
|
echo "${BASH_REMATCH[1]}"
|
|
else
|
|
echo 'null'
|
|
fi
|
|
}
|
|
|
|
# Clone the repository containing the static JSON files
|
|
clone() {
|
|
git clone "https://github.com/${org}/${data_repo}.git" "$data_repo"
|
|
}
|
|
|
|
# Configure git to use the supplied user when committing
|
|
configure_git() {
|
|
git config --global user.name "$username"
|
|
git config --global user.email "$email"
|
|
}
|
|
|
|
pr_input_updater_start() {
|
|
cat <<EOF
|
|
{
|
|
"body": "A [PokeAPI/api-data](${data_repo_url}) refresh has started. In ~45 minutes the staging branch of [PokeAPI/api-data](${data_repo_url}/tree/staging) will be pushed with the new generated data. <br><br> The staging branch will be deployed in our [staging environment]($staging_environment_url) and the entire API will be ready to review. <br><br> A Pull Request ([master](${data_repo_url}/tree/master)<-[staging](${data_repo_url}/tree/staging)) will be also created at [PokeAPI/api-data](${data_repo_url}/pulls) and assigned to the PokeAPI Core team to be reviewed. If approved and merged new data will soon be available worldwide at [pokeapi.co]($production_environment_url)."
|
|
}
|
|
EOF
|
|
}
|
|
|
|
pr_input_updater_end_success() {
|
|
cat <<EOF
|
|
{
|
|
"body": "The updater script has finished its job and has now opened a Pull Request towards [PokeAPI/api-data](${data_repo_url}/pulls) with the updated data. <br><br> The Pull Request can be seen deployed in our [staging environment]($staging_environment_url) when [CircleCI deploy]($deploy_circleci_status_url) will be finished (_check the start time of the last build_)."
|
|
}
|
|
EOF
|
|
}
|
|
|
|
pr_input_updater_end_no_deploy() {
|
|
cat <<EOF
|
|
{
|
|
"body": "This Pull Request won't be deployed since the label \`no-deploy\` was found in its meta-data."
|
|
}
|
|
EOF
|
|
}
|
|
|
|
pr_input_updater_end_failed() {
|
|
cat <<EOF
|
|
{
|
|
"body": "The updater script could not finish its job. Please check [CircleCI's builds]($engine_circleci_status_url) and [logs](${CIRCLE_BUILD_URL})."
|
|
}
|
|
EOF
|
|
}
|
|
|
|
pr_input_updater_end_no_new_data() {
|
|
cat <<EOF
|
|
{
|
|
"body": "The updater script finished its job and the generated data didn't change. The deploy was thus skipped. For further information check [CircleCI's builds]($engine_circleci_status_url) and [logs](${CIRCLE_BUILD_URL})."
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# If the job was started by a Pull Request and not by a cron job, add a comment to notify the users
|
|
notify_engine_pr() {
|
|
local -r allowed_events='start end_failed end_success end_no_deploy end_no_new_data'
|
|
if [[ "$allowed_events" == *"$1"* ]] && [[ "$CIRCLE_BRANCH" == 'master' ]]; then
|
|
engine_repo_pr_number=$(get_invokator_pr_number)
|
|
if [ "$engine_repo_pr_number" != "null" ] && [ -n "$CIRCLE_USERNAME" ]; then
|
|
curl -f -H "$auth_header" -X POST --data "$(pr_input_updater_$1)" "https://api.github.com/repos/$org/$engine_repo/issues/$engine_repo_pr_number/comments"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# Run the updater script (https://github.com/PokeAPI/api-data/blob/master/updater/cmd.bash) which will generate the new pokeapi data and push it to the api-data repository under a new branch
|
|
run_updater() {
|
|
engine_repo_pr_number=$(get_invokator_pr_number)
|
|
cd "$data_repo/updater" || cleanexit 'fail' "Failed to cd"
|
|
# Wait to be sure PokeAPI/pokeapi's master branch has been updated on Github with the lastest merged PR content
|
|
sleep 10
|
|
|
|
# Build the updater image
|
|
docker build -t pokeapi-updater .
|
|
if [ $? -ne 0 ]; then
|
|
cleanexit 'fail' "Failed to build the pokeapi-updater image"
|
|
fi
|
|
|
|
# Run the updater
|
|
docker network create pokeapi
|
|
docker run --privileged --network pokeapi --network-alias docker -e REPO_POKEAPI_CHECKOUT_OBJECT="$CIRCLE_SHA1" -e COMMIT_EMAIL="$email" -e COMMIT_NAME="$username" -e BRANCH_NAME="$branch_name" -e REPO_POKEAPI="https://github.com/$org/$engine_repo.git" -e REPO_DATA="https://$MACHINE_USER_GITHUB_API_TOKEN@github.com/$org/$data_repo.git" -e COMMIT_MESSAGE="[Updater Bot] Regenerate data from https://github.com/$org/$engine_repo/pull/$engine_repo_pr_number" -e COMMIT_AND_PUSH='true' -e RUN_AS='root' pokeapi-updater
|
|
return_code=$?
|
|
if [ "$return_code" -eq 2 ]; then
|
|
cleanexit 'no-new-data' "Generated data is the same as old data, skipping deploy"
|
|
elif [ "$return_code" -ne 0 ]; then
|
|
cleanexit 'fail' "Failed to run the pokeapi-updater container"
|
|
fi
|
|
|
|
cd ../.. || cleanexit 'fail' "Failed to cd"
|
|
}
|
|
|
|
# Check if the updater script has pushed the data to a new branch
|
|
check_remote_branch() {
|
|
# Wait for Github to update origin/${branch_name}
|
|
sleep 10
|
|
|
|
curl -f -H "$auth_header" -X GET "https://api.github.com/repos/$org/$data_repo/branches/$1"
|
|
if [ $? -ne 0 ]; then
|
|
cleanexit 'fail' "The updater script failed to push the new data"
|
|
fi
|
|
}
|
|
|
|
pr_input_alredy_open() {
|
|
cat <<EOF
|
|
{
|
|
"base": "$branch_name"
|
|
}
|
|
EOF
|
|
}
|
|
|
|
check_pr_already_open() {
|
|
data_repo_pr_number=$(curl -H "$auth_header" -X GET --data "$(pr_input_alredy_open)" "https://api.github.com/repos/$org/$data_repo/pulls" | jq '.[0].number')
|
|
echo "$data_repo_pr_number"
|
|
}
|
|
|
|
# Generate the input content that will be sent to Github's API to open a refresh-data PR towards PokeAPI/api-data
|
|
pr_input_content_with_pr_number() {
|
|
cat <<EOF
|
|
{
|
|
"title": "API data update from \`${org}/${engine_repo}#${1}\`",
|
|
"body": "Incoming data generated by [${org}/${engine_repo}](https://github.com/${org}/${engine_repo})'s CircleCI worker since Pull Request [#${1}](https://github.com/${org}/${engine_repo}/pull/${1}) was merged into the \`master\` branch. <br><br> The new data was generated using a copy of [${org}/${engine_repo}](https://github.com/${org}/${engine_repo}/commits/master) repository when the \`HEAD\` was pointing to \`${CIRCLE_SHA1}\`. <br><br> This Pull Request should have been deployed in our [staging environment]($staging_environment_url), check [here]($deploy_circleci_status_url) for the lastest build timestamp and status code.",
|
|
"head": "$branch_name",
|
|
"base": "master",
|
|
"assignees": [
|
|
"Naramsim"
|
|
],
|
|
"labels": [
|
|
"api-data-update"
|
|
]
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# Copy of pr_input_content_with_pr_number
|
|
# run when the script cannot detect the merged PR number that started the CircleCI job
|
|
pr_input_content_without_pr_number() {
|
|
cat <<EOF
|
|
{
|
|
"title": "API data update from \`${org}/${engine_repo}\`",
|
|
"body": "Incoming data generated by [${org}/${engine_repo}](https://github.com/${org}/${engine_repo})'s CircleCI worker. <br><br> The new data was generated using a copy of [${org}/${engine_repo}](https://github.com/${org}/${engine_repo}/commits/master) repository when the \`HEAD\` was pointing to \`${CIRCLE_SHA1}\`. <br><br> This Pull Request should have been deployed in our [staging environment]($staging_environment_url), check [here]($deploy_circleci_status_url) for the lastest build timestamp and status code.",
|
|
"head": "$branch_name",
|
|
"base": "master",
|
|
"assignees": [
|
|
"Naramsim"
|
|
],
|
|
"labels": [
|
|
"api-data-update"
|
|
]
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# Create/update a Pull Request to merge the branch recently pushed by the updater with the master branch
|
|
create_pr() {
|
|
engine_repo_pr_number=$(get_invokator_pr_number)
|
|
data_repo_pr_already_open_number=$(check_pr_already_open)
|
|
if [ "$data_repo_pr_already_open_number" != "null" ]; then
|
|
data_repo_pr_number="$data_repo_pr_already_open_number"
|
|
if [ "$engine_repo_pr_number" != "null" ]; then
|
|
data_repo_pr_number=$(curl -H "$auth_header" -X PATCH --data "$(pr_input_content_with_pr_number "$engine_repo_pr_number")" "https://api.github.com/repos/$org/$data_repo/pulls/$data_repo_pr_already_open_number" | jq '.number')
|
|
else
|
|
data_repo_pr_number=$(curl -H "$auth_header" -X PATCH --data "$(pr_input_content_without_pr_number)" "https://api.github.com/repos/$org/$data_repo/pulls/$data_repo_pr_already_open_number" | jq '.number')
|
|
fi
|
|
else
|
|
if [ "$engine_repo_pr_number" != "null" ]; then
|
|
data_repo_pr_number=$(curl -H "$auth_header" -X POST --data "$(pr_input_content_with_pr_number "$engine_repo_pr_number")" "https://api.github.com/repos/$org/$data_repo/pulls" | jq '.number')
|
|
else
|
|
data_repo_pr_number=$(curl -H "$auth_header" -X POST --data "$(pr_input_content_without_pr_number)" "https://api.github.com/repos/$org/$data_repo/pulls" | jq '.number')
|
|
fi
|
|
fi
|
|
if [[ "$data_repo_pr_number" = "null" ]]; then
|
|
cleanexit 'fail' "Could not create the Pull Request"
|
|
fi
|
|
echo "$data_repo_pr_number"
|
|
}
|
|
|
|
pr_input_assignees_and_labels() {
|
|
cat <<EOF
|
|
{
|
|
"assignees": [
|
|
"Naramsim"
|
|
],
|
|
"labels": [
|
|
"api-data-update"
|
|
]
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# Assign the PR to Naramsim and add a label
|
|
customize_pr() {
|
|
# Wait for Github to open the PR
|
|
sleep 10
|
|
|
|
data_repo_pr_number=$1
|
|
curl -H "$auth_header" -X PATCH --data "$(pr_input_assignees_and_labels)" "https://api.github.com/repos/$org/$data_repo/issues/$data_repo_pr_number"
|
|
if [ $? -ne 0 ]; then
|
|
echo "Could not add Assignees and Labes to the Pull Request"
|
|
fi
|
|
}
|
|
|
|
pr_input_reviewers() {
|
|
cat <<EOF
|
|
{
|
|
"reviewers": [
|
|
"Naramsim"
|
|
],
|
|
"team_reviewers": [
|
|
"core-team"
|
|
]
|
|
}
|
|
EOF
|
|
}
|
|
|
|
# Request the Core team to review the Pull Request
|
|
add_reviewers_to_pr() {
|
|
data_repo_pr_number=$1
|
|
curl -H "$auth_header" -X POST --data "$(pr_input_reviewers)" "https://api.github.com/repos/$org/$data_repo/pulls/$data_repo_pr_number/requested_reviewers"
|
|
if [ $? -ne 0 ]; then
|
|
echo "Could not add Reviewers to the Pull Request"
|
|
fi
|
|
}
|
|
|
|
assert_no_deploy_label
|
|
prepare
|
|
configure_git
|
|
clone
|
|
notify_engine_pr "start"
|
|
run_updater
|
|
check_remote_branch "$branch_name"
|
|
if [ "$CIRCLE_BRANCH" = 'master' ]; then
|
|
sleep 300 # 5 minutes, the time it takes for CircleCI's api-data script to generate the data and for CircleCI's deploy script to deploy it to the staging environment
|
|
check_remote_branch "$branch_name"
|
|
data_repo_pr_number=$(create_pr)
|
|
customize_pr "$data_repo_pr_number"
|
|
add_reviewers_to_pr "$data_repo_pr_number"
|
|
fi
|
|
cleanexit 'success' 'Done'
|