#!/bin/bash version=0.10 program=${0##*/} progdir=${0%/*} if [ "$progdir" = "$program" ]; then progdir="."; fi # ---------------------------------------------------------------------- function usage() { cat < COPYRIGHT Copyright 2007 The IETF Trust. EOF } # ---------------------------------------------------------------------- function die() { echo -e "\n$program: error: $*" > /dev/stderr exit 1 } function say() { if [ -n "$VERBOSE" ]; then echo -e "$*"; fi } function note() { if [ -n "$VERBOSE" ]; then echo -e "\n$*"; fi } # ---------------------------------------------------------------------- function version() { echo -e "$program $version" } # ---------------------------------------------------------------------- trap 'echo "$program($LINENO): Command failed with error code $? ([$$] $0 $*)"; exit 1' ERR # ---------------------------------------------------------------------- # Option parsing # Options shortopts=hnvV longopts=help,dry-run,verbose,version # Default values MSG="" PROJ=ietfdb VERFILE=ietf/__init__.py SETTINGS=ietf/settings.py do="" if [ "$(uname)" = "Linux" ]; then args=$(getopt -o "$shortopts" --long "$longopts" -n '$program' -- $SV "$@") if [ $? != 0 ] ; then die "Terminating..." >&2 ; exit 1 ; fi eval set -- "$args" sed="sed -r" else # Darwin, BSDs args=$(getopt -o$shortopts $SV $*) if [ $? != 0 ] ; then die "Terminating..." >&2 ; exit 1 ; fi set -- $args sed="sed -E" fi while true ; do case "$1" in -h| --help) usage; exit;; # Show this help, then exit -m| --message) MSG=$2; shift;; # Specify a commit message -n| --dry-run) do="echo ==>";; # Show what would be done -v| --verbose) VERBOSE=1;; # Be more talkative -V| --version) version; exit;; # Show program version, then exit --) shift; break;; *) die "Internal error, inconsistent option specification: '$1'";; esac shift done # ---------------------------------------------------------------------- # The program itself ARGMIN=1 if [ $# -lt $ARGMIN ]; then usage die "$# arguments found, $ARGMIN required" fi VER=$1 REPO=$(svn info | grep "^Repository Root:" | awk '{ print $3 }') RDATE=$(svn info | grep "^Last Changed Date:" | awk '{ print $4 "T" $5 $6 }') RURL=$(svn info | grep "^URL:" | awk '{ print $2 }') RDIR=${RURL#$REPO} DIR=${RDIR#/} if [ -z "$DIR" ]; then die "Couldn't find anything to release here" elif [ "${DIR%%/*}" = "trunk" ]; then SRC="trunk" elif [ "${DIR%%/*}" = "branch" ]; then tmp=${DIR#*/} # get rid of 'branch/' SRC="branch/${tmp%%/*}" # keep first subdir under branch/ fi note "Releasing from $SRC" note "Locating the root of the working copy ..." while [ "${#DIR}" -gt "${#SRC}" ]; do [ "$DIR" = "$prev" ] && die "Internal error" cd .. #note " now at $PWD" prev=$DIR DIR=${DIR%/*} done if [ "$DIR" != "$SRC" ]; then die "Couldn't find the root of your '$SRC' working copy" fi say " $DIR" REPO=${REPO%/} # remove trailing slash SRC=${SRC#/} # remove leading slash MAJOR=${VER%%.*} REST=${VER#*.} MINOR=${REST%%.*} MAINT=${REST#*.} VER="$(printf %d.%d.%d $MAJOR $MINOR $MAINT)" NEXT=$(( $MAINT + 1 )) DEV="$(printf %d.%d.%d.dev0 $MAJOR $MINOR $NEXT)" #cd $DIR ?? note "Collecting static files ..." $do ietf/manage.py collectstatic --noinput --ignore=bower.json --ignore='README.*' $do svn commit static/lib/ -m "Updated static files under static/lib/" # note "Checking that there's a recent test-crawler log" # touch -d $RDATE .svn/.latest-commit # TCLOG=$(ls -t ../test-crawl-*.log | head -n 1) # [ $TCLOG -nt .svn/.latest-commit ] || die "Looked for ../test-crawl-*.log, but didn't find one newer than the latest repository commit ($RDATE)" note "Checking that changelog information is available ..." changes=$( sed -n "/^$PROJ ($VER.*)/,/^ -- /p" changelog ) [ "$changes" ] || die "No changelog information for $VER found" #note "$changes" note "Upgrading the python library modules before checking migrations and running tests ..." pip freeze > minimum-requirements.txt pip install --upgrade -r requirements.txt note "Checking that we don't have any model changes that haven't been captured in migrations ..." ietf/manage.py makemigrations | tee /dev/stderr | grep -q "^No changes detected$" || die "Model changes without migrations found." note "Running the tests suite and writing release coverage data ..." $do ietf/manage.py test --settings=ietf.settings_releasetest --save-version-coverage=$VER note "Committing the release coverage data ..." $do svn commit release-coverage.json.gz -m "Code coverage data for release $VER" contributors=$(echo "$changes" | sed 's/\.[ \t\n]/ /'| tr -c "a-z0-9.@-" "\n" | sort | uniq | grep '@' | sed -r -e 's/^\.+//' -e 's/\.+$//' -e 's/^/-c /' || true) note "Setting the current time on the release notes in the changelog file ..." $do sed -r -i -e "1,/^ -- /s/([A-Za-z-]+ <[a-z0-9.-]+@[a-z0-9.-]+> ).*$/\1$(TZ=UTC date +'%d %b %Y %H:%M:%S %z')/" changelog say " $(grep -m1 "^ -- " changelog)" note "Verifying that version $VER doesn't already exist ..." $do svn info $REPO/tags/$VER 2>&1 | $do egrep -q "(Not a valid URL|URL .* non-existent)" || die "The tag '$VER' already exists (or there was an error testing for it)." say " Ok" note "Committing the changelog ..." $do svn commit changelog -m "Changelog entry for $VER" note "Verifying there's no uncommitted changes ..." $do svn st | grep "^[AMGRD] " && die "There seems to be uncommitted changes in this working copy" note "Updating the version info in $VERFILE and making sure'\$Rev\$' is Ok ..." $do sed -i -r -e "/^__version__/s/\"[.0-9]+(dev[0-9]+)?\"/\"$VER\"/" \ -e "/^__rev__/s/\".*\"/\"\$Rev:\$\"/" \ $VERFILE note "Updating the deployment settings in settings.py" $do sed -i -r -e 's/^DEBUG *= *.*$/DEBUG = False/' \ -e "s/^SERVER_MODE *= *.*\$/SERVER_MODE = 'production'/" \ $SETTINGS note "Committing version information for version $VER: $MSG ..." $do svn commit $VERFILE $SETTINGS -m "Set version info to release version $VER before branching. $MSG" note "Creating new tag 'tags/$VER' from $SRC ..." $do svn cp $REPO/$SRC $REPO/tags/$VER -m "Creating new tag 'tags/$VER' from $SRC" note "Updating version and revision info to indicate that the source and branch aren't releases ..." $do sed -i -r -e "/^__version__/s/\"[0-9.]*\"/\"$DEV\"/" \ -e "/^__rev__/s/\"\\\$Rev: (.*) \\\$\"/\"\$Rev:\$ (dev) Latest release: Rev. \1 \"/" \ $VERFILE note "Updating the deployment settings in settings.py to development mode ..." $do sed -i -r -e 's/^DEBUG *= *.*$/DEBUG = True/' \ -e "s/^SERVER_MODE *= *.*\$/SERVER_MODE = 'development'/" \ $SETTINGS note "Committing the updated version and deployment settings ..." $do svn commit $VERFILE $SETTINGS -m "Set version info and settings back to development mode" note "Creating new tag 'tags/dev/$DEV' from $SRC ..." $do svn cp $REPO/$SRC $REPO/tags/dev/$DEV -m "Creating new tag 'tags/dev/$DEV' from $SRC" $do svn update -q echo " Hi, This is an automatic notification about a new datatracker release, v$VER, generated when running the mkrelease script. Release notes: $changes The new version is available for installation through SVN checkout, with 'svn checkout https://svn.tools.ietf.org/svn/tools/$PROJ/tags/$VER' For development, check out the new development version instead: 'svn checkout https://svn.tools.ietf.org/svn/tools/$PROJ/tags/dev/$DEV' Regards, Henrik (via the mkrelease script) " > ~/src/db/release-mail-v$VER.txt cat ~/src/db/release-mail-v$VER.txt | $do mail -s "New datatracker release: v$VER" housley@vigilsec.com rjs@nostrum.com henrik@levkowetz.com -c glen@amsl.com -c fenner@fenron.net -c cmorgan@amsl.com -c avezza@amsl.com -c amorris@amsl.com -c smccammon@amsl.com -c kmoreland@amsl.com -c olau@iola.dk -c mlarson@amsl.com $contributors cat ~/src/db/release-mail-v$VER.txt | $do mail -s "New datatracker release: v$VER" iesg@ietf.org wgchairs@ietf.org codesprints@ietf.org -b henrik@levkowetz.com $do toolsfeed control changelog /www/tools.ietf.org/tools/atomfeed.xml $do toolpush /www/tools.ietf.org/tools/atomfeed.xml pip install -r minimum-requirements.txt