|
|
|
#!/bin/bash
|
|
|
|
#
|
|
|
|
# Usage: Copy this file to your repository's .git/hooks directory. Name it
|
|
|
|
# "pre-push" and set the executable bit.
|
|
|
|
#
|
|
|
|
# This hook aborts git push when the log message of any commits to be pushed
|
|
|
|
# starts with "WIP" (work in progress) or "stash", if the remote ref begins
|
|
|
|
# with "refs/for/". This prevents incomplete/unwanted commits from being
|
|
|
|
# pushed to Gerrit instances.
|
|
|
|
#
|
|
|
|
# When pushing to Gerrit, this hook also runs a Gradle build similar to
|
|
|
|
# Gerrit's presubmit, to help catch errors locally and reduce churn on CLs.
|
|
|
|
#
|
|
|
|
# Remaining comments are copied from .git/hooks/pre-push.sample.
|
|
|
|
#
|
|
|
|
# An example hook script to verify what is about to be pushed. Called by "git
|
|
|
|
# push" after it has checked the remote status, but before anything has been
|
|
|
|
# pushed. If this script exits with a non-zero status nothing will be pushed.
|
|
|
|
#
|
|
|
|
# This hook is called with the following parameters:
|
|
|
|
#
|
|
|
|
# $1 -- Name of the remote to which the push is being done
|
|
|
|
# $2 -- URL to which the push is being done
|
|
|
|
#
|
|
|
|
# If pushing without using a named remote those arguments will be equal.
|
|
|
|
#
|
|
|
|
# Information about the commits which are being pushed is supplied as lines to
|
|
|
|
# the standard input in the form:
|
|
|
|
#
|
|
|
|
# <local ref> <local sha1> <remote ref> <remote sha1>
|
|
|
|
|
|
|
|
remote="$1"
|
|
|
|
url="$2"
|
|
|
|
|
|
|
|
z40=0000000000000000000000000000000000000000
|
|
|
|
|
|
|
|
while read local_ref local_sha remote_ref remote_sha
|
|
|
|
do
|
|
|
|
# Handle delete
|
|
|
|
if [[ "$local_sha" = "$z40" ]]; then
|
|
|
|
echo "$local_ref, $remote_ref"
|
|
|
|
:
|
|
|
|
# Move along if we don't match refs/for/
|
|
|
|
elif [[ ! "$remote_ref" =~ ^refs/for/.+ ]]; then
|
|
|
|
echo "$local_ref, $remote_ref"
|
|
|
|
:
|
|
|
|
else
|
|
|
|
echo "$local_ref, $local_sha, $remote_ref, $remote_sha"
|
|
|
|
if [[ "$remote_sha" = "$z40" ]]; then
|
|
|
|
# New branch, examine all commits
|
|
|
|
branchname="${remote_ref#refs/for/}"
|
|
|
|
|
|
|
|
if [[ "$branchname" =~ % ]]; then
|
|
|
|
# Gerrit allows various push-options by appending them to the remote
|
|
|
|
# using "%<option>=<string>[,<option>=<string>...]". Strip this off.
|
|
|
|
# TODO: '%' is a valid character in branch names, so this could mangle
|
|
|
|
# the branch name. We should find a way to workaround that.
|
|
|
|
echo "NOTE: stripping Gerrit push-options beginning at '%'"
|
|
|
|
branchname=${branchname%%\%*}
|
|
|
|
fi
|
|
|
|
|
|
|
|
if git check-ref-format --allow-onelevel "$branchname"; then
|
|
|
|
range="${remote}/${branchname}..$local_sha"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ -z "$range" ]]; then
|
|
|
|
range="$local_sha"
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
# Update to existing branch, examine new commits
|
|
|
|
range="$remote_sha..$local_sha"
|
|
|
|
fi
|
|
|
|
|
|
|
|
echo "Examining $range"
|
|
|
|
|
|
|
|
# Check for WIP/stash commit
|
|
|
|
commit=`git rev-list -n 1 -i --grep '^WIP' --grep '^stash' "$range"`
|
|
|
|
if [[ -n "$commit" ]]; then
|
|
|
|
echo >&2 "Found WIP/stash commit in $local_ref, not pushing ($commit)"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
|
|
|
|
# At least one ref is going to Gerrit, so we should run additional checks
|
|
|
|
run_checks=true
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
if [[ -n "$run_checks" ]]; then
|
|
|
|
# pre-push usually executes in the repository root, but just to be safe...
|
|
|
|
cd "$(git rev-parse --show-toplevel)"
|
|
|
|
./gradlew --init-script gradle/init.gradle.kts --no-configuration-cache check
|
|
|
|
exit $?
|
|
|
|
fi
|
|
|
|
|
|
|
|
exit 0
|