blob: adac0c59f7c94e1fcd9207be2800e7f0ea0ccfda [file] [log] [blame]
x#!/usr/bin/env bash
#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
#
#
# The veralign script sets the appropriate versions in all of
# the package configuration files for all of the supported
# languages. It is used to prepare a release or move master
# forward to the next anticipated version.
#
# USAGE
# -----------------------------------------------------------
# usage: veralign.sh <oldVersion> <newVersion>
#
# EXAMPLE
# -----------------------------------------------------------
# $ ./veralign.sh 0.12.0 1.0.0
# $ ./veralign.sh 1.0.0 1.1.0
#
# IMPORTANT USAGE NOTE
# -----------------------------------------------------------
# Define the environment variable DRYRUN to have the script
# print out all matches to the oldVersion hilighted so that
# you can verify it will change the right things.
#
declare -A FILES
# These files require a manual touch:
FILES[CHANGES.md]=manual
FILES[debian/changelog]=manual
FILES[doap.rdf]=manual
# These files can be updated automatically:
FILES[ApacheThrift.nuspec]=simpleReplace
FILES[appveyor.yml]=simpleReplace
FILES[bower.json]=jsonReplace
FILES[CMakeLists.txt]=simpleReplace
FILES[compiler/cpp/src/thrift/version.h]=simpleReplace
FILES[configure.ac]=configureReplace
FILES[contrib/Rebus/Properties/AssemblyInfo.cs]=simpleReplace
FILES[contrib/thrift.spec]=simpleReplace
FILES[contrib/zeromq/csharp/AssemblyInfo.cs]=simpleReplace
FILES[contrib/thrift-maven-plugin/pom.xml]=pomReplace
FILES[doc/specs/idl.md]=simpleReplace
FILES[lib/d/src/thrift/base.d]=simpleReplace
FILES[lib/dart/pubspec.yaml]=pubspecReplace
FILES[lib/delphi/src/Thrift.pas]=simpleReplace
FILES[lib/erl/src/thrift.app.src]=simpleReplace
FILES[lib/haxe/haxelib.json]=simpleReplace
FILES[lib/java/gradle.properties]=simpleReplace
FILES[lib/js/package-lock.json]=jsonReplace
FILES[lib/js/package.json]=jsonReplace
FILES[lib/js/src/thrift.js]=simpleReplace
FILES[lib/lua/Thrift.lua]=simpleReplace
FILES[lib/netstd/Tests/Thrift.Tests/Thrift.Tests.csproj]=simpleReplace
FILES[lib/netstd/Tests/Thrift.PublicInterfaces.Compile.Tests/Thrift.PublicInterfaces.Compile.Tests.csproj]=simpleReplace
FILES[lib/netstd/Tests/Thrift.IntegrationTests/Thrift.IntegrationTests.csproj]=simpleReplace
FILES[lib/netstd/Thrift/Properties/AssemblyInfo.cs]=simpleReplace
FILES[lib/netstd/Thrift/Thrift.csproj]=simpleReplace
FILES[lib/ocaml/_oasis]=simpleReplace
FILES[lib/perl/lib/Thrift.pm]=simpleReplace
FILES[lib/py/setup.py]=simpleReplace
FILES[lib/rb/thrift.gemspec]=simpleReplace
FILES[lib/rs/Cargo.toml]=simpleReplace
FILES[lib/st/package.xml]=simpleReplace
FILES[lib/swift/Sources/Thrift.swift]=simpleReplace
FILES[lib/swift/Tests/ThriftTests/ThriftTests.swift]=simpleReplace
FILES[lib/ts/package-lock.json]=jsonReplace
FILES[lib/ts/package.json]=jsonReplace
FILES[package-lock.json]=jsonReplace
FILES[package.json]=jsonReplace
FILES[sonar-project.properties]=simpleReplace
FILES[test/dart/test_client/pubspec.yaml]=pubspecReplace
FILES[test/erl/src/thrift_test.app.src]=simpleReplace
FILES[test/netstd/Client/Client.csproj]=simpleReplace
FILES[test/netstd/Server/Server.csproj]=simpleReplace
FILES[Thrift.podspec]=simpleReplace
FILES[tutorial/dart/client/pubspec.yaml]=pubspecReplace
FILES[tutorial/dart/console_client/pubspec.yaml]=pubspecReplace
FILES[tutorial/dart/server/pubspec.yaml]=pubspecReplace
FILES[tutorial/delphi/DelphiClient/DelphiClient.dproj]=simpleReplace
FILES[tutorial/delphi/DelphiServer/DelphiServer.dproj]=simpleReplace
FILES[tutorial/netstd/Client/Client.csproj]=simpleReplace
FILES[tutorial/netstd/Interfaces/Interfaces.csproj]=simpleReplace
FILES[tutorial/netstd/Server/Server.csproj]=simpleReplace
FILES[tutorial/ocaml/_oasis]=simpleReplace
if [ ! -f "CHANGES.md" ]; then
>&2 echo "error: run veralign.sh while in the thrift root directory"
exit 1
fi
if [ $# -ne 2 ]; then
>&2 echo "usage: veralign.sh <oldVersion> <newVersion>"
exit 1
fi
jq --version 1>/dev/null 2>/dev/null
if [ $? -ne 0 ]; then
>&2 echo "error: the 'jq' package is not installed"
exit 1
fi
#
# validateVersion: check that a version matches the major.minor.patch
# format which is the lowest common denominator supported by all
# project systems.
# \param $1 the version
# \returns 0 if the version is compliant
#
function validateVersion
{
local result
local valid
valid=$(echo "$1" | sed '/^[[:digit:]]\+\.[[:digit:]]\+\.[[:digit:]]\+$/!{q22}')
result=$?
if [ $result -eq 22 ]; then
>&2 echo "error: version '$1' does not conform to the required major.minor.patch format"
return ${result}
fi
}
OLDVERSION=$1
NEWVERSION=$2
validateVersion "${OLDVERSION}" || exit $?
validateVersion "${NEWVERSION}" || exit $?
#
# escapeVersion: escape the version for use as a sed search
# \param $1 the version to escape
# \output the escaped string
# \returns 0
# \example VERSEARCH=$(escapeVersion "[1.0.0]"); echo $VERSEARCH; => "\[1\.0\.0\]"
#
function escapeVersion
{
echo "$(echo "$1" | sed 's/\./\\./g' | sed 's/\[/\\\[/g' | sed 's/\]/\\\]/g')"
}
# Set up verbose hilighting if running interactive
if [ "$(tput colors)" -ne 0 ]; then
reverse=$(tput rev)
red=$(tput setaf 1)
green=$(tput setaf 2)
yellow=$(tput setaf 3)
normal=$(tput sgr0)
fi
declare -A MANUAL
#
# manual: note that update of said file is manual
# \param $1 filename to do replacements on
# \returns 0
#
function manual
{
MANUAL["$1"]=""
return 0
}
#
# configureReplace: replace the AC_INIT field in configure.ac
# \param $1 filename to do replacements on
# \returns 0 on success
#
function configureReplace
{
replace "$1" "[thrift], [${OLDVERSION}]" "[thrift], [${NEWVERSION}]"
}
#
# jsonReplace: replace a specific version field in a JSON file
# must be a top level "version" field in the json structure
# \param $1 filename to do replacements on
# \returns 0 on success
#
function jsonReplace
{
local result
local output
if [ ! -z "$DRYRUN" ]; then
output=$(jq -e ".version" "$1")
else
output=$(jq -e ".version = \"${NEWVERSION}\"" "$1" > tmp.$$.json && mv tmp.$$.json "$1")
fi
result=$?
if [ $? -ne 0 ]; then
printf "%-60s | %5d | ${red}ERROR${normal}: version tag not found" "$1" "$count"
echo
return 1
elif [ ! -z "$DRYRUN" ]; then
output=${output%\"}
output=${output#\"}
printf "%-60s | %5d | MATCHES: version: \"${reverse}${green}${output}${normal}\"" "$1" 1
echo
return 0
fi
printf "%-60s | %5d | ${green}OK${normal}" "$1" 1
echo
return 0
}
#
# pubspecReplace: replace a specific version field in a YAML file
# must be a top level "version" field in the yaml structure
# did not find a package that preserves comments so this is
# somewhat brain-dead, but it gets the job done
# \param $1 filename to do replacements on
# \returns 0 on success
#
function pubspecReplace
{
replace "$1" "version: ${OLDVERSION}" "version: ${NEWVERSION}"
}
#
# pomReplace: replace a specific version field in a maven pom file
# must be a top level "version" field in the xml structure
# \param $1 filename to do replacements on
# \returns 0 on success
#
function pomReplace
{
replace "$1" "^ <version>${OLDVERSION}<\/version>" " <version>${NEWVERSION}<\/version>"
}
#
# replace: replace occurrences of one string with another
# the file specified must contain the old string at least once
# in order to be successful.
# \param $1 filename to do replacements on
# \param $2 the "old" string to be replaced
# \param $3 the "new" striing to replace it with
# \returns 0 on success
#
function replace
{
local result
local output
local oldString="$2"
local newString="$3"
local oldRegex=$(escapeVersion "${oldString}")
local count=$(grep -Ec "${oldRegex}" "$1")
local verbose
if [ $count -eq 0 ]; then
printf "%-60s | %5d | ${red}NOT FOUND${normal}: ${oldString}" "$1" 0
echo
return 1
elif [ ! -z "$DRYRUN" ]; then
printf "%-60s | %5d | MATCHES:" "$1" "$count"
echo
while read -r line; do
echo " > $(echo "$line" | sed "s/${oldRegex}/${reverse}${green}${oldString}${normal}/g")"
done < <(grep -E "${oldRegex}" "$1")
return 0
fi
output=$(sed -i "s/${oldRegex}/${newString}/g" "$1")
result=$?
if [ $result -ne 0 ]; then
printf "%-60s | %5d | ${red}ERROR${normal}: %s" "$1" "$count" "$output"
echo
return 1
fi
printf "%-60s | %5d | ${green}OK${normal}" "$1" "$count"
echo
return 0
}
#
# simpleReplace: replace occurrences of ${OLDVERSION} with ${NEWVERSION}
# the file specified must contain OLDVERSION at least once
# in order to be successful.
# \param $1 filename to do replacements on
# \param $2 the "old" string to be replaced
# \param $3 the "new" striing to replace it with
# \returns 0 on success
#
function simpleReplace
{
replace "$1" "${OLDVERSION}" "${NEWVERSION}"
}
echo ""
echo "Apache Thrift Version Alignment Tool"
echo "------------------------------------"
echo ""
echo "Previous Version: ${OLDVERSION}"
echo " New Version: ${NEWVERSION}"
echo ""
echo "-------------------------------------------------------------+-------+----------------------"
echo "Filename | Count | Status "
echo "-------------------------------------------------------------+-------+----------------------"
for file in $(echo "${!FILES[@]}" | sort); do
${FILES[$file]} $file || exit $?
done
echo
echo "Files that must be modified manually:"
echo
for manu in $(echo "${!MANUAL[@]}" | sort); do
echo " > ${yellow}${manu}${normal}"
done
exit 0