blob: 46ad2434c654ecc63e0fdf39356f821c4b0fe13c [file] [log] [blame]
#!/usr/bin/env bash
# -----------------------------------------------------------------------
# Copyright 2018-2023 Open Networking Foundation (ONF) and the ONF Contributors
#
# Licensed 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.
#
# github-release.sh
# builds (with make) and uploads release artifacts (binaries, etc.) to github
# given a tag also create checksums files and release notes from the commit
# message
# -----------------------------------------------------------------------
# 1) Staging to replace command github-release with gh.
# 2) gh auth / GIT_TOKEN handling may be needed
# -----------------------------------------------------------------------
# gh release create abc/1.2.3-rc2 --discussion-category "Announcements" --generate-notes hello.txt
# https://github.com/cli/cli/issues/4993
# -----------------------------------------------------------------------
##-------------------##
##---] GLOBALS [---##
##-------------------##
declare -g ARGV="$*" # archive for display
declare -g SCRIPT_VERSION='1.0' # git changeset needed
declare -g TRACE=0 # uncomment to set -x
declare -g VERBOSE=0 # trace github-release calls
declare -g RELEASE_TEMP
##--------------------##
##---] INCLUDES [---##
##--------------------#
# declare -g pgmdir="${0%/*}" # dirname($script)
echo "** ${0}: PWD=$(/bin/pwd)"
# -----------------------------------------------------------------------
# Jenkins must have checked out/copied the script -vs- repository
# -----------------------------------------------------------------------
# 17:56:27 [github-release_voltctl] $ /usr/bin/env bash /tmp/jenkins1043949650153731384.sh
# 17:56:27 ** /tmp/jenkins1043949650153731384.sh: PWD=/w/workspace/github-release_voltctl
# 17:56:27 5120009 4 drwxrwxr-x 4 jenkins jenkins 4096 Jan 25 22:56 .
# 17:56:27 5120010 4 drwxrwxr-x 9 jenkins jenkins 4096 Jan 25 22:56 ./voltctl
# 17:56:27 5120036 4 drwxrwxr-x 2 jenkins jenkins 4096 Jan 25 22:56 ./voltctl@tmp
# 17:56:27 /tmp/jenkins1043949650153731384.sh: line 44: /tmp/common/common.sh: No such file or directory
# -----------------------------------------------------------------------
# declare -a common_args=()
# common_args+=('--common-args-begin--')
# common_args+=('--traputils')
# common_args+=('--stacktrace')
# common_args+=('--tempdir')
# shellcheck disable=SC1091
# source "${pgmdir}/common/common.sh" "${common_args[@]}"
## -----------------------------------------------------------------------
## Intent: Output a log banner to identify the running script/version.
## -----------------------------------------------------------------------
## TODO:
## o SCRIPT_VERSION => git changeset for repo:ci-managment
## o echo "library version: ${env."library.libName.version"}"
# -----------------------------------------------------------------------
# 14:18:38 > git fetch --no-tags --progress -- https://gerrit.opencord.org/ci-management.git +refs/heads/*:refs/remotes/origin/* # timeout=10
# 14:18:39 Checking out Revision 50f6e0b97f449b32d32ec0e02d59642000351847 (master)
# -----------------------------------------------------------------------
function banner()
{
local iam="${0##*/}"
cat <<EOH
** -----------------------------------------------------------------------
** IAM: ${iam} :: ${FUNCNAME[0]}
** ARGV: ${ARGV}
** PWD: $(/bin/pwd)
** NOW: $(date '+%Y/%m/%d %H:%M:%S')
** VER: ${SCRIPT_VERSION:-'unknown'}
** -----------------------------------------------------------------------
EOH
return
}
## -----------------------------------------------------------------------
## Intent:
## Display available command versions
## A github release job is failing due to a command being mia.
## -----------------------------------------------------------------------
function displayCommands()
{
# which git || /bin/true
# git --version || /bin/true
# go version || /bin/true
# helm version || /bin/true
return
}
## -----------------------------------------------------------------------
## Intent: Display filesystem with a banner for logging
## -----------------------------------------------------------------------
function displayPwd()
{
local iam="${0##*/}"
echo "** ${iam}: ENTER"
cat <<EOM
** -----------------------------------------------------------------------
** IAM: $iam :: ${FUNCNAME[0]}
** PWD: $(/bin/pwd)
** -----------------------------------------------------------------------
EOM
find . -maxdepth 1 -ls
echo "** ${iam}: LEAVE"
return
}
## -----------------------------------------------------------------------
## Intent:
## -----------------------------------------------------------------------
function doDebug()
{
echo "** ${FUNCNAME[0]}: ENTER"
echo
echo "** PWD: $(/bin/pwd)"
echo "** make-pre: $(/bin/ls -l)"
echo
declare -p ARTIFACT_GLOB
declare -p RELEASE_TEMP
echo
echo "** ${FUNCNAME[0]}: ARTIFACT_GLOB=${ARTIFACT_GLOB}"
local artifact_glob="${ARTIFACT_GLOB%/*}"
declare -p artifact_glob
echo "** ${FUNCNAME[0]}: PWD=$(/bin/pwd)"
find "$artifact_glob" -print || /bin/true
copyToRelease
# Copy artifacts into the release temp dir
# shellcheck disable=SC2086
# cp -v "$ARTIFACT_GLOB" "$RELEASE_TEMP"
# echo "rsync -rv --checksum \"$artifact_glob/.\" \"$RELEASE_TEMP/.\""
# rsync -rv --checksum "$artifact_glob/." "$RELEASE_TEMP/."
echo "** ${FUNCNAME[0]}: LEAVE"
echo
return
}
## -----------------------------------------------------------------------
## Intent:
## -----------------------------------------------------------------------
function copyToRelease()
{
local iam="${FUNCNAME[0]}"
echo "** ${iam}: ENTER"
local artifact_glob="${ARTIFACT_GLOB%/*}"
echo "** ${iam}: $(declare -p ARTIFACT_GLOB) $(declare -p artifact_glob)"
# Copy artifacts into the release temp dir
# shellcheck disable=SC2086
# cp -v "$ARTIFACT_GLOB" "$RELEASE_TEMP"
echo "rsync -rv --checksum \"$artifact_glob/.\" \"$RELEASE_TEMP/.\""
rsync -rv --checksum "$artifact_glob/." "$RELEASE_TEMP/."
echo "** ${iam}: RELEASE_TEMP=${RELEASE_TEMP}"
find "$RELEASE_TEMP" -print
echo "** ${iam}: LEAVE"
echo
return
}
## -----------------------------------------------------------------------
## Intent:
## -----------------------------------------------------------------------
function github_release_pre()
{
local what="$1" ; shift
local user="$1" ; shift
local repo="$1" ; shift
local tag="$1" ; shift
local name="$1" ; shift
local descr="$1" ; shift
local iam="${FUNCNAME[0]}"
echo "** ${iam}: ENTER"
case "$what" in
gh)
declare -a cmd=()
cmd+=('gh')
# cmd+=('--verbose')
cmd+=('release')
cmd+=('create')
cmd+=('--user' "$user")
cmd+=('--repo' "$repo")
cmd+=('--title' "$name")
cmd+=('--descripton' "$descr")
cmd+=('--discussion-category' "Announcements")
# cmd+=('--latest') - auto based on date & ver
cmd+=('--verify-tag')
# --branch exists, omit switch for tag
cmd+=("$tag")
echo "** ${iam}: RUNNING " "${cmd[@]}"
"${cmd[@]}"
;;
*)
declare -a cmd=()
cmd+=('github-release')
# cmd+=('--verbose')
cmd+=('release')
cmd+=('--user' "$user")
cmd+=('--repo' "$repo")
cmd+=('--tag' "$tag")
cmd+=('--name' "$name")
cmd+=('--descripton' "$descr")
echo "** ${iam}: RUNNING " "${cmd[@]}"
"${cmd[@]}"
;;
esac
set +x
echo "** ${iam}: ENTER"
return
}
## -----------------------------------------------------------------------
## Intent:
## -----------------------------------------------------------------------
function github_release_view()
{
local what="$1" ; shift
local user="$1" ; shift
local repo="$1" ; shift
local tag="$1" ; shift
local name="$1" ; shift
local descr="$1" ; shift
local iam="${FUNCNAME[0]}"
echo "** ${iam}: ENTER"
case "$what" in
gh)
declare -a cmd=()
cmd+=('gh')
cmd+=('relase')
cmd+=('view')
cmd+=("$tag")
echo "** ${iam}: RUNNING " "${cmd[@]}"
"${cmd[@]}"
;;
*)
showReleaseInfo \
"$user"\
"$repo"\
"$tag"
;;
esac
echo "** ${iam}: LEAVE"
return
}
## -----------------------------------------------------------------------
## Intent:
## -----------------------------------------------------------------------
function showCommands()
{
local iam="${FUNCNAME[0]}"
echo "** ${iam}: ENTER"
declare -a cmds=()
cmds+=('github-release')
cmds+=('gh')
local cmd
for cmd in "${cmds[@]}";
do
echo
echo "Checking Command: $cmd"
echo "==========================================================================="
which -a "$cmd"
"$cmd" --version
done
echo "** ${iam}: LEAVE"
echo
return
}
## -----------------------------------------------------------------------
## Intent: Display info about a release stored on github.
## -----------------------------------------------------------------------
function showReleaseInfo()
{
local user="$1" ; shift
local repo="$1" ; shift
local tag=''
if [ $# -gt 0 ]; then
tag="$1" ; shift
fi
local iam="${FUNCNAME[0]}"
echo "** ${iam}: ENTER"
cat <<EOM
** -------------------------------------------------------------------
** ${iam}
** -------------------------------------------------------------------
EOM
declare -a args=()
[[ $VERBOSE -eq 1 ]] && args+=('--verbose')
args+=('info')
args+=('--user' "$user")
args+=('--repo' "$repo")
if [ ${#tag} -gt 0 ]; then
args+=('--tag' "$tag")
fi
echo "${iam}: github-relase info"
github-release "${args[@]}"
echo "** ${iam}: LEAVE"
return
}
##----------------##
##---] MAIN [---##
##----------------##
set -eu -o pipefail
iam="${0##*/}"
banner
# when not running under Jenkins, use current dir as workspace and a blank
# project name
WORKSPACE=${WORKSPACE:-.}
GERRIT_PROJECT=${GERRIT_PROJECT:-}
# Github organization (or user) this project is published on. Project name should
# be the same on both Gerrit and GitHub
GITHUB_ORGANIZATION=${GITHUB_ORGANIZATION:-}
# glob pattern relative to project dir matching release artifacts
# ARTIFACT_GLOB=${ARTIFACT_GLOB:-"release/*"} # stat -- release/* not found, literal string (?)
ARTIFACT_GLOB=${ARTIFACT_GLOB:-"release/."}
# Temporary staging directory to copy artifacts to
RELEASE_TEMP="$WORKSPACE/release"
mkdir -p "$RELEASE_TEMP"
# Use "release" as the default makefile target, can be a space separated list
RELEASE_TARGETS=${RELEASE_TARGETS:-release}
# check that we're on a semver released version, or exit
pushd "$GERRIT_PROJECT"
GIT_VERSION=$(git tag -l --points-at HEAD)
# match bare versions or v-prefixed golang style version
if [[ "$GIT_VERSION" =~ ^v?([0-9]+)\.([0-9]+)\.([0-9]+)$ ]]
then
echo "git has a SemVer released version tag: '$GIT_VERSION'"
echo "Building artifacts for GitHub release."
else
echo "No SemVer released version tag found, exiting..."
exit 0
fi
popd
# Set and handle GOPATH and PATH
export GOPATH=${GOPATH:-$WORKSPACE/go}
export PATH=$PATH:/usr/lib/go-1.12/bin:/usr/local/go/bin:$GOPATH/bin
# To support golang projects that require GOPATH to be set and code checked out there
# If $DEST_GOPATH is not an empty string:
# - create GOPATH within WORKSPACE, and destination directory within
# - set PATH to include $GOPATH/bin and the system go binaries
# - move project from $WORKSPACE/$GERRIT_PROJECT to new location in $GOPATH
# - start release process within that directory
DEST_GOPATH=${DEST_GOPATH:-}
if [ -n "$DEST_GOPATH" ]; then
mkdir -p "$GOPATH/src/$DEST_GOPATH"
release_path="$GOPATH/src/$DEST_GOPATH/$GERRIT_PROJECT"
mv "$WORKSPACE/$GERRIT_PROJECT" "$release_path"
else
release_path="$WORKSPACE/$GERRIT_PROJECT"
fi
if [ ! -f "$release_path/Makefile" ]; then
echo "Makefile not found at $release_path!"
exit 1
else
# shellcheck disable=SC2015
[[ -v TRACE ]] && { set -x; } || { set +x; } # SC2015 (shellcheck -x)
pushd "$release_path"
# Release description is sanitized version of the log message
RELEASE_DESCRIPTION="$(git log -1 --pretty=%B | tr -dc "[:alnum:]\n\r\.\[\]\:\-\\\/\`\' ")"
# build the release, can be multiple space separated targets
# shellcheck disable=SC2086
make "$RELEASE_TARGETS"
# doDebug # deterine why ARTIFACT_GLOB is empty
copyToRelease
# Are we failing on a literal string "release/*" ?
# cp -v "$ARTIFACT_GLOB" "$RELEASE_TEMP"
# echo "rsync -rv --checksum \"$artifact_glob/.\" \"$RELEASE_TEMP/.\""
# rsync -rv --checksum "$artifact_glob/." "$RELEASE_TEMP/."
showReleaseInfo \
"$GITHUB_ORGANIZATION"\
"$GERRIT_PROJECT"
showCommands
cat <<EOM
** -----------------------------------------------------------------------
** Create the release:
** 1) Create initial github release with download area.
** 2) Generate checksum.SHA256 for all released files.
** 3) Upload files to complete the release.
** 4) Display released info from github.
** -----------------------------------------------------------------------
EOM
# git auth login
# git auth logout
# Usage: github-release [global options] <verb> [verb options]
# create release
echo "** ${iam} Creating Release: $GERRIT_PROJECT - $GIT_VERSION"
github_release_pre 'gh'\
"$GITHUB_ORGANIZATION"\
"$GERRIT_PROJECT"\
"$GIT_VERSION"\
"$GERRIT_PROJECT - $GIT_VERSION"\
"$RELEASE_DESCRIPTION"
echo "** ${iam} Packaging release files"
pushd "$RELEASE_TEMP"
echo "** ${iam}: Files to release:"
readarray -t to_release < <(find . -mindepth 1 -maxdepth 1 -type f -print)
declare -p to_release
# Generate and check checksums
sha256sum -- * > checksum.SHA256
sha256sum -c < checksum.SHA256
echo
echo "** ${iam} Checksums(checksum.SHA256):"
cat checksum.SHA256
echo
echo "** ${iam} Upload files being released"
what='gh'
case "$what" in
gh)
gh release upload "$GIT_VERSION" "${to_release[@]}"
;;
*)
# upload all files to the release
for rel_file in *
do
echo "** ${iam} Uploading file: $rel_file"
github-release \
upload \
--user "$GITHUB_ORGANIZATION" \
--repo "$GERRIT_PROJECT" \
--tag "$GIT_VERSION" \
--name "$rel_file" \
--file "$rel_file"
done
;;
esac
popd
popd
echo "** ${iam} Display released info"
github_release_view 'gh'\
"$GITHUB_ORGANIZATION"\
"$GERRIT_PROJECT"\
"$GIT_VERSION"\
"$GERRIT_PROJECT - $GIT_VERSION"\
"$RELEASE_DESCRIPTION"
fi
# [SEE ALSO]
# -----------------------------------------------------------------------
# https://www.shellcheck.net/wiki/SC2236
# https://docs.github.com/en/rest/releases/releases?apiVersion=2022-11-28#create-a-release
# -----------------------------------------------------------------------
# https://cli.github.com/manual/gh_help_reference
# https://cli.github.com/manual/gh_release
# -----------------------------------------------------------------------
# [EOF]