mirror of
https://github.com/prometheus/prometheus
synced 2025-01-11 17:19:45 +00:00
0e24f246cc
sync_repo_files: Normalize usage of git_user
154 lines
5.0 KiB
Bash
Executable File
154 lines
5.0 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
# vim: ts=2 et
|
|
# Setting -x is absolutely forbidden as it could leak the GitHub token.
|
|
set -uo pipefail
|
|
|
|
# GITHUB_TOKEN required scope: repo.repo_public
|
|
|
|
git_mail="prometheus-team@googlegroups.com"
|
|
git_user="prombot"
|
|
branch="repo_sync"
|
|
commit_msg="Update common Prometheus files"
|
|
pr_title="Synchronize common files from prometheus/prometheus"
|
|
pr_msg="Propagating changes from prometheus/prometheus default branch."
|
|
orgs="prometheus prometheus-community"
|
|
|
|
GITHUB_TOKEN="${GITHUB_TOKEN:-}"
|
|
if [ -z "${GITHUB_TOKEN}" ]; then
|
|
echo -e "\e[31mGitHub token (GITHUB_TOKEN) not set. Terminating.\e[0m"
|
|
exit 1
|
|
fi
|
|
|
|
# List of files that should be synced.
|
|
SYNC_FILES="CODE_OF_CONDUCT.md LICENSE Makefile.common SECURITY.md"
|
|
|
|
# Go to the root of the repo
|
|
cd "$(git rev-parse --show-cdup)" || exit 1
|
|
|
|
source_dir="$(pwd)"
|
|
|
|
tmp_dir="$(mktemp -d)"
|
|
trap 'rm -rf "${tmp_dir}"' EXIT
|
|
|
|
get_default_branch(){
|
|
local url="https://api.github.com/repos/${1}"
|
|
curl --retry 5 --silent -u "${git_user}:${GITHUB_TOKEN}" "${url}" 2>/dev/null | jq -r .default_branch
|
|
}
|
|
|
|
fetch_repos() {
|
|
local url="https://api.github.com/users/${1}/repos?per_page=100"
|
|
curl --retry 5 --silent -u "${git_user}:${GITHUB_TOKEN}" "${url}" 2>/dev/null |
|
|
jq -r '.[] | select( .name != "prometheus" ) | .name'
|
|
}
|
|
|
|
push_branch() {
|
|
# stdout and stderr are redirected to /dev/null otherwise git-push could leak
|
|
# the token in the logs.
|
|
# Delete the remote branch in case it was merged but not deleted.
|
|
git push --quiet "https://${git_user}:${GITHUB_TOKEN}:@github.com/${1}" \
|
|
":${branch}" 1>/dev/null 2>&1
|
|
git push --quiet \
|
|
"https://${git_user}:${GITHUB_TOKEN}:@github.com/${1}" \
|
|
--set-upstream "${branch}" 1>/dev/null 2>&1
|
|
}
|
|
|
|
post_pull_request() {
|
|
post_template='{"title":"%s","base":"%s","head":"%s","body":"%s"}'
|
|
post_json="$(printf "${post_template}" "${pr_title}" "${2}" "${branch}" "${pr_msg}")"
|
|
curl --show-error --silent --fail \
|
|
-u "${git_user}:${GITHUB_TOKEN}" \
|
|
-d "${post_json}" \
|
|
"https://api.github.com/repos/${1}/pulls"
|
|
}
|
|
|
|
check_license() {
|
|
# Check to see if the input is an Apache license of some kind
|
|
echo "$1" | grep --quiet --no-messages --ignore-case 'Apache License'
|
|
}
|
|
|
|
process_repo() {
|
|
local org_repo="$1"
|
|
echo -e "\e[32mAnalyzing '${org_repo}'\e[0m"
|
|
|
|
default_branch="$(get_default_branch ${1})"
|
|
if [[ -z "${default_branch}" ]]; then
|
|
echo "Can't get the default branch."
|
|
return
|
|
fi
|
|
|
|
local needs_update=()
|
|
for source_file in ${SYNC_FILES}; do
|
|
source_checksum="$(sha256sum "${source_dir}/${source_file}" | cut -d' ' -f1)"
|
|
|
|
target_file="$(curl -s --fail "https://raw.githubusercontent.com/${org_repo}/${default_branch}/${source_file}")"
|
|
if [[ "${source_file}" == 'LICENSE' ]] && ! check_license "${target_file}" ; then
|
|
echo "LICENSE in ${org_repo} is not apache, skipping."
|
|
continue
|
|
fi
|
|
if [[ -z "${target_file}" ]]; then
|
|
echo "${source_file} doesn't exist in ${org_repo}"
|
|
case "${source_file}" in
|
|
CODE_OF_CONDUCT.md | SECURITY.md)
|
|
echo "${source_file} missing in ${org_repo}, force updating."
|
|
needs_update+=("${source_file}")
|
|
;;
|
|
esac
|
|
continue
|
|
fi
|
|
target_checksum="$(echo "${target_file}" | sha256sum | cut -d' ' -f1)"
|
|
if [ "${source_checksum}" == "${target_checksum}" ]; then
|
|
echo "${source_file} is already in sync."
|
|
continue
|
|
fi
|
|
needs_update+=("${source_file}")
|
|
done
|
|
|
|
if [[ "${#needs_update[@]}" -eq 0 ]] ; then
|
|
echo "No files need sync."
|
|
return
|
|
fi
|
|
|
|
# Clone target repo to temporary directory and checkout to new branch
|
|
git clone --quiet "https://github.com/${org_repo}.git" "${tmp_dir}/${org_repo}"
|
|
cd "${tmp_dir}/${org_repo}" || return 1
|
|
git checkout -b "${branch}" || return 1
|
|
|
|
# Update the files in target repo by one from prometheus/prometheus.
|
|
for source_file in ${needs_update[@]}; do
|
|
cp -f "${source_dir}/${source_file}" "./${source_file}"
|
|
done
|
|
|
|
if [ -n "$(git status --porcelain)" ]; then
|
|
git config user.email "${git_mail}"
|
|
git config user.name "${git_user}"
|
|
git add .
|
|
git commit -s -m "${commit_msg}"
|
|
if push_branch "${org_repo}"; then
|
|
post_pull_request "${org_repo}" "${default_branch}"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
for org in ${orgs}; do
|
|
mkdir -p "${tmp_dir}/${org}"
|
|
# Iterate over all repositories in ${org}. The GitHub API can return 100 items
|
|
# at most but it should be enough for us as there are less than 40 repositories
|
|
# currently.
|
|
fetch_repos "${org}" | while read -r repo; do
|
|
# Check if a PR is already opened for the branch.
|
|
prLink=$(curl --show-error --silent \
|
|
-u "${git_user}:${GITHUB_TOKEN}" \
|
|
"https://api.github.com/repos/${org}/${repo}/pulls?head=${repo}:${branch}" | jq '.[0].url')
|
|
if [[ "${prLink}" != "null" ]]; then
|
|
echo "Pull request already opened for branch '${branch}': ${prLink}"
|
|
echo "Either close it or merge it before running this script again!"
|
|
continue
|
|
fi
|
|
|
|
if ! process_repo "${org}/${repo}"; then
|
|
echo "Failed to process '${org}/${repo}'"
|
|
exit 1
|
|
fi
|
|
done
|
|
done
|