574 lines
16 KiB
Bash
Executable file
574 lines
16 KiB
Bash
Executable file
#!/bin/sh
|
|
#
|
|
# 2011-2022 Nico Schottelius (nico-cdist at schottelius.org)
|
|
# 2016-2019 Darko Poljak (darko.poljak at gmail.com)
|
|
#
|
|
# This file is part of cdist.
|
|
#
|
|
# cdist is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# cdist is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with cdist. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
#
|
|
# This file contains the heavy lifting found usually in the Makefile.
|
|
#
|
|
|
|
usage() {
|
|
printf "usage: %s TARGET [TARGET-ARGS...]
|
|
Available targets:
|
|
changelog-changes
|
|
changelog-version
|
|
check-date
|
|
check-unittest
|
|
ml-release
|
|
archlinux-release
|
|
pypi-release
|
|
release-git-tag
|
|
sign-git-release
|
|
release
|
|
test
|
|
test-remote
|
|
pycodestyle
|
|
pep8
|
|
check-pycodestyle
|
|
shellcheck-global-explorers
|
|
shellcheck-type-explorers
|
|
shellcheck-manifests
|
|
shellcheck-local-gencodes
|
|
shellcheck-remote-gencodes
|
|
shellcheck-bin
|
|
shellcheck-gencodes
|
|
shellcheck-types
|
|
shellcheck
|
|
shellcheck-type-files
|
|
shellcheck-with-files
|
|
shellcheck-build-helper
|
|
check-shellcheck
|
|
version-branch
|
|
version
|
|
target-version
|
|
clean
|
|
distclean\n" "$1"
|
|
}
|
|
|
|
basename="${0##*/}"
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
usage "${basename}"
|
|
exit 1
|
|
fi
|
|
|
|
option=$1; shift
|
|
|
|
SHELLCHECKCMD="shellcheck -s sh -f gcc -x"
|
|
# Skip SC2154 for variables starting with __ since such variables are cdist
|
|
# environment variables.
|
|
SHELLCHECK_SKIP=': __.*is referenced but not assigned.*\[SC2154\]'
|
|
SHELLCHECKTMP=".shellcheck.tmp"
|
|
|
|
# Change to checkout directory
|
|
basedir="${0%/*}/../"
|
|
cd "$basedir"
|
|
|
|
case "$option" in
|
|
changelog-changes)
|
|
if [ "$#" -eq 1 ]; then
|
|
start=$1
|
|
else
|
|
start="[[:digit:]]"
|
|
fi
|
|
|
|
end="[[:digit:]]"
|
|
|
|
awk -F: "BEGIN { start=0 }
|
|
{
|
|
if(start == 0) {
|
|
if (\$0 ~ /^$start/) {
|
|
start = 1
|
|
}
|
|
} else {
|
|
if (\$0 ~ /^$end/) {
|
|
exit
|
|
} else {
|
|
print \$0
|
|
}
|
|
}
|
|
}" "$basedir/docs/changelog"
|
|
;;
|
|
|
|
changelog-version)
|
|
# get version from changelog
|
|
grep '^[[:digit:]]' "$basedir/docs/changelog" | head -n1 | sed 's/:.*//'
|
|
;;
|
|
|
|
check-date)
|
|
# verify date in changelog is today
|
|
date_today="$(date +%Y-%m-%d)"
|
|
date_changelog=$(grep '^[[:digit:]]' "$basedir/docs/changelog" | head -n1 | sed 's/.*: //')
|
|
|
|
if [ "$date_today" != "$date_changelog" ]; then
|
|
printf "Date in changelog is not today\n"
|
|
printf "Changelog date: %s\n" "${date_changelog}"
|
|
exit 1
|
|
fi
|
|
;;
|
|
|
|
check-unittest)
|
|
"$0" test
|
|
;;
|
|
|
|
ml-release)
|
|
if [ $# -ne 1 ]; then
|
|
printf "%s ml-release version\n" "$0" >&2
|
|
exit 1
|
|
fi
|
|
|
|
version=$1; shift
|
|
|
|
(
|
|
cat << eof
|
|
Subject: cdist $version has been released
|
|
|
|
Hello .*,
|
|
|
|
cdist $version has been released with the following changes:
|
|
|
|
eof
|
|
|
|
"$0" changelog-changes "$version"
|
|
cat << eof
|
|
|
|
eof
|
|
) > mailinglist.tmp
|
|
;;
|
|
|
|
archlinux-release)
|
|
if [ $# -ne 1 ]; then
|
|
printf "%s archlinux-release version\n" "$0" >&2
|
|
exit 1
|
|
fi
|
|
version=$1; shift
|
|
|
|
ARCHLINUXTAR="cdist-${version}-1.src.tar.gz"
|
|
./PKGBUILD.in "${version}"
|
|
umask 022
|
|
mkaurball
|
|
burp -c system "${ARCHLINUXTAR}"
|
|
;;
|
|
|
|
pypi-release)
|
|
# Ensure that pypi release has the right version
|
|
"$0" version
|
|
|
|
make docs-clean
|
|
make docs
|
|
python3 setup.py sdist upload
|
|
;;
|
|
|
|
release-git-tag)
|
|
target_version=$($0 changelog-version)
|
|
if git rev-parse --verify "refs/tags/${target_version}" 2>/dev/null; then
|
|
printf "Tag for %s exists, aborting\n" "${target_version}"
|
|
exit 1
|
|
fi
|
|
printf "Enter tag description for %s: " "${target_version}"
|
|
read -r tagmessage
|
|
|
|
# setup for signed tags:
|
|
# gpg --fulL-gen-key
|
|
# gpg --list-secret-keys --keyid-format LONG
|
|
# git config --local user.signingkey <id>
|
|
# for exporting pub key:
|
|
# gpg --armor --export <id> > pubkey.asc
|
|
# gpg --output pubkey.gpg --export <id>
|
|
# show tag with signature
|
|
# git show <tag>
|
|
# verify tag signature
|
|
# git tag -v <tag>
|
|
#
|
|
# gpg verify signature
|
|
# gpg --verify <asc-file> <file>
|
|
# gpg --no-default-keyring --keyring <pubkey.gpg> --verify <asc-file> <file>
|
|
# Ensure gpg-agent is running.
|
|
GPG_TTY=$(tty)
|
|
export GPG_TTY
|
|
gpg-agent
|
|
|
|
git tag -s "$target_version" -m "$tagmessage"
|
|
git push --tags
|
|
;;
|
|
|
|
sign-git-release)
|
|
if [ $# -lt 2 ]
|
|
then
|
|
printf "usage: %s sign-git-release TAG TOKEN [ARCHIVE]\n" "$0"
|
|
printf " if ARCHIVE is not specified then it is created\n"
|
|
exit 1
|
|
fi
|
|
tag="$1"
|
|
if ! git rev-parse -q --verify "${tag}" >/dev/null 2>&1
|
|
then
|
|
printf "Tag \"%s\" not found.\n" "${tag}"
|
|
exit 1
|
|
fi
|
|
token="$2"
|
|
if [ $# -gt 2 ]
|
|
then
|
|
archivename="$3"
|
|
else
|
|
archivename="cdist-${tag}.tar"
|
|
git archive --prefix="cdist-${tag}/" -o "${archivename}" "${tag}" \
|
|
|| exit 1
|
|
# make sure target version is generated
|
|
"$0" target-version
|
|
tar -x -f "${archivename}" || exit 1
|
|
cp cdist/version.py "cdist-${tag}/cdist/version.py" || exit 1
|
|
tar -c -f "${archivename}" "cdist-${tag}/" || exit 1
|
|
rm -r -f "cdist-${tag}/"
|
|
gzip "${archivename}" || exit 1
|
|
archivename="${archivename}.gz"
|
|
fi
|
|
gpg --armor --detach-sign "${archivename}" || exit 1
|
|
|
|
project="ungleich-public%2Fcdist"
|
|
sed_cmd='s/^.*"markdown":"\([^"]*\)".*$/\1/'
|
|
|
|
# upload archive
|
|
response_archive=$(curl -f -X POST \
|
|
--http1.1 \
|
|
-H "PRIVATE-TOKEN: ${token}" \
|
|
-F "file=@${archivename}" \
|
|
"https://code.ungleich.ch/api/v4/projects/${project}/uploads" \
|
|
| sed "${sed_cmd}") || exit 1
|
|
|
|
# upload archive signature
|
|
response_archive_sig=$(curl -f -X POST \
|
|
--http1.1 \
|
|
-H "PRIVATE-TOKEN: ${token}" \
|
|
-F "file=@${archivename}.asc" \
|
|
"https://code.ungleich.ch/api/v4/projects/${project}/uploads" \
|
|
| sed "${sed_cmd}") || exit 1
|
|
|
|
# make release
|
|
changelog=$("$0" changelog-changes "$1" | sed 's/^[[:space:]]*//')
|
|
release_notes=$(
|
|
printf "%s\n\n%s\n\n**Changelog**\n\n%s\n" \
|
|
"${response_archive}" "${response_archive_sig}" "${changelog}"
|
|
)
|
|
curl -f -X POST \
|
|
-H "PRIVATE-TOKEN: ${token}" \
|
|
-F "description=${release_notes}" \
|
|
"https://code.ungleich.ch/api/v4/projects/${project}/repository/tags/${tag}/release" \
|
|
|| exit 1
|
|
|
|
# remove generated files (archive and asc)
|
|
if [ $# -eq 2 ]
|
|
then
|
|
rm -f "${archivename}"
|
|
fi
|
|
rm -f "${archivename}.asc"
|
|
;;
|
|
|
|
release)
|
|
set -e
|
|
target_version=$($0 changelog-version)
|
|
target_branch=$($0 version-branch)
|
|
|
|
printf "Beginning release process for %s\n" "${target_version}"
|
|
|
|
# First check everything is sane
|
|
"$0" check-date
|
|
"$0" check-unittest
|
|
"$0" check-pycodestyle
|
|
"$0" check-shellcheck
|
|
|
|
# Generate version file to be included in packaging
|
|
"$0" target-version
|
|
|
|
# Ensure the git status is clean, else abort
|
|
if ! git diff-index --name-only --exit-code HEAD ; then
|
|
printf "Unclean tree, see files above, aborting.\n"
|
|
exit 1
|
|
fi
|
|
|
|
# Ensure we are on the master branch
|
|
masterbranch=yes
|
|
if [ "$(git rev-parse --abbrev-ref HEAD)" != "master" ]; then
|
|
printf "Releases are happening from the master branch, aborting.\n"
|
|
|
|
printf "Enter the magic word to release anyway:"
|
|
read -r magicword
|
|
|
|
if [ "$magicword" = "iknowwhatido" ]; then
|
|
masterbranch=no
|
|
else
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
if [ "$masterbranch" = yes ]; then
|
|
# Ensure version branch exists
|
|
if ! git rev-parse --verify "refs/heads/${target_branch}" 2>/dev/null; then
|
|
git branch "$target_branch"
|
|
fi
|
|
|
|
# Merge master branch into version branch
|
|
git checkout "$target_branch"
|
|
git merge master
|
|
fi
|
|
|
|
# Verify that after the merge everything works
|
|
"$0" check-date
|
|
"$0" check-unittest
|
|
|
|
# Generate documentation (man and html)
|
|
# First, clean old generated docs
|
|
make docs-clean
|
|
make docs
|
|
|
|
#############################################################
|
|
# Everything green, let's do the release
|
|
|
|
# Tag the current commit
|
|
"$0" release-git-tag
|
|
|
|
# Also merge back the version branch
|
|
if [ "$masterbranch" = yes ]; then
|
|
git checkout master
|
|
git merge "$target_branch"
|
|
fi
|
|
|
|
# Publish git changes
|
|
# if you want to have mirror locally then uncomment this and comment below
|
|
# git push --mirror
|
|
git push
|
|
# push also new branch and set up tracking
|
|
git push -u origin "${target_branch}"
|
|
# fi
|
|
|
|
# Create and publish package for pypi
|
|
"$0" pypi-release
|
|
|
|
# sign git tag
|
|
printf "Enter upstream repository authentication token: "
|
|
read -r token
|
|
"$0" sign-git-release "${target_version}" "${token}"
|
|
|
|
# Announce change on ML
|
|
"$0" ml-release "${target_version}"
|
|
|
|
cat << eof
|
|
Manual steps post release:
|
|
- cdist-web
|
|
- send generated mailinglist.tmp mail
|
|
eof
|
|
;;
|
|
|
|
test)
|
|
if [ ! -f "cdist/version.py" ]
|
|
then
|
|
printf "cdist/version.py is missing, generate it first.\n"
|
|
exit 1
|
|
fi
|
|
|
|
PYTHONPATH="$(pwd -P)"
|
|
export PYTHONPATH
|
|
|
|
if [ $# -lt 1 ]; then
|
|
python3 -m cdist.test
|
|
else
|
|
python3 -m unittest "$@"
|
|
fi
|
|
;;
|
|
|
|
test-remote)
|
|
if [ ! -f "cdist/version.py" ]
|
|
then
|
|
printf "cdist/version.py is missing, generate it first.\n"
|
|
exit 1
|
|
fi
|
|
|
|
PYTHONPATH="$(pwd -P)"
|
|
export PYTHONPATH
|
|
|
|
python3 -m cdist.test.exec.remote
|
|
;;
|
|
|
|
pycodestyle|pep8)
|
|
pycodestyle "${basedir}" "${basedir}/bin/cdist"
|
|
;;
|
|
|
|
check-pycodestyle)
|
|
"$0" pycodestyle
|
|
printf "\\nPlease review pycodestyle report.\\n"
|
|
while true
|
|
do
|
|
printf "Continue (yes/no)?\n"
|
|
any=
|
|
read -r any
|
|
case "$any" in
|
|
yes)
|
|
break
|
|
;;
|
|
no)
|
|
exit 1
|
|
;;
|
|
*)
|
|
printf "Please answer with 'yes' or 'no' explicitly.\n"
|
|
;;
|
|
esac
|
|
done
|
|
;;
|
|
|
|
shellcheck-global-explorers)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/explorer -type f -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-type-explorers)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/type -type f -path "*/explorer/*" -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-manifests)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/type -type f -name manifest -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-local-gencodes)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/type -type f -name gencode-local -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-remote-gencodes)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/type -type f -name gencode-remote -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
# NOTE: shellcheck-scripts is kept for compatibility
|
|
shellcheck-bin|shellcheck-scripts)
|
|
# shellcheck disable=SC2086
|
|
${SHELLCHECKCMD} bin/cdist-dump bin/cdist-new-type > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-gencodes)
|
|
errors=false
|
|
"$0" shellcheck-local-gencodes || errors=true
|
|
"$0" shellcheck-remote-gencodes || errors=true
|
|
! $errors || exit 1
|
|
;;
|
|
|
|
shellcheck-types)
|
|
errors=false
|
|
"$0" shellcheck-type-explorers || errors=true
|
|
"$0" shellcheck-manifests || errors=true
|
|
"$0" shellcheck-gencodes || errors=true
|
|
! $errors || exit 1
|
|
;;
|
|
|
|
shellcheck)
|
|
errors=false
|
|
"$0" shellcheck-global-explorers || errors=true
|
|
"$0" shellcheck-types || errors=true
|
|
"$0" shellcheck-bin || errors=true
|
|
! $errors || exit 1
|
|
;;
|
|
|
|
shellcheck-type-files)
|
|
# shellcheck disable=SC2086
|
|
find cdist/conf/type -type f -path "*/files/*" -exec ${SHELLCHECKCMD} {} + | grep -v "${SHELLCHECK_SKIP}" > "${SHELLCHECKTMP}"
|
|
test ! -s "${SHELLCHECKTMP}" || { cat "${SHELLCHECKTMP}"; exit 1; }
|
|
;;
|
|
|
|
shellcheck-with-files)
|
|
errors=false
|
|
"$0" shellcheck || errors=true
|
|
"$0" shellcheck-type-files || errors=true
|
|
! $errors || exit 1
|
|
;;
|
|
|
|
shellcheck-build-helper)
|
|
${SHELLCHECKCMD} ./bin/cdist-build-helper
|
|
;;
|
|
|
|
check-shellcheck)
|
|
"$0" shellcheck
|
|
printf "\\nPlease review shellcheck report.\\n"
|
|
while true
|
|
do
|
|
printf "Continue (yes/no)?\n"
|
|
any=
|
|
read -r any
|
|
case "$any" in
|
|
yes)
|
|
break
|
|
;;
|
|
no)
|
|
exit 1
|
|
;;
|
|
*)
|
|
printf "Please answer with 'yes' or 'no' explicitly.\n"
|
|
;;
|
|
esac
|
|
done
|
|
;;
|
|
|
|
version-branch)
|
|
"$0" changelog-version | cut -d. -f '1,2'
|
|
;;
|
|
|
|
version)
|
|
printf "VERSION = \"%s\"\n" "$(git describe)" > cdist/version.py
|
|
;;
|
|
|
|
target-version)
|
|
target_version=$($0 changelog-version)
|
|
printf "VERSION = \"%s\"\n" "${target_version}" > cdist/version.py
|
|
;;
|
|
|
|
clean)
|
|
make clean
|
|
|
|
# Archlinux
|
|
rm -f cdist-*.pkg.tar.xz cdist-*.tar.gz
|
|
rm -rf pkg/ src/
|
|
|
|
rm -f MANIFEST PKGBUILD
|
|
rm -rf dist/
|
|
|
|
# Signed release
|
|
rm -f cdist-*.tar.gz
|
|
rm -f cdist-*.tar.gz.asc
|
|
|
|
# Temp files
|
|
rm -f ./*.tmp
|
|
rm -f ./.*.tmp
|
|
;;
|
|
|
|
distclean)
|
|
"$0" clean
|
|
rm -f cdist/version.py
|
|
;;
|
|
*)
|
|
printf "Unknown target: '%s'.\n" "${option}" >&2
|
|
usage "${basename}"
|
|
exit 1
|
|
;;
|
|
|
|
esac
|