cannam@62: #! /usr/bin/env bash cannam@62: cannam@62: set -euo pipefail cannam@62: cannam@62: if [ "$1" != "package" ]; then cannam@62: if (grep -r KJ_DBG c++/src | egrep -v '/debug(-test)?[.]' | grep -v 'See KJ_DBG\.$'); then cannam@62: echo '*** Error: There are instances of KJ_DBG in the code.' >&2 cannam@62: exit 1 cannam@62: fi cannam@62: cannam@62: if (egrep -r 'TODO\((now|soon)\)' *); then cannam@62: echo '*** Error: There are release-blocking TODOs in the code.' >&2 cannam@62: exit 1 cannam@62: fi cannam@62: fi cannam@62: cannam@62: doit() { cannam@62: echo "@@@@ $@" cannam@62: "$@" cannam@62: } cannam@62: cannam@62: get_version() { cannam@62: local VERSION=$(grep AC_INIT c++/configure.ac | sed -e 's/^[^]]*],\[\([^]]*\)].*$/\1/g') cannam@62: if [[ ! "$VERSION" =~ $1 ]]; then cannam@62: echo "Couldn't parse version: $VERSION" >&2 cannam@62: exit 1 cannam@62: fi cannam@62: echo "$VERSION" cannam@62: } cannam@62: cannam@62: get_release_version() { cannam@62: get_version '^[0-9]+[.][0-9]+[.][0-9]+(-rc[0-9]+|[.][0-9]+)?$' cannam@62: } cannam@62: cannam@62: update_version() { cannam@62: local OLD=$1 cannam@62: local NEW=$2 cannam@62: local BRANCH_DESC=$3 cannam@62: cannam@62: local OLD_REGEX=${OLD//./[.]} cannam@62: doit sed -i -e "s/$OLD_REGEX/$NEW/g" c++/configure.ac cannam@62: doit sed -i -e "s/set(VERSION.*)/set(VERSION $NEW)/g" c++/CMakeLists.txt cannam@62: cannam@62: local NEW_NOTAG=${NEW%%-*} cannam@62: declare -a NEW_ARR=(${NEW_NOTAG//./ }) cannam@62: doit sed -i -re " cannam@62: s/^#define CAPNP_VERSION_MAJOR [0-9]+\$/#define CAPNP_VERSION_MAJOR ${NEW_ARR[0]}/g; cannam@62: s/^#define CAPNP_VERSION_MINOR [0-9]+\$/#define CAPNP_VERSION_MINOR ${NEW_ARR[1]}/g; cannam@62: s/^#define CAPNP_VERSION_MICRO [0-9]+\$/#define CAPNP_VERSION_MICRO ${NEW_ARR[2]:-0}/g" \ cannam@62: c++/src/capnp/common.h cannam@62: cannam@62: local NEW_COMBINED=$(( ${NEW_ARR[0]} * 1000000 + ${NEW_ARR[1]} * 1000 + ${NEW_ARR[2]:-0 })) cannam@62: doit sed -i -re "s/^#if CAPNP_VERSION != [0-9]*\$/#if CAPNP_VERSION != $NEW_COMBINED/g" \ cannam@62: c++/src/*/*.capnp.h c++/src/*/*/*.capnp.h cannam@62: cannam@62: doit git commit -a -m "Set $BRANCH_DESC version to $NEW." cannam@62: } cannam@62: cannam@62: build_packages() { cannam@62: local VERSION=$1 cannam@62: local VERSION_BASE=${VERSION%%-*} cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Building C++ package..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: # make dist tarball and move into .. cannam@62: cd c++ cannam@62: doit autoreconf -i cannam@62: doit ./configure cannam@62: doit make -j6 distcheck cannam@62: doit mv capnproto-c++-$VERSION.tar.gz .. cannam@62: doit make distclean cannam@62: cannam@62: # build windows executables cannam@62: doit ./configure --host=i686-w64-mingw32 --with-external-capnp \ cannam@62: --disable-shared CXXFLAGS='-static-libgcc -static-libstdc++' cannam@62: doit make -j6 capnp.exe capnpc-c++.exe capnpc-capnp.exe cannam@62: doit i686-w64-mingw32-strip capnp.exe capnpc-c++.exe capnpc-capnp.exe cannam@62: doit mkdir capnproto-tools-win32-$VERSION cannam@62: doit mv capnp.exe capnpc-c++.exe capnpc-capnp.exe capnproto-tools-win32-$VERSION cannam@62: doit make maintainer-clean cannam@62: cannam@62: # repack dist tarball and win32 tools into win32 zip, with DOS line endings cannam@62: doit tar zxf ../capnproto-c++-$VERSION.tar.gz cannam@62: find capnproto-c++-$VERSION -name '*.c++' -o -name '*.h' -o -name '*.capnp' -o -name '*.md' -o -name '*.txt' | grep -v testdata | doit xargs unix2dos cannam@62: doit zip -r ../capnproto-c++-win32-$VERSION.zip capnproto-c++-$VERSION capnproto-tools-win32-$VERSION cannam@62: cannam@62: rm -rf capnproto-c++-$VERSION capnproto-tools-win32-$VERSION cannam@62: cd .. cannam@62: } cannam@62: cannam@62: cherry_pick() { cannam@62: shift cannam@62: if [ $# -gt 0 ]; then cannam@62: echo "=========================================================================" cannam@62: echo "Cherry-picking fixes" cannam@62: echo "=========================================================================" cannam@62: doit git cherry-pick "$@" cannam@62: fi cannam@62: } cannam@62: cannam@62: done_banner() { cannam@62: local VERSION=$1 cannam@62: local PUSH=$2 cannam@62: local FINAL=$3 cannam@62: echo "=========================================================================" cannam@62: echo "Done" cannam@62: echo "=========================================================================" cannam@62: echo "Ready to release:" cannam@62: echo " capnproto-c++-$VERSION.tar.gz" cannam@62: echo " capnproto-c++-win32-$VERSION.zip" cannam@62: echo "Don't forget to push changes:" cannam@62: echo " git push origin $PUSH" cannam@62: cannam@62: read -s -n 1 -p "Shall I push to git and upload to capnproto.org now? (y/N)" YESNO cannam@62: cannam@62: echo cannam@62: case "$YESNO" in cannam@62: y | Y ) cannam@62: doit git push origin $PUSH cannam@62: doit gce-ss copy-files capnproto-c++-$VERSION.tar.gz capnproto-c++-win32-$VERSION.zip \ cannam@62: fe:/var/www/capnproto.org cannam@62: cannam@62: if [ "$FINAL" = yes ]; then cannam@62: echo "=========================================================================" cannam@62: echo "Publishing docs" cannam@62: echo "=========================================================================" cannam@62: cd doc cannam@62: doit ./push-site.sh cannam@62: cd .. cannam@62: echo "=========================================================================" cannam@62: echo "Really done" cannam@62: echo "=========================================================================" cannam@62: fi cannam@62: cannam@62: echo "Release is available at:" cannam@62: echo " http://capnproto.org/capnproto-c++-$VERSION.tar.gz" cannam@62: ;; cannam@62: * ) cannam@62: echo "OK, do it yourself then." cannam@62: ;; cannam@62: esac cannam@62: } cannam@62: cannam@62: BRANCH=$(git rev-parse --abbrev-ref HEAD) cannam@62: cannam@62: case "${1-}:$BRANCH" in cannam@62: # ====================================================================================== cannam@62: candidate:master ) cannam@62: echo "New major release." cannam@62: cannam@62: if [ $# -gt 1 ]; then cannam@62: echo "Cannot cherry-pick when starting from master. Do it yourself." >&2 cannam@62: exit 1 cannam@62: fi cannam@62: cannam@62: HEAD_VERSION=$(get_version '^[0-9]+[.][0-9]+-dev$') cannam@62: RELEASE_VERSION=${HEAD_VERSION%%-dev}.0 cannam@62: cannam@62: echo "Version: $RELEASE_VERSION" cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Creating release branch..." cannam@62: echo "=========================================================================" cannam@62: doit git checkout -b release-$RELEASE_VERSION cannam@62: cannam@62: update_version $HEAD_VERSION $RELEASE_VERSION-rc1 "release branch" cannam@62: cannam@62: build_packages $RELEASE_VERSION-rc1 cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Updating version in master branch..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: doit git checkout master cannam@62: declare -a VERSION_ARR=(${RELEASE_VERSION//./ }) cannam@62: NEXT_VERSION=${VERSION_ARR[0]}.$((VERSION_ARR[1] + 1)) cannam@62: cannam@62: update_version $HEAD_VERSION $NEXT_VERSION-dev "mainlaine" cannam@62: cannam@62: done_banner $RELEASE_VERSION-rc1 "master release-$RELEASE_VERSION" no cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: candidate:release-* ) cannam@62: echo "New release candidate." cannam@62: OLD_VERSION=$(get_release_version) cannam@62: cannam@62: if [[ $OLD_VERSION == *-rc* ]]; then cannam@62: # New release candidate for existing release. cannam@62: cannam@62: RC=${OLD_VERSION##*-rc} cannam@62: BRANCH_VERSION=${OLD_VERSION%%-rc*} cannam@62: RC_VERSION=$BRANCH_VERSION-rc$(( RC + 1 )) cannam@62: cannam@62: echo "Version: $RC_VERSION" cannam@62: else cannam@62: # New micro release. cannam@62: cannam@62: declare -a VERSION_ARR=(${OLD_VERSION//./ }) cannam@62: BRANCH_VERSION=${VERSION_ARR[0]}.${VERSION_ARR[1]}.$((VERSION_ARR[2] + 1)) cannam@62: cannam@62: RC_VERSION=$BRANCH_VERSION-rc1 cannam@62: echo "Version: $RC_VERSION" cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Creating new release branch..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: doit git checkout -b release-$BRANCH_VERSION cannam@62: fi cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Updating version number to $RC_VERSION..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: update_version $OLD_VERSION $RC_VERSION "release branch" cannam@62: cannam@62: cherry_pick "$@" cannam@62: cannam@62: build_packages $RC_VERSION cannam@62: cannam@62: done_banner $RC_VERSION release-$BRANCH_VERSION no cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: final:release-* ) cannam@62: echo "Final release." cannam@62: OLD_VERSION=$(get_release_version) cannam@62: cannam@62: if [[ $OLD_VERSION != *-rc* ]]; then cannam@62: echo "Current version is already a final release. You need to create a new candidate first." >&2 cannam@62: exit 1 cannam@62: fi cannam@62: cannam@62: if [ $# -gt 1 ]; then cannam@62: echo "Cannot cherry-pick into final release. Make another candidate." >&2 cannam@62: exit 1 cannam@62: fi cannam@62: cannam@62: RC=${OLD_VERSION##*-rc} cannam@62: NEW_VERSION=${OLD_VERSION%%-rc*} cannam@62: cannam@62: echo "Version: $NEW_VERSION" cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Updating version number to $NEW_VERSION..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: doit sed -i -re "s/capnproto-c[+][+]-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-c++-$NEW_VERSION/g" doc/install.md cannam@62: doit sed -i -re "s/capnproto-c[+][+]-win32-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-c++-win32-$NEW_VERSION/g" doc/install.md cannam@62: doit sed -i -re "s/capnproto-tools-win32-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-tools-win32-$NEW_VERSION/g" doc/install.md cannam@62: update_version $OLD_VERSION $NEW_VERSION "release branch" cannam@62: cannam@62: doit git tag v$NEW_VERSION cannam@62: cannam@62: build_packages $NEW_VERSION cannam@62: cannam@62: done_banner $NEW_VERSION "v$NEW_VERSION release-$NEW_VERSION" yes cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: security:release-* ) cannam@62: echo "Security release." cannam@62: OLD_VERSION=$(get_release_version) cannam@62: cannam@62: if [[ $OLD_VERSION == *-rc* ]]; then cannam@62: echo "Security releases don't have candidates." >&2 cannam@62: exit 1 cannam@62: fi cannam@62: cannam@62: declare -a VERSION_ARR=(${OLD_VERSION//./ } 0) cannam@62: NEW_VERSION=${VERSION_ARR[0]}.${VERSION_ARR[1]}.${VERSION_ARR[2]}.$((VERSION_ARR[3] + 1)) cannam@62: cannam@62: echo "Version: $NEW_VERSION" cannam@62: cannam@62: echo "=========================================================================" cannam@62: echo "Updating version number to $NEW_VERSION..." cannam@62: echo "=========================================================================" cannam@62: cannam@62: doit sed -i -re "s/capnproto-c[+][+]-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-c++-$NEW_VERSION/g" doc/install.md cannam@62: doit sed -i -re "s/capnproto-c[+][+]-win32-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-c++-win32-$NEW_VERSION/g" doc/install.md cannam@62: doit sed -i -re "s/capnproto-tools-win32-[0-9]+[.][0-9]+[.][0-9]+([.][0-9]+)?\>/capnproto-tools-win32-$NEW_VERSION/g" doc/install.md cannam@62: update_version $OLD_VERSION $NEW_VERSION "release branch" cannam@62: cannam@62: cherry_pick "$@" cannam@62: cannam@62: doit git tag v$NEW_VERSION cannam@62: cannam@62: build_packages $NEW_VERSION cannam@62: cannam@62: done_banner $NEW_VERSION "v$NEW_VERSION release-$NEW_VERSION" yes cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: retry:release-* ) cannam@62: echo "Retrying release." cannam@62: OLD_VERSION=$(get_release_version) cannam@62: echo "Version: $OLD_VERSION" cannam@62: cannam@62: if [[ $OLD_VERSION == *-rc* ]]; then cannam@62: # We can add more cherry-picks when retrying a candidate. cannam@62: cherry_pick "$@" cannam@62: else cannam@62: if [ $# -gt 1 ]; then cannam@62: echo "Cannot cherry-pick into final release. Make another candidate." >&2 cannam@62: exit 1 cannam@62: fi cannam@62: fi cannam@62: cannam@62: OLD_VERSION=$(get_release_version) cannam@62: build_packages $OLD_VERSION cannam@62: cannam@62: if [[ $OLD_VERSION == *-rc* ]]; then cannam@62: BRANCH_VERSION=${OLD_VERSION%%-rc*} cannam@62: done_banner $OLD_VERSION release-$BRANCH_VERSION no cannam@62: else cannam@62: doit git tag v$OLD_VERSION cannam@62: done_banner $OLD_VERSION "v$OLD_VERSION release-$OLD_VERSION" no cannam@62: fi cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: package:* ) cannam@62: echo "Just building a package." cannam@62: build_packages $(get_version '.*') cannam@62: ;; cannam@62: cannam@62: # ====================================================================================== cannam@62: *:master ) cannam@62: echo "Invalid command for mainline branch. Only command is 'candidate'." >&2 cannam@62: exit 1 cannam@62: ;; cannam@62: cannam@62: *:release-* ) cannam@62: echo "Invalid command for release branch. Commands are 'candidate', 'final', and 'retry'." >&2 cannam@62: exit 1 cannam@62: ;; cannam@62: cannam@62: * ) cannam@62: echo "Not a master or release branch." >&2 cannam@62: exit 1 cannam@62: ;; cannam@62: esac