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