#! /bin/bash

function Recho() { echo -e "\e[0;31m""$@""\e[0m"; }
function Gecho() { echo -e "\e[0;32m""$@""\e[0m"; }
function Becho() { echo -e "\e[0;34m""$@""\e[0m"; }

function errexit() {
  exec >&2
  Recho "Error: $@"
  exit 1
}

function usage() {
  exec >&2
  cat <<EOF
  Usage:
	$(basename $0) [OPTIONS]

  Options:
	-h,-?,--help	This page.
	-n		Dryrun.

Prepare a new changes file entry preloaded with all commits since the last
changes tag and load it into \$EDITOR (vi). If the version file was changed,
optionally submitt and tag the new changes. Otherwise simply leave the changes
in place.

Don't forgett to push created tags as well: git push --tags

EOF
  exit 1
}

DRYRUN=0

case "$1" in
  -[n]*)
    DRYRUN=1
    ;;
  -[hH?]*)
    usage
    ;;
  --help)
    usage
    ;;
esac


export LC_ALL=""
export LANG="en"

function Edit()
{
  local FILE="$1"
  vi "$FILE"
  sed -i 's/ \+$//' "$FILE"
}

EMAIL="$(git config --get user.email)"

CHANGESFILE=$(ls package/*.changes)
test -f "$CHANGESFILE" || errexit "No changes file '$CHANGESFILE'"

VERSIONFILE="VERSION.cmake"
test -f "$VERSIONFILE" || errexit "No version file '$VERSIONFILE'"

function getversion() {
  cat "$VERSIONFILE" \
  | awk '
  function getnum() {
    gsub("^[^\"]*\"","")
    gsub("\".*$","")
  }
  /^ *SET *\( *VERSION_MAJOR *"[0-9]+" *\)/       {getnum();major=$0}
  /^ *SET *\( *VERSION_MINOR *"[0-9]+" *\)/       {getnum();minor=$0}
  /^ *SET *\( *VERSION_PATCH *"[0-9]+" *\)/       {getnum();patch=$0}
  /^# LAST RELEASED:/ {
      gsub("^.*RELEASED: *","");
      gsub(" +$","");
      lastrelease=$0
  }
  END {
    thisrelease = major"."minor"."patch
    printf "LAST_RELEASE='%s'\n", lastrelease
    printf "THIS_RELEASE='%s'\n", major"."minor"."patch
    printf "THIS_MINOR='%s'\n", minor
    printf "THIS_PATCH='%s'\n", patch
  }
  '
}

function setversion() {
  local KEY="$1"
  local VAL="$2"
  sed -i "s/^ *SET *( *${KEY} .*/SET(${KEY} \"${VAL}\")/" "$VERSIONFILE"
}

function sameVersion() {
  test "$LAST_RELEASE" == "$THIS_RELEASE"
}

function getchanges() {
  git log --no-merges --pretty=format:'- %s' "$LAST_RELEASE"..HEAD | grep -v 'po.tar.bz2\|Translated using Weblate\|weblate/master'
}

function newchangesentry() {
  echo "-------------------------------------------------------------------"
  echo "$(date) - $EMAIL"
  echo ""
  echo "$(getchanges)"
  sameVersion || {
    echo "- version $THIS_RELEASE"
  }
  echo ""
}

function is_fast_forward() {
  git fetch
  test "$(git rev-list HEAD..origin/$(git name-rev --name-only HEAD) --count)" == "0"
}

is_fast_forward || {
  Recho "!!!"
  Recho "!!! Branch is not fast-forward. Pull changes first."
  Recho "!!!"
  exit 7
}

git status --porcelain | grep '^[^ ?]' | grep -v "$VERSIONFILE\|$CHANGESFILE" && {
  Becho "!!! Files other than version and changes are added to the index."
  Becho "!!! Doing dryrun..."
  DRYRUN=1
}
if [ "$DRYRUN" == "1" ]; then
  eval $(getversion)
  newchangesentry
  sameVersion && {
    Becho "!!! Version is unchanged at $LAST_RELEASE."
  }
  exit 0
fi


# check version file
#
while true; do
  # $LAST_RELEASE
  # $THIS_RELEASE
  eval $(getversion)
  sameVersion && {
    newchangesentry
    Becho "!!! Version is unchanged at $LAST_RELEASE."
    read -n 1 -p "$(Gecho "(a)bort, (c)ontinue, (P) patch, (M) minor, (e)dit version [e]: ")" RES
    echo
    case "${RES:-e}" in
      [eE]*)
	Edit $VERSIONFILE
	continue
	;;
      [P])
	setversion VERSION_PATCH $(($THIS_PATCH + 1))
	eval $(getversion)
	continue
	;;
      [M])
	setversion VERSION_MINOR $(($THIS_MINOR + 1))
	setversion VERSION_PATCH 0
	eval $(getversion)
	continue
	;;
      [cC])
	Becho "!!! Leave $VERSIONFILE untouched"
	break
	;;
      *)
	errexit "aborted"
	;;
    esac
  }
  break
done

# prepare changes file
#
TMPFILE=$(mktemp)
trap " [ -f \"$TMPFILE\" ] && /bin/rm -f -- \"$TMPFILE\" " 0 1 2 3 13 15
{ newchangesentry; cat $CHANGESFILE; } >$TMPFILE

RES=e
while [ "$RES" == "e" ]; do
  Edit $TMPFILE
  echo
  awk '{print}/^----------/{n=n+1; if ( n == 2 ) exit 0; }' $TMPFILE
  read -n 1 -p "$(Gecho "(a)bort, (c)ontinue, (s)ubmitt, (e)dit : ")" RES
  echo
  case "$RES" in
    [eE]*)
      RES=e
      ;;
    [cCsS])
      Becho "!!! Store new $CHANGESFILE"
      mv $TMPFILE $CHANGESFILE
      chmod 644 $CHANGESFILE

      test "$RES" == "s" && {
	if [ "$LAST_RELEASE" == "$THIS_RELEASE" ]; then
	  git add "$CHANGESFILE" && git commit -m "changes"
	else
	  Becho "!!! Remember new version $THIS_RELEASE in $VERSIONFILE"
	  sed -i "s/^# LAST RELEASED:.*$/# LAST RELEASED: $THIS_RELEASE/" $VERSIONFILE
	  if git add "$CHANGESFILE" "$VERSIONFILE" \
		&& git commit -m "changes $THIS_RELEASE" \
		  && git tag -m "tagging $THIS_RELEASE" "$THIS_RELEASE" HEAD; then
	    Becho "!!!"
	    Becho "!!! Do not forget to push the commit and the tag: $(Gecho git push --tags origin HEAD)"
	    Becho "!!!"
	  else
	    Recho "!!!"
	    Recho "!!! Commit failed. Check manually. (git reset HEAD~)"
	    Recho "!!!"
	    exit 9
	  fi
	fi
      }
      ;;
    *)
      Becho "!!! Leave $CHANGESFILE untouched"
      ;;
  esac
done
